plover-gtk/transactionhelper.c
author J. Ali Harlow <ali@juiblex.co.uk>
Fri Mar 08 18:20:14 2019 +0000 (2019-03-08)
changeset 90 c12878ef2fe1
parent 76 7a98a5020c44
child 98 71127797ca73
permissions -rw-r--r--
Use configured rsvg command in app-manager
ali@24
     1
/*
ali@75
     2
 * Copyright (C) 2014, 2016, 2018  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
ali@24
    19
#include "config.h"
ali@24
    20
#include <stdlib.h>
ali@61
    21
#include <string.h>
ali@24
    22
#include <errno.h>
ali@24
    23
#include <gtk/gtk.h>
ali@24
    24
#include <plover/plover.h>
ali@24
    25
#include <plover/transaction.h>
ali@24
    26
#include <plover-gtk/transactionhelper.h>
ali@44
    27
#include "plover/uri-handler.h"
ali@24
    28
ali@24
    29
/*
ali@24
    30
 * A PloverTransactionHelper uses a GtkAssistant to help a user run a
ali@24
    31
 * transaction.
ali@24
    32
 */
ali@24
    33
ali@24
    34
G_DEFINE_TYPE(PloverTransactionHelper,plover_transaction_helper,G_TYPE_OBJECT)
ali@24
    35
ali@38
    36
enum plover_transaction_type {
ali@38
    37
    TRANSACTION_TYPE_NULL=0,
ali@38
    38
    TRANSACTION_TYPE_INSTALL=1UL<<0,
ali@38
    39
    TRANSACTION_TYPE_REMOVE=1UL<<1,
ali@38
    40
    TRANSACTION_TYPE_UPDATE=TRANSACTION_TYPE_INSTALL|TRANSACTION_TYPE_REMOVE
ali@38
    41
};
ali@38
    42
ali@38
    43
typedef struct _PloverTransactionHelperPrivate {
ali@38
    44
    enum plover_transaction_type transaction_type;
ali@38
    45
    gchar *default_prefix;
ali@38
    46
} PloverTransactionHelperPrivate;
ali@38
    47
ali@38
    48
#define PLOVER_TRANSACTION_HELPER_GET_PRIVATE(obj)\
ali@38
    49
                                G_TYPE_INSTANCE_GET_PRIVATE(obj,\
ali@38
    50
				  PLOVER_TYPE_TRANSACTION_HELPER,\
ali@38
    51
				  PloverTransactionHelperPrivate)
ali@38
    52
ali@24
    53
enum {
ali@24
    54
    CLOSE=0,
ali@24
    55
    N_SIGNALS
ali@24
    56
};
ali@24
    57
ali@24
    58
static guint signals[N_SIGNALS];
ali@24
    59
ali@86
    60
static PloverTransaction *
ali@86
    61
  plover_transaction_helper_new_transaction(PloverTransactionHelper *helper,
ali@86
    62
  GError **error);
ali@86
    63
ali@24
    64
static void plover_transaction_helper_finalize(PloverTransactionHelper *helper)
ali@24
    65
{
ali@38
    66
    PloverTransactionHelperPrivate *priv;
ali@38
    67
    priv=PLOVER_TRANSACTION_HELPER_GET_PRIVATE(helper);
ali@38
    68
    g_free(priv->default_prefix);
ali@24
    69
    g_free(helper->error_primary_text);
ali@24
    70
    g_free(helper->base);
ali@24
    71
    g_free(helper->unsatisfied);
ali@24
    72
    if (helper->comps)
ali@24
    73
	plover_comps_free(helper->comps);
ali@24
    74
    plover_vector_free(helper->report_adding);
ali@24
    75
    plover_vector_free(helper->report_removing);
ali@24
    76
}
ali@24
    77
ali@24
    78
static void plover_transaction_helper_dispose(PloverTransactionHelper *helper)
ali@24
    79
{
ali@24
    80
    g_clear_error(&helper->error);
ali@24
    81
    if (helper->error_dialog)
ali@24
    82
    {
ali@24
    83
	g_signal_handlers_disconnect_by_data(helper->error_dialog,helper);
ali@24
    84
	gtk_widget_destroy(helper->error_dialog);
ali@24
    85
	helper->error_dialog=NULL;
ali@24
    86
    }
ali@24
    87
    if (helper->assistant)
ali@24
    88
    {
ali@24
    89
	g_signal_handlers_disconnect_by_data(helper->assistant,helper);
ali@24
    90
	g_clear_object(&helper->assistant);
ali@24
    91
    }
ali@24
    92
    g_clear_object(&helper->ui);
ali@24
    93
    g_slist_foreach(helper->transactions,(GFunc)g_object_unref,NULL);
ali@24
    94
    g_slist_free(helper->transactions);
ali@24
    95
    helper->transactions=NULL;
ali@61
    96
    g_clear_object(&helper->alternate_installed);
ali@24
    97
    g_clear_object(&helper->installed);
ali@24
    98
    g_clear_object(&helper->upstream);
ali@24
    99
    g_clear_object(&helper->relocated_upstream);
ali@24
   100
}
ali@24
   101
ali@24
   102
static void
ali@24
   103
  plover_transaction_helper_class_init(PloverTransactionHelperClass *klass)
ali@24
   104
{
ali@24
   105
    GObjectClass *gobject_class=G_OBJECT_CLASS(klass);
ali@24
   106
    gobject_class->finalize=
ali@24
   107
      (void (*)(GObject *))plover_transaction_helper_finalize;
ali@24
   108
    gobject_class->dispose=
ali@24
   109
      (void (*)(GObject *))plover_transaction_helper_dispose;
ali@38
   110
    g_type_class_add_private(klass,sizeof(PloverTransactionHelperPrivate));
ali@24
   111
    signals[CLOSE]=g_signal_newv("close",
ali@24
   112
      G_TYPE_FROM_CLASS(klass),G_SIGNAL_RUN_LAST,NULL,NULL,NULL,
ali@24
   113
      g_cclosure_marshal_VOID__VOID,G_TYPE_NONE,0,NULL);
ali@24
   114
}
ali@24
   115
ali@24
   116
static void plover_transaction_helper_init(PloverTransactionHelper *helper)
ali@24
   117
{
ali@24
   118
    helper->report_adding=plover_vector_new();
ali@24
   119
    helper->report_removing=plover_vector_new();
ali@24
   120
}
ali@24
   121
ali@24
   122
static void plover_transaction_helper_assistant_cancel(GtkAssistant *assistant,
ali@24
   123
  PloverTransactionHelper *helper)
ali@24
   124
{
ali@24
   125
    gtk_widget_hide(GTK_WIDGET(helper->assistant));
ali@24
   126
    gtk_assistant_set_current_page(helper->assistant,0);
ali@24
   127
    g_signal_emit(helper,signals[CLOSE],0);
ali@24
   128
}
ali@24
   129
ali@24
   130
static void plover_transaction_helper_assistant_close(GtkAssistant *assistant,
ali@24
   131
  PloverTransactionHelper *helper)
ali@24
   132
{
ali@24
   133
    gtk_widget_hide(GTK_WIDGET(helper->assistant));
ali@24
   134
    gtk_assistant_set_current_page(helper->assistant,0);
ali@24
   135
    g_signal_emit(helper,signals[CLOSE],0);
ali@24
   136
}
ali@24
   137
ali@24
   138
static void
ali@24
   139
  plover_transaction_helper_prepare_confirm(PloverTransactionHelper *helper)
