BACKPORT: Merge version-tracking sort_deps code from current trunk (r2448)

This commit is contained in:
Steve McIntyre 2012-09-29 12:04:16 +00:00
parent 8f8e026611
commit 4244f204c7
4 changed files with 553 additions and 200 deletions

View File

@ -60,6 +60,7 @@ debian-cd (3.1.6) UNRELEASED-backport; urgency=low
* Explicitly add s390-tools and sysconfig-hardware for s390. * Explicitly add s390-tools and sysconfig-hardware for s390.
* Make grab_md5 more robust * Make grab_md5 more robust
* Remove -e bashisms from Makefile * Remove -e bashisms from Makefile
* Merge version-tracking sort_deps code from current trunk (r2448)
-- Raphaël Hertzog <hertzog@debian.org> Fri, 04 Feb 2011 09:59:21 +0100 -- Raphaël Hertzog <hertzog@debian.org> Fri, 04 Feb 2011 09:59:21 +0100

View File

@ -11,7 +11,7 @@ Vcs-Browser: http://svn.debian.org/wsvn/debian-cd/trunk?op=log
Package: debian-cd Package: debian-cd
Architecture: all Architecture: all
Depends: ${misc:Depends}, curl, perl, dpkg-dev, cpp, libdigest-md5-perl, tofrodos, apt (>= 0.3.11.1), make, genisoimage, lynx-cur | lynx, grep-dctrl, bc, libcompress-zlib-perl Depends: ${misc:Depends}, curl, perl, dpkg-dev, cpp, libdigest-md5-perl, tofrodos, apt (>= 0.3.11.1), make, genisoimage, lynx-cur | lynx, grep-dctrl, bc, libcompress-zlib-perl, libdpkg-perl
Recommends: hfsutils, netpbm, syslinux-common Recommends: hfsutils, netpbm, syslinux-common
Description: Tools for building (Official) Debian CD set Description: Tools for building (Official) Debian CD set
Debian-cd is the official tool for building Debian CD set since the potato Debian-cd is the official tool for building Debian CD set since the potato

View File

@ -32,6 +32,7 @@ options=" -q -o Dir::State::status=$APTTMP/$CODENAME-$ARCH/status \
-o Dir::Cache=$APTTMP/$CODENAME-$ARCH/apt-cache/ \ -o Dir::Cache=$APTTMP/$CODENAME-$ARCH/apt-cache/ \
-o Dir::Etc=$APTTMP/$CODENAME-$ARCH/apt/ \ -o Dir::Etc=$APTTMP/$CODENAME-$ARCH/apt/ \
-o APT::Cache::AllVersions=0 \ -o APT::Cache::AllVersions=0 \
-o APT::Cache::ShowVersion=1 \
-o APT::Architecture=$ARCH " -o APT::Architecture=$ARCH "
sections=main sections=main

View File

