pre-inst/pre-inst.c
author J. Ali Harlow <ali@juiblex.co.uk>
Tue Jun 29 10:09:34 2021 +0100 (2021-06-29)
changeset 108 b3d8e196dac8
parent 97 55ae076f393c
child 109 2947214c450e
permissions -rw-r--r--
Added tag 0.6.1 for changeset 6ae203c8b28d
ali@24
     1
/*
ali@96
     2
 * Copyright (C) 2014, 2020  J. Ali Harlow <ali@juiblex.co.uk>
ali@24
     3
 *
ali@24
     4
 * This program is free software; you can redistribute it and/or modify
ali@24
     5
 * it under the terms of the GNU General Public License as published by
ali@24
     6
 * the Free Software Foundation; either version 2 of the License, or
ali@24
     7
 * (at your option) any later version.
ali@24
     8
 *
ali@24
     9
 * This program is distributed in the hope that it will be useful,
ali@24
    10
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
ali@24
    11
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
ali@24
    12
 * GNU General Public License for more details.
ali@24
    13
 *
ali@24
    14
 * You should have received a copy of the GNU General Public License along
ali@24
    15
 * with this program; if not, write to the Free Software Foundation, Inc.,
ali@24
    16
 * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
ali@24
    17
 *
ali@24
    18
 * References:
ali@24
    19
 *	http://www.transmissionzero.co.uk/computing/win32-apps-with-mingw/
ali@24
    20
 */
ali@24
    21
ali@24
    22
#include "config.h"
ali@24
    23
#ifndef WIN32
ali@24
    24
#define _XOPEN_SOURCE 500
ali@24
    25
#endif
ali@24
    26
#include <stdlib.h>
ali@24
    27
#include <stdio.h>
ali@24
    28
#include <string.h>
ali@97
    29
#include <errno.h>
ali@24
    30
#include <lua.h>
ali@24
    31
#include <razor.h>
ali@24
    32
#include <plover/plover.h>
ali@99
    33
#include <plover/uri-handler.h>
ali@24
    34
#include <whelk/whelk.h>
ali@24
    35
#ifdef WIN32
ali@24
    36
#include <windows.h>
ali@24
    37
#include <process.h>
ali@24
    38
#include <commctrl.h>
ali@24
    39
#include "resource.h"
ali@24
    40
ali@24
    41
#ifndef FOF_NO_UI
ali@24
    42
#define FOF_NO_UI	(FOF_SILENT|FOF_NOCONFIRMATION|FOF_NOERRORUI|\
ali@24
    43
			FOF_NOCONFIRMMKDIR)
ali@24
    44
#endif
ali@24
    45
ali@24
    46
#else	/* WIN32 */
ali@24
    47
#include <ftw.h>
ali@24
    48
#endif	/* WIN32 */
ali@96
    49
#include "post.h"
ali@24
    50
ali@24
    51
#ifdef WIN32
ali@24
    52
/* Under WIN32, g_spawn requires a helper program which we'd rather avoid */
ali@24
    53
#undef USE_G_SPAWN
ali@24
    54
#else
ali@24
    55
#define USE_G_SPAWN
ali@24
    56
#endif
ali@24
    57
ali@24
    58
LUALIB_API int luaopen_posix(lua_State *L);
ali@24
    59
ali@24
    60
#ifdef WIN32
ali@24
    61
DWORD main_thread_id;
ali@24
    62
#endif
ali@24
    63
ali@24
    64
gchar *prefix;
ali@24
    65
ali@24
    66
int verify_and_fix(const char *root)
ali@24
    67
{
ali@24
    68
    return 0;
ali@24
    69
}
ali@24
    70
ali@24
    71
#ifdef WIN32
ali@24
    72
INT_PTR CALLBACK ProgressDialogProc(HWND dialog,UINT msg,WPARAM w_param,
ali@24
    73
  LPARAM l_param)
ali@24
    74
{
ali@24
    75
    HWND progress;
ali@24
    76
    DWORD style;
ali@24
    77
    switch (msg)
ali@24
    78
    {
ali@24
    79
	case WM_INITDIALOG:
ali@24
    80
	    progress=GetDlgItem(dialog,IDC_PROGRESS); 
ali@24
    81
	    style=GetWindowLong(progress,GWL_STYLE);
ali@24
    82
	    SetWindowLong(progress,GWL_STYLE,style|PBS_MARQUEE);
ali@24
    83
	    SendMessage(progress,PBM_SETMARQUEE,(WPARAM)TRUE,(LPARAM)30);
ali@24
    84
	    return (INT_PTR)TRUE;
ali@24
    85
    }
ali@24
    86
    return (INT_PTR)FALSE;
ali@24
    87
}
ali@24
    88