ali@24
   140
{
ali@24
   141
    gchar *package_list,*add,*remove,*s;
ali@24
   142
    GtkLabel *label;
ali@24
   143
    struct plover_vector *report;
ali@24
   144
    if (helper->report_adding->len)
ali@24
   145
    {
ali@24
   146
	plover_vector_sort(helper->report_adding);
ali@24
   147
	if (helper->report_adding_dependencies)
ali@24
   148
	{
ali@24
   149
	    report=plover_vector_dup(helper->report_adding);
ali@24
   150
	    if (helper->report_adding->len==1)
ali@24
   151
		plover_vector_append(report,"its dependencies");
ali@24
   152
	    else
ali@24
   153
		plover_vector_append(report,"their dependencies");
ali@24
   154
	    package_list=plover_vector_format_for_display(report);
ali@24
   155
	    plover_vector_free(report);
ali@24
   156
	}
ali@24
   157
	else
ali@24
   158
	    package_list=
ali@24
   159
	      plover_vector_format_for_display(helper->report_adding);
ali@24
   160
	add=g_strconcat("Packages to be installed or updated: ",package_list,
ali@24
   161
	  ".",NULL);
ali@24
   162
	g_free(package_list);
ali@24
   163
    }
ali@24
   164
    else
ali@24
   165
	add=NULL;
ali@24
   166
    if (helper->report_removing->len)
ali@24
   167
    {
ali@24
   168
	plover_vector_sort(helper->report_removing);
ali@24
   169
	if (helper->report_removing_dependants)
ali@24
   170
	{
ali@24
   171
	    report=plover_vector_dup(helper->report_removing);
ali@24
   172
	    if (helper->report_adding->len==1)
ali@24
   173
		plover_vector_append(report,"its dependants");
ali@24
   174
	    else
ali@24
   175
		plover_vector_append(report,"their dependants");
ali@24
   176
	    package_list=plover_vector_format_for_display(report);
ali@24
   177
	    plover_vector_free(report);
ali@24
   178
	}
ali@24
   179
	else
ali@24
   180
	    package_list=
ali@24
   181
	      plover_vector_format_for_display(helper->report_removing);
ali@24
   182
	remove=g_strconcat("Packages to be removed: ",package_list,".",NULL);
ali@24
   183
	g_free(package_list);
ali@24
   184
    }
ali@24
   185
    else
ali@24
   186
	remove=NULL;
ali@24
   187
    label=GTK_LABEL(gtk_builder_get_object(helper->ui,"SISummaryOfWork"));
ali@24
   188
    if (add && remove)
ali@24
   189
	s=g_strconcat("<b>Installation Summary</b>\n\n",remove,"\n\n",add,NULL);
ali@24
   190
    else if (add || remove)
ali@24
   191
	s=g_strconcat("<b>Installation Summary</b>\n\n",add?add:remove,NULL);
ali@24
   192
    else
ali@24
   193
	s=g_strdup("<b>Installation Summary</b>\n\nNo changes scheduled");
ali@24
   194
    gtk_label_set_markup(label,s);
ali@24
   195
    g_free(s);
ali@24
   196
    g_free(add);
ali@24
   197
    g_free(remove);
ali@24
   198
}
ali@24
   199
ali@24
   200
static void plover_transaction_helper_run(PloverTransactionHelper *helper);
ali@24
   201
ali@24
   202
static void plover_transaction_helper_callback(GObject *source,
ali@24
   203
  GAsyncResult *result,gpointer user_data)
ali@24
   204
{
ali@24
   205
    GError *error=NULL;
ali@24
   206
    PloverTransactionHelper *helper=user_data;
ali@24
   207
    PloverTransaction *transaction=PLOVER_TRANSACTION(source);
ali@24
   208
    if (!plover_transaction_commit_finish(transaction,result,&error))
ali@24
   209
    {
ali@24
   210
	plover_transaction_helper_set_error(helper,error,
ali@24
   211
	  "Software installation failed");
ali@24
   212
	g_error_free(error);
ali@24
   213
    }
ali@24
   214
    else
ali@24
   215
	plover_transaction_helper_run(helper);
ali@38
   216
    /*
ali@38
   217
     * There may be status updates queued by transaction as idle events.
ali@38
   218
     * Process them now before we disconnect so that we don't lose them.
ali@38
   219
     */
ali@38
   220
    while(g_main_context_pending(NULL))
ali@38
   221
	g_main_context_iteration(NULL,FALSE);
ali@24
   222
    g_signal_handlers_disconnect_by_data(transaction,helper);
ali@24
   223
    g_object_unref(transaction);
ali@24
   224
}
ali@24
   225
ali@24
   226
static void plover_transaction_helper_transaction_status_changed(
ali@24
   227
  PloverTransaction *transaction,const char *status,
ali@24
   228
  PloverTransactionHelper *helper)
ali@24
   229
{
ali@24
   230
    GtkProgressBar *bar;
ali@24
   231
    bar=GTK_PROGRESS_BAR(gtk_builder_get_object(helper->ui,"SIProgressBar"));
ali@24
   232
    gtk_progress_bar_set_text(bar,status);
ali@24
   233
}
ali@24
   234
ali@24
   235
static void plover_transaction_helper_run(PloverTransactionHelper *helper)
ali@24
   236
{
ali@24
   237
    PloverTransaction *transaction;
ali@24
   238
    GtkWidget *page;
ali@24
   239
    page=GTK_WIDGET(gtk_builder_get_object(helper->ui,"SIProgress"));
ali@24
   240
    if (helper->transactions)
ali@24
   241
    {
ali@24
   242
	if (helper->assistant)
ali@24
   243
	    gtk_assistant_set_page_complete(helper->assistant,page,FALSE);
ali@24
   244
	transaction=helper->transactions->data;
ali@24
   245
	helper->transactions=g_slist_delete_link(helper->transactions,
ali@24
   246
	  helper->transactions);
ali@24
   247
	g_signal_connect(transaction,"status-changed",
ali@24
   248
	  G_CALLBACK(plover_transaction_helper_transaction_status_changed),
ali@24
   249
	  helper);
ali@24
   250
	plover_transaction_commit_async(transaction,NULL,
ali@24
   251
	  plover_transaction_helper_callback,helper);
ali@24
   252
    }
ali@24
   253
    else if (helper->assistant)
ali@24
   254
	gtk_assistant_set_page_complete(helper->assistant,page,TRUE);
ali@24
   255
}
ali@24
   256
ali@24
   257
static gboolean plover_transaction_helper_pulse(gpointer user_data)
ali@24
   258
{
ali@24
   259
    PloverTransactionHelper *helper=user_data;
ali@24
   260
    GtkWidget *w;
ali@24
   261
    GtkProgressBar *bar;
ali@24
   262
    if (!helper->assistant)
ali@24
   263
	return FALSE;
ali@24
   264
    w=GTK_WIDGET(gtk_builder_get_object(helper->ui,"SIProgress"));
ali@24
   265
    bar=GTK_PROGRESS_BAR(gtk_builder_get_object(helper->ui,"SIProgressBar"));
ali@24
   266
    if (gtk_assistant_get_page_complete(helper->assistant,w))
ali@24
   267
    {
ali@24
   268
	gtk_progress_bar_set_fraction(bar,1.0);
ali@24
   269
	helper->pulse_handler=0;
ali@24
   270
	return FALSE;
ali@24
   271
    }
ali@24
   272
    else
ali@24
   273
    {
ali@24
   274
	gtk_progress_bar_pulse(bar);
ali@24
   275
	return TRUE;
ali@24
   276
    }
ali@24
   277
}
ali@24
   278
ali@24
   279
static void
ali@24
   280
  plover_transaction_helper_prepare_progress(PloverTransactionHelper *helper)
