[976] | 1 | #!/usr/bin/perl -w
|
---|
| 2 | #
|
---|
| 3 | # rpmbootstrap application, a debootstrap like for RPM distros
|
---|
| 4 | #
|
---|
| 5 | # $Id$
|
---|
| 6 | #
|
---|
[1534] | 7 | # Copyright B. Cornec 2010-2012
|
---|
| 8 | # Eric Anderson's changes are (c) Copyright 2012 Hewlett Packard
|
---|
[976] | 9 | # Provided under the GPL v2
|
---|
| 10 |
|
---|
| 11 | # Syntax: see at end
|
---|
| 12 |
|
---|
| 13 | use strict 'vars';
|
---|
| 14 | use Getopt::Long qw(:config auto_abbrev no_ignore_case);
|
---|
| 15 | use Data::Dumper;
|
---|
| 16 | use English;
|
---|
| 17 | use LWP::UserAgent;
|
---|
[984] | 18 | use File::Basename;
|
---|
| 19 | use File::Copy;
|
---|
| 20 | use File::Find;
|
---|
[976] | 21 | use ProjectBuilder::Version;
|
---|
| 22 | use ProjectBuilder::Base;
|
---|
| 23 | use ProjectBuilder::Env;
|
---|
| 24 | use ProjectBuilder::Conf;
|
---|
| 25 | use ProjectBuilder::Distribution;
|
---|
| 26 |
|
---|
| 27 | # Global variables
|
---|
| 28 | my %opts; # CLI Options
|
---|
| 29 |
|
---|
| 30 | =pod
|
---|
| 31 |
|
---|
| 32 | =head1 NAME
|
---|
| 33 |
|
---|
| 34 | rpmbootstrap - creates a chrooted RPM based distribution a la debootstrap, aka Virtual Environment (VE)
|
---|
| 35 |
|
---|
| 36 | =head1 DESCRIPTION
|
---|
| 37 |
|
---|
[1082] | 38 | rpmbootstrap creates a chroot environment (Virtual Environment or VE)
|
---|
| 39 | with a minimal distribution in it, suited for building packages for example.
|
---|
| 40 | It's very much like debootstrap but for RPM based distribution.
|
---|
| 41 | It aims at supporting all distributions supported by project-builder.org
|
---|
| 42 | (RHEL, RH, Fedora, OpeSUSE, SLES, Mandriva, ...)
|
---|
[976] | 43 |
|
---|
[1082] | 44 | It is inspired by work done by Steve Kemp for rinse (http://www.steve.org.uk/),
|
---|
[1326] | 45 | and similar to mock or febootstrap, but fully integrated with project-builder.org
|
---|
| 46 | (which also supports rinse and mock). Contrary to these, rpmbootstrap creates
|
---|
| 47 | an environment where the packages commands are usable after build, as described
|
---|
| 48 | hereafter.
|
---|
[976] | 49 |
|
---|
[1315] | 50 | rpmbootstrap works in 2 phases. The first one is used to download all
|
---|
| 51 | the required packages to have a working package management system in the
|
---|
| 52 | chroot working. This list of packages is stored in /etc/pb/pb.conf under
|
---|
| 53 | the rbsmindep parameter (aka rpmbootstrap minimal dependencies). Once the
|
---|
| 54 | packages have been downloaded from the mirror, they are extracted with
|
---|
| 55 | rpm2cpio. At that point you should be able to use yum on Fedora, urpmi
|
---|
| 56 | on Mandriva/Mageia and zypper on OpenSuSE.
|
---|
| 57 | The second phase uses exactly the previous mentioned tools to install
|
---|
| 58 | exactly the same package list to have a coherent RPM db at the end.
|
---|
| 59 |
|
---|
| 60 | rpmbootstrap has additional options to execute a post-install script
|
---|
| 61 | (-s) or to add packages (-a). Then pb can use the chroot to perform even
|
---|
| 62 | more actions in it.
|
---|
| 63 |
|
---|
[976] | 64 | =head1 SYNOPSIS
|
---|
| 65 |
|
---|
[1900] | 66 | rpmbootstrap [-vhmqpdk][-s script][-i image][-a pkg1[,pkg2,...]] distribution-version-arch [target-dir] [mirror [script]]
|
---|
[976] | 67 |
|
---|
[1082] | 68 | rpmbootstrap [--verbose][--help][--man][--quiet][--print-rpms][--download-only]
|
---|
[1900] | 69 | [--keep][--script script][--image image][--add pkg1,[pkg2,...]] distribution-version-arch [target-dir] [mirror [script]]
|
---|
[976] | 70 |
|
---|
| 71 | =head1 OPTIONS
|
---|
| 72 |
|
---|
| 73 | =over 4
|
---|
| 74 |
|
---|
| 75 | =item B<-v|--verbose>
|
---|
| 76 |
|
---|
| 77 | Print a brief help message and exits.
|
---|
| 78 |
|
---|
| 79 | =item B<-h|--help>
|
---|
| 80 |
|
---|
| 81 | Print a brief help message and exits.
|
---|
| 82 |
|
---|
| 83 | =item B<--man>
|
---|
| 84 |
|
---|
| 85 | Prints the manual page and exits.
|
---|
| 86 |
|
---|
| 87 | =item B<-q|--quiet>
|
---|
| 88 |
|
---|
| 89 | Do not print any output.
|
---|
| 90 |
|
---|
| 91 | =item B<-p|--print-rpms>
|
---|
| 92 |
|
---|
[1082] | 93 | Print the packages to be installed, and exit.
|
---|
| 94 | Note that a target directory must be specified so rpmbootstrap can determine
|
---|
| 95 | which packages should be installed, and to resolve dependencies.
|
---|
| 96 | The target directory will be deleted.
|
---|
[976] | 97 |
|
---|
| 98 | =item B<-d|--download-only>
|
---|
| 99 |
|
---|
| 100 | Download packages, but don't perform installation.
|
---|
| 101 |
|
---|
| 102 | =item B<-k|--keep>
|
---|
| 103 |
|
---|
| 104 | Keep packages in the cache dir for later reuse. By default remove them.
|
---|
| 105 |
|
---|
| 106 | =item B<-s|--script script>
|
---|
| 107 |
|
---|
| 108 | Name of the script you want to execute on the related VEs after the installation.
|
---|
[1082] | 109 | It is executed in host environment.
|
---|
| 110 | You can use the chroot command to execute actions in the VE.
|
---|
[976] | 111 |
|
---|
[1900] | 112 | =item B<-i|--image image>
|
---|
[976] | 113 |
|
---|
[1900] | 114 | Name of the ISO image or the docker image of the distribution you want to install on the related VE.
|
---|
[976] | 115 |
|
---|
[991] | 116 | =item B<-a|--add pkg1[,pkg2,...]>
|
---|
| 117 |
|
---|
[1082] | 118 | Additional packages to add from the distribution you want to install on the related VE
|
---|
| 119 | at the end of the chroot build.
|
---|
[991] | 120 |
|
---|
[1531] | 121 | =item B<--no-stop-on-error>
|
---|
| 122 |
|
---|
| 123 | Continue through errors with best effort.
|
---|
| 124 |
|
---|
[976] | 125 | =back
|
---|
| 126 |
|
---|
| 127 | =head1 ARGUMENTS
|
---|
| 128 |
|
---|
[1044] | 129 | =over 4
|
---|
| 130 |
|
---|
[976] | 131 | =item B<distribution-version-arch>
|
---|
| 132 |
|
---|
| 133 | Full name of the distribution that needs to be installed in the VE. E.g. fedora-11-x86_64.
|
---|
| 134 |
|
---|
| 135 | =item B<target-dir>
|
---|
| 136 |
|
---|
[1082] | 137 | This is the target directory under which the VE will be created.
|
---|
| 138 | Created on the fly if needed.
|
---|
| 139 | If none is given use the default directory hosting VE for project-builder.org
|
---|
| 140 | (Cf: vepath parameter in $HOME/.pbrc)
|
---|
[976] | 141 |
|
---|
[1044] | 142 | =back
|
---|
| 143 |
|
---|
[976] | 144 | =head1 EXAMPLE
|
---|
| 145 |
|
---|
| 146 | To setup a Fedora 12 distribution with an i386 architecture issue:
|
---|
| 147 |
|
---|
| 148 | rpmbootstrap fedora-12-i386 /tmp/fedora/12/i386
|
---|
| 149 |
|
---|
| 150 | =head1 WEB SITES
|
---|
| 151 |
|
---|
[1082] | 152 | The main Web site of the project is available at L<http://www.project-builder.org/>.
|
---|
| 153 | Bug reports should be filled using the trac instance of the project at L<http://trac.project-builder.org/>.
|
---|
[976] | 154 |
|
---|
| 155 | =head1 USER MAILING LIST
|
---|
| 156 |
|
---|
[1082] | 157 | Cf: L<http://www.mondorescue.org/sympa/info/pb-announce> for announces and
|
---|
| 158 | L<http://www.mondorescue.org/sympa/info/pb-devel> for the development of the pb project.
|
---|
[976] | 159 |
|
---|
| 160 | =head1 CONFIGURATION FILE
|
---|
| 161 |
|
---|
| 162 | Uses Project-Builder.org configuration file (/etc/pb/pb.conf or /usr/local/etc/pb/pb.conf)
|
---|
| 163 |
|
---|
| 164 | =head1 AUTHORS
|
---|
| 165 |
|
---|
| 166 | The Project-Builder.org team L<http://trac.project-builder.org/> lead by Bruno Cornec L<mailto:bruno@project-builder.org>.
|
---|
| 167 |
|
---|
| 168 | =head1 COPYRIGHT
|
---|
| 169 |
|
---|
| 170 | Project-Builder.org is distributed under the GPL v2.0 license
|
---|
| 171 | described in the file C<COPYING> included with the distribution.
|
---|
| 172 |
|
---|
| 173 | =cut
|
---|
| 174 |
|
---|
| 175 | # ---------------------------------------------------------------------------
|
---|
| 176 |
|
---|
[1531] | 177 | $Global::pb_stop_on_error = 1;
|
---|
[976] | 178 | my ($projectbuilderver,$projectbuilderrev) = pb_version_init();
|
---|
| 179 | my $appname = "rpmbootstrap";
|
---|
[983] | 180 | $ENV{'PBPROJ'} = $appname;
|
---|
[976] | 181 |
|
---|
| 182 | # Initialize the syntax string
|
---|
| 183 |
|
---|
| 184 | pb_syntax_init("$appname Version $projectbuilderver-$projectbuilderrev\n");
|
---|
[984] | 185 | pb_temp_init();
|
---|
[976] | 186 |
|
---|
| 187 | GetOptions("help|?|h" => \$opts{'h'},
|
---|
[1177] | 188 | "man|m" => \$opts{'man'},
|
---|
| 189 | "verbose|v+" => \$opts{'v'},
|
---|
| 190 | "quiet|q" => \$opts{'q'},
|
---|
| 191 | "log-files|l=s" => \$opts{'l'},
|
---|
| 192 | "script|s=s" => \$opts{'s'},
|
---|
| 193 | "print-rpms|p" => \$opts{'p'},
|
---|
| 194 | "download-only|d" => \$opts{'d'},
|
---|
| 195 | "keep|k" => \$opts{'k'},
|
---|
[1900] | 196 | "image|i=s" => \$opts{'i'},
|
---|
[1177] | 197 | "add|a=s" => \$opts{'a'},
|
---|
| 198 | "version|V=s" => \$opts{'V'},
|
---|
[1531] | 199 | "stop-on-error!" => \$Global::pb_stop_on_error,
|
---|
[976] | 200 | ) || pb_syntax(-1,0);
|
---|
| 201 |
|
---|
| 202 | if (defined $opts{'h'}) {
|
---|
| 203 | pb_syntax(0,1);
|
---|
| 204 | }
|
---|
| 205 | if (defined $opts{'man'}) {
|
---|
| 206 | pb_syntax(0,2);
|
---|
| 207 | }
|
---|
| 208 | if (defined $opts{'v'}) {
|
---|
| 209 | $pbdebug = $opts{'v'};
|
---|
| 210 | }
|
---|
| 211 | if (defined $opts{'q'}) {
|
---|
| 212 | $pbdebug=-1;
|
---|
| 213 | }
|
---|
| 214 | if (defined $opts{'l'}) {
|
---|
| 215 | open(pbLOG,"> $opts{'l'}") || die "Unable to log to $opts{'l'}: $!";
|
---|
| 216 | $pbLOG = \*pbLOG;
|
---|
| 217 | $pbdebug = 0 if ($pbdebug == -1);
|
---|
[1177] | 218 | }
|
---|
[976] | 219 | pb_log_init($pbdebug, $pbLOG);
|
---|
[981] | 220 | #pb_display_init("text","");
|
---|
[976] | 221 |
|
---|
[981] | 222 | #if (defined $opts{'s'}) {
|
---|
| 223 | #$pbscript = $opts{'s'};
|
---|
| 224 | #}
|
---|
| 225 | #if (defined $opts{'i'}) {
|
---|
[1900] | 226 | #$image = $opts{'i'};
|
---|
[981] | 227 | #}
|
---|
[976] | 228 |
|
---|
| 229 | # Get VE name
|
---|
| 230 | $ENV{'PBV'} = shift @ARGV;
|
---|
| 231 | die pb_syntax(-1,1) if (not defined $ENV{'PBV'});
|
---|
| 232 |
|
---|
| 233 | die "Needs to be run as root" if ($EFFECTIVE_USER_ID != 0);
|
---|
| 234 |
|
---|
| 235 | #
|
---|
| 236 | # Initialize distribution info from pb conf file
|
---|
| 237 | #
|
---|
[984] | 238 | pb_log(0,"Starting VE build for $ENV{'PBV'}\n");
|
---|
[1177] | 239 | my $pbos = pb_distro_get_context($ENV{'PBV'});
|
---|
[976] | 240 |
|
---|
| 241 | #
|
---|
| 242 | # Check target dir
|
---|
| 243 | # Create if not existent and use default if none given
|
---|
| 244 | #
|
---|
| 245 | pb_env_init_pbrc(); # to get content of HOME/.pbrc
|
---|
| 246 | my $vepath = shift @ARGV;
|
---|
| 247 |
|
---|
[982] | 248 | #
|
---|
| 249 | # Check for command requirements
|
---|
| 250 | #
|
---|
| 251 | my ($req,$opt) = pb_conf_get_if("oscmd","oscmdopt");
|
---|
[1128] | 252 | pb_check_requirements($req,$opt,$appname);
|
---|
[982] | 253 |
|
---|
[976] | 254 | if (not defined $vepath) {
|
---|
[1181] | 255 | my ($vestdpath) = pb_conf_get("vepath");
|
---|
[1531] | 256 | $vepath = pb_path_expand("$vestdpath->{'default'}/$pbos->{'name'}/$pbos->{'version'}/$pbos->{'arch'}") if (defined $vestdpath->{'default'});
|
---|
[976] | 257 | }
|
---|
| 258 |
|
---|
[981] | 259 | die pb_log(0,"No target-dir specified and no default vepath found in $ENV{'PBETC'}\n") if (not defined $vepath);
|
---|
[976] | 260 |
|
---|
[1815] | 261 | # Cleanup first
|
---|
| 262 | if ( -d $vepath) {
|
---|
| 263 | pb_rm_rf($vepath);
|
---|
| 264 | }
|
---|
| 265 | pb_mkdir_p($vepath);
|
---|
[976] | 266 |
|
---|
| 267 | #
|
---|
| 268 | # Get the package list to download, store them in a cache directory
|
---|
| 269 | #
|
---|
[1402] | 270 | my ($rbscachedir) = pb_conf_get_if("cachedir");
|
---|
[1181] | 271 | my ($pkgs,$mirror) = pb_distro_get_param($pbos,pb_conf_get("rbsmindep","rbsmirrorsrv"));
|
---|
[1802] | 272 | my ($updater) = pb_distro_get_param($pbos,pb_conf_get_if("rbsmirrorupd"));
|
---|
[1531] | 273 | die "No packages defined for $pbos->{name}-$pbos->{version}-$pbos->{arch}" unless $pkgs =~ /\w/;
|
---|
[976] | 274 |
|
---|
| 275 | my $cachedir = "/var/cache/rpmbootstrap";
|
---|
| 276 | $cachedir = $rbscachedir->{'default'} if (defined $rbscachedir->{'default'});
|
---|
[1402] | 277 | $cachedir = $rbscachedir->{$appname} if (defined $rbscachedir->{$appname});
|
---|
[976] | 278 |
|
---|
| 279 | # Point to the right subdir and create it if needed
|
---|
[1177] | 280 | $cachedir .= "/$pbos->{'name'}-$pbos->{'version'}-$pbos->{'arch'}";
|
---|
[976] | 281 | pb_mkdir_p($cachedir) if (! -d $cachedir);
|
---|
| 282 |
|
---|
| 283 | # Get the complete package name from the mirror
|
---|
| 284 | #
|
---|
| 285 | my $ua = LWP::UserAgent->new;
|
---|
| 286 | $ua->timeout(10);
|
---|
| 287 | $ua->env_proxy;
|
---|
| 288 |
|
---|
[1314] | 289 | die "No mirror defined for $pbos->{'name'}-$pbos->{'version'}-$pbos->{'arch'}" if ((not defined $mirror) || ($mirror =~ /^\t*$/));
|
---|
[1802] | 290 | my $resp;
|
---|
[1815] | 291 | my $repo;
|
---|
[1892] | 292 | my %list_pkg;
|
---|
[976] | 293 |
|
---|
[1815] | 294 | ($repo,$resp) = rbs_mirror_response($mirror);
|
---|
[1892] | 295 | foreach my $r (split(/\n/,$resp->as_string())) {
|
---|
| 296 | $list_pkg{$r} = $mirror;
|
---|
| 297 | }
|
---|
[1031] | 298 |
|
---|
[1802] | 299 | # If an update source is availble add it after so that these pkgs update the main ones
|
---|
| 300 | if (defined $updater) {
|
---|
[1815] | 301 | my $void;
|
---|
| 302 | ($void,$resp) = rbs_mirror_response($mirror.$updater);
|
---|
[1892] | 303 | foreach my $r (split(/\n/,$resp->as_string())) {
|
---|
| 304 | $list_pkg{$r} = $mirror.$updater;
|
---|
| 305 | }
|
---|
[1031] | 306 | }
|
---|
| 307 |
|
---|
[983] | 308 | # Manages architectures specificities
|
---|
[1177] | 309 | my $parch = $pbos->{'arch'};
|
---|
| 310 | $parch = "i[3456]86" if ($pbos->{'arch'} eq "i386");
|
---|
[983] | 311 |
|
---|
| 312 | # Get the list of packages and their URL in this hash
|
---|
| 313 | my %url;
|
---|
[1812] | 314 | my %done;
|
---|
[1892] | 315 | foreach my $l (keys %list_pkg) {
|
---|
[1802] | 316 | my ($url,$desc) = rbs_find_pkg($l,$parch,"pkg");
|
---|
| 317 | if (defined $url) {
|
---|
[1892] | 318 | $url{$url} = "$list_pkg{$l}/$desc";
|
---|
[1802] | 319 | } else {
|
---|
| 320 | pb_log(3,"not a package, maybe a dir containing packages\n");
|
---|
[1812] | 321 | ($url,$desc) = rbs_find_pkg($l,$parch,"dir");
|
---|
| 322 | if ((defined $desc) and (not defined $done{$desc})) {
|
---|
| 323 | my $response1 = $ua->get("$mirror/$desc");
|
---|
| 324 | # Check if we have a fedora 17/18 type of repo
|
---|
| 325 | if ($response1->is_success) {
|
---|
| 326 | $done{$desc} = "true";
|
---|
| 327 | foreach my $d (split(/\n/,$response1->as_string())) {
|
---|
| 328 | my ($url2,$desc2) = rbs_find_pkg($d,$parch,"pkg");
|
---|
| 329 | if (defined $url2) {
|
---|
| 330 | # Here we have the dir in which are packages
|
---|
| 331 | $url{$url2} = "$mirror/$desc/$desc2";
|
---|
| 332 | } else {
|
---|
| 333 | pb_log(3,"not a package, and not a dir containing packages\n");
|
---|
[1802] | 334 | }
|
---|
| 335 | }
|
---|
[1812] | 336 | } else {
|
---|
[1802] | 337 | }
|
---|
[983] | 338 | }
|
---|
| 339 | }
|
---|
[976] | 340 | }
|
---|
| 341 |
|
---|
[984] | 342 | #
|
---|
| 343 | # Prepare early the yum cache env for the VE in order to copy in it packages on the fly
|
---|
| 344 | #
|
---|
[990] | 345 | my $oscachedir = "/tmp";
|
---|
| 346 | my $osupdcachedir;
|
---|
| 347 | my $osupdname = "";
|
---|
| 348 |
|
---|
[1177] | 349 | if ($pbos->{'install'} =~ /yum/) {
|
---|
[990] | 350 | $oscachedir = "$vepath/var/cache/yum/core/packages/";
|
---|
| 351 | $osupdcachedir = "$vepath/var/cache/yum/updates-released/packages/";
|
---|
| 352 | $osupdname = "YUM";
|
---|
| 353 | # Recent Fedora release use a new yum cache dir
|
---|
[1177] | 354 | if (($pbos->{'name'} eq "fedora") && ($pbos->{'version'} > 8)) {
|
---|
| 355 | $oscachedir = "$vepath/var/cache/yum/$pbos->{'arch'}/$pbos->{'version'}/fedora/packages";
|
---|
| 356 | $osupdcachedir = "$vepath/var/cache/yum/$pbos->{'arch'}/$pbos->{'version'}/updates/packages";
|
---|
[990] | 357 | $osupdcachedir = "$vepath/var/cache/yum/updates-released/packages/";
|
---|
| 358 | }
|
---|
[1177] | 359 | } elsif ($pbos->{'install'} =~ /zypper/) {
|
---|
| 360 | $oscachedir = "$vepath/var/cache/zypp/packages/opensuse/suse/$pbos->{'arch'}";
|
---|
[990] | 361 | $osupdname = "Zypper";
|
---|
[1177] | 362 | } elsif ($pbos->{'install'} =~ /urpmi/) {
|
---|
[992] | 363 | $oscachedir = "$vepath/var/cache/urpmi/rpms";
|
---|
| 364 | $osupdname = "URPMI";
|
---|
[984] | 365 | }
|
---|
[991] | 366 | pb_log(1,"Setting up $osupdname cache in VE\n");
|
---|
[990] | 367 | pb_mkdir_p($oscachedir);
|
---|
| 368 | pb_mkdir_p($osupdcachedir) if (defined $osupdcachedir);
|
---|
[983] | 369 |
|
---|
[976] | 370 | # For each package to process, get it, put it in the cache dir
|
---|
| 371 | # and extract it in the target dir. If not asked to keep, remove it
|
---|
| 372 | # Just download if asked so.
|
---|
| 373 |
|
---|
[983] | 374 | my $warning = 0;
|
---|
| 375 | my $lwpkg ="";
|
---|
[1531] | 376 | my @installed_packages;
|
---|
| 377 |
|
---|
[1815] | 378 | # On systemd systems, the pkg needs to be first
|
---|
| 379 | # to have the correct links made first
|
---|
| 380 | if ($pkgs =~ /,filesystem,/) {
|
---|
| 381 | $pkgs =~ s/,filesystem,/,/;
|
---|
| 382 | $pkgs = "filesystem,".$pkgs;
|
---|
| 383 | }
|
---|
| 384 |
|
---|
[981] | 385 | foreach my $p (split(/,/,$pkgs)) {
|
---|
[1532] | 386 | $p =~ s/\s+//go;
|
---|
[983] | 387 | pb_log(1,"Processing package $p ...\n");
|
---|
| 388 | # Just print packages names if asked so.
|
---|
[984] | 389 | if (defined $url{$p}) {
|
---|
| 390 | if ($opts{'p'}) {
|
---|
[983] | 391 | pb_log(0,"$url{$p}\n");
|
---|
[984] | 392 | next;
|
---|
[983] | 393 | } else {
|
---|
[984] | 394 | # Now download if not already in cache
|
---|
| 395 | my $p1 = basename($url{$p});
|
---|
| 396 | if (! -f "$cachedir/$p1") {
|
---|
[1892] | 397 | pb_system("wget --tries=5 --quiet -O $cachedir/$p1-new $url{$p}","Downloading package $p1 ...");
|
---|
[1532] | 398 | rename("$cachedir/$p1-new", "$cachedir/$p1") || die "mv $cachedir/$p1-new $cachedir/$p1 failed: $!";
|
---|
[984] | 399 | } else {
|
---|
| 400 | pb_log(1,"Package $p1 already in cache\n");
|
---|
| 401 | }
|
---|
[1177] | 402 |
|
---|
[984] | 403 | # End if download only
|
---|
| 404 | if ($opts{'d'}) {
|
---|
| 405 | next;
|
---|
| 406 | }
|
---|
[1177] | 407 |
|
---|
[984] | 408 | #
|
---|
[990] | 409 | # Copy the cached .RPM files into the oscachedir directory, so that os doesn't need to download them again.
|
---|
[984] | 410 | #
|
---|
[990] | 411 | pb_log(1,"Link package into $oscachedir\n");
|
---|
[1177] | 412 | copy("$cachedir/$p1",$oscachedir) if (defined $oscachedir);
|
---|
| 413 | symlink("$oscachedir/$p1","$osupdcachedir/p1") if (defined $osupdcachedir);
|
---|
[984] | 414 |
|
---|
| 415 | # And extract it to the finale dir
|
---|
| 416 | pb_system("cd $vepath ; rpm2cpio $cachedir/$p1 | cpio -ivdum","Extracting package $p1 into $vepath");
|
---|
[1177] | 417 |
|
---|
[984] | 418 | # Remove cached package if not asked to keep
|
---|
| 419 | if (! $opts{'k'}) {
|
---|
| 420 | unlink("$cachedir/$p1");
|
---|
| 421 | }
|
---|
[1531] | 422 | push(@installed_packages, $p);
|
---|
[983] | 423 | }
|
---|
[984] | 424 | } else {
|
---|
| 425 | pb_log(0,"WARNING: unable to find URL for $p\n");
|
---|
| 426 | $warning++;
|
---|
| 427 | $lwpkg .= " $p";
|
---|
[976] | 428 | }
|
---|
[983] | 429 | }
|
---|
| 430 |
|
---|
| 431 | if ($warning ge 1) {
|
---|
[1177] | 432 | pb_log(0,"$warning WARNINGS found.\nMaybe you should review your package list for $pbos->{'name'}-$pbos->{'version'}-$pbos->{'arch'}\nand remove$lwpkg\n");
|
---|
[983] | 433 | }
|
---|
| 434 |
|
---|
[984] | 435 | # Stop here if we just print
|
---|
[983] | 436 | if ($opts{'p'}) {
|
---|
[1661] | 437 | pb_exit(0);
|
---|
[983] | 438 | }
|
---|
[984] | 439 |
|
---|
| 440 | # Now executes the VE finalization steps required for it to work correctly
|
---|
| 441 | pb_log(0,"VE post configuration\n");
|
---|
| 442 |
|
---|
| 443 | # yum needs that distro-release package be installed, so force it
|
---|
[1177] | 444 | if ($pbos->{'install'} =~ /yum/) {
|
---|
| 445 | my $ddir = $pbos->{'name'};
|
---|
[990] | 446 | foreach my $p1 (<$cachedir/($ddir|redhat)-release-*.rpm>) {
|
---|
| 447 | copy("$cachedir/$p1","$vepath/tmp");
|
---|
| 448 | pb_system("chroot $vepath rpm -ivh --force --nodeps /tmp/$p1","Forcing RPM installation of $p1");
|
---|
| 449 | unlink("$vepath/tmp/$p1");
|
---|
| 450 | }
|
---|
[1652] | 451 | # RedHat 6.2 has a buggy termcap setup and doesn't support usage of setarch, so still returns x86_64 in a chroot
|
---|
| 452 | } elsif (($pbos->{'name'} =~ /redhat/) && ( $pbos->{'version'} =~ /^6/)) {
|
---|
[1645] | 453 | pb_system("chroot $vepath ldconfig","Forcing ldconfig on $pbos->{'name'} $pbos->{'version'}");
|
---|
[1652] | 454 | pb_system("chroot $vepath echo 'arch_canon: x86_64: x86_64 1' >> /etc/rpmrc","Forcing ldconfig on $pbos->{'name'} $pbos->{'version'}");
|
---|
| 455 | pb_system("chroot $vepath echo 'arch_compat: x86_64: i386' >> /etc/rpmrc","Forcing ldconfig on $pbos->{'name'} $pbos->{'version'}");
|
---|
| 456 | pb_system("chroot $vepath echo 'buildarch_compat: x86_64: i386' >> /etc/rpmrc","Forcing ldconfig on $pbos->{'name'} $pbos->{'version'}");
|
---|
| 457 | pb_system("chroot $vepath echo 'buildarchtranslate: x86_64: i386' >> /etc/rpmrc","Forcing ldconfig on $pbos->{'name'} $pbos->{'version'}");
|
---|
[984] | 458 | }
|
---|
| 459 | #
|
---|
| 460 | # Make sure there is a resolv.conf file present, such that DNS lookups succeed.
|
---|
| 461 | #
|
---|
| 462 | pb_log(1,"Creating resolv.conf\n");
|
---|
| 463 | pb_mkdir_p("$vepath/etc");
|
---|
| 464 | copy("/etc/resolv.conf","$vepath/etc/");
|
---|
| 465 |
|
---|
| 466 | #
|
---|
| 467 | # BUGFIX:
|
---|
| 468 | #
|
---|
[1177] | 469 | if ((($pbos->{'name'} eq "centos") || ($pbos->{'name'} eq "rhel")) && ($pbos->{'version'} eq "5")) {
|
---|
[984] | 470 | pb_log(1,"BUGFIX for centos-5\n");
|
---|
| 471 | pb_mkdir_p("$vepath/usr/lib/python2.4/site-packages/urlgrabber.skx");
|
---|
| 472 | foreach my $i (<$vepath/usr/lib/python2.4/site-packages/urlgrabber/keepalive.*>) {
|
---|
[1531] | 473 | move($i,"$vepath/usr/lib/python2.4/site-packages/urlgrabber.skx/");
|
---|
[984] | 474 | }
|
---|
| 475 | }
|
---|
| 476 |
|
---|
| 477 | #
|
---|
| 478 | # /proc needed
|
---|
| 479 | #
|
---|
| 480 | pb_mkdir_p("$vepath/proc");
|
---|
[1564] | 481 | pb_system("mount -o bind /proc $vepath/proc","Mounting /proc") unless (-d "$vepath/proc/$$");
|
---|
[984] | 482 |
|
---|
| 483 | #
|
---|
| 484 | # Some devices may be needed
|
---|
| 485 | #
|
---|
[992] | 486 | pb_mkdir_p("$vepath/dev");
|
---|
[1029] | 487 | chmod 0755,"$vepath/dev";
|
---|
[984] | 488 | pb_system("mknod -m 644 $vepath/dev/random c 1 8","Creating $vepath/dev/random") if (! -c "$vepath/dev/random");
|
---|
| 489 | pb_system("mknod -m 644 $vepath/dev/urandom c 1 9","Creating $vepath/dev/urandom") if (! -c "$vepath/dev/urandom");
|
---|
| 490 | pb_system("mknod -m 666 $vepath/dev/zero c 1 5","Creating $vepath/dev/zero") if (! -c "$vepath/dev/zero");
|
---|
[1309] | 491 | pb_system("mknod -m 666 $vepath/dev/null c 1 3","Creating $vepath/dev/null") if (! -c "$vepath/dev/null");
|
---|
[984] | 492 |
|
---|
[1812] | 493 | # Where is bash
|
---|
| 494 | my $bash = "/usr/bin/bash";
|
---|
| 495 | if (! -x "$vepath/$bash" ) {
|
---|
| 496 | $bash = "/bin/bash";
|
---|
| 497 | if (! -x "$vepath/$bash" ) {
|
---|
| 498 | die "No bash found in usual places. Please report to dev team";
|
---|
| 499 | }
|
---|
| 500 | }
|
---|
| 501 |
|
---|
| 502 |
|
---|
[990] | 503 | my $minipkglist;
|
---|
| 504 |
|
---|
[1027] | 505 | pb_log(1,"Adapting $osupdname repository entries\n");
|
---|
[1177] | 506 | if ($pbos->{'install'} =~ /yum/) {
|
---|
[984] | 507 | #
|
---|
| 508 | # Force the architecture for yum
|
---|
| 509 | # The goal is to allow i386 chroot on x86_64
|
---|
| 510 | #
|
---|
| 511 | # FIX: Not sufficient to have yum working
|
---|
| 512 | # mirrorlist is not usable
|
---|
| 513 | # $releasever also needs to be filtered
|
---|
| 514 | # yum.conf as well
|
---|
[1531] | 515 | foreach my $i (<$vepath/etc/yum.repos.d/*.repo>,"$vepath/etc/yum.conf") {
|
---|
[1177] | 516 | pb_system("sed -i -e 's/\$basearch/$pbos->{'arch'}/g' $i","","quiet");
|
---|
| 517 | pb_system("sed -i -e 's/\$releasever/$pbos->{'version'}/g' $i","","quiet");
|
---|
[984] | 518 | pb_system("sed -i -e 's/^mirrorlist/#mirrorlist/' $i","","quiet");
|
---|
[1817] | 519 | pb_system("sed -i -e 's/^metalink/#metalink/' $i","","quiet");
|
---|
[1015] | 520 | # rather use neutral separators here
|
---|
[1815] | 521 | pb_system("sed -i -e 's|^#baseurl.*\$|baseurl=$repo|' $i","","quiet");
|
---|
[984] | 522 | }
|
---|
[990] | 523 | $minipkglist = "ldconfig yum passwd vim-minimal dhclient authconfig";
|
---|
[1177] | 524 | } elsif ($pbos->{'install'} =~ /zypper/) {
|
---|
[990] | 525 | pb_mkdir_p("$vepath/etc/zypp/repos.d");
|
---|
[1177] | 526 | open(REPO,"> $vepath/etc/zypp/repos.d/$pbos->{'name'}-$pbos->{'version'}") || die "Unable to create repo file";
|
---|
[990] | 527 | my $baseurl = dirname(dirname($mirror));
|
---|
[992] | 528 | # Setup the repo
|
---|
[1177] | 529 | if ($pbos->{'version'} eq "10.2") {
|
---|
[1812] | 530 | pb_system("chroot $vepath $bash -c \"yes | /usr/bin/zypper sa $baseurl $pbos->{'name'}-$pbos->{'version'}\"","Bootstrapping Zypper");
|
---|
[1299] | 531 | } else {
|
---|
[1603] | 532 | # don't care if remove fails if add succeeds.
|
---|
[1812] | 533 | pb_system("chroot $vepath $bash -c \"/usr/bin/zypper rr $pbos->{'name'}-$pbos->{'version'}\"","Bootstrapping Zypper","mayfail");
|
---|
| 534 | pb_system("chroot $vepath $bash -c \"/usr/bin/zypper ar $baseurl $pbos->{'name'}-$pbos->{'version'}\"","Bootstrapping Zypper");
|
---|
[990] | 535 | }
|
---|
[1299] | 536 | #print REPO << "EOF";
|
---|
| 537 | #[opensuse]
|
---|
| 538 | #name=$pbos->{'name'}-$pbos->{'version'}
|
---|
| 539 | #baseurl=$baseurl
|
---|
| 540 | #enabled=1
|
---|
| 541 | #gpgcheck=1
|
---|
| 542 | #
|
---|
| 543 | #EOF
|
---|
| 544 | close(REPO);
|
---|
[1603] | 545 | $minipkglist = "zypper aaa_base";
|
---|
[1531] | 546 | # TODO: Re-installing packages missing and necessary on opensuse 11.4 to get /etc/passwd created.
|
---|
[1177] | 547 | } elsif ($pbos->{'install'} =~ /urpmi/) {
|
---|
[992] | 548 | # Setup the repo
|
---|
[993] | 549 | my $baseurl = dirname(dirname(dirname($mirror)));
|
---|
[1812] | 550 | pb_system("chroot $vepath $bash -c \"urpmi.addmedia --distrib $baseurl\"","Bootstrapping URPMI");
|
---|
[1531] | 551 | # TODO here too ?
|
---|
[1894] | 552 | $minipkglist = "ldconfig urpmi bash passwd vim-minimal dhcp-client";
|
---|
[1645] | 553 | } elsif ($pbos->{'install'} =~ /\/rpm/) {
|
---|
| 554 | opendir(CDIR,$cachedir) || die "Unable to open directory $cachedir: $!";
|
---|
| 555 | foreach my $p (@installed_packages) {
|
---|
| 556 | foreach my $f (readdir(CDIR)) {
|
---|
[1894] | 557 | # find an rpm package ref name-ver-tag.arch.rpm
|
---|
| 558 | next if ($f =~ /^\./);
|
---|
[1645] | 559 | if ($f =~ /^$p-([^-]+)-([^-]+)\.(noarch|$parch)\.rpm$/) {
|
---|
| 560 | # Copy it to the chroot and reference it
|
---|
| 561 | copy("$cachedir/$f","$vepath/var/cache");
|
---|
| 562 | $minipkglist .= " /var/cache/$f";
|
---|
| 563 | last;
|
---|
| 564 | }
|
---|
| 565 | }
|
---|
| 566 | rewinddir(CDIR);
|
---|
| 567 | }
|
---|
| 568 | closedir(CDIR);
|
---|
[984] | 569 | }
|
---|
| 570 |
|
---|
| 571 | #
|
---|
[990] | 572 | # Run "install the necessary modules".
|
---|
| 573 | # No need for sudo here
|
---|
| 574 | #
|
---|
[1177] | 575 | $pbos->{'install'} =~ s/sudo//g;
|
---|
[1533] | 576 | if (((defined $ENV{http_proxy}) && ($ENV{http_proxy} ne '')) || ((defined $ENV{ftp_proxy}) && ($ENV{ftp_proxy} ne ''))) {
|
---|
| 577 | if ($pbos->{'name'} eq "opensuse") {
|
---|
| 578 | # For opensuse 11.4 or 12.1 -- one of them didn't work with http_proxy or HTTP_PROXY set.
|
---|
| 579 | open(PROXY, "> $vepath/etc/sysconfig/proxy") || die "can't open $vepath/etc/sysconfig/proxy: $!";
|
---|
| 580 | print PROXY "HTTP_PROXY=$ENV{http_proxy}\n" if ((defined $ENV{http_proxy}) && ($ENV{http_proxy} ne ''));
|
---|
| 581 | print PROXY "FTP_PROXY=$ENV{ftp_proxy}\n" if ((defined $ENV{ftp_proxy}) && ($ENV{ftp_proxy} ne ''));
|
---|
| 582 | close(PROXY);
|
---|
| 583 | }
|
---|
| 584 | }
|
---|
| 585 |
|
---|
| 586 | # Keep redhat variants from destroying nis domain on install
|
---|
[1812] | 587 | #pb_system("chroot $vepath $bash -e -c \"ln -snf /bin/true /bin/domainname\"");
|
---|
[1533] | 588 |
|
---|
| 589 | #if ($pbos->{'name'} =~ /fedora/i) { # hack to prevent fedora from destroying NIS settings on host
|
---|
| 590 | # open (AUTH, ">$vepath/etc/pam.d/system-auth-ac") || die "unable to open $vepath/etc/pam.d/system-auth-ac for write: $!";
|
---|
| 591 | # print AUTH "#pam_systemd -- will fix later in bootstrap\n";
|
---|
| 592 | # close(AUTH);
|
---|
| 593 | #}
|
---|
[1812] | 594 | pb_system("chroot $vepath $bash -c \"$pbos->{'install'} $minipkglist \"","Bootstrapping OS by running $pbos->{'install'} $minipkglist");
|
---|
[990] | 595 |
|
---|
[1531] | 596 | # CentOS6 will replace the yum.repos.d files; oddly it will leave the yum.conf file alone and make the new one ".rpmnew"
|
---|
| 597 | if (($pbos->{'name'} eq "centos") && ($pbos->{'version'} =~ /^6.*/)) {
|
---|
| 598 | pb_log(0,"Fixing $pbos->{'name'} $pbos->{'version'} bug for yum conf files");
|
---|
| 599 | foreach my $from (<$vepath/etc/yum.repos.d/*.rpmorig>) {
|
---|
| 600 | my $to = $from;
|
---|
| 601 | $to =~ s/.rpmorig$//;
|
---|
| 602 | pb_system("mv $from $to", "Recover $from");
|
---|
| 603 | }
|
---|
| 604 | }
|
---|
| 605 |
|
---|
[990] | 606 | #
|
---|
[984] | 607 | # make 'passwd' work.
|
---|
| 608 | #
|
---|
| 609 | pb_log(1,"Authfix\n");
|
---|
[1533] | 610 | # TODO: Not generic enough to be done like that IMHO
|
---|
| 611 | # In case it was changed during the install
|
---|
[1812] | 612 | #pb_system("chroot $vepath $bash -e -c \"ln -snf /bin/true /bin/domainname\"");
|
---|
| 613 | pb_system("chroot $vepath $bash -c \"if [ -x /usr/bin/authconfig ]; then /usr/bin/authconfig --enableshadow --update --nostart; fi\"","Calling authconfig");
|
---|
[984] | 614 |
|
---|
[991] | 615 | # Installed additional packages we were asked to
|
---|
| 616 | if (defined $opts{'a'}) {
|
---|
| 617 | $opts{'a'} =~ s/,/ /g;
|
---|
[1812] | 618 | pb_system("chroot $vepath $bash -c \"$pbos->{'install'} $opts{'a'} \"","Adding packages to OS by running $pbos->{'install'} $opts{'a'}");
|
---|
[991] | 619 | }
|
---|
| 620 |
|
---|
[984] | 621 | #
|
---|
| 622 | # Clean up
|
---|
| 623 | #
|
---|
| 624 | pb_log(1,"Cleaning up\n");
|
---|
[1177] | 625 | if ($pbos->{'install'} =~ /yum/) {
|
---|
[984] | 626 | pb_system("chroot $vepath /usr/bin/yum clean all","Cleaning yum");
|
---|
| 627 | }
|
---|
| 628 | pb_system("umount $vepath/proc","Unmounting /proc");
|
---|
| 629 | find(\&unlink_old_conf, $vepath);
|
---|
| 630 |
|
---|
| 631 | # Executes post-install step if asked for
|
---|
| 632 | if ($opts{'s'}) {
|
---|
| 633 | pb_system("$opts{'s'} $vepath","Executing the post-install script: $opts{'s'} $vepath");
|
---|
| 634 | }
|
---|
| 635 |
|
---|
[1531] | 636 | if ($warning > 0) {
|
---|
| 637 | pb_log(0,"\n\n\n\n$warning WARNINGS found.\nMaybe you should review your package list for $pbos->{'name'}-$pbos->{'version'}-$pbos->{'arch'}\nand remove$lwpkg\n");
|
---|
| 638 | pb_log(0,"waiting 60 seconds to give you a chance to notice.\n");
|
---|
| 639 | sleep(60);
|
---|
| 640 | }
|
---|
| 641 |
|
---|
[1661] | 642 | pb_exit();
|
---|
| 643 |
|
---|
[984] | 644 | # Function for File::Find
|
---|
| 645 | sub unlink_old_conf {
|
---|
| 646 |
|
---|
| 647 | unlink($_) if ($_ =~ /\.rpmorig$/);
|
---|
| 648 | unlink($_) if ($_ =~ /\.rpmnew$/);
|
---|
| 649 | }
|
---|
[1802] | 650 |
|
---|
| 651 | # Function to store packages found on Web pages
|
---|
| 652 |
|
---|
| 653 | sub rbs_find_pkg {
|
---|
| 654 |
|
---|
| 655 | my $l = shift;
|
---|
| 656 | my $parch = shift;
|
---|
| 657 | my $type = shift;
|
---|
| 658 |
|
---|
| 659 | pb_log(2,"Entering rbs_find_pkg with l=$l and parch=$parch\n");
|
---|
| 660 | # Find a href ref in first pos
|
---|
| 661 | if ($l =~ /<a href="([^<>]*)">([^<>]*)<\/a>/i) {
|
---|
| 662 | my $pkg = $1;
|
---|
| 663 | my $desc = $2;
|
---|
| 664 | pb_log(3,"Found desc URL $desc: ");
|
---|
| 665 | # find an rpm package ref name-ver-tag.arch.rpm
|
---|
| 666 | if (($type eq "pkg") && ($pkg =~ /(.+)-([^-]+)-([^-]+)\.(noarch|$parch)\.rpm$/)) {
|
---|
| 667 | pb_log(3,"package ($1 + $2 + $3 + $4)\n");
|
---|
| 668 | my $url = $1;
|
---|
| 669 | pb_log(2,"Exiting rbs_find_pkg with url=$url and desc=$desc\n");
|
---|
| 670 | return($url,$desc);
|
---|
| 671 | } elsif (($type eq "dir") && ($pkg =~ /([0-z])\//)) {
|
---|
| 672 | my $url = $1;
|
---|
| 673 | pb_log(3,"dir ($1)\n");
|
---|
| 674 | return($url,$1);
|
---|
| 675 | } else {
|
---|
| 676 | pb_log(3,"not a package\n");
|
---|
| 677 | }
|
---|
| 678 | }
|
---|
| 679 | pb_log(2,"Exiting rbs_find_pkg with undef\n");
|
---|
| 680 | return(undef,undef);
|
---|
| 681 | }
|
---|
| 682 |
|
---|
| 683 | sub rbs_mirror_response {
|
---|
| 684 |
|
---|
| 685 | my $mirror = shift;
|
---|
| 686 |
|
---|
| 687 | pb_log(0,"Downloading package list from $mirror ...\n");
|
---|
| 688 | my $response = $ua->get($mirror);
|
---|
| 689 | if (! $response->is_success) {
|
---|
| 690 | if ($mirror =~ /i386/) {
|
---|
| 691 | # Some distro have an i586 or i686 mirror dir instead for i386
|
---|
| 692 | warn "Unable to download package from $mirror for $pbos->{'name'}-$pbos->{'version'}-$pbos->{'arch'}.".$response->status_line;
|
---|
| 693 | $mirror =~ s|/i386/|/i586/|;
|
---|
| 694 | pb_log(0,"Downloading now package list from $mirror ...\n");
|
---|
| 695 | $response = $ua->get($mirror);
|
---|
| 696 | if (! $response->is_success) {
|
---|
| 697 | die "Unable to download package from $mirror for $pbos->{'name'}-$pbos->{'version'}-$pbos->{'arch'}".$response->status_line;
|
---|
| 698 | }
|
---|
| 699 | }
|
---|
| 700 | }
|
---|
| 701 | pb_log(3,"Mirror $mirror gave answer: ".Dumper($response->dump(maxlength => 0))."\n");
|
---|
| 702 |
|
---|
| 703 | # Try to find where the repodata structure is for later usage
|
---|
| 704 | my $repo = $mirror;
|
---|
| 705 | my $found = 0;
|
---|
| 706 | if ($pbos->{'install'} =~ /yum/) {
|
---|
| 707 | my $response1;
|
---|
| 708 | while ($found == 0) {
|
---|
| 709 | $response1 = $ua->get("$repo/repodata");
|
---|
| 710 | pb_log(2,"REPO analyzed: $repo\n");
|
---|
| 711 | if (! $response1->is_success) {
|
---|
| 712 | $repo = dirname($repo);
|
---|
| 713 |
|
---|
| 714 | # There is a limit to the loop, when / is reached and nothing found
|
---|
| 715 | my ($scheme, $account, $host, $port, $path) = pb_get_uri($repo);
|
---|
| 716 | die "Unable to find the repodata structure of the mirror $mirror\nPlease check the URL or warn the dev team.\n" if (($path =~ /^[\/]+$/) || ($path =~ /^$/));
|
---|
| 717 |
|
---|
| 718 | # / not reached, so looping
|
---|
| 719 | next;
|
---|
| 720 | } else {
|
---|
| 721 | # repodata found $repo is correct
|
---|
| 722 | $found = 1;
|
---|
| 723 | pb_log(2,"REPO found: $repo\n");
|
---|
| 724 | last;
|
---|
| 725 | }
|
---|
| 726 | }
|
---|
| 727 | }
|
---|
[1815] | 728 | return($repo,$response);
|
---|
[1802] | 729 | }
|
---|
| 730 |
|
---|