#endif
ali@24
    89
ali@24
    90
#ifdef WIN32
ali@24
    91
__stdcall
ali@24
    92
#endif
ali@24
    93
unsigned pre_install_thread(void *data)
ali@24
    94
{
ali@97
    95
    gboolean changed;
ali@24
    96
    int retval;
ali@95
    97
    const char *repository=data;
ali@46
    98
    gchar *s,*uri;
ali@24
    99
    char *install[]={"plover-gtkui",NULL};
ali@97
   100
    char **pkgs=install;
ali@97
   101
    struct comps *comps;
ali@97
   102
    struct comps_group *group;
ali@97
   103
    struct comps_requirement *pkg;
ali@97
   104
    struct plover_vector *packages=NULL;
ali@24
   105
    GError *error=NULL;
ali@94
   106
    plover__uri_handler_init();
ali@24
   107
    prefix=plover_pre_install_prefix();
ali@24
   108
    s=g_strconcat(prefix,"/var/log/pre-install",NULL);
ali@24
   109
    plover_log_open(s);
ali@24
   110
    g_free(s);
ali@24
   111
    s=g_strconcat(prefix,"/var/lib/razor",NULL);
ali@46
   112
    uri=razor_path_to_uri(s);
ali@24
   113
    g_free(s);
ali@46
   114
    razor_set_database_uri(uri);
ali@46
   115
    free(uri);
ali@24
   116
    if (verify_and_fix(prefix))
ali@24
   117
    {
ali@24
   118
	g_free(prefix);
ali@24
   119
	return -1;
ali@24
   120
    }
ali@97
   121
    s=g_strconcat(repository,"/repodata/comps.xml",NULL);
ali@97
   122
    comps=plover_comps_new_from_uri(s,&error);
ali@97
   123
    retval=!comps;
ali@97
   124
    g_free(s);
ali@97
   125
    if (!retval)
ali@97
   126
    {
ali@97
   127
	group=plover_comps_lookup_group(comps,"installer");
ali@97
   128
	if (group)
ali@97
   129
	{
ali@97
   130
	    packages=plover_vector_new();
ali@97
   131
	    do
ali@97
   132
	    {
ali@97
   133
		changed=FALSE;
ali@97
   134
		for(pkg=group->packages;pkg;pkg=pkg->next)
ali@97
   135
		{
ali@97
   136
		    if (plover_vector_contains(packages,pkg->name))
ali@97
   137
			continue;
ali@97
   138
		    if (pkg->type==COMPS_REQUIREMENT_DEFAULT ||
ali@97
   139
		      pkg->type==COMPS_REQUIREMENT_MANDATORY ||
ali@97
   140
		      pkg->type==COMPS_REQUIREMENT_CONDITIONAL &&
ali@97
   141
		      plover_vector_contains(packages,pkg->requires))
ali@97
   142
		    {
ali@97
   143
			changed=TRUE;
ali@97
   144
			plover_vector_append(packages,pkg->name);
ali@97
   145
		    }
ali@97
   146
		}
ali@97
   147
	    } while(changed);
ali@97
   148
	    pkgs=packages->strings;
ali@97
   149
	}
ali@97
   150
    }
ali@97
   151
    if (!retval)
ali@97
   152
	retval=!plover_install_uri(repository,prefix,pkgs,&error);
ali@24
   153
    if (!retval)
ali@95
   154
	retval=!plover_update_uri(repository,prefix,NULL,&error);
ali@97
   155
    if (packages)
ali@97
   156
	plover_vector_free(packages);
ali@97
   157
    if (comps)
ali@97
   158
	plover_comps_free(comps);
ali@24
   159
    if (error)
ali@24
   160
    {
ali@24
   161
	fprintf(stderr,"%s\n",error->message);
ali@24
   162
	g_error_free(error);
ali@24
   163
    }
ali@24
   164
#ifdef WIN32
ali@24
   165
    PostQuitMessage(retval);
ali@24
   166
    PostThreadMessage(main_thread_id,WM_QUIT,retval,0);
ali@24
   167
    _endthreadex(retval);
ali@24
   168
#endif
ali@24
   169
    return retval;
ali@24
   170
}
ali@24
   171
ali@24
   172
/*
ali@24
   173
 * The idea is that if pre_install() fails, update/setup should fall back
ali@24
   174
 * to console interfaces.
ali@24
   175
 */
ali@24
   176
#ifdef WIN32
ali@24
   177
HANDLE
ali@24
   178
#else
ali@24
   179
void *
ali@24
   180
#endif
ali@95
   181
