diff options
Diffstat (limited to 'gurpmi2')
-rwxr-xr-x | gurpmi2 | 373 |
1 files changed, 373 insertions, 0 deletions
diff --git a/gurpmi2 b/gurpmi2 new file mode 100755 index 00000000..04f6b13a --- /dev/null +++ b/gurpmi2 @@ -0,0 +1,373 @@ +#!/usr/bin/perl + +#- Copyright (C) 2005 Mandrakesoft + +use strict; +use warnings; + +BEGIN { #- set up a safe path and environment + $ENV{PATH} = "/sbin:/usr/sbin:/bin:/usr/bin:/usr/X11R6/bin"; + delete @ENV{qw(ENV BASH_ENV IFS CDPATH)}; +} + +#- This is needed because text printed by Gtk2 will always be encoded +#- in UTF-8; we first check if LC_ALL is defined, because if it is, +#- changing only LC_COLLATE will have no effect. +use POSIX qw(setlocale LC_ALL LC_COLLATE); +use locale; +BEGIN { + my $collation_locale = $ENV{LC_ALL}; + if ($collation_locale) { + $collation_locale =~ /UTF-8/ or setlocale(LC_ALL, "$collation_locale.UTF-8"); + } else { + $collation_locale = setlocale(LC_COLLATE); + $collation_locale =~ /UTF-8/ or setlocale(LC_COLLATE, "$collation_locale.UTF-8"); + } +} + +use urpm; +use urpm::msg qw(N); +use Gtk2; + +sub usage () { + print STDERR <<USAGE; +gurpmi version $urpm::VERSION +Usage : + gurpmi <rpm> [ <rpm>... ] +USAGE + exit 0; +} + +#- fatal gurpmi initialisation error (*not* fatal urpmi errors) +sub fatal { print STDERR "$_[0]\n"; exit 1 } + +sub quit () { Gtk2->main_quit } + +sub but ($) { " $_[0] " } + +#- GUI globals +my ($mainw, $mainbox); + +#- Gtk2 helper functions + +#- Replaces the contents of the main window with the specified box +#- (avoids popup multiplication) +sub change_mainw { + $mainw->remove($mainbox); + ($mainbox) = @_; + $mainw->add($mainbox); + $mainw->show_all; +} + +sub add_button_box { + my ($vbox, @buttons) = @_; + my $hbox = Gtk2::HButtonBox->new; + $vbox->pack_start($hbox, 0, 0, 0); + $hbox->set_layout('edge'); + $_->set_alignment(0.5, 0.5), $hbox->add($_) foreach @buttons; +} + +sub sync { + $mainw->show; + Gtk2->main_iteration while Gtk2->events_pending; +} + +sub new_label { + my ($msg) = @_; + my $label = Gtk2::Label->new($msg); + $label->set_line_wrap(1); + $label->set_alignment(0.5, 0.5); + if (($msg =~ tr/\n/\n/) > 5) { + my $sw = Gtk2::ScrolledWindow->new; + $sw->set_policy('never', 'automatic'); + $sw->add_with_viewport($label); + $sw->set_size_request(-1,200); + return $sw; + } else { + return $label; + } +} + +#- sets the window to a please-wait message +sub wait_label { + my $wait_vbox = Gtk2::VBox->new(0, 5); + my $wait_label = Gtk2::Label->new($_[0] || N("Please wait...")); + $wait_label->set_alignment(0.5, 0.5); + $wait_vbox->pack_start($wait_label, 1, 1, 0); + change_mainw($wait_vbox); + sync(); +} + +#- Parse command line +my (@all_rpms); +foreach (@ARGV) { + if (/^-/) { + /^--?[hv?]/ and usage; + fatal N("Unknown option %s", $_); + } + push @all_rpms, $_; +} +@all_rpms or fatal N("No packages specified"); + +$> and fatal N("Must be root"); + +#- Now, the graphical stuff. + +Gtk2->init; + +#- Create main window + +$mainw = Gtk2::Window->new('toplevel'); +$mainw->set_border_width(3); +$mainw->set_title(N("RPM installation")); +$mainw->signal_connect(destroy => \&quit); +$mainw->set_position('center'); +$mainw->set_modal(0); +$mainbox = Gtk2::VBox->new(0, 5); +$mainw->add($mainbox); + +#- Performs installation + +my $urpm = configure_urpm(); +my $state = {}; +my %requested = $urpm->register_rpms(@all_rpms); +$urpm->resolve_dependencies( + $state, + \%requested, + callback_choices => \&ask_choice, +); +my @ask_unselect = $urpm->unselected_packages($state); +@ask_unselect +? ask_continue(N( + "Some package requested cannot be installed:\n%s\nContinue?", + join "\n", $urpm->translate_why_unselected($state, sort @ask_unselect) + ), \&do_install) +: do_install(); + + +$mainw->show_all; +Gtk2->main; + +#- Creates and configure an urpm object for this application to use. +sub configure_urpm { + my $urpm = new urpm; + $urpm->{fatal} = sub { + Gtk2::MessageDialog->new($mainw, [qw(modal destroy-with-parent)], 'error', 'ok', $_[1])->run; + quit; + exit $_[0]; + }; + $urpm->{error} = sub { + my $w = Gtk2::MessageDialog->new($mainw, [qw(modal destroy-with-parent)], 'warning', 'ok', $_[0]); + $w->run; + $w->destroy; + }; + $urpm->exlock_rpm_db; + $urpm->shlock_urpmi_db; + $urpm->configure; + #- default options values + exists $urpm->{options}{$_} or $urpm->{options}{$_} = 1 foreach qw(post-clean verify-rpm split-length); + $urpm->{options}{'split-level'} = 20 unless exists $urpm->{options}{'split-level'}; + $urpm; +} + +#- Callback for choices +sub ask_choice { + my (undef, undef, undef, $choices) = @_; + my $radio; + my @radios = map { + $radio = Gtk2::RadioButton->new_with_label( + $radio ? $radio->get_group : undef, + (scalar $_->fullname) . "\n" . $_->summary + . ($_->flag_installed ? N(" (to upgrade)") : '') + . ($_->flag_upgrade ? N(" (to install)") : '') + ); + } @$choices; + my $vbox = Gtk2::VBox->new(0, 5); + my $label = Gtk2::Label->new(N("One of the following packages is needed:")); + $label->set_alignment(0.5, 0.5); + $vbox->pack_start($label, 1, 1, 0); + $vbox->pack_start($_, 1, 1, 0) foreach @radios; + my $cancel_button = Gtk2::Button->new(but N("_Cancel")); + my $choice_button = Gtk2::Button->new(but N("_Ok")); + $cancel_button->signal_connect(clicked => \&quit); + $choice_button->signal_connect(clicked => sub { + my $n = 0; + foreach (@radios) { last if $_->get_active; ++$n } + $choices->[$n]; + }); + add_button_box($vbox, $cancel_button, $choice_button); + change_mainw($vbox); + $radios[0]->set_active(1); +} + +sub ask_continue { + my ($msg, $nextclosure) = @_; + my $vbox = Gtk2::VBox->new(0, 5); + $vbox->pack_start(new_label($msg), 1, 1, 0); + my $continue_button = Gtk2::Button->new(but N("_Ok")); + my $quit_button = Gtk2::Button->new(but N("_Abort")); + $quit_button->signal_connect(clicked => \&quit); + $continue_button->signal_connect(clicked => sub { goto &$nextclosure }); + add_button_box($vbox, $quit_button, $continue_button); + change_mainw($vbox); +} + +sub ask_continue_blocking { + my ($msg) = @_; + my $w = Gtk2::MessageDialog->new($mainw, [qw(modal destroy-with-parent)], 'question', 'yes-no', $msg); + my $answer = $w->run; + $w->destroy; + quit if $answer eq 'no'; +} + +sub do_install { + wait_label; + my @ask_remove = $urpm->removed_packages($state); + @ask_remove + ? ask_continue(N( + "The following packages have to be removed for others to be upgraded:\n%s\nContinue?", + join "\n", $urpm->translate_why_removed($state, sort @ask_remove) + ), \&do_install_2) + : goto &do_install_2; +} + +sub do_install_2 { + my @to_install; + foreach my $pkg (sort { $a->name cmp $b->name } @{$urpm->{depslist}}[keys %{$state->{selected}}]) { + $pkg->arch ne 'src' and push @to_install, scalar $pkg->fullname; + } + $urpm->{nb_install} = @to_install; + @to_install > 1 + ? ask_continue(N( + "To satisfy dependencies, the following %d packages are going to be installed:\n%s\n", + scalar(@to_install), join "\n", @to_install + ), \&do_install_3) + : goto \&do_install_3; +} + +sub do_install_3 { + wait_label(N("Package installation...")); + my ($local_sources, $list) = $urpm->get_source_packages($state->{selected}); + $local_sources || $list or $urpm->{fatal}(3, N("unable to get source packages, aborting")); + my %sources = %$local_sources; + my %error_sources; + my $vbox = Gtk2::VBox->new(0, 5); + my $progress_label = Gtk2::Label->new('-'); + $vbox->pack_start($progress_label, 1, 1, 0); + my $progressbar = Gtk2::ProgressBar->new; + $progressbar->set_size_request(300, -1); + $vbox->pack_start($progressbar, 0, 0, 0); + change_mainw($vbox); + $urpm->copy_packages_of_removable_media($list, \%sources, + force_local => 1, + ask_for_medium => sub { + my $w = Gtk2::MessageDialog->new($mainw, [qw(modal destroy-with-parent)], 'warning', 'ok', + N("Please insert the medium named \"%s\" on device [%s]", $_[0], $_[1]) + ); + $w->run; + $w->destroy; + } + ); + $urpm->create_transaction( + $state, + split_level => $urpm->{options}{'split-level'}, + split_length => $urpm->{options}{'split-length'}, + ); + my ($nok, @errors); + foreach my $set (@{$state->{transaction} || []}) { + my (@transaction_list, %transaction_sources); + $urpm->prepare_transaction($set, $list, \%sources, \@transaction_list, \%transaction_sources); + $urpm->download_packages_of_distant_media( + \@transaction_list, + \%transaction_sources, + \%error_sources, + limit_rate => $urpm->{options}{'limit-rate'}, + compress => $urpm->{options}{compress}, + resume => $urpm->{options}{resume}, + force_local => 1, + callback => sub { + my ($mode, $file, $percent) = @_; + if ($mode eq 'start') { + $file =~ s|/*\s*$||; $file =~ s|.*/||; + $progress_label->set_label(N("Downloading package `%s'...", $file)); + select(undef, undef, undef, 0.1); #- hackish + } elsif ($mode eq 'progress') { + $progressbar->set_fraction($percent / 100); + } elsif ($mode eq 'end') { + $progressbar->set_fraction(1); + } + sync(); + }, + ); + my %transaction_sources_install = %{$urpm->extract_packages_to_install(\%transaction_sources) || {}}; + if (!$urpm->{options}{'verify-rpm'} || grep { $_->{'verify-rpm'} } @{$urpm->{media}}) { + my @bad_signatures = $urpm->check_sources_signatures(\%transaction_sources_install, \%transaction_sources, translate => 1); + if (@bad_signatures) { + ask_continue_blocking(N( + "The following packages have bad signatures:\n%s\n\nDo you want to continue installation ?", + (join "\n", @bad_signatures) + )); + } + } + #- check for local files. + if (my @missing = grep { m|^/| && ! -e $_ } values %transaction_sources_install, values %transaction_sources) { + $urpm->{error}(N("Installation failed, some files are missing:\n%s\nYou may want to update your urpmi database", + join "\n", map { s|([^:]*://[^/:\@]*:)[^/:\@]*(\@.*)|$1xxxx$2|; " $_" } @missing)); + ++$nok; + next; + } + if (keys(%transaction_sources_install) || keys(%transaction_sources)) { + @{$set->{remove} || []} and + $progress_label->set_label(N("removing %s", join(' ', @{$set->{remove} || []}))); + my $progress_nb; + my $total_nb = scalar grep { m|^/| } values %transaction_sources_install, values %transaction_sources; + my $callback_inst = sub { + my ($urpm, $type, $id, $subtype, $amount, $total) = @_; + my $pkg = defined $id && $urpm->{depslist}[$id]; + if ($subtype eq 'start') { + if ($type eq 'trans') { + $progress_label->set_label(N("Preparing...")); + } else { + $progress_label->set_label(N("Installing package `%s' (%s/%s)...", $pkg->name, ++$progress_nb, $urpm->{nb_install})); + } + } elsif ($subtype eq 'progress') { + $progressbar->set_fraction($amount / $total); + } + sync(); + }; + my @l = $urpm->install( + $set->{remove} || [], + \%transaction_sources_install, + \%transaction_sources, + 'fork' => @{$state->{transaction} || []} > 1, #- fork if multiple transaction + translate_message => 1, + oldpackage => $state->{oldpackage}, + callback_inst => $callback_inst, + callback_trans => $callback_inst, + ); + if (@l) { + $progress_label->set_label(N("Installation failed") . ":\n" . join("\n", map { "\t$_" } @l)); + ++$nok; + push @errors, @l; + } + } + } + $vbox = Gtk2::VBox->new(0, 5); + $progress_label = Gtk2::Label->new('-'); + $vbox->pack_start($progress_label, 1, 1, 0); + my $quit_button = Gtk2::Button->new(but N("_Done")); + $quit_button->signal_connect(clicked => \&quit); + add_button_box($vbox, $quit_button); + change_mainw($vbox); + if (values %error_sources) { + $progress_label->set_label(N("Installation failed, some files are missing:\n%s\nYou may want to update your urpmi database", + join "\n", map { s|([^:]*://[^/:\@]*:)[^/:\@]*(\@.*)|$1xxxx$2|; " $_" } values %error_sources)); + } elsif (@{$state->{transaction} || []} == 0 && @ask_unselect == 0) { + $progress_label->set_label(N("The package(s) are already installed")); + } else { + $progress_label->set_label(N("Installation finished")); + } + $urpm->unlock_urpmi_db; + $urpm->unlock_rpm_db; + $urpm->try_umounting_removables; +} |