ali@24
   281
{
ali@24
   282
    GError *error=NULL;
ali@24
   283
    GtkToggleButton *button;
ali@24
   284
    PloverTransaction *transaction;
ali@38
   285
    GSList *save_transactions;
ali@38
   286
    PloverTransactionHelperPrivate *priv;
ali@38
   287
    enum plover_transaction_type save_transaction_type;
ali@38
   288
    priv=PLOVER_TRANSACTION_HELPER_GET_PRIVATE(helper);
ali@24
   289
    button=GTK_TOGGLE_BUTTON(gtk_builder_get_object(helper->ui,
ali@24
   290
      "SIRemoveExisting"));
ali@24
   291
    if (gtk_toggle_button_get_active(button))
ali@24
   292
    {
ali@86
   293
	transaction=plover_transaction_helper_new_transaction(helper,&error);
ali@86
   294
	/*
ali@86
   295
	 * I think we want switch to the alternate installed set in the case of
ali@86
   296
	 * alternate_database_clashes, but not in the case of
ali@86
   297
	 * active_database_is_incompatible (see
ali@86
   298
	 * plover_transaction_helper_update_summary_page).
ali@86
   299
	 * Whether testing for helper->alternate_installed is sufficient I'm
ali@86
   300
	 * far from clear.
ali@86
   301
	 */
ali@86
   302
	if (helper->alternate_installed)
ali@86
   303
	    plover_transaction_set_installed(transaction,
ali@86
   304
	      helper->alternate_installed);
ali@86
   305
	if (transaction && !plover_transaction_remove(transaction,NULL,&error))
ali@86
   306
	{
ali@86
   307
	    g_object_unref(transaction);
ali@86
   308
	    transaction=NULL;
ali@86
   309
	}
ali@38
   310
	if (transaction)
ali@38
   311
	{
ali@38
   312
	    save_transactions=helper->transactions;
ali@38
   313
	    helper->transactions=NULL;
ali@38
   314
	    save_transaction_type=priv->transaction_type;
ali@38
   315
	    priv->transaction_type=0;
ali@38
   316
	    if (!plover_transaction_helper_add_transaction(helper,transaction,
ali@38
   317
	      NULL,PLOVER_TRANSACTION_HELPER_REPORT_REMOVE,&error))
ali@38
   318
	    {
ali@38
   319
		g_object_unref(transaction);
ali@38
   320
		transaction=NULL;
ali@38
   321
		helper->transactions=save_transactions;
ali@38
   322
		priv->transaction_type=save_transaction_type;
ali@38
   323
	    }
ali@38
   324
	    else
ali@38
   325
	    {
ali@38
   326
		g_slist_foreach(save_transactions,(GFunc)g_object_unref,NULL);
ali@38
   327
		g_slist_free(save_transactions);
ali@38
   328
	    }
ali@38
   329
	}
ali@24
   330
	if (!transaction)
ali@24
   331
	{
ali@24
   332
	    if (g_error_matches(error,PLOVER_POSIX_ERROR,ENOENT))
ali@24
   333
		g_clear_error(&error);
ali@24
   334
	    if (error)
ali@24
   335
	    {
ali@24
   336
		plover_transaction_helper_set_error(helper,error,
ali@38
   337
		  "Failed to remove existing packages");
ali@24
   338
		g_error_free(error);
ali@24
   339
		return;
ali@24
   340
	    }
ali@24
   341
	}
ali@24
   342
    }
ali@24
   343
    /*
ali@24
   344
     * Note that PloverTransaction does support cancelling a transaction, but
ali@24
   345
     * there are a number of challenges with using it:
ali@24
   346
     *	- cancellation is only supported during the file phase if razor
ali@24
   347
     *	  has atomic rollback,
ali@24
   348
     *  - cancellation is not supported during post-transaction scripts at all
ali@24
   349
     *    (since by the time the first script is started the atomic has already
ali@24
   350
     *    been committed) and these can take quite some time,
ali@24
   351
     *  - where a transaction has an embedded COMMIT, any rollback won't
ali@24
   352
     *    go back beyond this point.
ali@24
   353
     * To support user-cancel, then, we would need some mechanism to:
ali@24
   354
     *  - Comunicate that the operation is being cancelled and this may take
ali@24
   355
     *    some time,
ali@24
   356
     *  - Not allow cancellation at all after the last post-transaction script
ali@24
   357
     *    phase is started,
ali@24
   358
     *  - Report the partially completed transaction where cancellation
ali@24
   359
     *    occurred after a COMMIT point.
ali@24
   360
     * At present, this doesn't appear worth the effort.
ali@24
   361
     */
ali@24
   362
    if (helper->assistant)
ali@24
   363
	gtk_assistant_commit(helper->assistant);
ali@24
   364
    plover_transaction_helper_run(helper);
ali@24
   365
    helper->pulse_handler=g_timeout_add(100,plover_transaction_helper_pulse,
ali@24
   366
      helper);
ali@24
   367
}
ali@24
   368
ali@24
   369
static void plover_transaction_helper_assistant_prepare(GtkAssistant *assistant,
ali@24
   370
  GtkWidget *page,PloverTransactionHelper *helper)
ali@24
   371
{
ali@24
   372
    if (page==GTK_WIDGET(gtk_builder_get_object(helper->ui,"SIConfirm")))
ali@24
   373
	plover_transaction_helper_prepare_confirm(helper);
ali@24
   374
    else if (page==GTK_WIDGET(gtk_builder_get_object(helper->ui,"SIProgress")))
ali@24
   375
	plover_transaction_helper_prepare_progress(helper);
ali@24
   376
}
ali@24
   377
ali@24
   378
static void
ali@24
   379
  plover_transaction_helper_remove_existing_toggled(GtkToggleButton *button,
ali@24
   380
  PloverTransactionHelper *helper)
ali@24
   381
{
ali@24
   382
    GtkWidget *w;
ali@24
   383
    if (helper->assistant)
ali@24
   384
    {
ali@24
   385
	w=GTK_WIDGET(gtk_builder_get_object(helper->ui,"SIConfirm"));
ali@24
   386
	gtk_assistant_set_page_complete(helper->assistant,w,
ali@24
   387
	  gtk_toggle_button_get_active(button));
ali@24
   388
    }
ali@24
   389
}
ali@24
   390
ali@24
   391
PloverTransactionHelper *plover_transaction_helper_new(GtkBuilder *ui)
ali@24
   392
{
ali@24
   393
    gsize len;
ali@24
   394
    gchar *s,*directory,*contents;
ali@24
   395
    GError *error=NULL;
ali@24
   396
    GtkWidget *w;
ali@24
   397
    PloverTransactionHelper *helper;
ali@24
   398
    g_return_val_if_fail(ui == NULL || GTK_IS_BUILDER(ui),NULL);
ali@24
   399
    helper=PLOVER_TRANSACTION_HELPER(
ali@24
   400
      g_object_new(PLOVER_TYPE_TRANSACTION_HELPER,NULL));
ali@24
   401
    if (ui)
ali@24
   402
	helper->ui=g_object_ref(ui);
ali@24
   403
    else
ali@24
   404
	helper->ui=gtk_builder_new();
ali@24
   405
    helper->assistant=
ali@24
   406
      GTK_ASSISTANT(gtk_builder_get_object(helper->ui,"SoftwareInstallation"));
ali@24
   407
    if (!helper->assistant)
ali@24
   408
    {
ali@24
   409
	directory=g_strdup(g_getenv("PLOVER_DATADIR"));
ali@24
   410
	if (!directory)
ali@24
   411
	{
ali@24
   412
#ifdef WIN32
ali@24
   413
	    s=g_win32_get_package_installation_directory_of_module(NULL);
ali@24
   414
	    directory=g_build_filename(s,"share","plover",NULL);
ali@24
   415
	    g_free(s);
ali@24
   416
#else
ali@24
   417
	    directory=g_strdup(PLOVER_DATADIR);
ali@24
   418
#endif
ali@24
   419
	}
ali@24
   420
	s=g_build_filename(directory,"software-installation.ui",NULL);
ali@24
   421
	g_free(directory);
ali@24
   422
	(void)g_file_get_contents(s,&contents,&len,&error);
ali@24
   423
	g_free(s);
ali@24
   424
	if (!error)
ali@24
   425
	{
ali@24
   426
	    (void)gtk_builder_add_from_string(helper->ui,contents,len,&error);
ali@24
   427
	    g_free(contents);
ali@24
   428
	}
ali@24
   429
	if (error)
ali@24
   430
	{
ali@24
   431
	    g_critical("software-installation.ui: %s",error->message);
ali@24
   432
	    g_clear_error(&error);
ali@24
   433
	    g_set_error(&error,PLOVER_GENERAL_ERROR,PLOVER_GENERAL_ERROR_FAILED,
ali@24
   434
	      "Internal error (no user interface)");
ali@24
   435
	    plover_transaction_helper_set_error(helper,error,
ali@24
   436
	      "Can't start installer");
ali@24
   437
	    return helper;
ali@24
   438
	}
ali@24
   439
	helper->assistant=GTK_ASSISTANT(gtk_builder_get_object(helper->ui,
ali@24
   440
	  "SoftwareInstallation"));
ali@24
   441
    }
ali@24
   442
    if (!helper->assistant)
ali@24
   443
    {
ali@24
   444
	g_critical("\"SoftwareInstallation\" object not found");
ali@24
   445
	g_set_error(&error,PLOVER_GENERAL_ERROR,PLOVER_GENERAL_ERROR_FAILED,
ali@24
   446
	  "Internal error (missing wizard)");
ali@24
   447
	plover_transaction_helper_set_error(helper,error,
ali@24
   448
	  "Can't start installer");
ali@24
   449
	g_error_free(error);
ali@24
   450
	return helper;
ali@24
   451
    }
ali@24
   452
    else
ali@24
   453
	g_object_ref(helper->assistant);
ali@24
   454
    if (!GTK_IS_ASSISTANT(helper->assistant))
ali@24
   455
    {
ali@24
   456
	g_critical("\"SoftwareInstallation\" is not a GtkAssistant");
ali@24
   457
	g_set_error(&error,PLOVER_GENERAL_ERROR,PLOVER_GENERAL_ERROR_FAILED,
ali@24
   458
	  "Internal error (unexpected wizard type)");
ali@24
   459
	plover_transaction_helper_set_error(helper,error,
ali@24
   460
	  "Can't start installer");
ali@24
   461
	g_error_free(error);
ali@24
   462
	return helper;
ali@24
   463
    }
ali@24
   464
    g_signal_connect(helper->assistant,"cancel",
ali@24
   465
      G_CALLBACK(plover_transaction_helper_assistant_cancel),helper);
ali@24
   466
    g_signal_connect(helper->assistant,"close",
ali@24
   467
      G_CALLBACK(plover_transaction_helper_assistant_close),helper);
ali@24
   468
    g_signal_connect(helper->assistant,"prepare",
ali@24
   469
      G_CALLBACK(plover_transaction_helper_assistant_prepare),helper);
ali@24
   470
    w=GTK_WIDGET(gtk_builder_get_object(helper->ui,"SIRemoveExisting"));
ali@24
   471
    if (w)
ali@24
   472
	g_signal_connect(w,"toggled",
ali@24
   473
	  G_CALLBACK(plover_transaction_helper_remove_existing_toggled),helper);
ali@24
   474
    w=GTK_WIDGET(gtk_builder_get_object(helper->ui,"SIIntroduction"));
ali@24
   475
    if (w)
ali@24
   476
	gtk_assistant_set_page_complete(helper->assistant,w,TRUE);
ali@24
   477
    return helper;
ali@24
   478
}
ali@24
   479