pre_install(const char *repository)
ali@24
   182
{
ali@24
   183
#ifdef WIN32
ali@24
   184
    HANDLE retval;
ali@24
   185
#else
ali@24
   186
    void *retval;
ali@24
   187
#endif
ali@38
   188
    razor_set_lua_loader("posix",(void (*)())luaopen_posix);
ali@38
   189
    razor_set_lua_loader("whelk",(void (*)())luaopen_whelk);
ali@24
   190
#ifdef WIN32
ali@95
   191
    retval=(HANDLE)_beginthreadex(NULL,0,pre_install_thread,(void *)repository,
ali@95
   192
      0,NULL);
ali@24
   193
#else
ali@95
   194
    if (pre_install_thread((void *)repository))
ali@24
   195
	retval=NULL;
ali@24
   196
    else
ali@24
   197
	retval=(void *)1;		/* Non-NULL to indicate success */
ali@24
   198
#endif
ali@24
   199
    return retval;
ali@24
   200
}
ali@24
   201
ali@24
   202
#ifndef WIN32
ali@24
   203
int remove_ignore(const char *fpath,const struct stat *sb,int typeflag,
ali@24
   204
  struct FTW *ftwbuf)
ali@24
   205
{
ali@24
   206
    (void)remove(fpath);
ali@24
   207
    return 0;
ali@24
   208
}
ali@24
   209
#endif
ali@24
   210
ali@24
   211
gboolean deltree(const char *path)
ali@24
   212
{
ali@24
   213
#ifdef WIN32
ali@24
   214
    /* Based on g_local_file_trash() */
ali@24
   215
    SHFILEOPSTRUCTW op={0};
ali@24
   216
    gboolean success;
ali@24
   217
    wchar_t *wfilename;
ali@24
   218
    long len;
ali@24
   219
    wfilename=g_utf8_to_utf16(path,-1,NULL,&len,NULL);
ali@24
   220
    /* SHFILEOPSTRUCT.pFrom is double-zero-terminated */
ali@24
   221
    wfilename=g_renew(wchar_t,wfilename,len+2);
ali@24
   222
    wfilename[len+1]=0;
ali@24
   223
    op.wFunc=FO_DELETE;
ali@24
   224
    op.pFrom=wfilename;
ali@24
   225
    op.fFlags=FOF_NO_UI;
ali@24
   226
    success=!SHFileOperationW(&op);
ali@24
   227
    if (success && op.fAnyOperationsAborted)
ali@24
   228
	success=FALSE;
ali@24
   229
    g_free(wfilename);
ali@24
   230
    return success;
ali@24
   231
#else
ali@24
   232
    return nftw(path,remove_ignore,64,FTW_DEPTH|FTW_PHYS);
ali@24
   233
#endif
ali@24
   234
}
ali@24
   235
ali@24
   236
gboolean pre_uninstall(void)
ali@24
   237
{
ali@24
   238
    gboolean success;
ali@46
   239
    gchar *s,*uri;
ali@24
   240
    GError *error=NULL;
ali@38
   241
    razor_set_lua_loader("posix",(void (*)())luaopen_posix);
ali@38
   242
    razor_set_lua_loader("whelk",(void (*)())luaopen_whelk);
ali@24
   243
    prefix=plover_pre_install_prefix();
ali@24
   244
    s=g_strconcat(prefix,"/var/lib/razor",NULL);
ali@46
   245
    uri=razor_path_to_uri(s);
ali@24
   246
    g_free(s);
ali@46
   247
    razor_set_database_uri(uri);
ali@46
   248
    free(uri);
ali@24
   249
    success=plover_remove(NULL,&error);
ali@24
   250
    if (error)
ali@24
   251
    {
ali@24
   252
	fprintf(stderr,"%s\n",error->message);
ali@24
   253
	g_error_free(error);
ali@24
   254
    }
ali@24
   255
    deltree(prefix);
ali@24
   256
    return success;
ali@24
   257
}
ali@24
   258
ali@24
   259
#if defined(WIN32) && !defined(USE_G_SPAWN)
ali@24
   260
/* Based on glib's g_spawn_win32.c */
ali@24
   261
ali@24
   262
static gchar *
ali@24
   263
