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