2010-04-03 00:07:49 +00:00
# Sumavi Inc (C) 2010
#####################################################
# imgport will export and import xCAT stateless, statelite, and diskful templates.
# This will make it so that you can easily share your images with others.
# All your images are belong to us!
package xCAT_plugin::imgport ;
use strict ;
use warnings ;
2010-04-08 20:14:16 +00:00
#use xCAT::Table;
#use xCAT::Schema;
#use xCAT::NodeRange qw/noderange abbreviate_noderange/;
#use xCAT::Utils;
2010-04-03 00:07:49 +00:00
use Data::Dumper ;
use XML::Simple ;
use POSIX qw/strftime/ ;
use Getopt::Long ;
use File::Temp ;
use File::Copy ;
2010-04-08 20:14:16 +00:00
use File::Path qw/mkpath/ ;
use File::Basename ;
2010-04-03 00:07:49 +00:00
use Cwd ;
my $ requestcommand ;
2010-04-08 20:14:16 +00:00
$ ::VERBOSE = 0 ;
2010-04-03 00:07:49 +00:00
1 ;
#some quick aliases to table/value
my % shortnames = (
groups = > [ qw( nodelist groups ) ] ,
tags = > [ qw( nodelist groups ) ] ,
mgt = > [ qw( nodehm mgt ) ] ,
#switch => [qw(switch switch)],
) ;
#####################################################
# Return list of commands handled by this plugin
#####################################################
sub handled_commands
{
return {
imgexport = > "imgport" ,
imgimport = > "imgport" ,
} ;
}
#####################################################
# Process the command
#####################################################
sub process_request
{
#use Getopt::Long;
Getopt::Long:: Configure ( "bundling" ) ;
#Getopt::Long::Configure("pass_through");
Getopt::Long:: Configure ( "no_pass_through" ) ;
my $ request = shift ;
my $ callback = shift ;
$ requestcommand = shift ;
my $ command = $ request - > { command } - > [ 0 ] ;
my $ args = $ request - > { arg } ;
if ( $ command eq "imgexport" ) {
return xexport ( $ request , $ callback ) ;
} elsif ( $ command eq "imgimport" ) {
return ximport ( $ request , $ callback ) ;
} else {
print "Error: $command not found in export\n" ;
$ callback - > ( { error = > [ "Error: $command not found in this module." ] , errorcode = > [ 1 ] } ) ;
#return (1, "$command not found in sumavinode");
}
}
2010-04-08 01:40:57 +00:00
# extract the bundle, then add it to the osimage table. Basically the ying of the yang of the xexport
# function.
sub ximport {
my $ request = shift ;
my $ callback = shift ;
my % rsp ; # response
my $ help ;
my $ xusage = sub {
my $ ec = shift ;
push @ { $ rsp { data } } , "imgimport: Takes in an xCAT image bundle and defines it to xCAT so you can use it" ;
push @ { $ rsp { data } } , "Usage: " ;
push @ { $ rsp { data } } , "\timgimport [-h|--help] - displays this help message" ;
push @ { $ rsp { data } } , "\timgimport <image name> - imports image. Image should be an xCAT bundle" ;
if ( $ ec ) { $ rsp { errorcode } = $ ec ; }
$ callback - > ( \ % rsp ) ;
} ;
unless ( defined ( $ request - > { arg } ) ) { $ xusage - > ( 1 ) ; return ; }
@ ARGV = @ { $ request - > { arg } } ;
if ( $# ARGV eq - 1 ) {
$ xusage - > ( 1 ) ;
return ;
}
GetOptions (
'h|?|help' = > \ $ help ,
2010-05-21 06:44:28 +00:00
'v|verbose' = > \ $ ::VERBOSE ,
2010-04-08 01:40:57 +00:00
) ;
2010-04-03 00:07:49 +00:00
2010-04-08 01:40:57 +00:00
if ( $ help ) {
$ xusage - > ( 0 ) ;
return ;
}
# first extract the bundle
extract_bundle ( $ request , $ callback ) ;
}
# function to export your image. The image should already be in production, work well, and have
# no bugs. Lots of places will have problems because the image may not be in osimage table
# or they may have hardcoded things, or have post install scripts.
2010-04-03 00:07:49 +00:00
sub xexport {
my $ request = shift ;
my $ callback = shift ;
my % rsp ; # response
my $ help ;
my @ extra ;
my $ xusage = sub {
my $ ec = shift ;
push @ { $ rsp { data } } , "imgexport: Creates a tarball of an existing xCAT image" ;
2010-05-21 06:44:28 +00:00
push @ { $ rsp { data } } , "Usage: imgexport [-f] <image name> [directory] [[--extra <file:dir> ] ... ]" ;
2010-04-03 00:07:49 +00:00
if ( $ ec ) { $ rsp { errorcode } = $ ec ; }
$ callback - > ( \ % rsp ) ;
} ;
unless ( defined ( $ request - > { arg } ) ) { $ xusage - > ( 1 ) ; return ; }
@ ARGV = @ { $ request - > { arg } } ;
if ( $# ARGV eq - 1 ) {
$ xusage - > ( 1 ) ;
return ;
}
GetOptions (
'h|?|help' = > \ $ help ,
2010-04-08 20:14:16 +00:00
'extra=s' = > \ @ extra ,
'v|verbose' = > \ $ ::VERBOSE
2010-04-03 00:07:49 +00:00
) ;
if ( $ help ) {
$ xusage - > ( 0 ) ;
return ;
}
# ok, we're done with all that. Now lets actually start doing some work.
my $ img_name = shift @ ARGV ;
my $ dest = shift @ ARGV ;
2010-04-08 20:14:16 +00:00
my $ cwd = $ request - > { cwd } ; #getcwd;
$ cwd = $ cwd - > [ 0 ] ;
2010-04-03 00:07:49 +00:00
2010-04-08 20:14:16 +00:00
$ callback - > ( { data = > [ "Exporting $img_name to $cwd..." ] } ) ;
2010-04-03 00:07:49 +00:00
# check if all files are in place
my $ attrs = get_image_info ( $ img_name , $ callback , @ extra ) ;
#print Dumper($attrs);
unless ( $ attrs ) {
return 1 ;
}
# make manifest and tar it up.
2010-04-08 20:14:16 +00:00
make_bundle ( $ img_name , $ dest , $ attrs , $ callback , $ cwd ) ;
2010-04-03 00:07:49 +00:00
}
2010-04-08 01:40:57 +00:00
2010-04-03 00:07:49 +00:00
# verify the image and return the values
sub get_image_info {
my $ imagename = shift ;
my $ callback = shift ;
my @ extra = @ _ ;
my $ errors = 0 ;
my $ ostab = new xCAT:: Table ( 'osimage' , - create = > 1 ) ;
unless ( $ ostab ) {
$ callback - > (
{ error = > [ "Unable to open table 'osimage' What's going on here dude?" ] , errorcode = > 1 }
) ;
return 0 ;
}
( my $ attrs ) = $ ostab - > getAttribs ( { imagename = > $ imagename } , 'profile' , 'provmethod' , 'osvers' , 'osarch' ) ;
if ( ! $ attrs ) {
$ callback - > ( { error = > [ "Cannot find image \'$imagename\' from the osimage table." ] , errorcode = > [ 1 ] } ) ;
return 0 ;
}
unless ( $ attrs - > { provmethod } ) {
$ callback - > ( { error = > [ "The 'provmethod' field is not set for \'$imagename\' in the osimage table." ] , errorcode = > [ 1 ] } ) ;
$ errors + + ;
}
unless ( $ attrs - > { profile } ) {
$ callback - > ( { error = > [ "The 'profile' field is not set for \'$imagename\' in the osimage table." ] , errorcode = > [ 1 ] } ) ;
$ errors + + ;
}
unless ( $ attrs - > { osvers } ) {
$ callback - > ( { error = > [ "The 'osvers' field is not set for \'$imagename\' in the osimage table." ] , errorcode = > [ 1 ] } ) ;
$ errors + + ;
}
unless ( $ attrs - > { osarch } ) {
$ callback - > ( { error = > [ "The 'osarch' field is not set for \'$imagename\' in the osimage table." ] , errorcode = > [ 1 ] } ) ;
$ errors + + ;
}
unless ( $ attrs - > { provmethod } =~ /install|netboot|statelite/ ) {
$ callback - > ( { error = > [ "Exporting images with 'provemethod' " . $ attrs - > { provmethod } . " is not supported. Hint: install, netboot, or statelite" ] , errorcode = > [ 1 ] } ) ;
$ errors + + ;
}
2010-04-08 21:53:05 +00:00
$ attrs - > { imagename } = $ imagename ;
2010-04-03 00:07:49 +00:00
if ( $ errors ) {
return 0 ;
}
$ attrs = get_files ( $ imagename , $ callback , $ attrs ) ;
if ( $# extra > - 1 ) {
my $ ex = get_extra ( $ callback , @ extra ) ;
if ( $ ex ) {
$ attrs - > { extra } = $ ex ;
}
}
# if we get nothing back, then we couldn't find the files. How sad, return nuthin'
return $ attrs ;
}
# returns a hash of files
# extra {
# file => dir
# file => dir
# }
sub get_extra {
my $ callback = shift ;
my @ extra = @ _ ;
my $ extra ;
# make sure that the extra is formatted correctly:
foreach my $ e ( @ extra ) {
unless ( $ e =~ /:/ ) {
$ callback - > ( { error = > [ "Extra argument $e is not formatted correctly and will be ignored. Hint: </my/file/dir/file:to_install_dir>" ] , errorcode = > [ 1 ] } ) ;
next ;
}
my ( $ file , $ to_dir ) = split ( /:/ , $ e ) ;
unless ( - r $ file ) {
$ callback - > ( { error = > [ "Can not find Extra file $file. Argument will be ignored" ] , errorcode = > [ 1 ] } ) ;
next ;
}
#print "$file => $to_dir";
push @ { $ extra } , { 'src' = > $ file , 'dest' = > $ to_dir } ;
}
return $ extra ;
}
# well we check to make sure the files exist and then we return them.
sub get_files {
my $ imagename = shift ;
my $ errors = 0 ;
my $ callback = shift ;
my $ attrs = shift ; # we'll hopefully get a reference to it and modify this variable.
my @ arr ; # array of directory search paths
my $ template = '' ;
# todo is XCATROOT not going to be /opt/xcat/ in normal situations? We'll always
# assume it is for now
my $ xcatroot = "/opt/xcat" ;
# get the install root
my $ installroot = xCAT::Utils - > getInstallDir ( ) ;
unless ( $ installroot ) {
$ installroot = '/install' ;
}
my $ provmethod = $ attrs - > { provmethod } ;
2010-05-21 06:44:28 +00:00
my $ osvers = $ attrs - > { osvers } ;
2010-04-03 00:07:49 +00:00
# here's the case for the install. All we need at first is the
# template. That should do it.
if ( $ provmethod =~ /install/ ) {
# we need to get the template for this one!
@ arr = ( "$installroot/custom/install" , "$xcatroot/share/xcat/install" ) ;
2010-04-08 21:53:05 +00:00
my $ template = look_for_file ( 'tmpl' , $ callback , $ attrs , @ arr ) ;
2010-04-03 00:07:49 +00:00
unless ( $ template ) {
$ callback - > ( { error = > [ "Couldn't find install template for $imagename" ] , errorcode = > [ 1 ] } ) ;
$ errors + + ;
} else {
$ callback - > ( { data = > [ "$template" ] } ) ;
$ attrs - > { template } = $ template ;
}
$ attrs - > { media } = "required" ;
}
# for stateless I need to save the
# ramdisk
# the kernel
# the rootimg.gz
2010-05-21 06:44:28 +00:00
if ( $ osvers =~ /esx/ ) {
# don't do anything because these files don't exist for ESX stateless.
} elsif ( $ provmethod =~ /netboot/ ) {
2010-04-03 00:07:49 +00:00
@ arr = ( "$installroot/netboot" ) ;
# look for ramdisk
2010-04-08 21:53:05 +00:00
my $ ramdisk = look_for_file ( 'initrd.gz' , $ callback , $ attrs , @ arr ) ;
2010-04-03 00:07:49 +00:00
unless ( $ ramdisk ) {
$ callback - > ( { error = > [ "Couldn't find ramdisk (initrd.gz) for $imagename" ] , errorcode = > [ 1 ] } ) ;
$ errors + + ;
} else {
$ callback - > ( { data = > [ "$ramdisk" ] } ) ;
$ attrs - > { ramdisk } = $ ramdisk ;
}
# look for kernel
2010-04-08 21:53:05 +00:00
my $ kernel = look_for_file ( 'kernel' , $ callback , $ attrs , @ arr ) ;
2010-04-03 00:07:49 +00:00
unless ( $ kernel ) {
$ callback - > ( { error = > [ "Couldn't find kernel (kernel) for $imagename" ] , errorcode = > [ 1 ] } ) ;
$ errors + + ;
} else {
$ callback - > ( { data = > [ "$kernel" ] } ) ;
$ attrs - > { kernel } = $ kernel ;
}
# look for rootimg.gz
2010-04-08 21:53:05 +00:00
my $ rootimg = look_for_file ( 'rootimg.gz' , $ callback , $ attrs , @ arr ) ;
2010-04-03 00:07:49 +00:00
unless ( $ rootimg ) {
$ callback - > ( { error = > [ "Couldn't find rootimg (rootimg.gz) for $imagename" ] , errorcode = > [ 1 ] } ) ;
$ errors + + ;
} else {
$ callback - > ( { data = > [ "$rootimg" ] } ) ;
$ attrs - > { rootimg } = $ rootimg ;
}
}
if ( $ errors ) {
$ attrs = 0 ;
}
return $ attrs ;
}
# argument:
# type of file: This is usually the suffix of the file, or the file name.
# attributes: These are the paramaters you got from the osimage table in a hash.
# @dirs: Some search paths where we'll start looking for them.
# then we just return a string of the full path to where the file is.
# mostly because we just ooze awesomeness.
sub look_for_file {
my $ file = shift ;
2010-04-08 21:53:05 +00:00
my $ callback = shift ;
2010-04-03 00:07:49 +00:00
my $ attrs = shift ;
my @ dirs = @ _ ;
my $ r_file = '' ;
my $ profile = $ attrs - > { profile } ;
my $ arch = $ attrs - > { osarch } ;
my $ distname = $ attrs - > { osvers } ;
# go through the directories and look for the file. We hopefully will find it...
foreach my $ d ( @ dirs ) {
# widdle down rhel5.4, rhel5., rhel5, rhel, rhe, rh, r,
2010-04-08 21:53:05 +00:00
my $ dd = $ distname ; # dd is distro directory, or disco dave, whichever you prefer.
if ( $ dd =~ /win/ ) { $ dd = 'windows' } ;
2010-04-03 00:07:49 +00:00
until ( - r "$d/$dd" or not $ dd ) {
2010-04-08 21:53:05 +00:00
$ callback - > ( { data = > [ "not in $d/$dd..." ] } ) if $ ::VERBOSE ;
2010-04-03 00:07:49 +00:00
chop ( $ dd ) ;
}
if ( $ distname && $ file eq 'tmpl' ) {
2010-04-08 21:53:05 +00:00
$ callback - > ( { data = > [ "looking in $d/$dd..." ] } ) if $ ::VERBOSE ;
2010-04-03 00:07:49 +00:00
# now look for the file name: foo.rhel5.x86_64.tmpl
( - r "$d/$dd/$profile.$distname.$arch.tmpl" ) && ( return "$d/$dd/$profile.$distname.$arch.tmpl" ) ;
# now look for the file name: foo.rhel5.tmpl
( - r "$d/$dd/$profile.$distname.tmpl" ) && ( return "$d/$dd/$profile.$distname.tmpl" ) ;
# now look for the file name: foo.x86_64.tmpl
( - r "$d/$dd/$profile.$arch.tmpl" ) && ( return "$d/$dd/$profile.$arch.tmpl" ) ;
# finally, look for the file name: foo.tmpl
( - r "$d/$dd/$profile.tmpl" ) && ( return "$d/$dd/$profile.tmpl" ) ;
} else {
# this may find the ramdisk: /install/netboot/
( - r "$d/$dd/$arch/$profile/$file" ) && ( return "$d/$dd/$arch/$profile/$file" ) ;
}
}
# I got nothing man. Can't find it. Sorry 'bout that.
# returning nothing:
return '' ;
}
# here's where we make the tarball
sub make_bundle {
my $ imagename = shift ;
my $ dest = shift ;
my $ attribs = shift ;
my $ callback = shift ;
# tar ball is made in local working directory. Sometimes doing this in /tmp
# is bad. In the case of my development machine, the / filesystem was nearly full.
# so doing it in cwd is easy and predictable.
2010-04-08 20:14:16 +00:00
my $ dir = shift ;
#my $dir = getcwd;
2010-04-08 21:53:05 +00:00
# get rid of spaces and put in underlines.
$ imagename =~ s/\s+/_/g ;
2010-04-08 20:14:16 +00:00
# we may find that cwd doesn't work, so we use the request cwd.
2010-04-08 01:40:57 +00:00
my $ ttpath = mkdtemp ( "$dir/imgexport.$$.XXXXXX" ) ;
2010-04-08 20:14:16 +00:00
$ callback - > ( { data = > [ "Creating $ttpath..." ] } ) if $ ::VERBOSE ;
2010-04-03 00:07:49 +00:00
my $ tpath = "$ttpath/$imagename" ;
mkdir ( "$tpath" ) ;
chmod 0755 , $ tpath ;
# make manifest.xml file. So easy! This is why we like XML. I didn't like
# the idea at first though.
my $ xml = new XML:: Simple ( RootName = > 'xcatimage' ) ;
open ( FILE , ">$tpath/manifest.xml" ) or die "Could not open $tpath/manifest.xml" ;
2010-04-08 01:40:57 +00:00
print FILE $ xml - > XMLout ( $ attribs , noattr = > 1 , xmldecl = > '<?xml version="1.0"?>' ) ;
2010-04-03 00:07:49 +00:00
#print $xml->XMLout($attribs, noattr => 1, xmldecl => '<?xml version="1.0">');
close ( FILE ) ;
# these are the only files we copy in. (unless you have extras)
for my $ a ( "kernel" , "template" , "ramdisk" , "rootimg" ) {
if ( $ attribs - > { $ a } ) {
copy ( $ attribs - > { $ a } , $ tpath ) ;
}
}
# extra files get copied in the extra directory.
if ( $ attribs - > { extra } ) {
mkdir ( "$tpath/extra" ) ;
chmod 0755 , "$tpath/extra" ;
foreach ( @ { $ attribs - > { extra } } ) {
2010-06-16 06:23:18 +00:00
my $ fromf = $ _ - > { src } ;
print " $fromf\n" ;
if ( - d $ fromf ) {
print "fromf is a directory" ;
mkpath ( "$tpath/extra/$fromf" ) ;
`cp -a $fromf/* $tpath/extra/$fromf/` ;
} else {
copy ( $ fromf , "$tpath/extra" ) ;
}
2010-04-03 00:07:49 +00:00
}
}
# now get right below all this stuff and tar it up.
chdir ( $ ttpath ) ;
2010-04-08 21:53:05 +00:00
$ callback - > ( { data = > [ "Inside $ttpath." ] } ) ;
2010-04-03 00:07:49 +00:00
unless ( $ dest ) {
$ dest = "$dir/$imagename.tgz" ;
}
# if no absolute path specified put it in the cwd
unless ( $ dest =~ /^\// ) {
$ dest = "$dir/$dest" ;
}
$ callback - > ( { data = > [ "Compressing $imagename bundle. Please be patient." ] } ) ;
2010-04-08 20:14:16 +00:00
my $ rc ;
if ( $ ::VERBOSE ) {
$ callback - > ( { data = > [ "tar czvf $dest . " ] } ) ;
$ rc = system ( "tar czvf $dest . " ) ;
} else {
$ rc = system ( "tar czf $dest . " ) ;
}
2010-04-08 21:53:05 +00:00
$ callback - > ( { data = > [ "Done!" ] } ) ;
2010-04-03 00:07:49 +00:00
if ( $ rc ) {
$ callback - > ( { error = > [ "Failed to compress archive! (Maybe there was no space left?)" ] , errorcode = > [ 1 ] } ) ;
return ;
}
chdir ( $ dir ) ;
$ rc = system ( "rm -rf $ttpath" ) ;
if ( $ rc ) {
$ callback - > ( { error = > [ "Failed to clean up temp space $ttpath" ] , errorcode = > [ 1 ] } ) ;
return ;
}
}
2010-04-08 01:40:57 +00:00
sub extract_bundle {
my $ request = shift ;
2010-04-08 20:14:16 +00:00
#print Dumper($request);
2010-04-08 01:40:57 +00:00
my $ callback = shift ;
@ ARGV = @ { $ request - > { arg } } ;
my $ xml ;
my $ data ;
my $ datas ;
2010-04-08 20:14:16 +00:00
my $ error = 0 ;
2010-04-08 01:40:57 +00:00
my $ bundle = shift @ ARGV ;
# extract the image in temp path in cwd
2010-04-08 20:14:16 +00:00
my $ dir = $ request - > { cwd } ; #getcwd;
$ dir = $ dir - > [ 0 ] ;
#print Dumper($dir);
unless ( - r $ bundle ) {
$ bundle = "$dir/$bundle" ;
}
unless ( - r $ bundle ) {
$ callback - > ( { error = > [ "Can not find $bundle" ] , errorcode = > [ 1 ] } ) ;
return ;
}
2010-04-08 01:40:57 +00:00
my $ tpath = mkdtemp ( "$dir/imgimport.$$.XXXXXX" ) ;
2010-04-08 20:14:16 +00:00
$ callback - > ( { data = > [ "Unbundling image..." ] } ) ;
my $ rc ;
if ( $ ::VERBOSE ) {
$ callback - > ( { data = > [ "tar zxvf $bundle -C $tpath" ] } ) ;
$ rc = system ( "tar zxvf $bundle -C $tpath" ) ;
$ rc = system ( "tar zxvf $bundle -C $tpath" ) ;
} else {
$ rc = system ( "tar zxf $bundle -C $tpath" ) ;
}
2010-04-08 01:40:57 +00:00
if ( $ rc ) {
$ callback - > ( { error = > [ "Failed to extract bundle $bundle" ] , errorcode = > [ 1 ] } ) ;
}
# get all the files in the tpath. These should be all the image names.
my @ files = < $ tpath / * > ;
# go through each image directory. Find the XML and put it into the array. If there are any
# errors then the whole thing is over and we error and leave.
foreach my $ imgdir ( @ files ) {
2010-04-08 20:14:16 +00:00
#print "$imgdir \n";
2010-04-08 01:40:57 +00:00
unless ( - r "$imgdir/manifest.xml" ) {
$ callback - > ( { error = > [ "Failed to find manifest.xml file in image bundle" ] , errorcode = > [ 1 ] } ) ;
return ;
}
$ xml = new XML:: Simple ;
# get the data!
# put it in an eval string so that it
$ data = eval { $ xml - > XMLin ( "$imgdir/manifest.xml" ) } ;
if ( $@ ) {
2010-04-08 21:53:05 +00:00
$ callback - > ( { error = > [ "invalid manifest.xml file inside the bundle. Please verify the XML" ] , errorcode = > [ 1 ] } ) ;
2010-04-08 20:14:16 +00:00
#my $foo = $@;
#$foo =~ s/\n//;
#$callback->({error=>[$foo],errorcode=>[1]});
#foreach($@){
# last;
#}
2010-04-08 01:40:57 +00:00
return ;
}
2010-04-08 20:14:16 +00:00
#print Dumper($data);
2010-04-08 01:40:57 +00:00
#push @{$datas}, $data;
# now we need to import the files...
unless ( verify_manifest ( $ data , $ callback ) ) {
2010-04-08 20:14:16 +00:00
$ error + + ;
2010-04-08 01:40:57 +00:00
next ;
}
2010-04-08 20:14:16 +00:00
# check media first
unless ( check_media ( $ data , $ callback ) ) {
$ error + + ;
next ;
}
#import manifest.xml into xCAT database
unless ( set_config ( $ data , $ callback ) ) {
$ error + + ;
next ;
}
2010-04-08 01:40:57 +00:00
# now place files in appropriate directories.
2010-04-08 20:14:16 +00:00
unless ( make_files ( $ data , $ imgdir , $ callback ) ) {
$ error + + ;
next ;
}
my $ osimage = $ data - > { imagename } ;
$ callback - > ( { data = > [ "Successfully imported $osimage" ] } ) ;
2010-04-08 01:40:57 +00:00
}
2010-04-08 20:14:16 +00:00
# remove temp file only if there were no problems.
unless ( $ error ) {
$ rc = system ( "rm -rf $tpath" ) ;
if ( $ rc ) {
$ callback - > ( { error = > [ "Failed to clean up temp space $tpath" ] , errorcode = > [ 1 ] } ) ;
return ;
}
}
}
# return 1 for true 0 for false.
# need to make sure media is copied before importing image.
sub check_media {
my $ data = shift ;
my $ callback = shift ;
my $ rc = 0 ;
unless ( $ data - > { 'media' } ) {
$ rc = 1 ;
} elsif ( $ data - > { media } eq 'required' ) {
my $ os = $ data - > { osvers } ;
my $ arch = $ data - > { osarch } ;
my $ installroot = xCAT::Utils - > getInstallDir ( ) ;
unless ( $ installroot ) {
$ installroot = '/install' ;
}
unless ( - d "$installroot/$os/$arch" ) {
$ callback - > ( { error = > [ "This image requires that you first copy media for $os-$arch" ] , errorcode = > [ 1 ] } ) ;
} else {
$ rc = 1 ;
}
}
return $ rc ;
2010-04-08 01:40:57 +00:00
}
sub set_config {
my $ data = shift ;
my $ callback = shift ;
my $ ostab = xCAT::Table - > new ( 'osimage' , - create = > 1 , - autocommit = > 0 ) ;
my % keyhash ;
my $ osimage = $ data - > { imagename } ;
2010-04-08 20:14:16 +00:00
$ callback - > ( { data = > [ "Adding $osimage" ] } ) if $ ::VERBOSE ;
2010-04-08 01:40:57 +00:00
# now we make a quick hash of what we want to put into this
$ keyhash { provmethod } = $ data - > { provmethod } ;
$ keyhash { profile } = $ data - > { profile } ;
$ keyhash { osvers } = $ data - > { osvers } ;
$ keyhash { osarch } = $ data - > { osarch } ;
$ ostab - > setAttribs ( { imagename = > $ osimage } , \ % keyhash ) ;
$ ostab - > commit ;
2010-04-08 20:14:16 +00:00
return 1 ;
2010-04-08 01:40:57 +00:00
}
sub verify_manifest {
my $ data = shift ;
my $ callback = shift ;
my $ errors = 0 ;
# first make sure that the stuff is defined!
unless ( $ data - > { imagename } ) {
$ callback - > ( { error = > [ "The 'imagename' field is not defined in manifest.xml." ] , errorcode = > [ 1 ] } ) ;
$ errors + + ;
}
unless ( $ data - > { provmethod } ) {
$ callback - > ( { error = > [ "The 'provmethod' field is not defined in manifest.xml." ] , errorcode = > [ 1 ] } ) ;
$ errors + + ;
}
unless ( $ data - > { profile } ) {
$ callback - > ( { error = > [ "The 'profile' field is not defined in manifest.xml." ] , errorcode = > [ 1 ] } ) ;
$ errors + + ;
}
unless ( $ data - > { osvers } ) {
$ callback - > ( { error = > [ "The 'osvers' field is not defined in manifest.xml." ] , errorcode = > [ 1 ] } ) ;
$ errors + + ;
}
unless ( $ data - > { osarch } ) {
$ callback - > ( { error = > [ "The 'osarch' field is not defined in manifest.xml." ] , errorcode = > [ 1 ] } ) ;
$ errors + + ;
}
unless ( $ data - > { provmethod } =~ /install|netboot|statelite/ ) {
$ callback - > ( { error = > [ "Importing images with 'provemethod' " . $ data - > { provmethod } . " is not supported. Hint: install, netboot, or statelite" ] , errorcode = > [ 1 ] } ) ;
$ errors + + ;
}
# if the install method is used, then we need to have certain files in place.
if ( $ data - > { provmethod } =~ /install/ ) {
# we need to get the template for this one!
unless ( $ data - > { template } ) {
$ callback - > ( { error = > [ "The 'osarch' field is not defined in manifest.xml." ] , errorcode = > [ 1 ] } ) ;
$ errors + + ;
}
#$attrs->{media} = "required"; (need to do something to verify media!
2010-05-21 06:44:28 +00:00
} elsif ( $ data - > { osvers } =~ /esx/ ) {
$ callback - > ( { info = > [ 'this is an esx image' ] } ) ;
# do nothing for ESX
1 ;
2010-04-08 01:40:57 +00:00
} elsif ( $ data - > { provmethod } =~ /netboot|statelite/ ) {
unless ( $ data - > { ramdisk } ) {
$ callback - > ( { error = > [ "The 'ramdisk' field is not defined in manifest.xml." ] , errorcode = > [ 1 ] } ) ;
$ errors + + ;
}
unless ( $ data - > { kernel } ) {
$ callback - > ( { error = > [ "The 'kernel' field is not defined in manifest.xml." ] , errorcode = > [ 1 ] } ) ;
$ errors + + ;
}
unless ( $ data - > { rootimg } ) {
$ callback - > ( { error = > [ "The 'rootimg' field is not defined in manifest.xml." ] , errorcode = > [ 1 ] } ) ;
$ errors + + ;
}
}
if ( $ errors ) {
# we had problems, error and exit.
return 0 ;
}
# returning 1 means everything went good!
return 1 ;
}
sub make_files {
my $ data = shift ;
2010-04-08 20:14:16 +00:00
my $ imgdir = shift ;
2010-04-08 01:40:57 +00:00
my $ callback = shift ;
my $ os = $ data - > { osvers } ;
my $ arch = $ data - > { osarch } ;
my $ profile = $ data - > { profile } ;
2010-04-08 20:14:16 +00:00
my $ installroot = xCAT::Utils - > getInstallDir ( ) ;
unless ( $ installroot ) {
$ installroot = '/install' ;
}
2010-04-08 01:40:57 +00:00
if ( $ data - > { provmethod } =~ /install/ ) {
2010-04-08 20:14:16 +00:00
# you'll get a hash like this:
#$VAR1 = {
# 'provmethod' => 'install',
# 'profile' => 'all',
# 'template' => '/opt/xcat/share/xcat/install/centos/all.tmpl',
# 'imagename' => 'Default_Stateful',
# 'osarch' => 'x86_64',
# 'media' => 'required',
# 'osvers' => 'centos5.4'
# };
my $ template = basename ( $ data - > { template } ) ;
2010-04-08 21:53:05 +00:00
if ( $ os =~ /win/ ) {
$ os = 'windows' ;
}
2010-04-08 20:14:16 +00:00
my $ instdir = "$installroot/custom/$os/$arch" ;
#mkpath("$instdir", { verbose => 1, mode => 0755, error => \my $err });
mkpath ( "$instdir" , { verbose = > 1 , mode = > 0755 } ) ;
# it could be that the old one already exists, in this case back it up.
2010-04-08 01:40:57 +00:00
2010-04-08 20:14:16 +00:00
if ( - r "$instdir/$template" ) {
$ callback - > ( { data = > [ "$instdir/$template already exists. Moving to $instdir/$template.ORIG..." ] } ) ;
move ( "$instdir/$template" , "$instdir/$template.ORIG" ) ;
}
move ( "$imgdir/$template" , $ instdir ) ;
2010-04-08 01:40:57 +00:00
2010-05-21 06:44:28 +00:00
} elsif ( $ data - > { osvers } =~ /esx/ ) {
1 ;
2010-04-08 01:40:57 +00:00
} elsif ( $ data - > { provmethod } =~ /netboot|statelite/ ) {
2010-04-08 20:14:16 +00:00
# data will look something like this:
#$VAR1 = {
# 'provmethod' => 'netboot',
# 'profile' => 'compute',
# 'ramdisk' => '/install/netboot/centos5.4/x86_64/compute/initrd.gz',
# 'kernel' => '/install/netboot/centos5.4/x86_64/compute/kernel',
# 'imagename' => 'Default_Stateless_1265981465',
# 'osarch' => 'x86_64',
# 'extra' => [
# {
# 'dest' => '/install/custom/netboot/centos',
# 'src' => '/opt/xcat/share/xcat/netboot/centos/compute.centos5.4.pkglist'
# },
# {
# 'dest' => '/install/custom/netboot/centos',
# 'src' => '/opt/xcat/share/xcat/netboot/centos/compute.exlist'
# }
# ],
# 'osvers' => 'centos5.4',
# 'rootimg' => '/install/netboot/centos5.4/x86_64/compute/rootimg.gz'
# };
my $ kernel = basename ( $ data - > { kernel } ) ;
my $ ramdisk = basename ( $ data - > { ramdisk } ) ;
my $ rootimg = basename ( $ data - > { rootimg } ) ;
my $ netdir = "$installroot/netboot/$os/$arch/$profile" ;
mkpath ( "$netdir" , { verbose = > 1 , mode = > 0755 } ) ;
# save the old one off
foreach my $ f ( $ kernel , $ ramdisk , $ rootimg ) {
if ( - r "$netdir/$kernel" ) {
$ callback - > ( { data = > [ "Moving old $netdir/$f to $netdir/$f.ORIG..." ] } ) if $ ::VERBOSE ;
move ( "$netdir/$f" , "$netdir/$f.ORIG" ) ;
}
copy ( "$imgdir/$f" , $ netdir ) ;
}
# TODO: for statelite we should expand the rootimg and get the table values in place.
2010-04-08 01:40:57 +00:00
}
if ( $ data - > { extra } ) {
# have to copy extras
2010-04-08 20:14:16 +00:00
print "copying extras...\n" if $ ::VERBOSE ;
2010-06-16 06:23:18 +00:00
#if its just a hash then there is only one entry.
if ( ref ( $ data - > { extra } ) eq 'HASH' ) {
my $ ex = $ data - > { extra } ;
#my $f = basename($ex->{src});
my $ ff = $ ex - > { src } ;
my $ dest = $ ex - > { dest } ;
unless ( moveExtra ( $ callback , $ ff , $ dest , $ imgdir ) ) {
return 0 ;
}
# if its an array go through each item.
} else {
foreach ( @ { $ data - > { extra } } ) {
#my $f = basename($_->{src});
my $ ff = $ _ - > { src } ;
my $ dest = $ _ - > { dest } ;
unless ( moveExtra ( $ callback , $ ff , $ dest , $ imgdir ) ) {
return 0 ;
}
2010-04-08 20:14:16 +00:00
}
}
2010-04-08 01:40:57 +00:00
}
2010-04-08 20:14:16 +00:00
# return 1 meant everything was successful!
return 1 ;
2010-04-08 01:40:57 +00:00
}
2010-06-16 06:23:18 +00:00
sub moveExtra {
my $ callback = shift ;
my $ ff = shift ;
my $ dest = shift ;
my $ imgdir = shift ;
my $ f = basename ( $ ff ) ;
if ( - r "$dest/$f" ) {
$ callback - > ( { data = > [ "Moving old $dest/$f to $dest/$f.ORIG..." ] } ) if $ ::VERBOSE ;
move ( "$dest/$f" , "$dest/$f.ORIG" ) ;
}
# this extra file is a directory, so we are moving the directory over.
print "Is $imgdir/extra/$ff a directory or a file?\n" ;
if ( - d "$imgdir/extra/$ff" ) {
print "This is a directory\n" ;
unless ( - d $ dest ) {
unless ( mkpath ( $ dest ) ) {
$ callback - > ( { error = > [ "Failed to create $dest" ] , errorcode = > 1 } ) ;
return 0 ;
}
}
2010-06-16 06:47:39 +00:00
# this could cause some problems. This is one of the reasons we may not want to
# allow copying of directories.
system ( "cp -a $imgdir/extra/$ff/* $dest" ) ;
if ( $? ) {
$ callback - > ( { error = > [ "Failed to cp -a $imgdir/extra/$ff/* to $dest" ] , errorcode = > 1 } ) ;
2010-06-16 06:23:18 +00:00
return 0 ;
}
} else {
print "This is a file\n" ;
# this extra file is a file and we can just copy to the destination.
unless ( copy ( "$imgdir/extra/$f" , $ dest ) ) {
$ callback - > ( { error = > [ "Failed to copy $imgdir/extra/$f to $dest" ] , errorcode = > 1 } ) ;
return 0 ;
}
}
return 1 ;
}