win32_cmdline_quote(const char *string)
ali@24
   264
{
ali@24
   265
    const gchar *p=string;
ali@24
   266
    gchar *retval,*q;
ali@24
   267
    gint len=0;
ali@24
   268
    gboolean need_dblquotes=FALSE;
ali@24
   269
    while (*p)
ali@24
   270
    {
ali@24
   271
	if (*p==' ' || *p=='\t')
ali@24
   272
	    need_dblquotes=TRUE;
ali@24
   273
	else if (*p=='"')
ali@24
   274
	    len++;
ali@24
   275
	else if (*p=='\\')
ali@24
   276
	{
ali@24
   277
	    const gchar *pp=p;
ali@24
   278
	    while (*pp && *pp=='\\')
ali@24
   279
		pp++;
ali@24
   280
	    if (*pp=='"')
ali@24
   281
		len++;
ali@24
   282
	}
ali@24
   283
	len++;
ali@24
   284
	p++;
ali@24
   285
    }
ali@24
   286
    q=retval=g_malloc(len+need_dblquotes*2+1);
ali@24
   287
    p=string;
ali@24
   288
    if (need_dblquotes)
ali@24
   289
	*q++='"';
ali@24
   290
    while (*p)
ali@24
   291
    {
ali@24
   292
	if (*p=='"')
ali@24
   293
	    *q++='\\';
ali@24
   294
	else if (*p=='\\')
ali@24
   295
	{
ali@24
   296
	    const gchar *pp=p;
ali@24
   297
	    while (*pp && *pp=='\\')
ali@24
   298
		pp++;
ali@24
   299
	    if (*pp=='"')
ali@24
   300
		*q++='\\';
ali@24
   301
	}
ali@24
   302
	*q++=*p;
ali@24
   303
	p++;
ali@24
   304
    }
ali@24
   305
    if (need_dblquotes)
ali@24
   306
	*q++='"';
ali@24
   307
    *q++='\0';
ali@24
   308
    return retval;
ali@24
   309
}
ali@24
   310
ali@24
   311
/* Create a win32-style wide-character argv with suitable quoting */
ali@24
   312
wchar_t **win32_argv_import(char **argv,GError **error)
ali@24
   313
{
ali@24
   314
    int i,n;
ali@24
   315
    gchar *s;
ali@24
   316
    wchar_t **wargv;
ali@24
   317
    GError *tmp_error=NULL;
ali@24
   318
    n=g_strv_length(argv);
ali@24
   319
    wargv=g_new(wchar_t *,n+1);
ali@24
   320
    for(i=0;i<n;i++)
ali@24
   321
    {
ali@24
   322
	s=win32_cmdline_quote(argv[i]);
ali@24
   323
	wargv[i]=g_utf8_to_utf16(s,-1,NULL,NULL,&tmp_error);
ali@24
   324
	g_free(s);
ali@24
   325
	if (!wargv[i])
ali@24
   326
	{
ali@24
   327
	    g_set_error(error,G_SPAWN_ERROR,G_SPAWN_ERROR_FAILED,
ali@24
   328
	      "Invalid argument #%d: %s",i,tmp_error->message);
ali@24
   329
	    g_error_free(tmp_error);
ali@24
   330
	    for(i--;i>=0;i--)
ali@24
   331
		g_free(wargv[i]);
ali@24
   332
	    g_free(wargv);
ali@24
   333
	    return FALSE;
ali@24
   334
	}
ali@24
   335
    }
ali@24
   336
    wargv[i]=NULL;
ali@24
   337
    return wargv;
ali@24
   338
}
ali@24
   339
ali@24
   340
gboolean spawn_sync(char **argv,GError **error)
ali@24
   341
{
ali@24
   342
    wchar_t *wargv0,**wargv;
ali@24
   343
    gintptr rc;
ali@24
   344
    GError *tmp_error=NULL;
ali@24
   345
    wargv0=g_utf8_to_utf16(argv[0],-1,NULL,NULL,&tmp_error);
ali@24
   346
    if (!wargv0)
ali@24
   347
    {
ali@24
   348
	fprintf(stderr,"Conversion error in post\n");
ali@24
   349
	g_set_error(error,G_SPAWN_ERROR,G_SPAWN_ERROR_FAILED,
ali@24
   350
	  "Invalid program name: %s",tmp_error->message);
ali@24
   351
	g_error_free(tmp_error);
ali@24
   352
	return FALSE;
ali@24
   353
    }
ali@24
   354
    wargv=win32_argv_import(argv,error);
ali@24
   355
    if (!wargv)
ali@24
   356
    {
ali@24
   357
	fprintf(stderr,"Conversion error in post\n");
ali@24
   358
	g_free(wargv0);
ali@24
   359
	return FALSE;
ali@24
   360
    }
ali@24
   361
    errno=0;
ali@24
   362
    rc=_wspawnvp(P_WAIT,wargv0,(const wchar_t **)wargv);
ali@24
   363
    g_free(wargv0);
ali@24
   364
    g_strfreev((gchar **)wargv);
ali@24
   365
    if (rc==-1 && errno!=0)
ali@24
   366
    {
ali@24
   367
	fprintf(stderr,"Failed to start post command (%s)\n",g_strerror(errno));
ali@24
   368
	g_set_error(error,G_SPAWN_ERROR,G_SPAWN_ERROR_FAILED,
ali@24
   369
	  "Failed to execute post command: %s",g_strerror(errno));
ali@24
   370
	return FALSE;
ali@24
   371
    }
ali@24
   372
    if (rc!=EXIT_SUCCESS)
ali@24
   373
    {
ali@24
   374
	fprintf(stderr,"post command failed (%ld)\n",(long)rc);
ali@24
   375
	g_set_error(error,G_SPAWN_ERROR,G_SPAWN_ERROR_FAILED,
ali@24
   376
	  "Post command exited with code %ld",(long)rc);
ali@24
   377
	return FALSE;
ali@24
   378
    }
ali@24
   379
    return TRUE;
ali@24
   380
}
ali@24
   381
