05bad91461
git-svn-id: http://emdebian.org/svn/current@5691 563faec7-e20c-0410-992a-a66f704d0ccd
608 lines
17 KiB
Perl
Executable file
608 lines
17 KiB
Perl
Executable file
#!/usr/bin/perl
|
|
|
|
# Copyright (C) 2009 Neil Williams <codehelp@debian.org>
|
|
#
|
|
# This package is free software; you can redistribute it and/or modify
|
|
# it under the terms of the GNU General Public License as published by
|
|
# the Free Software Foundation; either version 3 of the License, or
|
|
# (at your option) any later version.
|
|
#
|
|
# This program is distributed in the hope that it will be useful,
|
|
# but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
# GNU General Public License for more details.
|
|
#
|
|
# You should have received a copy of the GNU General Public License
|
|
# along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
use IO::File;
|
|
use Config::Auto;
|
|
use File::Basename;
|
|
use Parse::Debian::Packages;
|
|
use strict;
|
|
use vars qw/ $progname $ourversion %scripts $dstrap $script $extra
|
|
@archives $deb $cachedir $config_str %packages $retval $str $retries
|
|
$dir $include $arch $foreign $suite $url $forceunpack $option %options
|
|
@e $sourcesname $libdir $dpkgdir @debootstrap %suites $mirror $etcdir
|
|
$repo @dirs @touch %sources $section %keys $host $key $value $type
|
|
$file $config $tidy /;
|
|
$progname = basename($0);
|
|
$ourversion = "0.0.3";
|
|
|
|
while( @ARGV ) {
|
|
$_= shift( @ARGV );
|
|
last if m/^--$/;
|
|
if (!/^-/) {
|
|
unshift(@ARGV,$_);
|
|
last;
|
|
}
|
|
elsif (/^(-\?|-h|--help|--version)$/) {
|
|
&usageversion();
|
|
exit( 0 );
|
|
}
|
|
elsif (/^(-f|--file)$/) {
|
|
$file = shift(@ARGV);
|
|
}
|
|
elsif (/^(-a|--arch)$/) {
|
|
$arch = shift(@ARGV);
|
|
}
|
|
elsif (/^(-d|--dir)$/) {
|
|
$dir = shift(@ARGV);
|
|
$dir .= ($dir =~ m:/$:) ? '' : "/";
|
|
}
|
|
elsif (/^(--tidy-up)$/) {
|
|
$tidy++;
|
|
}
|
|
else {
|
|
die "$progname: Unknown option $_.\n";
|
|
}
|
|
}
|
|
die ("Need a configuration file - use $progname -f\n")
|
|
if (not defined $file);
|
|
$config = Config::Auto::parse("$file");
|
|
%keys=();
|
|
foreach $key (%$config)
|
|
{
|
|
$type = lc($key) if (ref $key ne "HASH");
|
|
$value = $key if (ref $key eq "HASH");
|
|
$keys{$type} = $value;
|
|
}
|
|
%sources=();
|
|
%packages=();
|
|
%suites=();
|
|
%options=();
|
|
%scripts=();
|
|
foreach $section (sort keys %keys)
|
|
{
|
|
if ($section eq "general")
|
|
{
|
|
$arch = $keys{$section}{'arch'} if (not defined $arch);
|
|
$retries = $keys{$section}{'retries'};
|
|
$dir = $keys{$section}{'directory'} if (not defined $dir);
|
|
$forceunpack = lc($keys{$section}{'forceunpack'});
|
|
@debootstrap = split(' ', lc($keys{$section}{'debootstrap'}));
|
|
}
|
|
else
|
|
{
|
|
$sources{$section}=$keys{$section}{'source'};
|
|
$packages{$section}=$keys{$section}{'packages'};
|
|
$suites{$section}=$keys{$section}{'suite'};
|
|
$scripts{$section}=$keys{$section}{'script'};
|
|
$options{$section}=$keys{$section}{'options'};
|
|
}
|
|
}
|
|
print "$progname $ourversion using $file\n";
|
|
$host = `dpkg-architecture -qDEB_BUILD_ARCH`;
|
|
chomp ($host);
|
|
die ("$progname is not currently able to support native operation.\n")
|
|
if ($host eq $arch);
|
|
# don't let debootstrap do second-stage
|
|
$foreign = "--foreign"; # always set
|
|
$cachedir = "var/cache/apt/"; # archives
|
|
$libdir = "var/lib/apt/"; # lists
|
|
$etcdir = "etc/apt/"; # sources
|
|
$dpkgdir = "var/lib/dpkg/"; # state
|
|
|
|
mkdir ("$dir") if (not -d "$dir");
|
|
system ("mkdir -p ${dir}${cachedir}");
|
|
system ("mkdir -p ${dir}${libdir}");
|
|
system ("mkdir -p ${dir}${dpkgdir}");
|
|
system ("mkdir -p ${dir}etc/apt/sources.list.d/");
|
|
@dirs = qw/ alternatives info parts updates/;
|
|
@touch = qw/ diversion statoverride status lock/;
|
|
foreach my $dpkgd (@dirs) {
|
|
if (not -d "${dir}${dpkgdir}$dpkgd") {
|
|
mkdir "${dir}${dpkgdir}$dpkgd";
|
|
}
|
|
}
|
|
foreach my $file (@touch) {
|
|
utime(time, time, "${dir}${dpkgdir}/$file") or (
|
|
open(F, ">${dir}${dpkgdir}/$file") && close F )
|
|
}
|
|
unlink ("${dir}etc/apt/sources.list.d/sources.list")
|
|
if (-f "${dir}etc/apt/sources.list.d/sources.list");
|
|
unlink ("${dir}etc/apt/sources.list")
|
|
if (-f "${dir}etc/apt/sources.list");
|
|
|
|
foreach $repo (sort keys %suites)
|
|
{
|
|
if (not -e "${dir}${cachedir}") {
|
|
mkdir "${dir}${cachedir}";
|
|
}
|
|
if (not -e "$dir/${libdir}lists") {
|
|
mkdir "$dir/${libdir}lists";
|
|
}
|
|
if (not -e "$dir/${libdir}lists/partial") {
|
|
mkdir "$dir/${libdir}lists/partial";
|
|
}
|
|
if (not -e "$dir/${cachedir}archives") {
|
|
mkdir "$dir/${cachedir}archives";
|
|
}
|
|
if (not -e "$dir/${cachedir}archives/partial") {
|
|
mkdir "$dir/${cachedir}archives/partial";
|
|
}
|
|
if (-d "${dir}etc/apt/")
|
|
{
|
|
open (SOURCES, ">>${dir}etc/apt/sources.list.d/sources.list")
|
|
or die "Cannot open sources list $!";
|
|
$mirror = $sources{$repo};
|
|
$suite = $suites{$repo};
|
|
print SOURCES<<END;
|
|
deb $mirror $suite main
|
|
deb-src $mirror $suite main
|
|
END
|
|
close SOURCES;
|
|
}
|
|
}
|
|
|
|
$config_str = '';
|
|
$config_str .= " -o Apt::Architecture=$arch";
|
|
$config_str .= " -o Apt::Get::AllowUnauthenticated=true";
|
|
$config_str .= " -o Apt::Get::Download-Only=true";
|
|
$config_str .= " -o Apt::Install-Recommends=false";
|
|
$config_str .= " -o Dir=$dir";
|
|
$config_str .= " -o Dir::Etc=${dir}${etcdir}";
|
|
$sourcesname = "sources.list.d/$repo.sources.list";
|
|
$config_str .= " -o Dir::Etc::SourceList=${dir}${etcdir}$sourcesname";
|
|
$config_str .= " -o Dir::State=${dir}${libdir}";
|
|
$config_str .= " -o Dir::State::Status=${dir}${dpkgdir}/status";
|
|
$config_str .= " -o Dir::Cache=${dir}${cachedir}";
|
|
system ("apt-get $config_str update");
|
|
$str = join (' ', values %packages);
|
|
chomp($str);
|
|
$str .= " ";
|
|
my $required = &get_required_debs;
|
|
$str .= join (' ', @$required);
|
|
chomp($str);
|
|
print "apt-get -y $config_str install $str\n";
|
|
$retval = system ("apt-get -y $config_str install $str");
|
|
die ("apt download failed. Exit value: ".($retval/256)."\n")
|
|
if ($retval != 0);
|
|
|
|
foreach $dstrap (@debootstrap)
|
|
{
|
|
$url = $sources{$dstrap};
|
|
$suite = $suites{$dstrap};
|
|
$script = $scripts{$dstrap};
|
|
$option = $options{$dstrap};
|
|
$extra = $packages{$dstrap};
|
|
$include = '';
|
|
if (($extra ne '') and ($option !~ /no-resolve-deps/))
|
|
{
|
|
@e = split(' ', $extra);
|
|
$include = "--include ";
|
|
$include .= join (',', @e);
|
|
}
|
|
&write_script ($dstrap);
|
|
$str = "/usr/sbin/debootstrap $option $include --arch $arch $foreign";
|
|
$str .= " --keep-debootstrap-dir $suite $dir $url $script";
|
|
print "$str\n";
|
|
$retval = system ($str);
|
|
while ($retval != 0 and $retries > 0)
|
|
{
|
|
print "Problem - trying again, $retries left. ".($retval/250)."\n$!\n";
|
|
sleep 1;
|
|
$retval = system ("$str");
|
|
$retries--;
|
|
}
|
|
}
|
|
|
|
&force_unpack if ($forceunpack eq "true");
|
|
system ("touch ${dir}${libdir}lists/lock");
|
|
(not defined $tidy) ? system ("apt-get $config_str update") : &tidy_apt;
|
|
system ("rm -rf ${dir}debootstrap");
|
|
print "\nMultistrap system installed successfully in $dir.\n\n";
|
|
exit 0;
|
|
|
|
sub force_unpack
|
|
{
|
|
my %unpack=();
|
|
opendir (DEBS, "${dir}${cachedir}archives/")
|
|
or die ("Cannot read apt archives directory.\n");
|
|
@archives=grep(/.*\.deb$/, readdir DEBS);
|
|
closedir (DEBS);
|
|
print "I: Calculating obsolete packages\n";
|
|
foreach $deb (sort @archives)
|
|
{
|
|
my $version = `dpkg -f ${dir}${cachedir}archives/$deb Version`;
|
|
my $package = `dpkg -f ${dir}${cachedir}archives/$deb Package`;
|
|
chomp ($version);
|
|
chomp ($package);
|
|
if (exists $unpack{$package})
|
|
{
|
|
my $test=system("dpkg --compare-versions $unpack{$package} '<<' $version");
|
|
$test /= 256;
|
|
# unlink version in $unpack if 0
|
|
# unlink $deb (current one) if 1
|
|
if ($test == 0)
|
|
{
|
|
my $old = $deb;
|
|
$old =~ s/$version/$unpack{$package}/;
|
|
print "I: Removing $old\n";
|
|
unlink "${dir}${cachedir}archives/$old";
|
|
next;
|
|
}
|
|
else
|
|
{
|
|
print "I: Removing $deb\n";
|
|
unlink "${dir}${cachedir}archives/$deb";
|
|
}
|
|
}
|
|
$unpack{$package}=$version;
|
|
}
|
|
open (LOCK, ">${dir}${libdir}lists/lock");
|
|
close (LOCK);
|
|
opendir (DEBS, "${dir}${cachedir}archives/")
|
|
or die ("Cannot read apt archives directory.\n");
|
|
@archives=grep(/.*\.deb$/, readdir DEBS);
|
|
closedir (DEBS);
|
|
my $old = `pwd`;
|
|
chomp ($old);
|
|
chdir ("${dir}");
|
|
foreach $deb (sort @archives)
|
|
{
|
|
print "I: Extracting $deb...\n";
|
|
system ("ar -p \"./${cachedir}archives/$deb\" data.tar.gz | zcat | tar -xf -");
|
|
my $ver=`dpkg -f ./${cachedir}archives/$deb Version`;
|
|
my $pkg=`dpkg -f ./${cachedir}archives/$deb Package`;
|
|
chomp ($ver);
|
|
chomp ($pkg);
|
|
mkdir ("./tmp");
|
|
my $tmpdir = `mktemp -p ./tmp -d -t multistrap.XXXXXX`;
|
|
chomp ($tmpdir);
|
|
mkdir ("./${tmpdir}/listing");
|
|
system ("ar -p \"./${cachedir}archives/$deb\" data.tar.gz > ./${tmpdir}/listing/data.tar.gz");
|
|
my $datatar = `tar -tzf ./${tmpdir}/listing/data.tar.gz`;
|
|
my @lines = split("\n", $datatar);
|
|
open (LIST, ">>./${dpkgdir}info/${pkg}.list");
|
|
foreach my $l (@lines)
|
|
{
|
|
chomp ($l);
|
|
$l =~ s:^\.::;
|
|
$l =~ s:^/$:/\.:;
|
|
$l =~ s:/$::;
|
|
print LIST "$l\n";
|
|
}
|
|
close (LIST);
|
|
system ("rm -rf ./${tmpdir}/listing");
|
|
system ("dpkg -e ./${cachedir}archives/$deb ${tmpdir}/");
|
|
opendir (MAINT, "./${tmpdir}");
|
|
my @maint=grep(!m:\.\.?:, readdir (MAINT));
|
|
closedir (MAINT);
|
|
open (AVAIL, ">>./${dpkgdir}available");
|
|
open (STATUS, ">>./${dpkgdir}status");
|
|
foreach my $mscript (@maint)
|
|
{
|
|
rename "./${tmpdir}/$mscript", "./${dpkgdir}info/$pkg.$mscript";
|
|
if ( $mscript eq "control" )
|
|
{
|
|
open (MSCRIPT, "./${dpkgdir}info/$pkg.$mscript");
|
|
my @scr=<MSCRIPT>;
|
|
close (MSCRIPT);
|
|
my @avail = grep(!/^$/, @scr);
|
|
print AVAIL @avail;
|
|
print STATUS @avail;
|
|
print AVAIL "\n";
|
|
print STATUS "Status: install ok unpacked\n";
|
|
unlink ("./${dpkgdir}info/$mscript");
|
|
}
|
|
}
|
|
close (AVAIL);
|
|
if ( -f "./${dpkgdir}info/$pkg.conffiles")
|
|
{
|
|
print STATUS "Conffiles:\n";
|
|
print " -> Processing conffiles for $pkg\n";
|
|
open (CONF, "./${dpkgdir}info/$pkg.conffiles");
|
|
my @lines=<CONF>;
|
|
close (CONF);
|
|
foreach my $line (@lines)
|
|
{
|
|
chomp ($line);
|
|
my $md5=`md5sum ./$line | cut -d" " -f1`;
|
|
chomp ($md5);
|
|
print STATUS " $line $md5\n";
|
|
}
|
|
}
|
|
print STATUS "\n";
|
|
close (STATUS);
|
|
system ("rm -rf ./${tmpdir}");
|
|
}
|
|
chdir ("$old");
|
|
print "I: Unpacking complete.\n";
|
|
}
|
|
|
|
sub tidy_apt
|
|
{
|
|
print "I: Tidying up apt cache and list data.\n";
|
|
opendir (DEBS, "${dir}${libdir}lists/")
|
|
or die ("Cannot read apt lists directory.\n");
|
|
my @lists=grep(!m:\.\.?$:, readdir DEBS);
|
|
closedir (DEBS);
|
|
foreach my $file (@lists)
|
|
{
|
|
next if (-d $file);
|
|
unlink ("${dir}${libdir}lists/$file");
|
|
}
|
|
opendir (DEBS, "${dir}${cachedir}/")
|
|
or die ("Cannot read apt cache directory/.\n");
|
|
my @files=grep(!m:\.\.?$:, readdir DEBS);
|
|
closedir (DEBS);
|
|
foreach my $file (@files)
|
|
{
|
|
next if (-d $file);
|
|
next unless ($file =~ /\.bin$/);
|
|
unlink ("${dir}${cachedir}$file");
|
|
}
|
|
if ($forceunpack eq "true")
|
|
{
|
|
opendir (DEBS, "${dir}${cachedir}/archives/")
|
|
or die ("Cannot read apt archives directory/.\n");
|
|
my @files=grep(!m:\.\.?$:, readdir DEBS);
|
|
closedir (DEBS);
|
|
foreach my $file (@files)
|
|
{
|
|
next if (-d $file);
|
|
next unless ($file =~ /\.deb$/);
|
|
unlink ("${dir}${cachedir}archives/$file");
|
|
}
|
|
}
|
|
}
|
|
|
|
sub get_required_debs
|
|
{
|
|
# emulate required="$(get_debs Priority: required)"
|
|
# from debootstrap/functions
|
|
# needs to be run after the first apt-get install so that
|
|
# Packages files exist
|
|
my @required=();
|
|
my @debs=();
|
|
opendir (PKGS, "${dir}${libdir}lists/")
|
|
or die ("Cannot open ${dir}${libdir}lists/ directory. $!\n");
|
|
my @lists=grep(/_Packages$/, readdir (PKGS));
|
|
closedir (PKGS);
|
|
# only read Packages files from the debootstrap entries where
|
|
# options do not include --no-resolve-deps
|
|
foreach my $strap (@debootstrap)
|
|
{
|
|
next if ($options{$strap} =~ /no-resolve-deps/);
|
|
my $s = lc($strap);
|
|
foreach my $l (@lists)
|
|
{
|
|
next unless ($l =~ /$s/);
|
|
push (@required, $l);
|
|
}
|
|
}
|
|
foreach my $file (@required)
|
|
{
|
|
my $fh = IO::File->new("${dir}${libdir}lists/$file");
|
|
my $parser = Parse::Debian::Packages->new( $fh );
|
|
while (my %package = $parser->next)
|
|
{
|
|
next unless $package{'Priority'} eq "required";
|
|
push @debs, $package{'Package'};
|
|
}
|
|
}
|
|
return \@debs;
|
|
}
|
|
|
|
sub write_script
|
|
{
|
|
($dstrap) = @_;
|
|
$extra = $packages{$dstrap};
|
|
$script = $scripts{$dstrap};
|
|
return if ($script eq '');
|
|
open (SCRIPT, ">$script") or die ("Cannot open $script. $!\n");
|
|
print SCRIPT<<END;
|
|
mirror_style release
|
|
download_style apt
|
|
finddebs_style hardcoded
|
|
variants - buildd fakechroot minbase
|
|
|
|
work_out_debs () {
|
|
required="$extra"
|
|
}
|
|
|
|
first_stage_install () {
|
|
extract \$required
|
|
}
|
|
|
|
second_stage_install () {
|
|
info BASESUCCESS "Base system installed successfully."
|
|
}
|
|
END
|
|
close (SCRIPT);
|
|
}
|
|
|
|
sub usageversion {
|
|
print(STDERR <<END)
|
|
$progname version $ourversion
|
|
|
|
Usage:
|
|
$progname [-a ARCH] [-d DIR] -f CONFIG_FILE
|
|
$progname -?|-h|--help|--version
|
|
|
|
Command:
|
|
-f|--file CONFIG_FILE: path the the multistrap configuration file.
|
|
|
|
Options:
|
|
-a|--arch ARCHITECTURE: override the configuration file architecture.
|
|
-d|--dir PATH: override the configuration file directory.
|
|
-?|-h|--help: print this usage message and exit
|
|
--version: print this usage message and exit
|
|
|
|
$progname extends debootstrap to provide support for multiple
|
|
repositories, using a configuration file to specify the relevant suites,
|
|
debootstrap options, architecture, extra packages and the mirror to use
|
|
for each repository.
|
|
|
|
Example configuration:
|
|
[General]
|
|
arch=arm
|
|
directory=/opt/multistrap/
|
|
retries=5
|
|
# extract all downloaded archives as well as those
|
|
# unpacked by debootstrap.
|
|
forceunpack=false
|
|
# the order of sections is important.
|
|
# debootstraps are unpacked in this sequence.
|
|
debootstrap=Debian
|
|
|
|
[Debian]
|
|
packages=
|
|
source=http://ftp.uk.debian.org/debian
|
|
suite=lenny
|
|
script=
|
|
options=
|
|
|
|
This will result in a completely normal debootstrap of Debian lenny from
|
|
the specified mirror, for ARM in /opt/multistrap/.
|
|
|
|
'Architecture' and 'directory' can be overridden on the command line.
|
|
|
|
Specify a package to extend the debootstap to include that package and
|
|
all dependencies.
|
|
|
|
Specify a script only if the suite is not currently supported by
|
|
debootstrap (which in turn are based on Debian and Ubuntu suite names).
|
|
|
|
Useful options can include --no-resolve-deps if one of the repositories
|
|
contains packages that are also in any of the other repositories but at
|
|
a lower version. This prevents debootstrap unpacking the older version
|
|
of the duplicated package.
|
|
|
|
General settings:
|
|
|
|
The order of repositories specified in the debootstrap (under General),
|
|
determines which repository is unpacked first - this is important if one
|
|
repository is incomplete or contains older versions.
|
|
|
|
'directory' specifies the top level directory where the debootstrap
|
|
will be created - it is not packed into a .tgz once complete.
|
|
|
|
'retries' determines how many times debootstrap should be allowed to
|
|
restart (the same value is used for all repositories) to allow for
|
|
temporary network issues.
|
|
|
|
If 'arch' does not match the build architecture (determined using
|
|
`dpkg-architecture -qDEB_BUILD_ARCH`, --foreign is passed to
|
|
each debootstrap.
|
|
|
|
END
|
|
|| die "$progname: failed to write usage: $!\n";
|
|
}
|
|
|
|
=pod
|
|
|
|
=head1 Name
|
|
|
|
em_multistrap - extends debootstrap for multiple repository support
|
|
|
|
=head1 Synopsis
|
|
|
|
em_multistrap [-a ARCH] [-d DIR] -f CONFIG_FILE
|
|
em_multistrap -?|-h|--help|--version
|
|
|
|
=head1 Description
|
|
|
|
em_multistrap extends debootstrap to provide support for multiple
|
|
repositories, using a configuration file to specify the relevant suites,
|
|
debootstrap options, architecture, extra packages and the mirror to use
|
|
for each repository.
|
|
|
|
Example configuration:
|
|
|
|
[General]
|
|
arch=arm
|
|
directory=/opt/multistrap/
|
|
retries=5
|
|
# extract all downloaded archives as well as those
|
|
# unpacked by debootstrap.
|
|
forceunpack=false
|
|
# the order of sections is important.
|
|
# debootstraps are unpacked in this sequence.
|
|
debootstrap=Debian
|
|
|
|
[Debian]
|
|
packages=
|
|
source=http://ftp.uk.debian.org/debian
|
|
suite=lenny
|
|
script=
|
|
options=
|
|
|
|
This will result in a completely normal debootstrap of Debian lenny from
|
|
the specified mirror, for ARM in /opt/multistrap/.
|
|
|
|
'Architecture' and 'directory' can be overridden on the command line.
|
|
|
|
Section names are case-insensitive.
|
|
|
|
Specify a package to extend the debootstap to include that package and
|
|
all dependencies.
|
|
|
|
Specify a script only if the suite is not currently supported by
|
|
debootstrap (which in turn are based on Debian and Ubuntu suite names).
|
|
|
|
Useful options can include --no-resolve-deps if one of the repositories
|
|
contains packages that are also in any of the other repositories but at
|
|
a lower version. This prevents debootstrap unpacking the older version
|
|
of the duplicated package.
|
|
|
|
=head1 General settings:
|
|
|
|
The order of repositories specified in the debootstrap (under General),
|
|
determines which repository is unpacked first - this is important if one
|
|
repository is incomplete or contains older versions.
|
|
|
|
'directory' specifies the top level directory where the debootstrap
|
|
will be created - it is not packed into a .tgz once complete.
|
|
|
|
'retries' determines how many times debootstrap should be allowed to
|
|
restart (the same value is used for all repositories) to allow for
|
|
temporary network issues.
|
|
|
|
If 'arch' does not match the build architecture (determined using
|
|
`dpkg-architecture -qDEB_BUILD_ARCH`, --foreign is passed to
|
|
each debootstrap.
|
|
|
|
Note that em_multistrap deliberately turns off Install-Recommends.
|
|
|
|
As with debootstrap, em_multistrap will continue after errors although
|
|
you may want to purge $directory/var/cache/apt/archives from time to
|
|
time.
|
|
|
|
em_multistrap does not currently implement machine:variant support
|
|
but the build directory is not packed up at the end of the run so
|
|
other scripts can be used to implement customisations.
|
|
|
|
=head1 Native mode disabled
|
|
|
|
em_multistrap is not intended for native support, it was developed for
|
|
cross architeeture support. In order for multiple repositories to be
|
|
used, em_multistrap sets the --foreign option with debootstrap even if
|
|
debootstrap would be able to complete the installation of each single
|
|
repository. The reason for this is so that only the packages selected
|
|
by apt are actually unpacked.
|
|
|
|
Currently, em_multistrap disables native support.
|
|
|
|
=cut
|