ali@24
   480
PloverPackageSet *
ali@24
   481
  plover_transaction_helper_get_installed(PloverTransactionHelper *helper)
ali@24
   482
{
ali@61
   483
    gchar *s,*saved_database_uri;
ali@61
   484
    char *install_root,*local_database,*active_database,*alternate_database;
ali@61
   485
    const char *prefix;
ali@61
   486
    struct comps *comps;
ali@61
   487
    PloverPackageSet *alternate_installed,*installed;
ali@61
   488
    GError *error=NULL;
ali@61
   489
    struct razor_error *razor_error=NULL;
ali@24
   490
    g_return_val_if_fail(PLOVER_IS_TRANSACTION_HELPER(helper),NULL);
ali@61
   491
    if (!helper->installed)
ali@61
   492
    {
ali@61
   493
	comps=plover_transaction_helper_get_comps(helper,NULL);
ali@61
   494
	if (!comps)
ali@61
   495
	{
ali@61
   496
	    g_warning("plover_transaction_helper_get_installed: No comps");
ali@61
   497
	    return NULL;
ali@61
   498
	}
ali@61
   499
	install_root=getenv("RAZOR_ROOT");
ali@61
   500
	if (!install_root)
ali@61
   501
	    install_root="file:/";
ali@61
   502
	prefix=plover_transaction_helper_get_prefix(helper,NULL);
ali@61
   503
	if (prefix)
ali@61
   504
	{
ali@61
   505
	    s=g_strconcat(prefix,"/var/lib/razor",NULL);
ali@61
   506
	    local_database=razor_path_relative_to_uri(install_root,*s=='/'?s+1:s,
ali@61
   507
	      &razor_error);
ali@61
   508
	    g_free(s);
ali@61
   509
	    if (!local_database)
ali@61
   510
	    {
ali@61
   511
		g_warning("plover_transaction_helper_get_installed: %s",
ali@61
   512
		  razor_error_get_msg(razor_error));
ali@61
   513
		razor_error_free(razor_error);
ali@61
   514
		return NULL;
ali@61
   515
	    }
ali@61
   516
	}
ali@61
   517
	else
ali@61
   518
	    local_database=NULL;
ali@61
   519
	switch(comps->database)
ali@61
   520
	{
ali@61
   521
	    case COMPS_DATABASE_DISTRIBUTION_LOCAL:
ali@61
   522
		active_database=local_database;
ali@61
   523
		alternate_database=NULL;
ali@61
   524
		break;
ali@61
   525
	    case COMPS_DATABASE_GLOBAL:
ali@61
   526
		active_database=NULL;
ali@61
   527
		alternate_database=local_database;
ali@61
   528
		break;
ali@61
   529
	}
ali@61
   530
	saved_database_uri=g_strdup(razor_get_database_uri());
ali@61
   531
	if (prefix)
ali@61
   532
	{
ali@61
   533
	    razor_set_database_uri(alternate_database);
ali@61
   534
	    alternate_installed=plover_package_set_new();
ali@61
   535
	    if (!plover_package_set_open(alternate_installed,install_root,TRUE,
ali@61
   536
	      &error))
ali@61
   537
	    {
ali@61
   538
		g_object_unref(alternate_installed);
ali@61
   539
		g_warning("plover_transaction_helper_get_installed: %s",
ali@61
   540
		  error->message);
ali@61
   541
		g_error_free(error);
ali@61
   542
		free(local_database);
ali@61
   543
		razor_set_database_uri(saved_database_uri);
ali@61
   544
		g_free(saved_database_uri);
ali@61
   545
		return NULL;
ali@61
   546
	    }
ali@61
   547
	}
ali@61
   548
	else
ali@61
   549
	    alternate_installed=NULL;
ali@61
   550
	razor_set_database_uri(active_database);
ali@61
   551
	free(local_database);
ali@61
   552
	installed=plover_package_set_new();
ali@61
   553
	if (plover_package_set_open(installed,install_root,TRUE,&error))
ali@61
   554
	{
ali@61
   555
	    helper->alternate_installed=alternate_installed;
ali@61
   556
	    helper->installed=installed;
ali@61
   557
	}
ali@61
   558
	else
ali@61
   559
	{
ali@61
   560
	    g_object_unref(installed);
ali@61
   561
	    if (alternate_installed)
ali@61
   562
		g_object_unref(alternate_installed);
ali@61
   563
	    g_warning("plover_transaction_helper_get_installed: %s",error->message);
ali@61
   564
	    g_error_free(error);
ali@61
   565
	}
ali@61
   566
	razor_set_database_uri(saved_database_uri);
ali@61
   567
	g_free(saved_database_uri);
ali@61
   568
    }
ali@24
   569
    return helper->installed;
ali@24
   570
}
ali@24
   571
ali@24
   572
void plover_transaction_helper_set_installed(PloverTransactionHelper *helper,
ali@24
   573
  PloverPackageSet *installed)
ali@24
   574
{
ali@24
   575
    g_return_if_fail(PLOVER_IS_TRANSACTION_HELPER(helper));
ali@24
   576
    g_return_if_fail(PLOVER_IS_PACKAGE_SET(installed));
ali@24
   577
    g_return_if_fail(helper->installed == NULL);
ali@61
   578
    g_clear_object(&helper->alternate_installed);
ali@24
   579
    helper->installed=g_object_ref(installed);
ali@24
   580
}
ali@24
   581
ali@24
   582
PloverRepository *
ali@24
   583
  plover_transaction_helper_get_upstream(PloverTransactionHelper *helper,
ali@24
   584
  GError **error)
ali@24
   585
{
ali@24
   586
    const char *base;
ali@24
   587
    g_return_val_if_fail(PLOVER_IS_TRANSACTION_HELPER(helper),NULL);
ali@24
   588
    if (!helper->upstream)
ali@24
   589
    {
ali@24
   590
	base=plover_transaction_helper_get_base(helper);
ali@24
   591
	helper->upstream=plover_repository_new_from_yum(base,error);
ali@24
   592
    }
ali@24
   593
    return helper->upstream;
ali@24
   594
}
ali@24
   595
ali@24
   596
void plover_transaction_helper_set_upstream(PloverTransactionHelper *helper,
ali@24
   597
  PloverRepository *upstream)
ali@24
   598
{
ali@24
   599
    g_return_if_fail(PLOVER_IS_TRANSACTION_HELPER(helper));
ali@24
   600
    g_return_if_fail(PLOVER_IS_REPOSITORY(upstream));
ali@24
   601
    g_return_if_fail(helper->upstream == NULL);
ali@24
   602
    helper->upstream=g_object_ref(upstream);
ali@24
   603
}
ali@24
   604
ali@24
   605
const char *plover_transaction_helper_get_base(PloverTransactionHelper *helper)
ali@24
   606
{
ali@24
   607
    g_return_val_if_fail(PLOVER_IS_TRANSACTION_HELPER(helper),NULL);
ali@24
   608
    return helper->base;
ali@24
   609
}
ali@24
   610
ali@24
   611
void plover_transaction_helper_set_base(PloverTransactionHelper *helper,
ali@24
   612
  const char *base)
ali@24
   613
{
ali@24
   614
    g_return_if_fail(PLOVER_IS_TRANSACTION_HELPER(helper));
ali@24
   615
    g_return_if_fail(helper->transactions == NULL);
ali@24
   616
    g_free(helper->base);
ali@24
   617
    helper->base=g_strdup(base);
ali@24
   618
}
ali@24
   619
ali@24
   620
struct comps *
ali@24
   621
  plover_transaction_helper_get_comps(PloverTransactionHelper *helper,
ali@24
   622
  GError **error)
ali@24
   623
{
ali@24
   624
    gchar *s;
ali@24
   625
    g_return_val_if_fail(PLOVER_IS_TRANSACTION_HELPER(helper),NULL);
ali@75
   626
    if (!helper->comps && helper->base)
ali@24
   627
    {
ali@24
   628
	s=g_strconcat(helper->base,"/repodata/comps.xml",NULL);
ali@24
   629
	helper->comps=plover_comps_new_from_file(s);
ali@24
   630
	if (!helper->comps)
ali@24
   631
	    g_set_error(error,PLOVER_GENERAL_ERROR,
ali@24
   632
	      PLOVER_GENERAL_ERROR_FAILED,"%s: %s",s,g_strerror(errno));
ali@24
   633
	g_free(s);
ali@24
   634
    }
ali@24
   635
    return helper->comps;
ali@24
   636
}
ali@24
   637