@ -1,6 +1,6 @@
#!/usr/bin/perl -w #!/usr/bin/perl -w
# #
# Copyright 1999 Raphaël Hertzog <hertzog@debian.org> # Copyright 1999 Raphaël Hertzog <hertzog@debian.org>
# See the README file for the license # See the README file for the license
# #
# This script takes 1 argument on input : # This script takes 1 argument on input :
@ -13,31 +13,47 @@
# per-disc basis now. # per-disc basis now.
use strict; use strict;
use Data::Dumper;
use Dpkg::Version;
my $list = shift; my $listfile = shift;
my $nonfree = $ENV{'NONFREE'} || 0; my $nonfree = read_env('NONFREE', 0);
my $extranonfree = $ENV{'EXTRANONFREE'} || 0; my $extranonfree = read_env('EXTRANONFREE', 0);
my $force_firmware = $ENV{'FORCE_FIRMWARE'} || 0; my $force_firmware = read_env('FORCE_FIRMWARE', 0);
my $local = $ENV{'LOCAL'} || 0; my $local = read_env('LOCAL', 0);
my $complete = $ENV{'COMPLETE'} || 0; my $complete = read_env('COMPLETE', 0);
my $norecommends = $ENV{'NORECOMMENDS'} || 1; my $norecommends = read_env('NORECOMMENDS', 1);
my $nosuggests = $ENV{'NOSUGGESTS'} || 1; my $nosuggests = read_env('NOSUGGESTS', 1);
my $verbose = read_env('VERBOSE', 0);
my $max_pkg_size = read_env('MAX_PKG_SIZE', 9999999999999);
my $apt = "$ENV{'BASEDIR'}/tools/apt-selection"; my $apt = "$ENV{'BASEDIR'}/tools/apt-selection";
my $adir = "$ENV{'APTTMP'}/$ENV{'CODENAME'}-$ENV{'ARCH'}"; my $adir = "$ENV{'APTTMP'}/$ENV{'CODENAME'}-$ENV{'ARCH'}";
my $arch = "$ENV{'ARCH'}"; my $arch = "$ENV{'ARCH'}";
my $dir = "$ENV{'TDIR'}/$ENV{'CODENAME'}"; my $dir = "$ENV{'TDIR'}/$ENV{'CODENAME'}";
my $verbose = $ENV{'VERBOSE'} || 0;
my $max_pkg_size = $ENV{'MAX_PKG_SIZE'} || 9999999999999;
my @output; my $force_unstable_tasks = read_env('FORCE_SID_TASKSEL', 0);
my $tasks_packages = read_env('TASKS_PACKAGES',
"$ENV{'MIRROR'}/dists/sid/main/binary-$ENV{'ARCH'}/Packages.gz");
my @output = ();
$| = 1; # Autoflush for debugging $| = 1; # Autoflush for debugging
open(LOG, ">$dir/sort_deps.$arch.log") open(LOG, ">$dir/sort_deps.$arch.log")
|| die "Can't write in $dir/sort_deps.$arch.log !\n"; || die "Can't write in $dir/sort_deps.$arch.log !\n";
sub read_env {
my $env_var = shift;
my $default = shift;
if (exists($ENV{$env_var})) {
return $ENV{$env_var};
}
# else
return $default;
}
sub msg { sub msg {
my $level = shift; my $level = shift;
if ($verbose >= $level) { if ($verbose >= $level) {
@ -54,7 +70,7 @@ msg(0, "Running sort_deps to sort packages for $arch:\n");
msg(1, "====================================================================== msg(1, "======================================================================
Here are the settings you've chosen for making the list: Here are the settings you've chosen for making the list:
Architecture: $arch Architecture: $arch
List of prefered packages: $list List of prefered packages: $listfile
Output file: $dir/packages.$arch Output file: $dir/packages.$arch
"); ");
msg(1, "Complete selected packages with all the rest: "); msg(1, "Complete selected packages with all the rest: ");
@ -63,6 +79,10 @@ msg(1, "Include non-free packages: ");
msg(1, yesno($nonfree)."\n"); msg(1, yesno($nonfree)."\n");
msg(1, "Force inclusion of firmware packages: "); msg(1, "Force inclusion of firmware packages: ");
msg(1, yesno($force_firmware)."\n"); msg(1, yesno($force_firmware)."\n");
msg(1, "Ignore Recommends: ");
msg(1, yesno($norecommends)."\n");
msg(1, "Ignore Suggests: ");
msg(1, yesno($nosuggests)."\n");
msg(1, "====================================================================== msg(1, "======================================================================
"); ");
@ -73,30 +93,31 @@ open(AVAIL, "$apt cache dumpavail |") || die "Can't fork : $!\n";
my ($p, $re); my ($p, $re);
while (defined($_=<AVAIL>)) { while (defined($_=<AVAIL>)) {
next if not m/^Package: (\S+)\s*$/m; next if not m/^Package: (\S+)\s*$/m;
$p = $1; if (!$force_unstable_tasks || $1 !~ /^task-/) {
$included{$p} = 0; parse_package($_);
$packages{$p}{"Package"} = $p;
foreach $re (qw(Version Priority Section Filename Size MD5sum)) {
(m/^$re: (\S+)\s*$/m and $packages{$p}{$re} = $1)
|| msg(1, "Header field '$re' missing for package '$p'.\n");
} }
$packages{$p}{"Depends"} = [];
$packages{$p}{"Suggests"} = [];
$packages{$p}{"Recommends"} = [];
$packages{$p}{"IsUdeb"} = ($packages{$p}{"Filename"} =~ /.udeb$/) ? 1 : 0;
$packages{$p}{"IsFirmware"} = ($packages{$p}{"Filename"} =~ /(firmware|microcode)/) ? 1 : 0;
if ($packages{$p}{"Section"} =~ /contrib\//) {
$packages{$p}{"Component"} = "contrib";
} elsif ($packages{$p}{"Section"} =~ /non-free\//) {
$packages{$p}{"Component"} = "non-free";
} elsif ($packages{$p}{"IsUdeb"}) {
$packages{$p}{"Component"} = "main-installer";
} else {
$packages{$p}{"Component"} = "main";
}
} }
close AVAIL or die "apt-cache failed : $@ ($!)\n"; close AVAIL or die "apt-cache failed : $@ ($!)\n";
# Read in the extra (new/unstable) tasks packages
if ($force_unstable_tasks) {
my $num = 0;
if ($tasks_packages =~ /\.gz$/) {
open(AVAIL, "zcat $tasks_packages |") || die "Can't zcat $tasks_packages : $!\n";
} else {
open(AVAIL, "< $tasks_packages") || die "Can't open $tasks_packages for reading: $!\n";
}
while (defined($_=<AVAIL>)) {
next if not m/^Package: (\S+)\s*$/m;
if ($1 =~ /^task-/) {
parse_package($_);
$num++;
}
}
close AVAIL or die "reading unstable tasks failed : $@ ($!)\n";
msg(0, " Read $num tasks packages from $tasks_packages\n");
}
$/ = $oldrs; $/ = $oldrs;
# Get the list of excluded packages # Get the list of excluded packages
@ -131,7 +152,7 @@ foreach (keys %excluded) {
close (STATS); close (STATS);
# Browse the list of packages to include # Browse the list of packages to include
my ($output_size, $size) = (0, 0, 0); my ($output_size, $size) = (0, 0);
my %cds; my %cds;
# Generate a dependency tree for each package # Generate a dependency tree for each package
@ -142,7 +163,7 @@ while (@list) {
$ENV{'LC_ALL'} = 'C'; # Required since apt is now translated $ENV{'LC_ALL'} = 'C'; # Required since apt is now translated
open (APT, "$apt cache depends @pkg |") || die "Can't fork : $!\n"; open (APT, "$apt cache depends @pkg |") || die "Can't fork : $!\n";
my (@res) = (<APT>); my (@res) = (<APT>);
close APT or die "« apt-cache depends » failed ... \n" . close APT or die "'apt-cache depends failed ... \n" .
"you must have apt >= 0.3.11.1 !\n"; "you must have apt >= 0.3.11.1 !\n";
# Getting rid of conflicts/replaces/provides # Getting rid of conflicts/replaces/provides
my $i = 0; my $i = 0;
@ -152,8 +173,10 @@ while (@list) {
if ($res[$i] !~ m/^(\S+)\s*$/) { if ($res[$i] !~ m/^(\S+)\s*$/) {
msg(0, "UNEXPECTED: Line `$res[$i]' while parsing " . msg(0, "UNEXPECTED: Line `$res[$i]' while parsing " .
"end of deptree from '$p'\n"); "end of deptree from '$p'\n");
die "sort_deps failed! :-(\n";
} }
$p = lc $1; $i++; $p = lc $1;
$i++;
msg(2, " Dependency tree of `$p' ...\n"); msg(2, " Dependency tree of `$p' ...\n");
read_depends (\$i, \@res, $p); read_depends (\$i, \@res, $p);
} }
@ -181,7 +204,7 @@ msg(0, " S/R/I/B packages take $output_size bytes\n");
# Now start to look for packages wanted by the user ... # Now start to look for packages wanted by the user ...
msg(0, " Adding the rest of the requested packages\n"); msg(0, " Adding the rest of the requested packages\n");
open (LIST, "< $list") || die "Can't open $list : $!\n"; open (LIST, "< $listfile") || die "Can't open $listfile : $!\n";
while (defined($_=<LIST>)) { while (defined($_=<LIST>)) {
chomp; chomp;
next if m/^\s*$/; next if m/^\s*$/;
@ -199,7 +222,7 @@ while (defined($_=<LIST>)) {
# nevertheless ... this may be removed once the udebs have a # nevertheless ... this may be removed once the udebs have a
# better depencency system # better depencency system
if ($packages{$_}{"IsUdeb"}) { if ($packages{$_}{"IsUdeb"}) {
add_to_output($packages{$_}{"Size"}, [$_]); add_to_output($_);
} else { } else {
add_package ($_, ! $norecommends, ! $nosuggests); add_package ($_, ! $norecommends, ! $nosuggests);
} }
@ -251,7 +274,7 @@ if ($extranonfree and (! $nonfree))
# include and if COMPLETE=0 there's a chance that the package # include and if COMPLETE=0 there's a chance that the package
# will not get included in any CD ... so I'm checking the complete # will not get included in any CD ... so I'm checking the complete
# list again # list again
open (LIST, "< $list") || die "Can't open $list : $!\n"; open (LIST, "< $listfile") || die "Can't open $listfile : $!\n";
while (defined($_=<LIST>)) { while (defined($_=<LIST>)) {
chomp; chomp;
next if m/^\s*$/; next if m/^\s*$/;
@ -320,17 +343,87 @@ close LOG;
## END OF MAIN ## END OF MAIN
## BEGINNING OF SUBS ## BEGINNING OF SUBS
sub parse_package {
my $p;
m/^Package: (\S+)\s*$/m and $p = $1;
$included{$p} = 0;
$packages{$p}{"Package"} = $p;
foreach $re (qw(Version Priority Section Filename Size MD5sum)) {
(m/^$re: (\S+)\s*$/m and $packages{$p}{$re} = $1)
|| msg(1, "Header field '$re' missing for package '$p'.\n");
}
$packages{$p}{"Depends"} = [];
$packages{$p}{"Suggests"} = [];
$packages{$p}{"Recommends"} = [];
$packages{$p}{"IsUdeb"} = ($packages{$p}{"Filename"} =~ /.udeb$/) ? 1 : 0;
$packages{$p}{"IsFirmware"} = ($packages{$p}{"Filename"} =~ /(firmware|microcode)/) ? 1 : 0;
if ($packages{$p}{"Section"} =~ /contrib\//) {
$packages{$p}{"Component"} = "contrib";
} elsif ($packages{$p}{"Section"} =~ /non-free\//) {
$packages{$p}{"Component"} = "non-free";
} elsif ($packages{$p}{"IsUdeb"}) {
$packages{$p}{"Component"} = "main-installer";
} else {
$packages{$p}{"Component"} = "main";
}
}
sub dump_depend {
my $tmpin = shift;
my %d;
my $ret = "";
if ("ARRAY" eq ref($tmpin)) {
my @array = @{$tmpin};
foreach (@array) {
%d = %$_;
$ret .= $d{"Package"};
if ($d{"CmpOp"} ne "") {
$ret .= " (" . $d{"CmpOp"} . " " . $d{"Version"} . ")";
}
$ret .= " ";
}
} elsif ("HASH" eq ref($tmpin)) {
%d = %$tmpin;
$ret .= $d{"Package"};
if ($d{"CmpOp"} ne "") {
$ret .= " (" . $d{"CmpOp"} . " " . $d{"Version"} . ")";
}
} else {
die "dump_depend: $tmpin is neither an array nor a hash!\n";
}
return $ret;
}
sub dump_or_list {
my $out_type = shift;
my $elt = shift;
my @or = @$elt;
if (scalar @or == 1) {
msg(1, " $out_type: " . dump_depend($or[0]) . "\n");
} else {
msg(1, " $out_type: OR (");
foreach my $t (@or) {
msg(1, dump_depend($t) . " ");
}
msg(1, ")\n");
}
}
sub read_depends { sub read_depends {
my $i = shift; # Ref my $i = shift; # Ref
my $lines = shift; # Ref my $lines = shift; # Ref
my $pkg = shift; # string my $pkg = shift; # string
my $types = "(?:Pre)?Depends|Suggests|Recommends|Replaces|Conflicts|Breaks|Enhances"; my $types = "(?:Pre)?Depends|Suggests|Recommends|Replaces|Conflicts|Breaks|Enhances";
my (@dep, @rec, @sug); my (@dep, @rec, @sug);
my ($type, $or, $elt); my ($type, $or);
while ($lines->[$$i] =~ m/^\s([\s\|])($types):/) { while ($lines->[$$i] =~ m/^\s([\s\|])($types):/) {
$type = $2; $or = $1; $type = $2; $or = $1;
# Get rid of replaces and conflicts ... # Get rid of replaces, conflicts and any other fields we don't
# care about...
if (($type eq "Replaces") or if (($type eq "Replaces") or
($type eq "Conflicts") or ($type eq "Conflicts") or
($type eq "Breaks") or ($type eq "Breaks") or
@ -341,42 +434,144 @@ sub read_depends {
} }
next; next;
} }
my $out_type = $type;
$out_type =~ s/^Pre//; # PreDepends are like Depends for me
# Check the kind of depends : or, virtual, normal # Check the kind of depends : or, virtual, normal
if ($or eq '|') { if ($or eq '|') {
$elt = read_ordepends ($i, $lines); my $elt = read_ordepends ($i, $lines);
dump_or_list($out_type, \@$elt);
push @{$packages{$pkg}{$out_type}}, $elt;
} elsif ($lines->[$$i] =~ m/^\s\s$type: <([^>]+)>/) { } elsif ($lines->[$$i] =~ m/^\s\s$type: <([^>]+)>/) {
$elt = read_virtualdepends ($i, $lines); my $elt = read_virtualdepends ($i, $lines);
} elsif ($lines->[$$i] =~ m/^\s\s$type: (\S+)/) { foreach my $t (@$elt) {
$elt = $1; $$i++; msg(1, " $out_type: " . dump_depend($t) . " <virt>\n");
}
push @{$packages{$pkg}{$out_type}}, $elt;
} elsif ($lines->[$$i] =~ m/^\s\s$type: (\S+)( \((\S+) (\S+)\))*/) {
my @or;
my %elt;
$elt{"Package"} = $1;
if (defined $2) {
$elt{"CmpOp"} = $3;
$elt{"Version"} = $4;
} else {
$elt{"CmpOp"} = "";
$elt{"Version"} = "";
}
push @or, \%elt;
$$i++;
# Special case for packages providing not # Special case for packages providing not
# truely virtual packages # truly virtual packages
if ($lines->[$$i] =~ m/^\s{4}/) { if ($lines->[$$i] =~ m/^\s{4}/) {
$elt = [ $elt ]; while ($lines->[$$i] =~ m/\s{4}(\S+)( \((\S+) (\S+)\))*/) {
while ($lines->[$$i] =~ m/\s{4}(\S+)/) { my %elt1;
push @{$elt}, $1; $elt1{"Package"} = $1;
if (defined $2) {
$elt1{"CmpOp"} = $3;
$elt1{"Version"} = $4;
} else {
$elt1{"CmpOp"} = "";
$elt1{"Version"} = "";
}
push @or, \%elt1;
$$i++; $$i++;
} }
} }
dump_or_list($out_type, \@or);
push @{$packages{$pkg}{$out_type}}, \@or;
} else { } else {
msg(0, "ERROR: Unknown depends line : $lines->[$$i]\n"); msg(0, "ERROR: Unknown depends line : $lines->[$$i]\n");
foreach ($$i - 3 .. $$i + 3) { foreach ($$i - 3 .. $$i + 3) {
msg(0, " ", $lines->[$_]); msg(0, " ", $lines->[$_]);
} }
} }
$type =~ s/^Pre//; # PreDepends are like Depends for me
next if dep_satisfied($elt);
push @{$packages{$pkg}{$type}}, $elt;
} }
} }
# Big matrix of tests. Check to see if the available version of a
# package matches what we're requesting in a dependency relationship
sub check_versions {
my $wanted = shift;
my $op = shift;
my $available = shift;
# Trivial check - if we don't care about versioning, anything will
# do!
if ($op eq "") {
return 1;
}
# Ask the dpkg perl code to compare the version strings
my $comp = version_compare($available, $wanted);
if ($op eq "<=") {
if ($comp == -1 || $comp == 0) {
return 1;
}
} elsif ($op eq ">=") {
if ($comp == 0 || $comp == 1) {
return 1;
}
} elsif ($op eq "<<") {
if ($comp == -1) {
return 1;
}
} elsif ($op eq ">>") {
if ($comp == 1) {
return 1;
}
} elsif ($op eq "=") {
if ($comp == 0) {
return 1;
}
# Not sure this ("!") actually exists!
# Mentioned in apt sources, but not in debian policy
# No harm done by checking for it, though...
} elsif ($op eq "!") {
if ($comp == -1 || $comp == 1) {
return 1;
}
}
# else
return 0;
}
# Check if a specific dependency package is installed already
sub dep_pkg_included {
my $p = shift;
my %d = %$p;
my $pn = $d{"Package"};
if ($included{$pn}) {
if (check_versions($d{"Version"}, $d{"CmpOp"}, $packages{$pn}{"Version"})) {
msg(1, " $pn is included already, acceptable version " . $packages{$pn}{"Version"} . "\n");
return 1;
} else {
msg(1, " $pn is included already, but invalid version " . $packages{$pn}{"Version"} . "\n");
}
}
# else
return 0;
}
# Check to see if a dependency is satisfied, either a direct
# dependency or any one of an OR array
sub dep_satisfied { sub dep_satisfied {
my $p = shift; my $p = shift;
if (ref $p) {
if ("ARRAY" eq ref $p) {
foreach (@{$p}) { foreach (@{$p}) {
return 1 if $included{$_}; if (dep_pkg_included($_)) {
return 1;
} }
}
} elsif ("HASH" eq ref $p) {
return dep_pkg_included($p);
} else { } else {
return $included{$p}; die "dep_satisfied: $p is neither a hash nor an array!\n";
} }
return 0; return 0;
} }
@ -385,11 +580,21 @@ sub read_ordepends {
my $i = shift; my $i = shift;
my $lines = shift; my $lines = shift;
my @or = (); my @or = ();
my ($val,$dep, $last) = ('','',0); my ($val, $dep, $last) = ('','',0);
my ($op, $version);
chomp $lines->[$$i];
while ($lines->[$$i] while ($lines->[$$i]
=~ m/^\s([\s\|])((?:Pre)?Depends|Suggests|Recommends): (\S+)/) { =~ m/^\s([\s\|])((?:Pre)?Depends|Suggests|Recommends): (\S+)( \((\S+) (\S+)\))*/) {
$val = $3; $val = $3;
if (defined $4) {
$op = $5;
$version = $6;
} else {
$op = "";
$version = "";
}
$last = 1 if $1 ne '|'; #Stop when no more '|' $last = 1 if $1 ne '|'; #Stop when no more '|'
if ($val =~ m/^<.*>$/) { if ($val =~ m/^<.*>$/) {
$dep = read_virtualdepends ($i, $lines); $dep = read_virtualdepends ($i, $lines);
@ -399,11 +604,26 @@ sub read_ordepends {
push @or, $dep; push @or, $dep;
} }
} else { } else {
push @or, $val; $$i++; my %elt;
# Hack for packages providing not a truely $elt{"Package"} = $val;
$elt{"CmpOp"} = $op;
$elt{"Version"} = $version;
push @or, \%elt;
$$i++;
# Hack for packages providing not a truly
# virtual package # virtual package
while ($lines->[$$i] =~ m/^\s{4}(\S+)/) { while ($lines->[$$i] =~ m/^\s{4}(\S+)( \((\S+) (\S+)\))*/) {
push @or, $1; my %elt1;
$elt1{"Package"} = $1;
if (defined $2) {
$elt1{"CmpOp"} = $3;
$elt1{"Version"} = $4;
} else {
$elt1{"CmpOp"} = "";
$elt1{"Version"} = "";
}
msg(1, " " . dump_depend(\%elt1) . "\n");
push @or, \%elt1;
$$i++; $$i++;
} }
} }
@ -425,14 +645,28 @@ sub read_virtualdepends {
$$i++ $$i++
} }
# Now look at the alternatives on the following lines # Now look at the alternatives on the following lines
while ($lines->[$$i] =~ m/^\s{4}(\S+)/) { while ($lines->[$$i] =~ m/^\s{4}(\S+)( \((\S+) (\S+)\))*/) {
push @or, $1; my %elt;
$elt{"Package"} = $1;
if (defined $2) {
$elt{"CmpOp"} = $3;
$elt{"Version"} = $4;
} else {
$elt{"CmpOp"} = "";
$elt{"Version"} = "";
}
push @or, \%elt;
$$i++; $$i++;
} }
if (@or) { if (@or) {
return \@or; return \@or;
} else { } else {
return $virtual; my %elt;
$elt{"Package"} = $virtual;
$elt{"CmpOp"} = "";
$elt{"Version"} = "";
push @or, \%elt;
return \@or;
} }
} }
@ -463,7 +697,7 @@ sub add_package {
return; return;
} }
msg(3, " \@dep before checklist = @dep\n"); msg(3, " \@dep before checklist = " . dump_depend(\@dep) . "\n");
# Check if all packages are allowed (fail if one cannot) # Check if all packages are allowed (fail if one cannot)
($ok, $reasons) = check_list (\@dep, 1); ($ok, $reasons) = check_list (\@dep, 1);
@ -473,11 +707,12 @@ sub add_package {
return; return;
} }
msg(3, " \@dep after checklist = @dep\n"); msg(3, " \@dep after checklist = " . dump_depend(\@dep) . "\n");
if ($add_rec) { if ($add_rec) {
#TODO: Look for recommends (not yet included !!) #TODO: Look for recommends (not yet included !!)
add_recommends (\@dep); add_recommends (\@dep, $p);
msg(3, " \@dep after add_recommends = " . dump_depend(\@dep) . "\n");
# Check again but doesn't fail if one of the package cannot be # Check again but doesn't fail if one of the package cannot be
# installed, just ignore it (it will be removed from @dep) # installed, just ignore it (it will be removed from @dep)
($ok, $reasons) = check_list (\@dep, 0); ($ok, $reasons) = check_list (\@dep, 0);
@ -485,12 +720,13 @@ sub add_package {
msg(0, "UNEXPECTED: It shouldn't fail here !\n"); msg(0, "UNEXPECTED: It shouldn't fail here !\n");
return; return;
} }
msg(3, " \@dep after checklist2 = @dep\n"); msg(3, " \@dep after checklist2 = " . dump_depend(\@dep) . "\n");
} }
if ($add_sug) { if ($add_sug) {
#TODO: Look for suggests (not yet included !!) #TODO: Look for suggests (not yet included !!)
add_suggests (\@dep); add_suggests (\@dep, $p);
msg(3, " \@dep after add_suggests = " . dump_depend(\@dep) . "\n");
# Check again but doesn't fail if one of the package cannot be # Check again but doesn't fail if one of the package cannot be
# installed, just ignore it (it will be removed from @dep) # installed, just ignore it (it will be removed from @dep)
($ok, $reasons) = check_list (\@dep, 0); ($ok, $reasons) = check_list (\@dep, 0);
@ -498,105 +734,184 @@ sub add_package {
msg(0, "UNEXPECTED: It shouldn't fail here !\n"); msg(0, "UNEXPECTED: It shouldn't fail here !\n");
return; return;
} }
msg(3, " \@dep after checklist3 = @dep\n"); msg(3, " \@dep after checklist3 = " . dump_depend(\@dep) . "\n");
} }
# All packages are ok, now check for the size issue # All packages are ok, now list them out and add sizes
$size = get_size (\@dep); foreach my $t (@dep) {
add_to_output ($size, \@dep); my %t = %$t;
my $pkgname = $t{"Package"};
add_to_output($pkgname);
}
} }
sub accepted { sub accepted {
my $p = shift; my $p = shift;
return not $excluded{$p} if (exists $excluded{$p}); return not $excluded{$p} if (exists $excluded{$p});
# Return false for a non-existant package ... # Return false for a non-existent package ...
msg(1, "WARNING: $p cannot be accepted, it doesn't exist ...\n"); msg(1, "WARNING: $p cannot be accepted, it doesn't exist ...\n");
return 0; return 0;
} }
sub add_suggests { sub add_suggests {
my $list = shift; my $deps_list = shift;
my $pkg = shift;
my @parents = ($pkg);
my $p; # = shift; my $p; # = shift;
my @copy = @{$list}; # A copy is needed since I'll modify the array my @copy = @{$deps_list}; # A copy is needed since I'll modify the array
foreach $p (@copy) { foreach $p (@copy) {
add_missing($list, $packages{$p}{"Suggests"}, $p); my %t = %$p;
my $pkgname = $t{"Package"};
add_missing($deps_list, $packages{$pkgname}{"Suggests"}, \%t, 1, \@parents);
} }
} }
sub add_recommends { sub add_recommends {
my $list = shift; my $deps_list = shift;
my $pkg = shift;
my @parents = ($pkg);
my $p; # = shift; my $p; # = shift;
my @copy = @{$list}; # A copy is needed since I'll modify the array my @copy = @{$deps_list}; # A copy is needed since I'll modify the array
foreach $p (@copy) { foreach $p (@copy) {
add_missing($list, $packages{$p}{"Recommends"}, $p); my %t = %$p;
my $pkgname = $t{"Package"};
add_missing($deps_list, $packages{$pkgname}{"Recommends"}, \%t, 1, \@parents);
} }
} }
sub get_missing { sub get_missing {
my $p = shift; my $p = shift;
my @list = (); my @deps_list = ();
my @parents = ();
my %t;
my $dep_text;
if (not add_missing (\@list, $packages{$p}{"Depends"}, $p)) { $t{"Package"} = $p;
$t{"CmpOp"} = "";
$t{"Version"} = "";
if (not add_missing (\@deps_list, $packages{$p}{"Depends"}, \%t, 0, \@parents)) {
return (); return ();
} }
remove_entry($p, \@list); # Explicitly move the package itself to the end of the list,
push @list, $p; # i.e. *after* all its dependencies
return (@list); remove_entry(\%t, \@deps_list);
push @deps_list, \%t;
return (@deps_list);
} }
# Recursive function adding to the # Recursive function adding packages to our list
sub add_missing { sub add_missing {
my $list = shift; my $list = shift;
my $new = shift; my $new = shift;
my $pkgin = shift; my $pkgin = shift;
my @backup = @{$list}; my @backup = @{$list};
my $ok = 1; my $ok = 1;
my $soft_depend = shift;
my $parents = shift;
my $pkgname;
my (%pkgin);
# Check all dependencies if (ref $pkgin eq "HASH") {
foreach (@{$new}) { %pkgin = %$pkgin;
if (ref) {
my $textout = "";
foreach my $orpkg (@{$_}) {
$textout .= "$orpkg ";
}
msg(3, " $pkgin Dep: ( OR $textout)\n");
} else { } else {
msg(3, " $pkgin Dep: $_\n"); die "add_missing passed a non-hash";
} }
next if dep_satisfied ($_);
push(@{$parents}, $pkgin{"Package"});
#msg(3, " add_missing: parents atm @{$parents}\n");
# Check all dependencies
foreach my $thisdep (@{$new}) {
my $textout = "";
$pkgname = $pkgin{"Package"};
# Print out status
if ("ARRAY" eq ref($thisdep)) {
if (scalar(@{$thisdep} > 1)) {
$textout = "(OR ";
}
foreach my $orpkg (@{$thisdep}) {
$textout .= dump_depend($orpkg) . " ";
}
if (scalar(@{$thisdep} > 1)) {
$textout .= ")";
}
} elsif ("HASH" eq ref($thisdep)) {
$textout = dump_depend($thisdep);
} else {
die "add_missing: $thisdep should be an array or hash!\n";
}
msg(3, " $pkgname Dep: $textout soft_depend $soft_depend\n");
# Bail out early if we can!
if (dep_satisfied ($thisdep)) {
next;
}
# Still work to do...
# If it's an OR # If it's an OR
if (ref) { if ("ARRAY" eq ref($thisdep)) {
my $or_ok = 0; my $or_ok = 0;
# Loop over each package in the OR
# We always add the first package in the OR to allow # First check all the OR packages up-front with no
# APT to figure out which is the better one to install # recursion. If *any* one of them is already installed, it
# for any combination of packages that have similar # will do.
# alternative dependencies, but in different order. foreach my $pkg (@{$thisdep}) {
# Having the first alternative available should be good my %t = %$pkg;
# enough for all cases we care about. my $pkgname = $t{"Package"};
foreach my $pkg (@{$_}) {
next if not accepted ($pkg); # Already installed?
# If the package is already included if (dep_satisfied($pkg)) {
# then don't worry msg(3, " OR relationship already installed: " . dump_depend($pkg) . "\n");
if ($included{$pkg}) {
$or_ok = 1; $or_ok = 1;
last; last;
} }
# Pulled in already somewhere above us in the
# depth-first search? (yes, we have to cope with
# circular dependencies here...)
if (is_in ($t{"Package"}, $parents) &&
check_versions($t{"Version"}, $t{"CmpOp"}, $packages{$pkgname}{"Version"})) {
msg(3, " OR relationship already satisfied by parent " . dump_depend($pkg) . "\n");
$or_ok = 1;
last;
}
# else
msg(3, " " . dump_depend($pkg) . " not already installed\n");
}
# If we don't have any of the OR packages, then start
# again and try them in order. We always add the first
# package in the OR to allow APT to figure out which is
# the better one to install for any combination of
# packages that have similar alternative dependencies, but
# in different order. Having the first alternative
# available should be good enough for all cases we care
# about.
if (not $or_ok) {
msg(3, " OR relationship not already satisfied, looking at alternatives in order\n");
foreach my $pkg (@{$thisdep}) {
my %t = %$pkg;
my $pkgname = $t{"Package"};
if (not accepted($pkgname)) {
next;
}
# Check we don't already have the package # Check we don't already have the package
if (is_in ($pkg, $list)) { if (is_in_dep_list($pkg, $list)) {
$or_ok = 1; $or_ok = 1;
last; last;
# Otherwise try to add it # Otherwise try to add it
} else { } else {
# Stop after the first package that is # Stop after the first
# package that is
# added successfully # added successfully
push (@{$list}, $pkg); push (@{$list}, $pkg);
if (add_missing ($list, $packages{$pkg}{"Depends"}, $pkg)) { if (add_missing ($list, $packages{$pkgname}{"Depends"}, $pkg, $soft_depend, $parents)) {
$or_ok = 1; $or_ok = 1;
remove_entry($pkg, $list); remove_entry($pkg, $list);
push @{$list}, $pkg; push @{$list}, $pkg;
@ -606,38 +921,65 @@ sub add_missing {
} }
} }
} }
}
$ok &&= $or_ok; $ok &&= $or_ok;
if (not $ok) { if (not $ok) {
msg(1, " $pkgin failed, couldn's satisfy OR dep\n"); $pkgname = $pkgin{"Package"};
if ($soft_depend) {
msg(1, " $pkgname failed, couldn't satisfy OR dep (but it's a soft dep, so ignoring...)\n");
$ok = 1;
} else {
msg(1, " $pkgname failed, couldn't satisfy OR dep\n");
}
} }
# Else it's a simple dependency # Else it's a simple dependency
} else { } else {
msg(1, " Looking at adding $_ to satisfy dep\n"); my %t = %{$thisdep};
if (not exists $packages{lc $_}) { my $pt = dump_depend(\%t);
msg(1, " $_ doesn't exist...\n"); msg(1, " Looking at adding $pt to satisfy dep\n");
msg(1, " $pkgin failed, couldn't satisfy dep on $_\n");
if (not exists $packages{lc $t{"Package"}}) {
msg(1, " $pt doesn't exist...\n");
if ($soft_depend) {
msg(1, " soft dep: $pkgname ok, despite missing dep on $pt\n");
$ok = 1;
} else {
msg(1, " $pkgname failed, couldn't satisfy dep on $pt\n");
$ok = 0; $ok = 0;
last; last;
} }
next if $included{lc $_}; # Already included, don't worry }
next if is_in (lc $_, $list); if (dep_satisfied(\%t)) {
push @{$list}, lc $_; msg(1, " $pt already included\n");
if (not add_missing ($list, $packages{lc $_}{"Depends"}, lc $_)) { next; # Already included, don't worry
msg(1, "couldn't add $_ ...\n"); }
msg(1, "$pkgin failed, couldn't satisfy dep on $_\n"); if (is_in_dep_list(\%t, $list)) {
msg(1, " $pt already in dep list\n");
next;
}
push @{$list}, \%t;
if (not add_missing ($list, $packages{$t{"Package"}}{"Depends"}, \%t, $soft_depend, $parents)) {
my $pkgname = $pkgin{"Package"};
msg(1, "couldn't add $pt ...\n");
if ($soft_depend) {
msg(1, "soft dep: $pkgname ok, despite missing dep on $pt\n");
$ok = 1;
} else {
msg(1, "$pkgname failed, couldn't satisfy dep on $pt\n");
pop @{$list}; pop @{$list};
$ok = 0; $ok = 0;
} }
remove_entry(lc $_, $list); }
push @{$list}, lc $_; remove_entry(\%t, $list);
push @{$list}, \%t;
} }
} }
# If a problem has come up, then restore the original list # If a problem has come up, then restore the original list
if (not $ok) { if (not $ok) {
@{$list} = @backup; @{$list} = @backup;
} }
if (not is_in(lc $pkgin, $list)) { if (not is_in_dep_list(\%pkgin, $list)) {
push @{$list}, lc $pkgin; push @{$list}, \%pkgin;
} }
return $ok; return $ok;
} }
@ -652,15 +994,36 @@ sub is_in {
return 0; return 0;
} }
# Check if a package dependency is already in a dependency list
sub is_in_dep_list {
my $hash = shift;
my $array = shift;
my %t = %$hash;
foreach my $key (@{$array}) {
my %a = %$key;
if ($a{"Package"} eq $t{"Package"}) {
my $pn = $a{"Package"};
if (check_versions($t{"Version"}, $t{"CmpOp"}, $packages{$pn}{"Version"})) {
return 1;
}
}
}
return 0;
}
# Remove an antry from @{$array} # Remove an antry from @{$array}
sub remove_entry { sub remove_entry {
my $value = shift; my $tmp1 = shift;
my $array = shift; my $array = shift;
my $entries = scalar(@{$array}); my $entries = scalar(@{$array});
my $i; my $i;
my %t1 = %$tmp1;
for ($i=0; $i < $entries; $i++) { for ($i=0; $i < $entries; $i++) {
if (@{$array}[$i] eq $value) { my $tmp2 = @{$array}[$i];
my %t2 = %$tmp2;
if ($t1{"Package"} eq $t2{"Package"}) {
splice(@{$array}, $i, 1); splice(@{$array}, $i, 1);
$i--; $i--;
$entries--; $entries--;
@ -668,16 +1031,6 @@ sub remove_entry {
} }
} }
# The size of a group of packages
sub get_size {
my $arrayref = shift;
my $size = 0;
foreach (@{$arrayref}) {
$size += $packages{$_}{"Size"};
}
return $size;
}
# Check a list of packages # Check a list of packages
sub check_list { sub check_list {
my $ref = shift; my $ref = shift;
@ -685,31 +1038,34 @@ sub check_list {
my $ok = 1; my $ok = 1;
my @to_remove = (); my @to_remove = ();
my $reasons = ""; my $reasons = "";
foreach (@{$ref}) { foreach my $thispkg (@{$ref}) {
if (not exists $excluded{$_}) { my %t = %$thispkg;
msg(1," $_ has been refused because it doesn't exist ...\n"); my $pkgname = $t{"Package"};
if (not exists $excluded{$pkgname}) {
msg(1," $pkgname has been refused because it doesn't exist ...\n");
$ok = 0; $ok = 0;
push @to_remove, $_; push @to_remove, $thispkg;
$reasons = $reasons . " noexist"; $reasons = $reasons . " noexist";
next; next;
} }
if (not accepted($_)) { if (not accepted($pkgname)) {
msg(1," $_ has been refused because of $excluded{$_} ...\n"); msg(1," $pkgname has been refused because of $excluded{$pkgname} ...\n");
$ok = 0; $ok = 0;
push @to_remove, $_; push @to_remove, $thispkg;
$reasons = $reasons . " " . $excluded{$_}; $reasons = $reasons . " " . $excluded{$pkgname};
next; next;
} }
if ($included{$_}) { if ($included{$pkgname}) {
msg(1, msg(1, " $pkgname has already been included.\n");
" $_ has already been included in CD $included{$_}.\n"); push @to_remove, $thispkg;
push @to_remove, $_;
$reasons = $reasons . " alreadyinc"; $reasons = $reasons . " alreadyinc";
next; next;
} }
} }
foreach my $removed (@to_remove) { foreach my $removed (@to_remove) {
msg(2, " Removing $removed ... ($reasons )\n"); my %t = %$removed;
my $pkgname = $t{"Package"};
msg(2, " Removing $pkgname ... ($reasons )\n");
@{$ref} = grep { $_ ne $removed } @{$ref}; @{$ref} = grep { $_ ne $removed } @{$ref};
} }
return ($fail ? $ok : 1, $reasons); return ($fail ? $ok : 1, $reasons);
@ -717,17 +1073,12 @@ sub check_list {
# Add packages to the output list # Add packages to the output list
sub add_to_output { sub add_to_output {
my $size = shift; my $pkgname = shift;
my $ref = shift; my $size = $packages{$pkgname}{"Size"};
msg(2, " \$output_size = $output_size, \$size = $size\n");
$output_size += $size; $output_size += $size;
$included{$pkgname} = 1;
foreach my $pkg (@{$ref}) { push(@output, $pkgname);
$included{$pkg} = 1;
}
push(@output, @{$ref});
} }
sub yesno { sub yesno {