ali@24
   382
#endif /* defined(WIN32) && !defined(USE_G_SPAWN) */
ali@24
   383
ali@96
   384
gboolean pre_install_spawn_sync(char **argv,GError **error)
ali@96
   385
{
ali@96
   386
    GError *tmp_error=NULL;
ali@96
   387
#ifdef USE_G_SPAWN
ali@96
   388
    gchar *standard_output,*standard_error;
ali@96
   389
    int exit_status;
ali@96
   390
    if (!g_spawn_sync(NULL,argv,NULL,G_SPAWN_SEARCH_PATH,NULL,NULL,
ali@96
   391
      &standard_output,&standard_error,&exit_status,&tmp_error))
ali@96
   392
    {
ali@96
   393
	fprintf(stderr,"Failed to start post command\n");
ali@96
   394
	g_propagate_prefixed_error(error,tmp_error,"%s: ",argv[0]);
ali@96
   395
	return FALSE;
ali@96
   396
    }
ali@96
   397
    if (standard_output && *standard_output)
ali@96
   398
    {
ali@96
   399
	printf("Output from post command %s:\n",argv[0]);
ali@96
   400
	fputs(standard_output,stdout);
ali@96
   401
    }
ali@96
   402
    g_free(standard_output);
ali@96
   403
    if (standard_error && *standard_error)
ali@96
   404
    {
ali@96
   405
	printf("Error output from post command %s:\n",argv[0]);
ali@96
   406
	fputs(standard_error,stdout);
ali@96
   407
    }
ali@96
   408
    g_free(standard_error);
ali@96
   409
    if (!g_spawn_check_exit_status(exit_status,&tmp_error))
ali@96
   410
    {
ali@96
   411
	fprintf(stderr,"post command failed\n");
ali@96
   412
	g_propagate_prefixed_error(error,tmp_error,"%s: ",argv[0]);
ali@96
   413
	return FALSE;
ali@96
   414
    }
ali@96
   415
#else
ali@96
   416
    if (!spawn_sync(argv,&tmp_error))
ali@96
   417
    {
ali@96
   418
	g_propagate_prefixed_error(error,tmp_error,"%s: ",argv[0]);
ali@96
   419
	return FALSE;
ali@96
   420
    }
ali@96
   421
#endif
ali@96
   422
    return TRUE;
ali@96
   423
}
ali@96
   424
ali@24
   425
/*
ali@24
   426
 * Run a command after completing request.
ali@24
   427
 *
ali@24
   428
 * Command may refer to %INSTALL_PREFIX% which will be replaced by the
ali@95
   429
 * (first) install prefix used, by %TEST_RESULT% which will be replaced
ali@80
   430
 * by either "pass" or "fail" depending as to whether the request succeeded
ali@95
   431
 * or not and/or by %REPOSITORY% which will be replaced by the URI of the
ali@95
   432
 * repository used. Command may also include double quotes which will be used
ali@95
   433
 * to affect how the command is split into arguments much like a shell does.
ali@24
   434
 */
ali@95
   435
gboolean run_post(int argc,char **argv,gboolean test_result,
ali@95
   436
  const char *repository,GError **error)