ali@24
   638
const char *
ali@24
   639
  plover_transaction_helper_get_prefix(PloverTransactionHelper *helper,
ali@24
   640
  GError **error)
ali@24
   641
{
ali@24
   642
    const char *prefix;
ali@24
   643
    struct comps *comps;
ali@38
   644
    PloverTransactionHelperPrivate *priv;
ali@24
   645
    g_return_val_if_fail(PLOVER_IS_TRANSACTION_HELPER(helper),NULL);
ali@61
   646
    g_return_val_if_fail(helper->base != NULL || plover_transaction_helper_get_installed(helper) != NULL,NULL);
ali@38
   647
    priv=PLOVER_TRANSACTION_HELPER_GET_PRIVATE(helper);
ali@24
   648
    if (helper->base)
ali@24
   649
    {
ali@24
   650
	comps=plover_transaction_helper_get_comps(helper,error);
ali@24
   651
	if (!comps)
ali@24
   652
	    return NULL;
ali@38
   653
	g_free(priv->default_prefix);
ali@61
   654
	priv->default_prefix=plover_comps_get_default_prefix(comps);
ali@38
   655
	return priv->default_prefix;
ali@24
   656
    }
ali@24
   657
    prefix=plover_package_set_guess_prefix(helper->installed,error);
ali@24
   658
    return prefix;
ali@24
   659
}
ali@24
   660
ali@61
   661
#if 0
ali@24
   662
static int plover_transaction_helper_package_count(void)
ali@24
   663
{
ali@24
   664
    int count=0;
ali@24
   665
    char *install_root;
ali@24
   666
    struct razor_set *set;
ali@24
   667
    struct razor_package *package;
ali@24
   668
    struct razor_package_iterator *pi;
ali@24
   669
    install_root=getenv("RAZOR_ROOT");
ali@24
   670
    if (!install_root)
ali@24
   671
	install_root="";
ali@24
   672
    set=razor_root_open_read_only(install_root,NULL);
ali@24
   673
    if (set)
ali@24
   674
    {
ali@24
   675
	pi=razor_package_iterator_create(set);
ali@24
   676
	while (razor_package_iterator_next(pi,&package,RAZOR_DETAIL_LAST))
ali@24
   677
	    count++;
ali@24
   678
	razor_package_iterator_destroy(pi);
ali@24
   679
	razor_set_unref(set);
ali@24
   680
    }
ali@24
   681
    return count;
ali@24
   682
}
ali@61
   683
#endif
ali@61
   684
ali@61
   685
static gboolean prefix_clashes(const char *prefix,const char *alt)
ali@61
   686
{
ali@61
   687
    return g_str_has_prefix(prefix,alt) &&
ali@61
   688
      (prefix[strlen(alt)]=='\0' || prefix[strlen(alt)]=='/');
ali@61
   689
}
ali@24
   690
ali@76
   691
static void
ali@76
   692
  plover_transaction_helper_update_summary_page(PloverTransactionHelper *helper,
ali@24
   693
  GError **error)
ali@24
   694
{
ali@61
   695
    int i,remove_count=0;
ali@61
   696
    gboolean alternate_database_clashes=FALSE;
ali@61
   697
    gboolean active_database_is_incompatible=FALSE;
ali@61
   698
    char *local_database,*active_database,*alternate_database;
ali@61
   699
    const char *alternate_prefix;
ali@24
   700
    gchar *prefix=NULL,*s;
ali@24
   701
    struct comps *comps=NULL;
ali@38
   702
    GtkWidget *container,*summary,*page;
ali@24
   703
    GtkButton *button;
ali@24
   704
    GtkLabel *label;
ali@76
   705
    g_return_if_fail(PLOVER_IS_TRANSACTION_HELPER(helper));
ali@61
   706
    comps=plover_transaction_helper_get_comps(helper,error);
ali@76
   707
    if (comps)
ali@76
   708
	prefix=plover_comps_get_default_prefix(comps);
ali@76
   709
    else
ali@76
   710
	prefix=NULL;
ali@24
   711
    button=GTK_BUTTON(gtk_builder_get_object(helper->ui,"SIRemoveExisting"));
ali@24
   712
    gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(button),FALSE);
ali@24
   713
    container=GTK_WIDGET(gtk_builder_get_object(helper->ui,
ali@24
   714
      "SIIncompatibleInstallation"));
ali@38
   715
    summary=GTK_WIDGET(gtk_builder_get_object(helper->ui,"SISummaryOfWork"));
ali@24
   716
    page=GTK_WIDGET(gtk_builder_get_object(helper->ui,"SIConfirm"));
ali@61
   717
    if (helper->check_vendor && prefix && helper->alternate_installed)
ali@61
   718
    {
ali@61
   719
	alternate_prefix=
ali@61
   720
	  plover_package_set_guess_prefix(helper->alternate_installed,NULL);
ali@61
   721
	if (alternate_prefix && prefix_clashes(prefix,alternate_prefix))
ali@61
   722
	{
ali@61
   723
	    alternate_database_clashes=TRUE;
ali@61
   724
	    remove_count=g_slist_length(
ali@61
   725
	      plover_package_set_get_packages(helper->alternate_installed));
ali@61
   726
	}
ali@61
   727
    }
ali@61
   728
    /*
ali@61
   729
     * Rather than try to be too clever, we only deal with one thing
ali@61
   730
     * at a time. That means that if the alternate database clashes
ali@61
   731
     * there's no point checking if the active database is compatible.
ali@61
   732
     */
ali@61
   733
    if (!alternate_database_clashes)
ali@61
   734
    {
ali@61
   735
	if (helper->check_vendor && prefix &&
ali@61
   736
	  !plover_package_set_files_match_prefix(helper->installed,prefix))
ali@61
   737
	{
ali@61
   738
	    active_database_is_incompatible=TRUE;
ali@61
   739
	    remove_count=
ali@61
   740
	      g_slist_length(plover_package_set_get_packages(helper->installed));
ali@61
   741
	}
ali@61
   742
    }
ali@61
   743
    if (alternate_database_clashes || active_database_is_incompatible)
ali@24
   744
    {
ali@76
   745
	g_assert(comps!=NULL);
ali@24
   746
	label=GTK_LABEL(gtk_builder_get_object(helper->ui,
ali@24
   747
	  "SIIncompatibleInstallationLabel"));
ali@61
   748
	if (alternate_database_clashes)
ali@61
   749
	    s=g_strdup_printf("<b>Incompatible Installation</b>\n\n"
ali@61
   750
	      "There is an existing installation under %s\n"
ali@61
   751
	      "which is not compatible with this distribution. In order\n"
ali@61
   752
	      "to continue, the existing installation must be uninstalled.",
ali@61
   753
	      comps->vendor);
ali@61
   754
	else /* active_database_is_incompatible */
ali@61
   755
	    s=g_strdup_printf("<b>Incompatible Installation</b>\n\n"
ali@61
   756
	      "The existing installation is not from %s.\n In order "
ali@61
   757
	      "to continue, all the existing packages must be removed.",
ali@61
   758
	      comps->vendor);
ali@24
   759
	gtk_label_set_markup(label,s);
ali@24
   760
	g_free(s);
ali@61
   761
	s=g_strdup_printf("Remove %d existing package%s",remove_count,
ali@61
   762
	  remove_count==1?"":"s");
ali@24
   763
	gtk_button_set_label(button,s);
ali@24
   764
	g_free(s);
ali@24
   765
	gtk_widget_show(container);
ali@38
   766
	gtk_widget_hide(summary);
ali@24
   767
	if (helper->assistant)
ali@24
   768
	    gtk_assistant_set_page_complete(helper->assistant,page,FALSE);
ali@24
   769
    }
ali@24
   770
    else
ali@24
   771
    {
ali@24
   772
	gtk_widget_hide(container);
ali@38
   773
	gtk_widget_show(summary);
ali@24
   774
	if (helper->assistant)
ali@24
   775
	    gtk_assistant_set_page_complete(helper->assistant,page,TRUE);
ali@24
   776
    }
ali@38
   777
    g_free(prefix);
ali@24
   778
}
ali@24
   779
ali@24
   780
void plover_transaction_helper_set_check_vendor(PloverTransactionHelper *helper,
ali@24
   781
  gboolean check_vendor)
ali@24
   782
{
ali@24
   783
    g_return_if_fail(PLOVER_IS_TRANSACTION_HELPER(helper));
ali@24
   784
    if (helper->check_vendor!=check_vendor)
ali@24
   785
    {
ali@24
   786
	helper->check_vendor=check_vendor;
ali@24
   787
	if (helper->transactions)
ali@76
   788
	    plover_transaction_helper_update_summary_page(helper,NULL);
ali@24
   789
    }
ali@24
   790
}
ali@24
   791
