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