ali@24
   437
{
ali@24
   438
    int i,post_argc;
ali@24
   439
    char *s;
ali@24
   440
    gchar *expanded;
ali@24
   441
    gchar **post_argv;
ali@24
   442
    GError *tmp_error=NULL;
ali@87
   443
    if (argc<2)
ali@24
   444
    {
ali@24
   445
	g_set_error_literal(error,G_FILE_ERROR,G_FILE_ERROR_NOENT,
ali@24
   446
	  "--post: No command given");
ali@24
   447
	return FALSE;
ali@24
   448
    }
ali@24
   449
    printf("Running post command: %s\n",argv[1]);
ali@24
   450
    if (!g_shell_parse_argv(argv[1],&post_argc,&post_argv,&tmp_error))
ali@24
   451
    {
ali@24
   452
	g_propagate_prefixed_error(error,tmp_error,"%s: ",argv[1]);
ali@24
   453
	return FALSE;
ali@24
   454
    }
ali@24
   455
    for(i=0;i<post_argc;i++)
ali@24
   456
    {
ali@24
   457
	s=strstr(post_argv[i],"%INSTALL_PREFIX%");
ali@24
   458
	if (s)
ali@24
   459
	{
ali@24
   460
	    *s='\0';
ali@24
   461
	    s+=strlen("%INSTALL_PREFIX%");
ali@24
   462
	    expanded=g_strconcat(post_argv[i],prefix,s,NULL);
ali@24
   463
	    g_free(post_argv[i]);
ali@24
   464
	    post_argv[i]=expanded;
ali@24
   465
	}
ali@24
   466
	s=strstr(post_argv[i],"%TEST_RESULT%");
ali@24
   467
	if (s)
ali@24
   468
	{
ali@24
   469
	    *s='\0';
ali@24
   470
	    s+=strlen("%TEST_RESULT%");
ali@24
   471
	    expanded=g_strconcat(post_argv[i],test_result?"pass":"fail",s,NULL);
ali@24
   472
	    g_free(post_argv[i]);
ali@24
   473
	    post_argv[i]=expanded;
ali@24
   474
	}
ali@95
   475
	s=strstr(post_argv[i],"%REPOSITORY%");
ali@95
   476
	if (s)
ali@95
   477
	{
ali@95
   478
	    *s='\0';
ali@95
   479
	    s+=strlen("%REPOSITORY%");
ali@95
   480
	    expanded=g_strconcat(post_argv[i],repository,s,NULL);
ali@95
   481
	    g_free(post_argv[i]);
ali@95
   482
	    post_argv[i]=expanded;
ali@95
   483
	}
ali@24
   484
    }
ali@96
   485
    return pre_install_spawn_sync(post_argv,error);
ali@24
   486
}
ali@24
   487
ali@24
   488
#ifdef WIN32
ali@95
   489
DWORD win32_pre_install_gui(char *repository)
ali@24
   490
{
ali@24
   491
    HANDLE thread;
ali@24
   492
    INITCOMMONCONTROLSEX icc={0,};
ali@24
   493
    MSG msg;
ali@24
   494
    DWORD retval;
ali@24
   495
    main_thread_id=GetCurrentThreadId();
ali@95
   496
    thread=(HANDLE)pre_install(repository);
ali@24
   497
    if (!thread)
ali@24
   498
	return EXIT_FAILURE;
ali@24
   499
    icc.dwSize=sizeof(icc);
ali@24
   500
    icc.dwICC=ICC_WIN95_CLASSES;
ali@24
   501
    InitCommonControlsEx(&icc);
ali@24
   502
    DialogBox(GetModuleHandle(NULL),MAKEINTRESOURCE(IDD_PROGRESSDIALOG),NULL,
ali@24
   503
      &ProgressDialogProc);
ali@24
   504
    while(GetMessage(&msg,NULL,0,0)>0)
ali@24
   505
    {
ali@24
   506
	TranslateMessage(&msg);
ali@24
   507
	DispatchMessage(&msg);
ali@24
   508
    }
ali@24
   509
    WaitForSingleObject(thread,INFINITE);
ali@24
   510
    GetExitCodeThread(thread,&retval);
ali@24
   511
    CloseHandle(thread);
ali@24
   512
    return retval;
ali@24
   513
}
ali@24
   514
#endif	/* WIN32 */
ali@24
   515
ali@95
   516