ali@24
   792
/*
ali@75
   793
 * If plover_transaction_helper_add_transaction() fails with an error
ali@24
   794
 * of PLOVER_GENERAL_ERROR,PLOVER_GENERAL_ERROR_REQUIREMENTS_NOT_MET
ali@24
   795
 * then plover_transaction_helper_get_unsatisfied() can be used to
ali@24
   796
 * retrieve a textual description of the problem.
ali@24
   797
 */
ali@24
   798
ali@24
   799
const char *
ali@24
   800
  plover_transaction_helper_get_unsatisfied(PloverTransactionHelper *helper)
ali@24
   801
{
ali@24
   802
    g_return_val_if_fail(PLOVER_IS_TRANSACTION_HELPER(helper),NULL);
ali@24
   803
    return helper->unsatisfied;
ali@24
   804
}
ali@24
   805
ali@38
   806
#define PLOVER_TRANSACTION_HELPER_IS_VALID_REPORT_ACTION(action) \
ali@38
   807
  ((action)==PLOVER_TRANSACTION_HELPER_REPORT_INSTALL || \
ali@38
   808
  (action)==PLOVER_TRANSACTION_HELPER_REPORT_REMOVE || \
ali@38
   809
  (action)==PLOVER_TRANSACTION_HELPER_REPORT_UPDATE)
ali@38
   810
ali@24
   811
gboolean
ali@24
   812
  plover_transaction_helper_add_transaction(PloverTransactionHelper *helper,
ali@24
   813
  PloverTransaction *transaction,struct plover_vector *report_packages,
ali@38
   814
  PloverTransactionHelperReportAction report_action,GError **error)
ali@24
   815
{
ali@24
   816
    int i,count;
ali@24
   817
    gboolean other_packages;
ali@24
   818
    const char *s,*name;
ali@24
   819
    enum razor_install_action action;
ali@24
   820
    struct razor_install_iterator *ii;
ali@38
   821
    struct razor_set *report_set;
ali@24
   822
    struct razor_package *package;
ali@24
   823
    struct plover_vector *tasked_packages;
ali@38
   824
    PloverTransactionHelperPrivate *priv;
ali@38
   825
    GtkWidget *w;
ali@24
   826
    g_return_val_if_fail(PLOVER_IS_TRANSACTION_HELPER(helper),FALSE);
ali@24
   827
    g_return_val_if_fail(PLOVER_IS_TRANSACTION(transaction),FALSE);
ali@38
   828
    g_return_val_if_fail(PLOVER_TRANSACTION_HELPER_IS_VALID_REPORT_ACTION(report_action),FALSE);
ali@38
   829
    g_return_val_if_fail(plover_transaction_get_system_set(transaction)!=NULL,FALSE);
ali@38
   830
    priv=PLOVER_TRANSACTION_HELPER_GET_PRIVATE(helper);
ali@24
   831
    g_free(helper->unsatisfied);
ali@24
   832
    helper->unsatisfied=NULL;
ali@24
   833
    if (!plover_transaction_resolve(transaction,error))
ali@24
   834
    {
ali@24
   835
	s=plover_transaction_get_unsatisfied(transaction);
ali@24
   836
	helper->unsatisfied=g_strdup(s);
ali@24
   837
	return FALSE;
ali@24
   838
    }
ali@24
   839
    ii=plover_transaction_get_install_iterator(transaction,error);
ali@24
   840
    if (!ii)
ali@24
   841
	return FALSE;
ali@38
   842
    if (report_action==PLOVER_TRANSACTION_HELPER_REPORT_REMOVE)
ali@38
   843
	report_set=plover_transaction_get_system_set(transaction);
ali@38
   844
    else
ali@38
   845
	report_set=plover_transaction_get_next_set(transaction,error);
ali@38
   846
    if (!report_set)
ali@24
   847
	return FALSE;
ali@24
   848
    tasked_packages=plover_vector_new();
ali@24
   849
    other_packages=FALSE;
ali@24
   850
    while (razor_install_iterator_next(ii,&package,&action,&count))
ali@24
   851
    {
ali@38
   852
	if (action==report_action || action==RAZOR_INSTALL_ACTION_ADD &&
ali@38
   853
	  report_action==PLOVER_TRANSACTION_HELPER_REPORT_UPDATE)
ali@24
   854
	{
ali@38
   855
	    razor_package_get_details(report_set,package,RAZOR_DETAIL_NAME,
ali@38
   856
	      &name,RAZOR_DETAIL_LAST);
ali@24
   857
	    if (!report_packages ||
ali@24
   858
	      plover_vector_contains(report_packages,name))
ali@24
   859
		plover_vector_append(tasked_packages,name);
ali@24
   860
	    else
ali@24
   861
		other_packages=TRUE;
ali@24
   862
	}
ali@24
   863
    }
ali@24
   864
    if (!tasked_packages->len)
ali@24
   865
    {
ali@24
   866
	/*
ali@24
   867
	 * If there are no reportable packages tasked for action there
ali@24
   868
	 * shouldn't by any packages at all, but let's be paranoid.
ali@24
   869
	 */
ali@24
   870
	other_packages=FALSE;
ali@24
   871
	razor_install_iterator_rewind(ii);
ali@24
   872
	while (razor_install_iterator_next(ii,&package,&action,&count))
ali@24
   873
	{
ali@24
   874
	    if (action==report_action)
ali@24
   875
	    {
ali@38
   876
		razor_package_get_details(report_set,package,RAZOR_DETAIL_NAME,
ali@38
   877
		  &name,RAZOR_DETAIL_LAST);
ali@24
   878
		plover_vector_append(tasked_packages,name);
ali@24
   879
	    }
ali@24
   880
	}
ali@24
   881
    }
ali@24
   882
    if (!tasked_packages->len)
ali@24
   883
    {
ali@24
   884
	g_set_error(error,PLOVER_GENERAL_ERROR,
ali@24
   885
	  PLOVER_GENERAL_ERROR_NO_WORK,"Transaction includes no %s actions",
ali@38
   886
	  report_action==PLOVER_TRANSACTION_HELPER_REPORT_REMOVE?
ali@38
   887
	  "remove":"add");
ali@24
   888
	plover_vector_free(tasked_packages);
ali@24
   889
	return FALSE;
ali@24
   890
    }
ali@24
   891
    if (!helper->transactions)
ali@76
   892
	plover_transaction_helper_update_summary_page(helper,error);
ali@24
   893
    g_object_ref(transaction);
ali@24
   894
    helper->transactions=g_slist_append(helper->transactions,transaction);
ali@38
   895
    if (report_action==PLOVER_TRANSACTION_HELPER_REPORT_REMOVE)
ali@24
   896
    {
ali@38
   897
	priv->transaction_type|=TRANSACTION_TYPE_REMOVE;
ali@38
   898
	for(i=0;i<tasked_packages->len;i++)
ali@38
   899
	{
ali@38
   900
	    s=tasked_packages->strings[i];
ali@38
   901
	    if (!plover_vector_contains(helper->report_removing,s))
ali@38
   902
		plover_vector_append(helper->report_removing,s);
ali@38
   903
	}
ali@38
   904
	helper->report_removing_dependants|=other_packages;
ali@38
   905
    }
ali@38
   906
    else
ali@38
   907
    {
ali@38
   908
	if (report_action==PLOVER_TRANSACTION_HELPER_REPORT_UPDATE)
ali@38
   909
	    priv->transaction_type|=TRANSACTION_TYPE_REMOVE;
ali@38
   910
	priv->transaction_type|=TRANSACTION_TYPE_INSTALL;
ali@24
   911
	for(i=0;i<tasked_packages->len;i++)
ali@24
   912
	{
ali@24
   913
	    s=tasked_packages->strings[i];
ali@24
   914
	    if (!plover_vector_contains(helper->report_adding,s))
ali@24
   915
		plover_vector_append(helper->report_adding,s);
ali@24
   916
	}
ali@24
   917
	helper->report_adding_dependencies|=other_packages;
ali@24
   918
    }
ali@38
   919
    w=GTK_WIDGET(gtk_builder_get_object(helper->ui,"SIProgressLabel"));
ali@38
   920
    switch(priv->transaction_type)
ali@24
   921
    {
ali@38
   922
	case TRANSACTION_TYPE_INSTALL:
ali@38
   923
	    gtk_label_set_markup(GTK_LABEL(w),
ali@38
   924
	      "<b>Installing the Software</b>\n\n"
ali@38
   925
	      "Please wait while the Installation Assistant "
ali@38
   926
	      "installs the software.\n"
ali@38
   927
	      "This may take several minutes.");
ali@38
   928
	    break;
ali@38
   929
	case TRANSACTION_TYPE_REMOVE:
ali@38
   930
	    gtk_label_set_markup(GTK_LABEL(w),
ali@38
   931
	      "<b>Removing Packages</b>\n\n"
ali@38
   932
	      "Please wait while the Installation Assistant "
ali@38
   933
	      "removes packages.\n"
ali@38
   934
	      "This may take several minutes.");
ali@38
   935
	    break;
ali@38
   936
	default:
ali@38
   937
	case TRANSACTION_TYPE_UPDATE:
ali@38
   938
	    gtk_label_set_markup(GTK_LABEL(w),
ali@38
   939
	      "<b>Updating the Software</b>\n\n"
ali@38
   940
	      "Please wait while the Installation Assistant "
ali@38
   941
	      "updates the software.\n"
ali@38
   942
	      "This may take several minutes.");
ali@38
   943
	    break;
ali@24
   944
    }
ali@24
   945
    plover_vector_free(tasked_packages);
ali@24
   946
    return TRUE;
ali@24
   947
}
ali@24
   948
