diff options
Diffstat (limited to '')
-rwxr-xr-x | tasksel.pl | 948 |
1 files changed, 948 insertions, 0 deletions
diff --git a/tasksel.pl b/tasksel.pl new file mode 100755 index 0000000..007ecd1 --- /dev/null +++ b/tasksel.pl @@ -0,0 +1,948 @@ +#!/usr/bin/perl +# Debian task selector, mark II. +# Copyright 2004-2011 by Joey Hess <joeyh@debian.org>. +# Licensed under the GPL, version 2 or higher. +use 5.014; +use Locale::gettext; +use Getopt::Long; +use warnings; +use strict; +textdomain('tasksel'); + +my $debconf_helper="/usr/lib/tasksel/tasksel-debconf"; +my $testdir="/usr/lib/tasksel/tests"; +my $packagesdir="/usr/lib/tasksel/packages"; +my $descdir="/usr/share/tasksel/descs"; +my $localdescdir="/usr/local/share/tasksel/descs"; +my $statusfile="/var/lib/dpkg/status"; +my $infodir="/usr/lib/tasksel/info"; + +# This boolean indicates whether we are in dry-run (no-do) mode. More +# specifically, it disables the actual running of commands by the +# &run() function. +my $testmode=0; + +my $taskpackageprefix="task-"; + +sub warning { + print STDERR "tasksel: @_\n"; +} + +sub error { + print STDERR "tasksel: @_\n"; + exit 1; +} + +# my $statuscode = &run("ls", "-l", "/tmp"); +# => 0 +# Run a shell command except in test mode, and returns its exit code. +# Prints the command in test mode. Parameters should be pre-split for +# system. +sub run { + if ($testmode) { + print join(" ", @_)."\n"; + return 0; + } + else { + return system(@_) >> 8; + } +} + +# my @paths = &list_task_descs(); +# => ("/path/to/debian-tasks.desc", "/some/other/taskfile.desc") +# Get the list of desc files. +sub list_task_descs { + # Setting DEBIAN_TASKS_ONLY is a way for the Debian installer + # to tell tasksel to only use the Debian tasks (from + # tasksel-data). + if ($ENV{DEBIAN_TASKS_ONLY}) { + return glob("$descdir/debian-tasks.desc"); + } + else { + return glob("$descdir/*.desc"), glob("$localdescdir/*.desc"); + } +} + +# &read_task_desc("/path/to/taskfile.desc"); +# => ( +# { +# task => "gnome-desktop", +# parent => "desktop", +# relevance => 1, +# key => [task-gnome-desktop"], +# section => "user", +# test-default-desktop => "3 gnome", +# sortkey => 1desktop-01 +# }, +# ... +# ) +# Returns a list of hashes; hash values are arrays for multi-line fields. +sub read_task_desc { + my $desc=shift; + + # %tasks maps the name of each task (the Task: field) to its + # %%data information (that maps each key to value(s), see the + # %"while" loop below). + my %tasks; + + open (DESC, "<$desc") || die "Could not open $desc for reading: $!"; + local $/="\n\n"; + while (defined($_ = <DESC>)) { + # %data will contain the keys/values of the current + # stanza. + # + # The keys are stored lowercase. + # + # A single-line value is stored as a scalar "line1"; a + # multi-line value is stored as a ref to array + # ["line1", "line2"]. + # + # $data{relevance} is set to 5 if not otherwise + # specified in the stanza. + my %data; + + my @lines=split("\n"); + while (@lines) { + my $line=shift(@lines); + if ($line=~/^([^ ]+):(?: (.*))?/) { + my ($key, $value)=($1, $2); + $key=lc($key); + if (@lines && $lines[0] =~ /^\s+/) { + # multi-line field + my @values; + + # Ignore the first line if it is empty. + if (defined $value && length $value) { + push @values, $value; + } + + while (@lines && $lines[0] =~ /^\s+(.*)/) { + push @values, $1; + shift @lines; + } + $data{$key}=[@values]; + } + else { + $data{$key}=$value; + } + } + else { + warning "$desc: in stanza $.: warning: parse error, ignoring line: $line"; + } + } + $data{relevance}=5 unless exists $data{relevance}; + if (exists $data{task}) { + $tasks{$data{task}} = \%data; + } + } + close DESC; + + my @ret; + # In this loop, we simultaneously: + # + # - enrich the %data structures of all tasks with a + # ->{sortkey} field + # + # - and collect them into @ret. + foreach my $task (keys %tasks) { + my $t=$tasks{$task}; + if (exists $t->{parent} && exists $tasks{$t->{parent}}) { + # This task has a "Parent:" task. For example: + # + # Task: sometask + # Relevance: 3 + # Parent: parenttask + # + # Task: parenttask + # Relevance: 6 + # + # In this case, we set the sortkey to "6parenttask-03". + # + # XXX TODO: support correct sorting when + # Relevance is 10 or more (e.g. package + # education-tasks). + $t->{sortkey}=$tasks{$t->{parent}}->{relevance}.$t->{parent}."-0".$t->{relevance}; + } + else { + # This task has no "Parent:" task. For example: + # + # Task: sometask + # Relevance: 3 + # + # In this case, we set the sortkey to "3sometask-00". + $t->{sortkey}=$t->{relevance}.$t->{task}."-00"; + } + push @ret, $t; + } + return @ret; +} + +# &all_tasks(); +# => ( +# { +# task => "gnome-desktop", +# parent => "desktop", +# relevance => 1, +# key => [task-gnome-desktop"], +# section => "user", +# test-default-desktop => "3 gnome", +# sortkey => 1desktop-01 +# }, +# ... +# ) +# Loads info for all tasks, and returns a set of task structures. +sub all_tasks { + my %seen; + # Filter out duplicates: only the first occurrence of each + # task name is taken into account. + grep { $seen{$_->{task}}++; $seen{$_->{task}} < 2 } + map { read_task_desc($_) } list_task_descs(); +} + + +# my %apt_available = %_info_avail() +# => ( +# "debian-policy" => { priority => "optional", section => "doc" }, +# ... +# ) +# +# Call "apt-cache dumpavail" and collect the output information about +# package name, priority and section. +sub _info_avail { + my %ret = (); + # Might be better to use the perl apt bindings, but they are not + # currently in base. + open (AVAIL, "apt-cache dumpavail|"); + local $_; + my ($package, $section, $priority); + while (<AVAIL>) { + chomp; + if (not $_) { + # End of stanza + if (defined $package && defined $priority && defined $section) { + $ret{$package} = { + "priority" => $priority, + "section" => $section, + }; + } + } + elsif (/^Package: (.*)/) { + $package = $1; + } + elsif (/^Priority: (.*)/) { + $priority = $1; + } + elsif (/^Section: (.*)/) { + $section = $1; + } + } + close AVAIL; + return %ret; +} + +# my @installed = &list_installed(); +# => ("emacs", "vim", ...) +# Returns a list of all installed packages. +# This is not memoised and will run dpkg-query at each invocation. +# See &package_installed() for memoisation. +sub list_installed { + my @list; + open (LIST, q{LANG=C dpkg-query -W -f='${Package} ${Status}\n' |}); + while (<LIST>) { + # Each line looks like this: + # "adduser install ok installed" + if (/^([^ ]+) .* installed$/m) { + push @list, $1; + } + } + close LIST; + return @list; +} + +my %_info_avail_cache; + +# my $apt_available = &info_avail(); +# => { +# "debian-policy" => { priority => "optional", section => "doc" }, +# ... +# } +# Returns a hash of all available packages. Memoised. +sub info_avail { + my $package = shift; + if (!%_info_avail_cache) { + %_info_avail_cache = _info_avail(); + } + return \%_info_avail_cache; +} + +# if (&package_avail("debian-policy")) { ... } +# Given a package name, checks to see if it's installed or available. +# Memoised. +sub package_avail { + my $package = shift; + return info_avail()->{$package} || package_installed($package); +} + +# Memoisation for &package_installed(). +my %installed_pkgs; + +# if (&package_installed("debian-policy")) { ... } +# Given a package name, checks to see if it's installed. Memoised. +sub package_installed { + my $package=shift; + + if (! %installed_pkgs) { + foreach my $pkg (list_installed()) { + $installed_pkgs{$pkg} = 1; + } + } + + return $installed_pkgs{$package}; +} + +# if (&task_avail($task)) { ... } +# Given a task hash, checks that all of its key packages are installed or available. +# Returns true if all key packages are installed or available. +# Returns false if any of the key packages is not. +sub task_avail { + local $_; + my $task=shift; + if (! ref $task->{key}) { + return 1; + } + else { + foreach my $pkg (@{$task->{key}}) { + if (! package_avail($pkg)) { + return 0; + } + } + return 1; + } +} + +# if (&task_installed($task)) { ... } +# Given a task hash, checks to see if it is already installed. +# All of its key packages must be installed. Other packages are not checked. +sub task_installed { + local $_; + my $task=shift; + if (! ref $task->{key}) { + return 0; # can't tell with no key packages + } + else { + foreach my $pkg (@{$task->{key}}) { + if (! package_installed($pkg)) { + return 0; + } + } + return 1; + } +} + +# my @packages = &task_packages($task); +# Given a task hash, returns a list of all available packages in the task. +# +# It is the list of "Key:" packages, plus the packages indicated +# through the "Packages:" field. +sub task_packages { + my $task=shift; + + # The %list hashtable is used as a set: only its keys matter, + # the value is irrelevant. + my %list; + + # "Key:" packages are always included. + if (ref $task->{key}) { + # $task->{key} is not a line but a reference (to an + # array of lines). + map { $list{$_}=1 } @{$task->{key}}; + } + + if (! defined $task->{packages}) { + # No "Packages:" field. + # only key + } + elsif ($task->{packages} eq 'standard') { + # Special case of "Packages: standard" + # + # The standard packages are the non-library ones in + # "main" which priority is required, important or + # standard. + # + # We add all standard packages to %list, except the + # ones that are already installed. + my %info_avail=%{info_avail()}; + while (my ($package, $info) = each(%info_avail)) { + my ($priority, $section) = ($info->{priority}, $info->{section}); + if (($priority eq 'required' || + $priority eq 'important' || + $priority eq 'standard') && + # Exclude packages in non-main and library sections + $section !~ /^lib|\// && + # Exclude already installed packages + !package_installed($package)) { + $list{$package} = 1; + } + } + } + else { + # external method + my ($method, @params); + + # "Packages:" requests to run a program and use its + # output as the names of packages. + # + # There are basically two forms: + # + # Packages: myprogram + # + # Runs /usr/lib/tasksel/packages/myprogram TASKNAME + # + # Packages: myprogram + # arg1 + # arg2... + # + # Runs /usr/lib/tasksel/packages/myprogram TASKNAME arg1 arg2... + # + # The tasksel package provides the simple "list" + # program which simply outputs its arguments. + if (ref $task->{packages}) { + @params=@{$task->{packages}}; + $method=shift @params; + } + else { + $method=$task->{packages}; + } + + map { $list{$_}=1 } + grep { package_avail($_) } + split(' ', `$packagesdir/$method $task->{task} @params`); + } + + return keys %list; +} + +# &task_test($task, $new_install, $display_by_default, $install_by_default); +# Given a task hash, runs any test program specified in its data, and sets +# the _display and _install fields to 1 or 0 depending on its result. +# +# If _display is true, _install means the default proposal shown to +# the user, who can modify it. If _display is false, _install says +# what to do, without asking the user. +sub task_test { + my $task=shift; + my $new_install=shift; + $task->{_display} = shift; # default + $task->{_install} = shift; # default + $ENV{NEW_INSTALL}=$new_install if defined $new_install; + # Each task may define one or more tests in the form: + # + # Test-PROGRAM: ARGUMENTS... + # + # Each of the programs will be run like this: + # + # /usr/lib/tasksel/tests/PROGRAM TASKNAME ARGUMENTS... + # + # If $new_install is true, the NEW_INSTALL environment + # variable is set for invoking the program. + # + # The return code of the invocation then indicates what to set: + # + # 0 - don't display, but install it + # 1 - don't display, don't install + # 2 - display, mark for installation + # 3 - display, don't mark for installation + # anything else - don't change the values of _display or _install + foreach my $test (grep /^test-.*/, keys %$task) { + $test=~s/^test-//; + if (-x "$testdir/$test") { + my $ret=system("$testdir/$test", $task->{task}, split " ", $task->{"test-$test"}) >> 8; + if ($ret == 0) { + $task->{_display} = 0; + $task->{_install} = 1; + } + elsif ($ret == 1) { + $task->{_display} = 0; + $task->{_install} = 0; + } + elsif ($ret == 2) { + $task->{_display} = 1; + $task->{_install} = 1; + } + elsif ($ret == 3) { + $task->{_display} = 1; + $task->{_install} = 0; + } + } + } + + delete $ENV{NEW_INSTALL}; + return $task; +} + +# &hide_enhancing_tasks($task); +# +# Hides a task and marks it not to be installed if it enhances other +# tasks. +# +# Returns $task. +sub hide_enhancing_tasks { + my $task=shift; + if (exists $task->{enhances} && length $task->{enhances}) { + $task->{_display} = 0; + $task->{_install} = 0; + } + return $task; +} + +# &getdescriptions(@tasks); +# +# Looks up the descriptions of a set of tasks, returning a new list +# with the ->{shortdesc} fields filled in. +# +# Ideally, the .desc file would indicate a description of each task, +# which would be retrieved quickly. For missing Description fields, +# we fetch the data with "apt-cache show task-TASKNAME...", which +# takes longer. +# +# @tasks: list of references, each referencing a task data structure. +# +# Each data structured is enriched with a ->{shortdesc} field, +# containing the localized short description. +# +# Returns @tasks. +sub getdescriptions { + my @tasks=@_; + + # If the task has a description field in the task desc file, + # just use it, looking up a translation in gettext. + @tasks = map { + if (defined $_->{description}) { + $_->{shortdesc}=dgettext("debian-tasks", $_->{description}->[0]); + } + $_; + } @tasks; + + # Otherwise, a more expensive apt-cache query is done, + # to use the descriptions of task packages. + my @todo = grep { ! defined $_->{shortdesc} } @tasks; + if (@todo) { + open(APT_CACHE, "apt-cache show ".join(" ", map { $taskpackageprefix.$_->{task} } @todo)." |") || die "apt-cache show: $!"; + local $/="\n\n"; + while (defined($_ = <APT_CACHE>)) { + my ($name)=/^Package: $taskpackageprefix(.*)$/m; + my ($description)=/^Description-(?:[a-z][a-z](?:_[A-Z][A-Z])?): (.*)$/m; + ($description)=/^Description: (.*)$/m + unless defined $description; + if (defined $name && defined $description) { + @tasks = map { + if ($_->{task} eq $name) { + $_->{shortdesc}=$description; + } + $_; + } @tasks; + } + } + close APT_CACHE; + } + + return @tasks; +} + +# &task_to_debconf(@tasks); +# => "task1, task2, task3" +# Converts a list of tasks into a debconf list of the task short +# descriptions. +sub task_to_debconf { + join ", ", map { format_description_for_debconf($_) } getdescriptions(@_); +} + +# my $debconf_string = &format_description_for_debconf($task); +# => "... GNOME" +# Build a string for making a debconf menu item. +# If the task has a parent task, "... " is prepended. +sub format_description_for_debconf { + my $task=shift; + my $d=$task->{shortdesc}; + $d=~s/,/\\,/g; + $d="... ".$d if exists $task->{parent}; + return $d; +} + +# my $debconf_string = &task_to_debconf_C(@tasks); +# => "gnome-desktop, kde-desktop" +# Converts a list of tasks into a debconf list of the task names. +sub task_to_debconf_C { + join ", ", map { $_->{task} } @_; +} + +# my @my_tasks = &list_to_tasks("task1, task2, task3", @tasks); +# => ($task1, $task2, $task3) +# Given a first parameter that is a string listing task names, and then a +# list of task hashes, returns a list of hashes for all the tasks +# in the list. +sub list_to_tasks { + my $list=shift; + my %lookup = map { $_->{task} => $_ } @_; + return grep { defined } map { $lookup{$_} } split /[, ]+/, $list; +} + +# my @sorted_tasks = &order_for_display(@tasks); +# Orders a list of tasks for display. +# The tasks are ordered according to the ->{sortkey}. +sub order_for_display { + sort { + $a->{sortkey} cmp $b->{sortkey} + || 0 || + $a->{task} cmp $b->{task} + } @_; +} + +# &name_to_task($taskname, &all_tasks()); +# &name_to_task("gnome-desktop", &all_tasks()); +# => { +# task => "gnome-desktop", +# parent => "desktop", +# relevance => 1, +# key => [task-gnome-desktop"], +# section => "user", +# test-default-desktop => "3 gnome", +# sortkey => 1desktop-01 +# } +# Given a set of tasks and a name, returns the one with that name. +sub name_to_task { + my $name=shift; + return (grep { $_->{task} eq $name } @_)[0]; +} + +# &task_script($task, "preinst") or die; +# Run the task's (pre|post)(inst|rm) script, if there is any. +# Such scripts are located under /usr/lib/tasksel/info/. +sub task_script { + my $task=shift; + my $script=shift; + + my $path="$infodir/$task.$script"; + if (-e $path && -x _) { + my $ret=run($path); + if ($ret != 0) { + warning("$path exited with nonzero code $ret"); + return 0; + } + } + return 1; +} + +# &usage; +# Print the usage. +sub usage { + print STDERR gettext(q{tasksel [OPTIONS...] [COMMAND...] + Commands: + install TASK... install tasks + remove TASK... uninstall tasks + --task-packages=TASK list packages installed by TASK; can be repeated + --task-desc=TASK print the description of a task + --list-tasks list tasks that would be displayed and exit + Options: + -t, --test dry-run: don't really change anything + --new-install automatically install some tasks + --debconf-apt-progress="ARGUMENTS..." + provide additional arguments to debconf-apt-progress(1) +}); +} + +# Process command line options and return them in a hash. +sub getopts { + my %ret; + Getopt::Long::Configure ("bundling"); + if (! GetOptions(\%ret, "test|t", "new-install", "list-tasks", + "task-packages=s@", "task-desc=s", + "debconf-apt-progress=s")) { + usage(); + exit(1); + } + # Special case apt-like syntax. + if (@ARGV) { + my $cmd = shift @ARGV; + if ($cmd eq "install") { + $ret{cmd_install} = \@ARGV; + } + elsif ($cmd eq "remove") { + $ret{cmd_remove} = \@ARGV; + } + else { + usage(); + exit 1; + } + } + $testmode=1 if $ret{test}; # set global + return %ret; +} + +# &interactive($options, @tasks); +# Ask the user and mark tasks to install or remove accordingly. +# The tasks are enriched with ->{_install} or ->{_remove} set to true accordingly. +sub interactive { + my $options = shift; + my @tasks = @_; + + if (! $options->{"new-install"}) { + # Don't install hidden tasks if this is not a new install. + map { $_->{_install} = 0 } grep { $_->{_display} == 0 } @tasks; + } + + my @list = order_for_display(grep { $_->{_display} == 1 } @tasks); + if (@list) { + if (! $options->{"new-install"}) { + # Find tasks that are already installed. + map { $_->{_installed} = task_installed($_) } @list; + # Don't install new tasks unless manually selected. + map { $_->{_install} = 0 } @list; + } + else { + # Assume that no tasks are installed, to ensure + # that complete tasks get installed on new + # installs. + map { $_->{_installed} = 0 } @list; + } + my $question="tasksel/tasks"; + if ($options->{"new-install"}) { + $question="tasksel/first"; + } + my @default = grep { $_->{_display} == 1 && ($_->{_install} == 1 || $_->{_installed} == 1) } @tasks; + my $tmpfile=`mktemp`; + chomp $tmpfile; + my $ret=system($debconf_helper, $tmpfile, + task_to_debconf_C(@list), + task_to_debconf(@list), + task_to_debconf_C(@default), + $question) >> 8; + if ($ret == 30) { + exit 10; # back up + } + elsif ($ret != 0) { + error "debconf failed to run"; + } + open(IN, "<$tmpfile"); + $ret=<IN>; + if (! defined $ret) { + die "tasksel canceled\n"; + } + chomp $ret; + close IN; + unlink $tmpfile; + + # Set _install flags based on user selection. + map { $_->{_install} = 0 } @list; + foreach my $task (list_to_tasks($ret, @tasks)) { + if (! $task->{_installed}) { + $task->{_install} = 1; + } + $task->{_selected} = 1; + } + foreach my $task (@list) { + if (! $task->{_selected} && $task->{_installed}) { + $task->{_remove} = 1; + } + } + } + + # When a $task Enhances: a @group_of_tasks, it means that + # $task can only be installed if @group_of_tasks are also + # installed; and if @group_of_tasks is installed, it is an + # incentive to also install $task. + # + # For example, consider this task: + # + # Task: amharic-desktop + # Enhances: desktop, amharic + # + # The task amharic-desktop installs packages that make + # particular sense if the user wants both a desktop and the + # amharic language environment. Conversely, if + # amharic-desktop is selected (e.g. by preseeding), then it + # automatically also selects tasks "desktop" and "amharic". + + # If an enhancing task is already marked for + # install, probably by preseeding, mark the tasks + # it enhances for install. + foreach my $task (grep { $_->{_install} && exists $_->{enhances} && + length $_->{enhances} } @tasks) { + map { $_->{_install}=1 } list_to_tasks($task->{enhances}, @tasks); + } + + # Select enhancing tasks for install. + # XXX FIXME ugly hack -- loop until enhances settle to handle + # chained enhances. This is ugly and could loop forever if + # there's a cycle. + my $enhances_needswork=1; + + # %tested is the memoization of the below calls to + # %&task_test(). + my %tested; + + # Loop as long as there is work to do. + while ($enhances_needswork) { + $enhances_needswork=0; + + # Loop over all unselected tasks that enhance one or + # more things. + foreach my $task (grep { ! $_->{_install} && exists $_->{enhances} && + length $_->{enhances} } @tasks) { + # TODO: the computation of %tasknames could be + # done once and for all outside of this nested + # loop, saving some redundant work. + my %tasknames = map { $_->{task} => $_ } @tasks; + + # @deps is the list of tasks enhanced by $task. + # + # Basically, if all the deps are installed, + # and tests say that $task can be installed, + # then mark it to install. Otherwise, don't + # install it. + my @deps=map { $tasknames{$_} } split ", ", $task->{enhances}; + + if (grep { ! defined $_ } @deps) { + # task enhances an unavailable or + # uninstallable task + next; + } + + if (@deps) { + # FIXME: isn't $orig_state always + # false, given that the "for" loop + # above keeps only $tasks that do + # not have $_->{_install}? + my $orig_state=$task->{_install}; + + # Mark enhancing tasks for install if their + # dependencies are met and their test fields + # mark them for install. + if (! exists $tested{$task->{task}}) { + $ENV{TESTING_ENHANCER}=1; + task_test($task, $options->{"new-install"}, 0, 1); + delete $ENV{TESTING_ENHANCER}; + $tested{$task->{task}}=$task->{_install}; + } + else { + $task->{_install}=$tested{$task->{task}}; + } + + foreach my $dep (@deps) { + if (! $dep->{_install}) { + $task->{_install} = 0; + } + } + + if ($task->{_install} != $orig_state) { + # We have made progress: + # continue another round. + $enhances_needswork=1; + } + } + } + } + +} + +sub main { + my %options=getopts(); + my @tasks_remove; + my @tasks_install; + + # Options that output stuff and don't need a full processed list of + # tasks. + if (exists $options{"task-packages"}) { + my @tasks=all_tasks(); + foreach my $taskname (@{$options{"task-packages"}}) { + my $task=name_to_task($taskname, @tasks); + if ($task) { + print "$_\n" foreach task_packages($task); + } + } + exit(0); + } + elsif ($options{"task-desc"}) { + my $task=name_to_task($options{"task-desc"}, all_tasks()); + if ($task) { + # The Description looks like this: + # + # Description: one-line short description + # Longer description, + # possibly spanning + # multiple lines. + # + # $extdesc will contain the long description, + # reformatted to one line. + my $extdesc=join(" ", @{$task->{description}}[1..$#{$task->{description}}]); + print dgettext("debian-tasks", $extdesc)."\n"; + exit(0); + } + else { + fprintf STDERR ("Task %s has no description\n", $options{"task-desc"}); + exit(1); + } + } + + # This is relatively expensive, get the full list of available tasks and + # mark them. + my @tasks=map { hide_enhancing_tasks($_) } map { task_test($_, $options{"new-install"}, 1, 0) } + grep { task_avail($_) } all_tasks(); + + if ($options{"list-tasks"}) { + map { $_->{_installed} = task_installed($_) } @tasks; + @tasks=getdescriptions(@tasks); + # TODO: use printf() instead of print for correct column alignment + print "".($_->{_installed} ? "i" : "u")." ".$_->{task}."\t".$_->{shortdesc}."\n" + foreach order_for_display(grep { $_->{_display} } @tasks); + exit(0); + } + + if ($options{cmd_install}) { + @tasks_install = map { name_to_task($_, @tasks) } @{$options{cmd_install}}; + } + elsif ($options{cmd_remove}) { + @tasks_remove = map { name_to_task($_, @tasks) } @{$options{cmd_remove}}; + } + else { + interactive(\%options, @tasks); + + # Add tasks to install + @tasks_install = grep { $_->{_install} } @tasks; + # Add tasks to remove + @tasks_remove = grep { $_->{_remove} } @tasks; + } + + my @cmd; + if (-x "/usr/bin/debconf-apt-progress") { + @cmd = "debconf-apt-progress"; + push @cmd, split(' ', $options{'debconf-apt-progress'}) + if exists $options{'debconf-apt-progress'}; + push @cmd, "--"; + } + push @cmd, qw{apt-get -q -y -o APT::Install-Recommends=true -o APT::Get::AutomaticRemove=true -o Acquire::Retries=3 install}; + + # And finally, act on selected tasks. + if (@tasks_install || @tasks_remove) { + foreach my $task (@tasks_remove) { + push @cmd, map { "$_-" } task_packages($task); + task_script($task->{task}, "prerm"); + } + foreach my $task (@tasks_install) { + push @cmd, task_packages($task); + task_script($task->{task}, "preinst"); + } + my $ret=run(@cmd); + if ($ret != 0) { + error gettext("apt-get failed")." ($ret)"; + } + foreach my $task (@tasks_remove) { + task_script($task->{task}, "postrm"); + } + foreach my $task (@tasks_install) { + task_script($task->{task}, "postinst"); + } + } +} + +main(); |