gchar *pre_install_default_repository(const char *argv0)
ali@94
   517
{
ali@94
   518
    size_t length;
ali@94
   519
    void *contents;
ali@94
   520
    gchar *path;
ali@94
   521
    gchar *s,*uri;
ali@94
   522
    struct razor_error *tmp_error=NULL;
ali@94
   523
    /*
ali@95
   524
     * The default repository is the executable itself if it's an archive
ali@94
   525
     * or otherwise the directory in which the executable is stored.
ali@94
   526
     */
ali@94
   527
    path=plover_get_program(argv0);
ali@94
   528
    uri=razor_path_to_uri(path);
ali@95
   529
    free(path);
ali@94
   530
    s=g_strconcat(uri,"/repodata/comps.xml",NULL);
ali@94
   531
    contents=razor_uri_get_contents(s,&length,FALSE,&tmp_error);
ali@94
   532
    g_free(s);
ali@94
   533
    if (contents)
ali@94
   534
	razor_uri_free_contents(contents,length);
ali@94
   535
    else
ali@94
   536
    {
ali@94
   537
	if (razor_error_matches(tmp_error,RAZOR_GENERAL_ERROR,
ali@94
   538
	  RAZOR_GENERAL_ERROR_UNSUPPORTED_ARCHIVE))
ali@94
   539
	{
ali@94
   540
	    path=plover_get_program_directory(argv0);
ali@95
   541
	    uri=razor_path_to_uri(path);
ali@95
   542
	    free(path);
ali@94
   543
	}
ali@94
   544
	razor_error_free(tmp_error);
ali@94
   545
    }
ali@95
   546
    return uri;
ali@94
   547
}
ali@94
   548
ali@96
   549
gboolean pre_install_post(int argc,char **argv,gboolean success,
ali@96
   550
  const char *repository)
ali@96
   551
{
ali@96
   552
    gchar *s,*uri;
ali@96
   553
    struct post *post=NULL;
ali@96
   554
    GError *error=NULL;
ali@96
   555
    if (!argv && !success)
ali@96
   556
    {
ali@96
   557
	/*
ali@96
   558
	 * If no --post command has been given, then we want to run the
ali@96
   559
	 * program specified in %INSTALL_PREFIX%/etc/pre-inst.post
ali@96
   560
	 * However, if the pre-install failed, then that file isn't likely
ali@96
   561
	 * to exist (and it's requirements even less so) so we issue an error
ali@96
   562
	 * instead. Unfortunately, it's hard to include the error message
ali@96
   563
	 * here. It will have been logged in the plover log, but that's not
ali@96
   564
	 * much help to users.
ali@96
   565
	 */
ali@96
   566
#ifndef WIN32
ali@96
   567
	fprintf(stderr,
ali@96
   568
	  "Installation failed: Failed to unpack the main installer\n");
ali@96
   569
#else
ali@96
   570
	MessageBox(NULL,"Failed to unpack the main installer",
ali@96
   571
	  "Installation failed",MB_ICONERROR|MB_OK);
ali@96
   572
#endif
ali@96
   573
	return FALSE;
ali@96
   574
    }
ali@96
   575
    if (!argv)
ali@96
   576
    {
ali@96
   577
	post=pre_install_post_new(repository,prefix);
ali@96
   578
	uri=razor_path_to_uri(prefix);
ali@96
   579
	s=g_strconcat(uri,"/etc/pre-inst.post",NULL);
ali@96
   580
	g_free(uri);
ali@96
   581
	pre_install_post_load_uri(post,s,&error);
ali@96
   582
	if (error)
ali@96
   583
	    g_debug("Failed to load post configuration from pre-inst.post: %s",
ali@96
   584
	      error->message);
ali@96
   585
	g_free(s);
ali@96
   586
	if (g_error_matches(error,PLOVER_POSIX_ERROR,ENOENT))
ali@96
   587
	{
ali@96
   588
	    /*
ali@96
   589
	     * If no post configuration file exists, that's not an error;
ali@96
   590
	     * there simply is no post action configured.
ali@96
   591
	     */
ali@96
   592
	    g_clear_error(&error);
ali@96
   593
	}
ali@96
   594
    }
ali@96
   595
    if (post && post->argc>0)
ali@96
   596
	pre_install_spawn_sync(post->argv,&error);
ali@96
   597
    else if (argv)
ali@96
   598
	run_post(argc,argv,success,repository,&error);
ali@96
   599
    if (post)
ali@96
   600
	pre_install_post_free(post);
ali@96
   601
    if (error)
ali@96
   602
    {
ali@96
   603
#ifndef WIN32
ali@96
   604
	fprintf(stderr,"Error in post: %s\n",error->message);
ali@96
   605
#else
ali@96
   606
	MessageBox(NULL,error->message,"Error in post",MB_ICONERROR|MB_OK);
ali@96
   607
#endif
ali@96
   608
	g_error_free(error);
ali@96
   609
	success=FALSE;
ali@96
   610
    }
ali@96
   611
    return success;
ali@96
   612
}
ali@96
   613
ali@24
   614