ali@24
   949
static PloverTransaction *
ali@24
   950
  plover_transaction_helper_new_transaction(PloverTransactionHelper *helper,
ali@24
   951
  GError **error)
ali@24
   952
{
ali@24
   953
    gboolean ok;
ali@24
   954
    const char *base,*prefix;
ali@24
   955
    GError *tmp_error=NULL;
ali@24
   956
    PloverTransaction *transaction;
ali@24
   957
    g_return_val_if_fail(PLOVER_IS_TRANSACTION_HELPER(helper),NULL);
ali@61
   958
    g_return_val_if_fail(plover_transaction_helper_get_installed(helper) != NULL,NULL);
ali@24
   959
    prefix=plover_transaction_helper_get_prefix(helper,&tmp_error);
ali@24
   960
    if (tmp_error)
ali@24
   961
    {
ali@24
   962
	g_propagate_error(error,tmp_error);
ali@24
   963
	return NULL;
ali@24
   964
    }
ali@24
   965
    transaction=plover_transaction_new();
ali@24
   966
    plover_transaction_set_prefix(transaction,prefix);
ali@24
   967
    plover_transaction_set_installed(transaction,helper->installed);
ali@24
   968
    if (helper->upstream)
ali@24
   969
	ok=plover_transaction_set_upstream(transaction,helper->upstream,error);
ali@24
   970
    else
ali@24
   971
    {
ali@24
   972
	base=plover_transaction_helper_get_base(helper);
ali@24
   973
	ok=plover_transaction_set_upstream_from_yum(transaction,base,error);
ali@24
   974
    }
ali@24
   975
    if (!ok)
ali@24
   976
    {
ali@24
   977
	g_object_unref(transaction);
ali@24
   978
	transaction=NULL;
ali@24
   979
    }
ali@24
   980
    return transaction;
ali@24
   981
}
ali@24
   982
ali@24
   983
struct plover_vector *plover_transaction_helper_group_get_default_packages(
ali@24
   984
  PloverTransactionHelper *helper,const char *group,GError **error)
ali@24
   985
{
ali@24
   986
    gboolean changed;
ali@24
   987
    struct comps *comps;
ali@24
   988
    struct comps_group *grp;
ali@24
   989
    struct comps_requirement *pkg;
ali@24
   990
    struct plover_vector *default_packages;
ali@24
   991
    g_return_val_if_fail(PLOVER_IS_TRANSACTION_HELPER(helper),FALSE);
ali@24
   992
    comps=plover_transaction_helper_get_comps(helper,error);
ali@24
   993
    if (!comps)
ali@24
   994
	return NULL;
ali@24
   995
    grp=plover_comps_lookup_group(comps,group);
ali@24
   996
    if (!grp)
ali@24
   997
    {
ali@24
   998
	g_set_error(error,PLOVER_GENERAL_ERROR,
ali@24
   999
	  PLOVER_GENERAL_ERROR_FAILED,"%s: group not found",group);
ali@24
  1000
	return NULL;
ali@24
  1001
    }
ali@24
  1002
    default_packages=plover_vector_new();
ali@24
  1003
    do
ali@24
  1004
    {
ali@24
  1005
	changed=FALSE;
ali@24
  1006
	for(pkg=grp->packages;pkg;pkg=pkg->next)
ali@24
  1007
	{
ali@24
  1008
	    if (plover_vector_contains(default_packages,pkg->name))
ali@24
  1009
		continue;
ali@24
  1010
	    if (pkg->type==COMPS_REQUIREMENT_DEFAULT ||
ali@24
  1011
	      pkg->type==COMPS_REQUIREMENT_MANDATORY ||
ali@38
  1012
	      pkg->type==COMPS_REQUIREMENT_CONDITIONAL && pkg->requires &&
ali@24
  1013
	      plover_vector_contains(default_packages,pkg->requires))
ali@24
  1014
	    {
ali@24
  1015
		changed=TRUE;
ali@24
  1016
		plover_vector_append(default_packages,pkg->name);
ali@24
  1017
	    }
ali@24
  1018
	}
ali@24
  1019
    } while(changed);
ali@24
  1020
    return default_packages;
ali@24
  1021
}
ali@24
  1022
ali@24
  1023
/*
ali@24
  1024
 * Returns TRUE if there is work to be done or FALSE if the packages are
ali@24
  1025
 * already installed or on error.
ali@24
  1026
 */
ali@24
  1027
gboolean
ali@24
  1028
  plover_transaction_helper_install_packages(PloverTransactionHelper *helper,
ali@24
  1029
  struct plover_vector *packages,GError **error)
ali@24
  1030
{
ali@24
  1031
    gboolean retval;
ali@24
  1032
    PloverTransaction *transaction;
ali@24
  1033
    g_return_val_if_fail(PLOVER_IS_TRANSACTION_HELPER(helper),FALSE);
ali@24
  1034
    g_return_val_if_fail(packages != NULL,FALSE);
ali@24
  1035
    if (!packages->len)
ali@24
  1036
    {
ali@24
  1037
	g_set_error(error,PLOVER_GENERAL_ERROR,
ali@24
  1038
	  PLOVER_GENERAL_ERROR_NO_WORK,"No packages listed to be installed");
ali@24
  1039
	return FALSE;
ali@24
  1040
    }
ali@24
  1041
    transaction=plover_transaction_helper_new_transaction(helper,error);
ali@24
  1042
    if (!transaction)
ali@24
  1043
	return FALSE;
ali@24
  1044
    if (!plover_transaction_install(transaction,packages->strings,error))
ali@24
  1045
    {
ali@24
  1046
	g_object_unref(transaction);
ali@24
  1047
	return FALSE;
ali@24
  1048
    }
ali@24
  1049
    retval=plover_transaction_helper_add_transaction(helper,transaction,
ali@38
  1050
      packages,PLOVER_TRANSACTION_HELPER_REPORT_INSTALL,error);
ali@24
  1051
    g_object_unref(transaction);
ali@24
  1052
    return retval;
ali@24
  1053
}
ali@24
  1054
ali@24
  1055
/*
ali@24
  1056
 * Returns TRUE if there is work to be done or FALSE if the group is
ali@24
  1057
 * already installed or on error.
ali@24
  1058
 */
ali@24
  1059
gboolean
ali@24
  1060
  plover_transaction_helper_install_group(PloverTransactionHelper *helper,
ali@24
  1061
  const char *group,GError **error)
ali@24
  1062
{
ali@24
  1063
    gboolean retval;
ali@24
  1064
    struct plover_vector *selected_packages;
ali@24
  1065
    g_return_val_if_fail(PLOVER_IS_TRANSACTION_HELPER(helper),FALSE);
ali@24
  1066
    selected_packages=plover_transaction_helper_group_get_default_packages(
ali@24
  1067
      helper,group,error);
ali@24
  1068
    if (!selected_packages)
ali@24
  1069
	return FALSE;
ali@24
  1070
    if (!selected_packages->len)
ali@24
  1071
    {
ali@24
  1072
	g_set_error(error,PLOVER_GENERAL_ERROR,
ali@24
  1073
	  PLOVER_GENERAL_ERROR_FAILED,"%s: no default packages",group);
ali@24
  1074
	plover_vector_free(selected_packages);
ali@24
  1075
	return FALSE;
ali@24
  1076
    }
ali@24
  1077
    retval=plover_transaction_helper_install_packages(helper,selected_packages,
ali@24
  1078
      error);
ali@24
  1079
    plover_vector_free(selected_packages);
ali@24
  1080
    return retval;
ali@24
  1081
}
ali@24
  1082
ali@24
  1083
/*
ali@24
  1084
 * Returns TRUE if there is work to be done or FALSE if the group is
ali@24
  1085
 * not installed or on error.
ali@24
  1086
 */
ali@24
  1087
