2
0
mirror of https://github.com/xcat2/xcat-core.git synced 2025-05-21 19:22:05 +00:00

1027 lines
36 KiB
Perl

package xCAT_plugin::statelite;
BEGIN
{
$::XCATROOT = $ENV{'XCATROOT'} ? $ENV{'XCATROOT'} : '/opt/xcat';
}
use lib "$::XCATROOT/lib/perl";
use xCAT::Table;
use Getopt::Long;
use File::Basename;
use File::Path;
use File::Copy;
use File::Find;
use Cwd;
use File::Temp;
use xCAT::Utils qw(genpassword);
use xCAT::TableUtils qw(get_site_attribute);
use xCAT::SvrUtils;
use xCAT::PasswordUtils;
use Data::Dumper;
Getopt::Long::Configure("bundling");
Getopt::Long::Configure("pass_through");
my $cmdname = "liteimg";
my $statedir = ".statelite";
my $verbose = "0";
sub handled_commands {
return {
$cmdname => "statelite"
}
}
# function to handle request. Basically, get the information
# about the image and then do the action on it. Is that vague enough?
sub process_request {
my $request = shift;
my $callback = shift;
my $doreq = shift;
#my $sitetab = xCAT::Table->new('site');
#my $ent = $sitetab->getAttribs({key=>'installdir'},['value']);
my @entries = xCAT::TableUtils->get_site_attribute("installdir");
my $t_entry = $entries[0];
my $installroot = "/install";
# get /install directory
if (defined($t_entry)) {
$installroot = $t_entry;
}
# if not defined, error out... or should we set it as default?
unless ($installroot) {
$callback->({ error => ["No installdir defined in site table"], errorcode => [1] });
return;
}
@ARGV = @{ $request->{arg} };
my $argc = scalar @ARGV;
if ($argc == 0) {
$callback->({ info => ["$cmdname -h # this help message\n$cmdname -v # version\n$cmdname -V # verbose\n$cmdname -p <profile> -a <architecture> -o <OS>\n$cmdname imagename"] });
return;
}
my $rootfstype;
my $exlistloc; # borrowed from packimage.pm
my $osver;
my $arch;
my $profile;
my $rootimg_dir;
my $exlist; # it is used when rootfstype = ramdisk
my $destdir;
my $imagename;
my $dotorrent;
GetOptions(
"rootfstype|t=s" => \$rootfstype,
"profile|p=s" => \$profile,
"arch|a=s" => \$arch,
"osver|o=s" => \$osver,
"help|h" => \$help,
"tracker" => \$dotorrent,
"version|v" => \$version,
"verbose|V" => \$verbose
);
if ($version) {
my $version = xCAT::Utils->Version();
$callback->({ info => [$version] });
return;
}
if ($help) {
$callback->({ info => ["$cmdname -h # this help message\n$cmdname -v # version\n$cmdname -V # verbose\n$cmdname [-p profile] [-a architecture] [-o OS]\n$cmdname imagename"] });
return;
}
# if they only gave us the image name:
if (@ARGV > 0) {
$imagename = $ARGV[0];
if ($arch or $osver or $profile) {
$callback->({ error => ["-o, -p and -a options are not allowed when a image name is specified."], errorcode => [1] });
return;
}
#load the module in memory
eval { require("$::XCATROOT/lib/perl/xCAT/Table.pm") };
if ($@) {
$callback->({ error => [$@], errorcode => [1] });
return;
}
#get the info from the osimage and linux
my $osimagetab = xCAT::Table->new('osimage', -create => 1);
if (!$osimagetab) {
# os image doesn't exist in table.
$callback->({ error => ["The osimage table cannot be opened."], errorcode => [1] });
return;
}
# open the linux image table to get more attributes... for later.
my $linuximagetab = xCAT::Table->new('linuximage', -create => 1);
if (!$linuximagetab) {
$callback->({ error => ["The linuximage table cannot be opened."], errorcode => [1] });
return;
}
# get the os, arch, and profile from the image name table.
(my $ref) = $osimagetab->getAttribs({ imagename => $imagename }, 'rootfstype', 'osvers', 'osarch', 'profile', 'provmethod');
if (!$ref) {
$callback->({ error => ["Cannot find image \'$imagename\' from the osimage table."], errorcode => [1] });
return;
}
my $provmethod = $ref->{'provmethod'};
if ($provmethod !~ /statelite/) {
$callback->({ error => ["Please make sure that osimage.provmethod is set to statelite before calling this command."], errorcode => [1] });
return;
}
$rootfstype = $ref->{'rootfstype'};
$osver = $ref->{'osvers'};
$arch = $ref->{'osarch'};
$profile = $ref->{'profile'};
# get the exlist and rootimgdir attributes
(my $ref1) = $linuximagetab->getAttribs({ imagename => $imagename }, 'exlist', 'rootimgdir');
unless ($ref1) {
$callback->({ error => [qq{Cannot find image '$imagename' from the osimage table.}], errorcode => [1] });
}
$destdir = $ref1->{'rootimgdir'};
$exlistloc = $ref1->{'exlist'};
$rootimg_dir = "$destdir/rootimg";
} # end of case where they give us osimage.
unless ($osver and $arch and $profile) {
$callback->({ error => ["osimage.osvers, osimage.osarch, and osimage.profile and must be specified for the image $imagename in the database, or you must specify -o os -p profile -a arch."], errorcode => [1] });
return;
}
unless ($destdir) {
$destdir = "$installroot/netboot/$osver/$arch/$profile";
$rootimg_dir = "$destdir/rootimg";
}
my $oldpath = cwd();
# now we have all the info we need:
# - rootimg_dir
# - osver
# - arch
# - profile
my $rsp;
push @{ $rsp->{data} }, "Modifying $rootimg_dir ...";
xCAT::MsgUtils->message("I", $rsp, $callback);
#copy $installroot/postscripts into the image at /xcatpost
if (-e "$rootimg_dir/xcatpost") {
system("rm -rf $rootimg_dir/xcatpost");
}
system("mkdir -p $rootimg_dir/xcatpost");
system("cp -r $installroot/postscripts/* $rootimg_dir/xcatpost/");
# update rw to ro for sles
updateFstab($rootimg_dir,$profile,$arch);
#get the root password for the node
my $pass = xCAT::PasswordUtils::crypt_system_password();
if (!defined($pass)) {
$pass = 'cluster';
}
my $oldmask = umask(0077);
my $shadow;
open($shadow, "<", "$rootimg_dir/etc/shadow");
my @shadents = <$shadow>;
close($shadow);
open($shadow, ">", "$rootimg_dir/etc/shadow");
print $shadow "root:$pass:13880:0:99999:7:::\n";
foreach (@shadents) {
unless (/^root:/) {
print $shadow "$_";
}
}
close($shadow);
umask($oldmask);
my $distname = $osver;
unless (-r "$::XCATROOT/share/xcat/netboot/$distname/" or not $distname) {
chop($distname);
}
unless ($distname) {
$callback->({ error => ["Unable to find $::XCATROOT/share/xcat/netboot directory for $osver"], errorcode => [1] });
return;
}
unless ($imagename) {
#store the image in the DB
my @ret = xCAT::SvrUtils->update_tables_with_diskless_image($osver, $arch, $profile, 'statelite');
if ($ret[0]) {
$callback->({ error => [ "Error when updating the osimage tables: " . $ret[1] ] });
}
$imagename = "$osver-$arch-statelite-$profile";
$exlistloc = xCAT::SvrUtils->get_exlist_file_name("$installroot/custom/netboot/$distname", $profile, $osver, $arch);
unless ($exlistloc) {
$exlistloc = xCAT::SvrUtils->get_exlist_file_name("$::XCATROOT/share/xcat/netboot/$distname", $profile, $osver, $arch);
}
}
#sync fils configured in the synclist to the rootimage
$syncfile = xCAT::SvrUtils->getsynclistfile(undef, $osver, $arch, $profile, "netboot", $imagename);
if (defined($syncfile) && -d $rootimg_dir) {
my @filelist = split ',', $syncfile;
foreach my $synclistfile (@filelist) {
if ( -f $synclistfile) {
print "sync files from $synclistfile to the $rootimg_dir\n";
`$::XCATROOT/bin/xdcp -i $rootimg_dir -F $synclistfile`;
}
}
}
# check if the file "litefile.save" exists or not
# if it doesn't exist, then we get the synclist, and run liteMe
# if it exists, it means "liteimg" has run more than one time, we need to compare with the synclist
my @listSaved;
if (-e "$rootimg_dir/.statelite/litefile.save") {
open SAVED, "$rootimg_dir/.statelite/litefile.save";
# store all its contents to @listSaved;
while (<SAVED>) {
chomp $_;
push @listSaved, $_;
}
close SAVED;
}
my %hashSaved = ();
if (parseLiteFiles(\@listSaved, \%hashSaved)) {
$callback->({ error => ["parseLiteFiles failed for listSaved!"] });
return;
}
# now get the files for the node
my @synclist = xCAT::Utils->runcmd("ilitefile $imagename", 0, 1);
unless (@synclist) {
$callback->({ error => ["There are no files to sync for $imagename. You have to have some files read/write filled out in the synclist table."], errorcode => [1] });
return;
}
my $listNew = $synclist[0];
# for compatible reason, replace "tmpfs,rw" with "link" option in xCAT 2.5 or higher
for (@{$listNew}) {
s/tmpfs,rw/link/;
}
# the directory/file in litefile table must be the absolute path ("/***")
foreach my $entry (@$listNew) {
my @tmp = split(/\s+/, $entry);
# check the validity of the option
if ($tmp[1] !~ /^(tmpfs|persistent|localdisk|rw|ro|con|link|tmpfs,rw|link,ro|link,persistent|link,con)$/) {
$callback->({ error => [qq{ $tmp[2] has invalid option. The valid options: tmpfs persistent localdisk rw ro con link tmpfs,rw link,ro link,persistent link,con}], errorcode => [1] });
return;
}
unless ($tmp[2] =~ m{^/}) {
$callback->({ error => [qq{ $tmp[2] is not one absolute path. }], errorcode => [1] });
return;
}
if ($tmp[1] =~ m{con} and $tmp[2] =~ m{/$}) {
$callback->({ error => [qq{ $tmp[2] is directory, don't use "con" as its option }], errorcode => [1] });
return;
}
}
my %hashNew = ();
if (parseLiteFiles($listNew, \%hashNew)) {
$callback->({ error => ["parseLiteFiles failed for listNew!"] });
return;
}
# validate the options for all litefile entries
# if there is any scenario not supported, the command exits
foreach my $entry (keys %hashNew) {
my @tmp = split(/\s+/, $entry);
my $f = $tmp[1];
if ($hashNew{$entry}) {
if ($tmp[0] =~ m/ro$/ or $tmp[0] =~ m/con$/) {
$callback->({ error => [qq{the directory "$f" should not be with "ro" or "con" as its option}], errorcode => [1] });
return;
}
foreach my $child (@{ $hashNew{$entry} }) {
my @tmpc = split(/\s+/, $child);
my $fc = $tmpc[1];
if ($tmp[0] =~ m/link/) {
if ($tmpc[0] eq "link,ro") {
$callback->({ error => [qq{Based on the option of $f, $fc should not use "link,ro" as its option}], errorcode => [1] });
return;
}
if ($tmpc[0] !~ m/link/) {
$callback->({ error => [qq{Based on the option of $f, $fc can only use "link"-based options}], errorcode => [1] });
return;
}
} else {
if ($tmpc[0] =~ m/link/) {
# The /etc/mtab is a specific file which can only be handled by link option.
# It need to be existed in rootimage and during the runing of statelite,
# and after running of statelite, it need to be linked to /proc/mount
if ($tmpc[1] != "/etc/mtab") {
$callback->({ error => [qq{Based on the option of $f, $fc should not use "link"-based options}], errorcode => [1] });
return;
}
}
}
if (($tmp[0] =~ m{persistent}) and ($tmpc[0] !~ m{persistent})) {
# TODO: if the parent is "persistent", the child can be ro/persistent/rw/con
$callback->({ error => ["$fc should have persistent option like $f "], errorcode => [1] });
return;
}
}
}
}
# backup the file/directory before recovering the files in litefile.save
unless (-d "$rootimg_dir/.statebackup") {
if (-e "$rootimg_dir/.statebackup") {
xCAT::Utils->runcmd("rm $rootimg_dir/.statebackup", 0, 1);
}
$verbose && $callback->({ info => ["mkdir $rootimg_dir/.statebackup"] });
xCAT::Utils->runcmd("mkdir $rootimg_dir/.statebackup", 0, 1);
}
#delete useless rootimg/tmp/dracut.* files
#fix copy many dracut.* files cost too much time in liteimg
$verbose && $callback->({ info => ["removing \"$rootimg_dir/tmp/dracut.*\""] });
unlink glob "$rootimg_dir/tmp/dracut.*";
# recovery the files in litefile.save if necessary
foreach my $line (keys %hashSaved) {
my @oldentry = split(/\s+/, $line);
my $f = $oldentry[1];
my @newentries = grep /\s+$f$/, @{$listNew};
my @entry;
if (scalar @newentries == 1) {
@entry = split(/\s+/, $newentries[0]);
}
# backup the children to .statebackup
if ($hashSaved{$line}) {
my $childrenRef = $hashSaved{$line};
unless (-d "$rootimg_dir/.statebackup$f") {
xCAT::Utils->runcmd("rm -rf $rootimg_dir/.statebackup$f", 0, 1) if (-e "$rootimg_dir/.statebackup$f");
$verbose && $callback->({ info => ["mkdir $rootimg_dir/.statebackup$f"] });
xCAT::Utils->runcmd("mkdir -p $rootimg_dir/.statebackup$f");
}
foreach my $child (@{$childrenRef}) {
my @tmpc = split(/\s+/, $child);
my $name = $rootimg_dir . $tmpc[1];
if (-e $name) {
$verbose && $callback->({ info => ["cp -r -a $name $rootimg_dir/.statebackup$f"] });
xCAT::Utils->runcmd("cp -r -a $name $rootimg_dir/.statebackup$f");
}
}
}
# there's one parent directory, whose option is different from the old one
unless ($entry[1] eq $oldentry[0]) {
recoverFiles($rootimg_dir, \@oldentry, $callback);
# if its children items exist, we need to copy the backup files from .statebackup to the rootfs,
if ($hashSaved{$line}) {
$verbose && $callback->({ info => ["$f has child file/directory in the litefile table."] });
my $childrenRef = $hashSaved{$line};
foreach my $child (@{$childrenRef}) {
# recover them from .statebackup to $rootimg_dir
my @tmpc = split(/\s+/, $child);
my $name = $tmpc[1];
my @newentries = grep /\s+$name$/, @{listNew};
my $destf = $rootimg_dir . $name;
my $srcf = $rootimg_dir . "/.statebackup" . $name;
if (-e $destf) {
$verbose && $callback->({ info => ["rm -rf $destf"] });
xCAT::Utils->runcmd("rm -rf $destf", 0, 1);
}
# maybe the dir of $destf doesn't exist, so we will create one
my $dirDestf = dirname $destf;
unless (-d $dirDestf) {
$verbose && $callback->({ info => ["mkdir -p $dirDestf"] });
xCAT::Utils->runcmd("mkdir -p $dirDestf", 0, 1);
}
if (-e $srcf) {
$verbose && $callback->({ info => ["recovering from $srcf to $destf"] });
xCAT::Utils->runcmd("cp -r -a $srcf $destf", 0, 1);
}
}
}
}
# recover the children
if ($hashSaved{$line}) {
$verbose && $callback->({ info => ["$f has child file/directory in the litefile table."] });
my $childrenRef = $hashSaved{$line};
foreach my $child (@{$childrenRef}) {
my @tmpc = split(/\s+/, $child);
my $name = $tmpc[1];
my @newentries = grep /\s+$name$/, @{$listNew};
my @entry;
if (scalar @newentries == 1) {
@entry = split(/\s+/, $newentries[0]);
}
unless ($tmpc[0] eq $entry[1]) {
recoverFiles($rootimg_dir, \@tmpc, $callback);
}
}
}
}
# remove .statebackup
$verbose && $callback->({ info => ["remove .statebackup"] });
xCAT::Utils->runcmd("rm -rf $rootimg_dir/.statebackup", 0, 1);
# then store the @synclist to litefile.save
#system("cp $rootimg_dir/.statelite/litefile.save $rootimg_dir/.statelite/litefile.save1");
open SAVED, ">$rootimg_dir/.statelite/litefile.save";
foreach my $line (@{$listNew}) {
print SAVED "$line\n";
}
close SAVED;
liteMe($rootimg_dir, \%hashNew, $callback);
# now stick the rc file in:
# this is actually a pre-rc file because it gets run before the node boots up all the way.
$verbose && $callback->({ info => ["put the statelite rc file to $rootimg_dir/etc/init.d/"] });
# rh5,rh6.1 to rh6.4 use rc.statelite.ppc.redhat, otherwise use rc.statelite
if (($osver =~ m/^rh[a-zA-Z]*5/) or ($osver =~ m/^rh[a-zA-Z]*6(\.)?[1-4]$/) and $arch eq "ppc64") { # special case for redhat5/6.x on PPC64
system("cp -a $::XCATROOT/share/xcat/netboot/add-on/statelite/rc.statelite.ppc.redhat $rootimg_dir/etc/init.d/statelite");
} else {
system("cp -a $::XCATROOT/share/xcat/netboot/add-on/statelite/rc.statelite $rootimg_dir/etc/init.d/statelite");
}
# newly-introduced code for the rootfs with "ramdisk" as its type
if ($rootfstype eq "ramdisk") {
my $xcat_packimg_tmpfile = "/tmp/xcat_packimg.$$";
my $excludestr = "find . ";
my $includestr;
if ($exlistloc) {
my @excludeslist = split ',', $exlistloc;
foreach my $exlistlocname (@excludeslist) {
my $exlist;
my $excludetext;
open($exlist, "<", $exlistlocname);
system("echo -n > $xcat_packimg_tmpfile");
while (<$exlist>) {
$excludetext .= $_;
}
close($exlist);
#handle the #INLCUDE# tag recursively
my $idir = dirname($exlistlocname);
my $doneincludes = 0;
while (not $doneincludes) {
$doneincludes = 1;
if ($excludetext =~ /#INCLUDE:[^#^\n]+#/) {
$doneincludes = 0;
$excludetext =~ s/#INCLUDE:([^#^\n]+)#/include_file($1,$idir)/eg;
}
}
my @tmp = split("\n", $excludetext);
foreach (@tmp) {
chomp $_;
s/\s*#.*//; #-- remove comments
next if /^\s*$/; #-- skip empty lines
if (/^\+/) {
s/^\+//; #remove '+'
$includestr .= "-path '" . $_ . "' -o ";
} else {
s/^\-//; #remove '-' if any
$excludestr .= "'!' -path '" . $_ . "' -a ";
}
}
}
}
$excludestr =~ s/-a $//;
if ($includestr) {
$includestr =~ s/-o $//;
$includestr = "find . " . $includestr;
}
print "\nexcludestr=$excludestr\n\n includestr=$includestr\n\n"; # debug
# some rpms like atftp mount the rootimg/proc to /proc, we need to make sure rootimg/proc is free of junk
# before packaging the image
system("umount $rootimg_dir/proc");
#put the image name, uuid and timestamp into diskless image when it is packed.
$verbose && $callback->({ data => ["add image info to xcatinfo file"] });
`echo IMAGENAME="'$imagename'" > $rootimg_dir/opt/xcat/xcatinfo`;
my $uuid = `uuidgen`;
chomp $uuid;
`echo IMAGEUUID="'$uuid'" >> $rootimg_dir/opt/xcat/xcatinfo`;
my $timestamp = `date`;
chomp $timestamp;
`echo TIMESTAMP="'$timestamp'" >> $rootimg_dir/opt/xcat/xcatinfo`;
my $verb = "Packing";
my $temppath;
my $oldmask;
$callback->({ data => ["$verb contents of $rootimg_dir ..."] });
unlink("$destdir/rootimg-statelite.gz");
my $compress = "gzip";
#use "pigz" as the compress tool instead of gzip if "pigz" exist
my $ispigz = system("bash -c 'type -p pigz' >/dev/null 2>&1");
if ($ispigz == 0) {
$compress = "pigz";
}
$callback->({ info => ["compress method: $compress"] });
if ($exlistloc) {
chdir("$rootimg_dir");
system("$excludestr >> $xcat_packimg_tmpfile");
if ($includestr) {
system("$includestr >> $xcat_packimg_tmpfile");
}
$excludestr = "cat $xcat_packimg_tmpfile |cpio -H newc -o | $compress -c - > ../rootimg-statelite.gz";
} else {
$excludestr = "find . |cpio -H newc -o | $compress -c - > ../rootimg-statelite.gz";
}
$oldmask = umask 0077;
chdir("$rootimg_dir");
xCAT::Utils->runcmd("$excludestr");
chmod 0644, "$destdir/rootimg-statelite.gz";
if ($dotorrent) {
my $currdir = getcwd;
chdir($destdir);
unlink("rootimg-statelite.gz.metainfo");
system("ctorrent -t -u $dotorrent -l 1048576 -s rootimg-statelite.gz.metainfo rootimg.gz");
chmod 0644, "rootimg-statelite.gz.metainfo";
chdir($currdir);
}
umask $oldmask;
system("rm -f $xcat_packimg_tmpfile");
$callback->({ data => ["$verb contents of $rootimg_dir done."] });
}
chdir($oldpath);
}
sub liteMe {
my $rootimg_dir = shift;
my $hashNewRef = shift;
my $callback = shift;
unless (-d $rootimg_dir) {
$callback->({ error => ["no rootimage dir"], errorcode => [1] });
return;
}
unless (-d "$rootimg_dir/$statedir") {
# snapshot directory for tmpfs and persistent data.
if (-e "$rootimg_dir/$statedir") {
xCAT::Utils->runcmd("rm -rf $rootimg_dir/$statedir", 0, 1);
}
$callback->({ info => ["creating $rootimg_dir/$statedir"] });
xCAT::Utils->runcmd("mkdir -p $rootimg_dir/$statedir", 0, 1);
}
unless (-d "$rootimg_dir/$statedir/tmpfs") {
xCAT::Utils->runcmd("mkdir -p $rootimg_dir/$statedir/tmpfs", 0, 1);
}
foreach my $line (keys %{$hashNewRef}) {
liteItem($rootimg_dir, $line, 0, $callback);
if ($hashNewRef->{$line}) { # there're children
my $childrenRef = $hashNewRef->{$line};
foreach my $child (@{$childrenRef}) {
liteItem($rootimg_dir, $child, 1, $callback);
}
}
}
$callback->({ info => ["Modifying $rootimg_dir done."] });
# end loop, synclist should now all be in place.
}
sub getRelDir {
my $f = shift;
$f = dirname($f);
if ($f eq "/") {
return ".";
}
my $l = "";
my @arr = split("/", $f);
foreach (1 .. $#arr) {
$l .= "../";
}
chop($l); # get rid of last /
return $l
}
sub include_file
{
my $file = shift;
my $idir = shift;
my @text = ();
unless ($file =~ /^\//) {
$file = $idir . "/" . $file;
}
open(INCLUDE, $file) || \
return "#INCLUDEBAD:cannot open $file#";
while (<INCLUDE>) {
chomp($_);
s/\s+$//; #remove trailing spaces
next if /^\s*$/; #-- skip empty lines
push(@text, $_);
}
close(INCLUDE);
return join("\n", @text);
}
=head3 parseLiteFiles
In the liteentry table, one directory and its sub-items (including sub-directory and entries) can co-exist;
In order to handle such a scenario, one hash is generated to show the hirarachy relationship
For example, one array with entry names is used as the input:
my @entries = (
"imagename persistent /var/",
"imagename tempfs /var/tmp/",
"imagename link /root/",
"imagename link /root/.bashrc",
"imagename link /root/test/",
"imagename link /root/second/third",
"imagename tempfs /etc/resolv.conf",
"imagename tempfs /var/run/"
);
Then, one hash will generated as:
%hashentries = {
'persistent /var/' => [
'tempfs /var/tmp/',
'tempfs /var/run/'
],
'tempfs /etc/resolv.conf' => undef,
'link /root/' => [
'link /root/.bashrc',
'link /root/test/',
'link /root/second/third"
]
};
Arguments:
one array with entrynames,
one hash to hold the entries parsed
Returns:
0 if sucucess
1 if fail
=cut
sub parseLiteFiles {
my ($flref, $dhref) = @_;
my @entries = @{$flref};
foreach (@entries) {
my $entry = $_;
my @str = split /\s+/, $entry;
shift @str; # remove the imgname in @entries
$entry = join "\t", @str;
my $file = $str[1];
chop $file if ($file =~ m{/$});
unless (exists $dhref->{"$entry"}) {
my $parent = dirname $file;
my @res;
my $found = 0;
while ($parent ne "/") {
# to see whether $parent exists in @entries or not
$parent .= "/" unless ($parent =~ m/\/$/);
@res = grep { $_ =~ m/\Q$parent\E$/ } @entries;
$found = scalar @res;
last if ($found == 1);
$parent = dirname $parent;
}
if ($found == 1) { # $parent is found in @entries
# handle $res[0];
my @tmpresentry = split /\s+/, $res[0];
shift @tmpresentry; # remove the imgname in @tmpresentry
$res[0] = join "\t", @tmpresentry;
chop $parent;
my @keys = keys %{$dhref};
my $kfound = grep { $_ =~ m/\Q$res[0]\E$/ } @keys;
if ($kfound eq 0) {
$dhref->{ $res[0] } = [];
}
push @{ $dhref->{"$res[0]"} }, $entry;
} else {
$dhref->{"$entry"} = ();
}
}
}
return 0;
}
=head3
recoverFiles
=cut
sub recoverFiles {
my ($rootimg_dir, $oldentry, $callback) = @_;
$f = $oldentry->[1];
#$callback->({info => ["! updating $f ..."]});
if ($oldentry->[0] =~ m{^link}) {
my $target = $rootimg_dir . $f;
if (-l $target) { #not one directory
my $location = readlink $target;
# if it is not linked from tmpfs, it should have been modified by the .postinstall file
if ($location =~ /\.statelite\/tmpfs/) {
xCAT::Utils->runcmd("rm -rf $target", 0, 1);
my $default = $rootimg_dir . "/.default" . $f;
if (-e $default) {
xCAT::Utils->runcmd("cp -r -a $default $target", 0, 1);
} else { # maybe someone deletes the copy in .default directory
xCAT::Utils->runcmd("touch $target", 0, 1);
}
}
} else {
chop $target;
if (-l $target) {
my $location = readlink $target;
if ($location =~ /\.statelite\/tmpfs/) {
xCAT::Utils->runcmd("rm -rf $target", 0, 1);
my $default = $rootimg_dir . "/.default" . $f;
if (-e $default) {
xCAT::Utils->runcmd("cp -r -a $default $target", 0, 1);
} else {
xCAT::Utils->runcmd("mkdir $target", 0, 1);
}
}
}
}
$target = $rootimg_dir . "/.statelite/tmpfs" . $f;
xCAT::Utils->runcmd("rm -rf $target", 0, 1);
} else {
# shouldn't copy back from /.default, maybe the user has replaced the file/directory in .postinstall file
my $default = $rootimg_dir . $f;
xCAT::Utils->runcmd("rm -rf $default", 0, 1); # TODO: not sure whether it's necessary right now
}
return 0;
}
=head3
liteItem
=cut
sub liteItem {
my ($rootimg_dir, $item, $isChild, $callback) = @_;
my @entry = split(/\s+/, $item);
my $f = $entry[1]; # file name
my $rif = $rootimg_dir . $f; # the file's location in rootimg_dir
my $d = dirname($f);
if ($entry[0] =~ m/link/) {
# 1. copy original contents if they exist to .default directory
# 2. remove file
# 3. create symbolic link to .statelite
# the /etc/mtab should be handled every time even the parent /etc/ has been handled.
# if adding /etc/ to litefile, only tmpfs should be used.
if ($entry[1] eq "/etc/mtab") {
#
# In RHEL7 /etc/mtab is a symlink to /proc/self/mounts and not a regular file.
# If that's the case, then we skip forcing /etc/mtab into the .statelite path
# and symlink'ing /etc/mtab to that file. Otherwise, since the OS isn't updating
# /etc/mtab commands like "df" will not be happy.
#
my $ret = `readlink $rootimg_dir/etc/mtab`;
chomp($ret);
if ($? == 0 and ($ret eq '/proc/self/mounts')) {
$verbose && $callback->({ info => ["skipping /etc/mtab (symlink to /proc/self/mounts)"] });
next;
}
$isChild = 0;
}
if ($isChild == 0) {
#check if the file has been moved to .default by its parent or by last liteimg, if yes, then do nothing
my $ret = `readlink -m $rootimg_dir$f`;
if ($? == 0) {
if ($ret =~ /$rootimg_dir\/.default/) {
$verbose && $callback->({ info => ["do nothing for file $f"] });
next;
}
}
# copy the file to /.defaults
if (-f "$rif" or -d "$rif") {
# if its already a link then leave it alone.
unless (-l $rif) {
# mk the directory if it doesn't exist:
unless (-d "$rootimg_dir/.default$d") {
$verbose && $callback->({ info => ["mkdir -p $rootimg_dir/.default$d"] });
system("mkdir -p $rootimg_dir/.default$d");
}
# copy the file in place.
$verbose && $callback->({ info => ["cp -r -a $rif $rootimg_dir/.default$d"] });
system("cp -r -a $rif $rootimg_dir/.default$d");
# remove the real file
$verbose && $callback->({ info => ["rm -rf $rif"] });
system("rm -rf $rif");
}
} else {
# in this case the file doesn't exist in the image so we create something to it.
# here we're modifying the read/only image
unless (-d "$rootimg_dir$d") {
$verbose && $callback->({ info => ["mkdir -p $rootimg_dir$d"] });
system("mkdir -p $rootimg_dir$d");
}
unless (-d "$rootimg_dir/.default$d") {
$verbose && $callback->({ info => ["mkdir -p $rootimg_dir/.default$d"] });
system("mkdir -p $rootimg_dir/.default$d");
}
# now make touch the file:
if ($f =~ /\/$/) {
# if its a directory, make the directory in .default
$verbose && $callback->({ info => ["mkdir -p $rootimg_dir/.default$f"] });
system("mkdir -p $rootimg_dir/.default$f");
} else {
# if its just a file then link it.
$verbose && $callback->({ info => ["touch $rootimg_dir/.default$f"] });
system("touch $rootimg_dir/.default$f");
}
}
# now create the spot in tmpfs
$verbose && $callback->({ info => ["mkdir -p $rootimg_dir/$statedir/tmpfs$d"] });
system("mkdir -p $rootimg_dir/$statedir/tmpfs$d");
# now for each file, create a symbollic link to /.statelite/tmpfs/
# strip the last / if its a directory for linking, but be careful!
# doing ln -sf ../../tmp <blah>../tmp when tmp is a directory creates 50 levels of links.
# have to do:
# ln -sf ../../tmp <blah>../../tmp/ <- note the / on the end!
if ($f =~ /\/$/) {
$f =~ s/\/$//g;
}
# now do the links.
# link the .default file to the .statelite file and the real file to the .statelite file.
# ../ for tmpfs
# ../ for .statelite
# the rest are for the paths in the file.
my $l = getRelDir($f);
$verbose && $callback->({ info => ["ln -sf ../../$l/.default$f $rootimg_dir/$statedir/tmpfs$f"] });
system("ln -sfn ../../$l/.default$f $rootimg_dir/$statedir/tmpfs/$f");
$verbose && $callback->({ info => ["ln -sf $l/$statedir/tmpfs$f $rootimg_dir$f"] });
system("ln -sfn $l/$statedir/tmpfs$f $rootimg_dir$f");
} else {
# since its parent directory has been linked to .default and .statelite/tmpfs/,
# what we need to do is only to check whether it exists in .default directory
if ($f =~ m{/$}) { # one directory
unless (-d "$rootimg_dir/.default$f") {
if (-e "$rootimg_dir/.default$f") {
xCAT::Utils->runcmd("rm -rf $rootimg_dir/.default$f", 0, 1);
}
$verbose && $callback->({ info => ["mkdir -p $rootimg_dir/.default$f"] });
xCAT::Utils->runcmd("mkdir -p $rootimg_dir/.default$f", 0, 1);
}
} else { # only one file
my $fdir = dirname($f);
unless (-d "$rootimg_dir/.default$fdir") {
$verbose && $callback->({ info => ["mkdir -p $rootimg_dir/.default$fdir"] });
xCAT::Utils->runcmd("mkdir -p $rootimg_dir/.default$fdir", 0, 1);
}
unless (-e "$rootimg_dir/.default$f") {
$verbose && $callback->({ info => ["touch $rootimg_dir/.default$f"] });
xCAT::Utils->runcmd("touch $rootimg_dir/.default$f", 0, 1);
}
}
}
} else {
# if no such file like $rif, create one
unless (-e "$rif") {
if ($f =~ m{/$}) {
$verbose && $callback->({ info => ["mkdir -p $rif"] });
system("mkdir -p $rif");
} else {
# check whether its directory exists or not
my $rifdir = $rootimg_dir . $d;
unless (-e $rifdir) {
$verbose && $callback->({ info => ["mkdir $rifdir"] });
mkdir($rifdir);
}
$verbose && $callback->({ info => ["touch $rif"] });
system("touch $rif");
}
}
unless (-e "$rootimg_dir/.default$d") {
$verbose && $callback->({ info => ["mkdir -p $rootimg_dir/.default$d"] });
system("mkdir -p $rootimg_dir/.default$d");
}
# copy the file to /.defaults
$verbose && $callback->({ info => ["cp -r -a $rif $rootimg_dir/.default$d"] });
system("cp -r -a $rif $rootimg_dir/.default$d");
}
}
=head3
updateFstab
=cut
sub updateFstab {
$rootimg_dir = shift;
$profile = shift;
$arch = shift;
my $rootfs_name = $profile . "_" . $arch;
my $tfstab;
open($tfstab, "<", "$rootimg_dir/etc/fstab");
my @fsdents = <$tfstab>;
close($tfstab);
open($tfstab, ">", "$rootimg_dir/etc/fstab");
foreach my $line (@fsdents) {
if ( $line =~ /^$rootfs_name/ ) {
$line =~ s/\brw\b/ro/;
}
print $tfstab $line;
}
close($tfstab);
}
1;