int main(int argc,char **argv)
ali@24
   615
{
ali@80
   616
    gboolean success,uninstall=FALSE,enable_post=FALSE;
ali@24
   617
    GError *error=NULL;
ali@95
   618
    gchar *path=NULL,*repository=NULL;
ali@80
   619
    GOptionContext *context;
ali@80
   620
    GOptionEntry options[]={
ali@95
   621
	{"repository",0,0,G_OPTION_ARG_STRING,&repository,
ali@95
   622
	  "Repository location","uri"},
ali@80
   623
	{"path",0,0,G_OPTION_ARG_FILENAME,&path,
ali@80
   624
	  "Repository path","path"},
ali@80
   625
	{"uninstall",'u',0,G_OPTION_ARG_NONE,&uninstall,
ali@80
   626
	  "Uninstall all packages",NULL},
ali@80
   627
	{"post",0,0,G_OPTION_ARG_NONE,&enable_post,
ali@80
   628
	  "Run command after request is processed",NULL},
ali@80
   629
	{NULL}
ali@80
   630
    };
ali@24
   631
#ifdef WIN32
ali@24
   632
    /*
ali@24
   633
     * pre-inst is normally a GUI application, but rpm scripts may well
ali@24
   634
     * call console applications and it looks ugly if console windows keep
ali@24
   635
     * popping up. Avoid this by allocating our own console and hiding it.
ali@24
   636
     * Note:
ali@24
   637
     *	- If pre-inst is a console application (typically for debugging),
ali@24
   638
     *    then skip this step.
ali@24
   639
     *  - Call ShowWindow twice to negate special handling on first call.
ali@24
   640
     */
ali@24
   641
    if (!GetConsoleWindow())
ali@24
   642
    {
ali@24
   643
	AllocConsole();
ali@24
   644
	ShowWindow(GetConsoleWindow(),SW_HIDE);
ali@24
   645
	ShowWindow(GetConsoleWindow(),SW_HIDE);
ali@24
   646
    }
ali@24
   647
#endif
ali@31
   648
    plover_exception_handler_init();
ali@80
   649
    context=g_option_context_new("[command] - install the main installer");
ali@80
   650
    g_option_context_add_main_entries(context,options,NULL);
ali@80
   651
    g_option_context_set_description(context,
ali@80
   652
      "If --post is specified, then the command to run and its arguments\n"
ali@80
   653
      "should be listed at the end of the command line.\n"
ali@80
   654
      "Command may refer to %INSTALL_PREFIX% which will be replaced by the\n"
ali@95
   655
      "(first) install prefix used, by %TEST_RESULT% which will be\n"
ali@80
   656
      "replaced by either \"pass\" or \"fail\" depending as to whether the\n"
ali@95
   657
      "request succeeded or not and/or by %REPOSITORY% which will be\n"
ali@95
   658
      "replaced by the URI of the repository used. Command may also include\n"
ali@95
   659
      "double quotes which will be used to affect how the command is split\n"
ali@95
   660
      "into arguments much like a shell does.");
ali@80
   661
    g_option_context_set_strict_posix(context,TRUE);
ali@80
   662
    g_option_context_set_ignore_unknown_options(context,TRUE);
ali@80
   663
    if (!g_option_context_parse(context,&argc,&argv,&error))
ali@24
   664
    {
ali@80
   665
	g_printerr("pre-install: %s\n",error->message);
ali@99
   666
	g_printerr("Use \"%s --help\" for help\n",argv[0]);
ali@80
   667
	exit(1);
ali@80
   668
    }
ali@95
   669
    if (repository && path)
ali@95
   670
    {
ali@95
   671
	g_printerr("pre-install: "
ali@95
   672
	  "Only one of --repository and --path can be specified\n");
ali@95
   673
	exit(1);
ali@95
   674
    }
ali@95
   675
    if (path)
ali@95
   676
	repository=razor_path_to_uri(path);
ali@95
   677
    else if (!repository)
ali@95
   678
	repository=pre_install_default_repository(argv[0]);
ali@80
   679
    if (uninstall)
ali@24
   680
	success=pre_uninstall();
ali@80
   681
    else
ali@80
   682
    {
ali@80
   683
#ifdef WIN32
ali@95
   684
	success=win32_pre_install_gui(repository)==EXIT_SUCCESS;
ali@80
   685
#else
ali@95
   686
	success=!!pre_install(repository);
ali@80
   687
#endif
ali@24
   688
    }
ali@96
   689
    if (enable_post)
ali@96
   690
	success=pre_install_post(argc,argv,success,repository);
ali@96
   691
    else if (!uninstall)
ali@96
   692
	success=pre_install_post(0,NULL,success,repository);
ali@24
   693
#ifdef WIN32
ali@24
   694
    return success?EXIT_SUCCESS:EXIT_FAILURE;
ali@24
   695
#else
ali@24
   696
    return success?0:1;
ali@24
   697
#endif
ali@24
   698
}