gboolean plover_transaction_helper_remove_group(PloverTransactionHelper *helper,
ali@24
  1088
  const char *group,GError **error)
ali@24
  1089
{
ali@24
  1090
    gboolean retval;
ali@24
  1091
    struct plover_vector *selected_packages;
ali@24
  1092
    PloverTransaction *transaction;
ali@24
  1093
    g_return_val_if_fail(PLOVER_IS_TRANSACTION_HELPER(helper),FALSE);
ali@61
  1094
    g_return_val_if_fail(plover_transaction_helper_get_installed(helper) != NULL,FALSE);
ali@24
  1095
    selected_packages=plover_transaction_helper_group_get_default_packages(
ali@24
  1096
      helper,group,error);
ali@24
  1097
    if (!selected_packages)
ali@24
  1098
	return FALSE;
ali@24
  1099
    if (!selected_packages->len)
ali@24
  1100
    {
ali@24
  1101
	g_set_error(error,PLOVER_GENERAL_ERROR,
ali@24
  1102
	  PLOVER_GENERAL_ERROR_FAILED,"%s: no default packages",group);
ali@24
  1103
	plover_vector_free(selected_packages);
ali@24
  1104
	return FALSE;
ali@24
  1105
    }
ali@24
  1106
    transaction=plover_transaction_new();
ali@24
  1107
    plover_transaction_set_installed(transaction,helper->installed);
ali@24
  1108
    if (!plover_transaction_remove(transaction,selected_packages->strings,
ali@24
  1109
      error))
ali@24
  1110
    {
ali@24
  1111
	plover_vector_free(selected_packages);
ali@24
  1112
	g_object_unref(transaction);
ali@24
  1113
	return FALSE;
ali@24
  1114
    }
ali@24
  1115
    retval=plover_transaction_helper_add_transaction(helper,transaction,
ali@38
  1116
      NULL,PLOVER_TRANSACTION_HELPER_REPORT_REMOVE,error);
ali@24
  1117
    g_object_unref(transaction);
ali@24
  1118
    plover_vector_free(selected_packages);
ali@24
  1119
    return retval;
ali@24
  1120
}
ali@24
  1121
ali@24
  1122
/*
ali@24
  1123
 * Returns TRUE if there is work to be done or FALSE if all updates have
ali@24
  1124
 * already been applied or on error.
ali@24
  1125
 */
ali@24
  1126
gboolean plover_transaction_helper_update(PloverTransactionHelper *helper,
ali@24
  1127
  GError **error)
ali@24
  1128
{
ali@24
  1129
    gboolean retval;
ali@24
  1130
    PloverTransaction *transaction;
ali@24
  1131
    g_return_val_if_fail(PLOVER_IS_TRANSACTION_HELPER(helper),FALSE);
ali@24
  1132
    transaction=plover_transaction_helper_new_transaction(helper,error);
ali@24
  1133
    if (!transaction)
ali@24
  1134
	return FALSE;
ali@24
  1135
    if (!plover_transaction_update(transaction,NULL,error))
ali@24
  1136
    {
ali@24
  1137
	g_object_unref(transaction);
ali@24
  1138
	return FALSE;
ali@24
  1139
    }
ali@24
  1140
    retval=plover_transaction_helper_add_transaction(helper,transaction,
ali@38
  1141
      NULL,PLOVER_TRANSACTION_HELPER_REPORT_UPDATE,error);
ali@24
  1142
    g_object_unref(transaction);
ali@24
  1143
    return retval;
ali@24
  1144
}
ali@24
  1145
ali@24
  1146
gboolean plover_transaction_helper_get_visible(PloverTransactionHelper *helper)
ali@24
  1147
{
ali@24
  1148
    g_return_val_if_fail(PLOVER_IS_TRANSACTION_HELPER(helper),FALSE);
ali@24
  1149
    if (helper->error_dialog)
ali@24
  1150
	return TRUE;
ali@24
  1151
    else if (!helper->assistant)
ali@24
  1152
	return FALSE;
ali@24
  1153
    else
ali@24
  1154
	return gtk_widget_get_visible(GTK_WIDGET(helper->assistant));
ali@24
  1155
}
ali@24
  1156
ali@24
  1157
void plover_transaction_helper_present(PloverTransactionHelper *helper)
ali@24
  1158
{
ali@24
  1159
    g_return_if_fail(PLOVER_IS_TRANSACTION_HELPER(helper));
ali@24
  1160
    if (helper->error_dialog)
ali@24
  1161
	gtk_window_present(GTK_WINDOW(helper->error_dialog));
ali@24
  1162
    else if (helper->assistant)
ali@24
  1163
	gtk_window_present(GTK_WINDOW(helper->assistant));
ali@24
  1164
}
ali@24
  1165
ali@24
  1166
static void
ali@24
  1167
  plover_transaction_helper_error_dialog_response(GtkDialog *error_dialog,
ali@24
  1168
  int response_id,PloverTransactionHelper *helper)
ali@24
  1169
{
ali@24
  1170
    g_signal_handlers_disconnect_by_data(error_dialog,helper);
ali@24
  1171
    if ((GtkWidget *)error_dialog==helper->error_dialog)
ali@24
  1172
    {
ali@24
  1173
	gtk_widget_destroy(helper->error_dialog);
ali@24
  1174
	helper->error_dialog=NULL;
ali@24
  1175
	if (helper->assistant)
ali@24
  1176
	{
ali@24
  1177
	    gtk_widget_hide(GTK_WIDGET(helper->assistant));
ali@24
  1178
	    gtk_assistant_set_current_page(helper->assistant,0);
ali@24
  1179
	}
ali@24
  1180
	g_signal_emit(helper,signals[CLOSE],0);
ali@24
  1181
    }
ali@24
  1182
}
ali@24
  1183
ali@24
  1184
const char *plover_transaction_helper_get_error(PloverTransactionHelper *helper,
ali@24
  1185
  const GError **error)
ali@24
  1186
{
ali@24
  1187
    g_return_if_fail(PLOVER_IS_TRANSACTION_HELPER(helper));
ali@24
  1188
    if (!helper->error_dialog)
ali@24
  1189
	return NULL;
ali@24
  1190
    if (error)
ali@24
  1191
	*error=helper->error;
ali@24
  1192
    return helper->error_primary_text;
ali@24
  1193
}
ali@24
  1194
ali@24
  1195
void plover_transaction_helper_set_error(PloverTransactionHelper *helper,
ali@24
  1196
  const GError *error,const char *primary_text)
ali@24
  1197
{
ali@24
  1198
    GtkMessageType type;
ali@24
  1199
    GtkWindow *window;
ali@24
  1200
    g_return_if_fail(PLOVER_IS_TRANSACTION_HELPER(helper));
ali@24
  1201
    g_return_if_fail(error != NULL);
ali@24
  1202
    g_return_if_fail(primary_text != NULL);
ali@24
  1203
    if (helper->pulse_handler)
ali@24
  1204
    {
ali@24
  1205
	g_source_remove(helper->pulse_handler);
ali@24
  1206
	helper->pulse_handler=0;
ali@24
  1207
    }
ali@24
  1208
    if (helper->error_dialog)
ali@24
  1209
    {
ali@24
  1210
	gtk_widget_destroy(helper->error_dialog);
ali@24
  1211
	helper->error_dialog=NULL;
ali@24
  1212
    }
ali@24
  1213
    g_free(helper->error_primary_text);
ali@24
  1214
    helper->error_primary_text=g_strdup(primary_text);
ali@24
  1215
    g_clear_error(&helper->error);
ali@24
  1216
    helper->error=g_error_copy(error);
ali@24
  1217
    if (g_error_matches(error,PLOVER_GENERAL_ERROR,
ali@24
  1218
      PLOVER_GENERAL_ERROR_NO_WORK))
ali@24
  1219
	type=GTK_MESSAGE_INFO;
ali@24
  1220
    else
ali@24
  1221
	type=GTK_MESSAGE_ERROR;
ali@24
  1222
    if (helper->assistant)
ali@24
  1223
	window=GTK_WINDOW(helper->assistant);
ali@24
  1224
    else
ali@24
  1225
    	window=NULL;
ali@24
  1226
    helper->error_dialog=gtk_message_dialog_new(window,
ali@24
  1227
      GTK_DIALOG_DESTROY_WITH_PARENT,type,GTK_BUTTONS_CLOSE,primary_text);
ali@24
  1228
    gtk_message_dialog_format_secondary_text(
ali@24
  1229
      GTK_MESSAGE_DIALOG(helper->error_dialog),error->message);
ali@24
  1230
    gtk_widget_show(helper->error_dialog);
ali@24
  1231
    g_signal_connect(helper->error_dialog,"response",
ali@24
  1232
      G_CALLBACK(plover_transaction_helper_error_dialog_response),helper);
ali@24
  1233
}