328 lines
8.8 KiB
Perl

# IBM(c) 2007 EPL license http://www.eclipse.org/legal/epl-v10.html
package xCAT_plugin::partimage;
BEGIN
{
$::XCATROOT = $ENV{'XCATROOT'} ? $ENV{'XCATROOT'} : '/opt/xcat';
}
use lib "$::XCATROOT/lib/perl";
use Storable qw(dclone);
use Sys::Syslog;
use File::Temp qw/tempdir/;
use xCAT::Table;
use xCAT::Utils;
use xCAT::MsgUtils;
use xCAT::Template;
use xCAT::Postage;
use Data::Dumper;
use Getopt::Long;
Getopt::Long::Configure("bundling");
Getopt::Long::Configure("pass_through");
use File::Path;
use File::Copy;
my $pImageDir = "$::XCATROOT/share/xcat/install/partimage";
my $tftpDir = "/tftpboot";
my @cpiopid;
sub handled_commands
{
return {
mkinstall => "nodetype:os=imagecapture|imagerestore",
};
}
sub process_request
{
my $request = shift;
my $callback = shift;
my $doreq = shift;
my $distname = undef;
my $arch = undef;
my $path = undef;
if($request->{command}->[0] eq 'mkinstall'){
return mkinstall($request, $callback, $doreq);
}else{
errorSig($callback, "Error in partimage plugin, shouldn't call anything other than mkinstall function");
}
}
sub errorSig {
$callback = shift;
$msg = shift;
$callback->({
error => ["$msg"],
errorcode => [1]
});
}
# function used for creating boot attributes for partimage functionality.
sub mkinstall {
# we should be passed in the request which has the list of nodes
my $request = shift;
# The callback function for what to print out and do with results.
my $callback = shift;
# doreq isn't used here, but we shift for good measure.
my $doreq = shift;
my @nodes = @{$request->{node}};
my $node;
# read the nodetype table for our nodes.
my $ostab = xCAT::Table->new('nodetype');
my %doneimgs;
my $mode;
# go through each node:
# 1. Verify that nodetype table is filled out and accurate
# 2. Create auto file by subbing through variables.
foreach $node (@nodes){
my $osinst;
my $ent = $ostab->getNodeAttribs($node, ['profile', 'os', 'arch']);
# Verify that the user configured the tables like we expect them to.
# if not, then give them a stern warning.
unless ($ent->{os}){
errorSig($callback, "No OS defined in nodetype for $node. OS types for partimage should be imagecapture or imagerestore");
next;
}
unless($ent->{arch}){
errorSig($callback, "No arch defined in nodetype for $node. only x86 and x86_64 supported");
next;
}
unless($ent->{profile}){
errorSig($callback, "No profile defined in nodetype for $node. Default profile types are found in $pImageDir");
next;
}
my $os = $ent->{os};
my $arch = $ent->{arch};
my $profile = $ent->{profile};
if($os eq "imagecapture"){
$mode = "capture";
}
elsif($os eq "imagerestore"){
$mode = "restore";
}else{
errorSig($callback, "Unrecognized mode: $os. Should be imagecapture or imagerestore for partimage");
next;
}
# This is step 2. Take the template file and sub in the variables
# so that there is a file in /install/autoinst/$node for each node
my $tmplfile=get_tmpl_file_name("/install/custom/install/partimage", $profile, $os, $arch);
if (! $tmplfile) { $tmplfile=get_tmpl_file_name($pImageDir, $profile, $os, $arch); }
unless ( -r "$tmplfile"){
errorSig($callback, "No profile template exists for $profile");
next;
}
my $instDir = xCAT::Utils->get_site_attribute("installdir");
if(! defined($instDir)){
$instDir = "/install";
}
#Call the Template class to do substitution to produce the file
my $tmperr = xCAT::Template->subvars(
$tmplfile,
"$instDir/autoinst/$node",
$node
);
# if there were problems during substitution then let the user know
# about it.
if ($tmperr){
$callback->({
node => [{
name => [$node],
error => [$tmperr],
errorcode => [1]
}]
});
next;
}
# part 3. Create the image capture/restore image
# this is just a repackaging of the existing xCAT nbfs
# partimage uses the 32 bit one. No problems if its 64 bit
# the 32 bit will still work. This way we can capture 64 bit
# and 64 bit os's. What a clever tool!
unless ($doneimgs{"$os|$arch"}){
mkpath("$tftpDir/xcat/partimage/x86");
mkpath("$instDir/partimage");
# verify ramdisk is there
if(-r "$tftpDir/xcat/nbfs.x86.gz"){
copy("$tftpDir/xcat/nbfs.x86.gz", "$tftpDir/xcat/partimage/x86/");
}else{
errorSig($callback, "$tftpDir/xcat/nbfs.x86.gz was not found, please run 'mknb x86' first.");
}
# verify that kernel is there
if(-r "$tftpDir/xcat/nbk.x86"){
copy("$tftpDir/xcat/nbk.x86", "$tftpDir/xcat/partimage/x86/");
}else{
errorSig($callback, "$tftpDir/xcat/nbk.x86 was not found, please run 'mknb x86' first.");
}
# now get the image supped up
mkPartImage($node,$tftpDir,$callback);
$doneimgs{"$os|$arch"} = 1;
}
# part 4. Create the boot parameters for the machine.
#
# if there was an error then setBootParams will return 1;
# if there is no error it will return 0.
if(setBootParams($node,$callback, $instDir, $mode)){
next;
}
}
}
sub mkPartImage{
# modify nbfs.tgz
my $node = shift;
my $tftpDir = shift;
my $callback = shift;
my $pDataDir = "$pImageDir/data";
my $pNbDir = "$tftpDir/xcat/partimage/x86/tmp";
# $callback->({data=>["Creating nbfs.$arch.gz in $tftpdir/xcat"]});
print "$pNbDir";
# remove this directory if it exists.
if(-d $pNbDir){
rmtree("$pNbDir");
# errorSig($callback, "could not remove $pNbDir");
}
if(-r "$pNbDir/../pnbfs.x86.gz"){
unless(unlink("$pNbDir/../pnbfs.x86.gz") == 0 ){
errorSig($callback, "could not remove $pNbDir/../pnbfs.x86.gz\n");
}
}
mkpath($pNbDir);
$callback=>({data=>["cd $pNbDir; gunzip -c ../nbfs.x86.gz | cpio -id"]});
system("cd $pNbDir; gunzip -c ../nbfs.x86.gz | cpio -id");
# Tput files
# /usr/bin/tput
copy("$pDataDir/tput", "$pNbDir/usr/bin/");
system("chmod 755 $pNbDir/usr/bin/");
# /lib/libdl.so.2
copy("$pDataDir/lib/libdl.so.2", "$pNbDir/lib/");
# /usr/lib/libncursesw.so.5
copy("$pDataDir/lib/libncursesw.so.5", "$pNbDir/usr/lib/");
copy("$pDataDir/lib/ld-linux.so.2", "$pNbDir/lib/");
system("chmod 755 $pNbDir/lib/ld-linux.so.2");
copy("$pDataDir/lib/libc.so.6", "$pNbDir/lib/");
system("chmod 755 $pNbDir/lib/libc.so.6");
copy("$pDataDir/sfdisk", "$pNbDir/sbin");
system("chmod 755 $pNbDir/sbin/sfdisk");
copy("$pDataDir/e2fsck", "$pNbDir/sbin");
system("chmod 755 $pNbDir/sbin/e2fsck");
copy("$pDataDir/partimage", "$pNbDir/bin/");
system("chmod 755 $pNbDir/bin/partimage");
copy("$pDataDir/partimage.sh", "$pNbDir/etc/init.d/S80partimage.sh");
unlink("$pNbDir/etc/init.d/S99xcat.sh");
# we make a new nbfs.x86.gz and call it pnbfs. Its really just
# nbfs with some other things added and taken away.
system("cd $pNbDir; find . | cpio -o -H newc | gzip -9 > $pNbDir/../pnbfs.x86.gz");
# remove all the junk after compressing the file.
rmtree("$pNbDir");
}
# set the boot parameters of the node. If we return 1 there is an error
# if we return 0 everything went according to plan.
sub setBootParams{
my $node = shift;
my $callback = shift;
my $instDir = shift;
my $mode = shift;
# get all the info we need for the node:
# ms, NIC, etc.
my $restab = xCAT::Table->new('noderes');
my $bptab = xCAT::Table->new('bootparams',-create=>1);
my $hmtab = xCAT::Table->new('nodehm');
my $ent =
$restab->getNodeAttribs(
$node,['nfsserver', 'primarynic', 'installnic']
);
my $sent =
$hmtab->getNodeAttribs(
$node, ['serialport', 'serialspeed', 'serialflow']
);
unless ($ent and $ent->{nfsserver}){
errorSig($callback, "No noderes.nfsserver for $node defined");
$callback->("No noderes.nfsserver for $node defined");
return 1;
}
my $kernel = "xcat/partimage/x86/nbk.x86";
my $initrd = "xcat/partimage/x86/pnbfs.x86.gz";
my $kcmdline = "quiet";
# console stuff
if (defined $sent->{serialport}){
unless ($sent->{serialspeed}){
errorSig($callback, "serialport defined, but no serialspeed for $node in nodehm table");
return 1;
}
$kcmdline .=
" console=ttyS"
. $sent->{serialport} . ","
. $sent->{serialspeed};
if($sent and ($sent->{serialflow} =~ /(ctsrts|cts|hard)/)){
$kcmdline .= "n8r";
}
}
# xcatd stuff
# get xCATd port:
my $xPort = xCAT::Utils->get_site_attribute("xcatdport");
$kcmdline .= " xcatd=" . $ent->{nfsserver} . ":" . $xPort;
$kcmdline .= " pcfg=http://" . $ent->{nfsserver} . "/$instDir/autoinst/$node";
$kcmdline .= " mode=$mode";
$bptab->setNodeAttribs(
$node, {
kernel => $kernel,
initrd => $initrd,
kcmdline => $kcmdline
}
);
return 0;
}
sub get_tmpl_file_name {
my $base=shift;
my $profile=shift;
my $os=shift;
my $arch=shift;
if (-r "$base/$profile.$os.$arch.tmpl") {
return "$base/$profile.$os.$arch.tmpl";
}
elsif (-r "$base/$profile.$arch.tmpl") {
return "$base/$profile.$arch.tmpl";
}
elsif (-r "$base/$profile.$os.tmpl") {
return "$base/$profile.$os.tmpl";
}
elsif (-r "$base/$profile.tmpl") {
return "$base/$profile.tmpl";
}
return "";
}
1;