mirror of
				https://github.com/xcat2/xcat-core.git
				synced 2025-10-26 08:55:24 +00:00 
			
		
		
		
	git-svn-id: https://svn.code.sf.net/p/xcat/code/xcat-core/trunk@6645 8638fb3e-16cb-4fca-ae20-7b5d299a9bcd
		
			
				
	
	
		
			879 lines
		
	
	
		
			24 KiB
		
	
	
	
		
			Perl
		
	
	
	
	
	
			
		
		
	
	
			879 lines
		
	
	
		
			24 KiB
		
	
	
	
		
			Perl
		
	
	
	
	
	
| # 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;
 | |
| #use xCAT::Table;
 | |
| #use xCAT::Schema;
 | |
| #use xCAT::NodeRange qw/noderange abbreviate_noderange/;
 | |
| #use xCAT::Utils;
 | |
| use Data::Dumper;
 | |
| use XML::Simple;
 | |
| use POSIX qw/strftime/;
 | |
| use Getopt::Long;
 | |
| use File::Temp;
 | |
| use File::Copy;
 | |
| use File::Path qw/mkpath/;
 | |
| use File::Basename;
 | |
| use Cwd;
 | |
| my $requestcommand;
 | |
| $::VERBOSE = 0;
 | |
| 
 | |
| 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");
 | |
| 	}
 | |
| }
 | |
| 
 | |
| # 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,
 | |
| 		'v|verbose' => \$::VERBOSE,
 | |
| 	);
 | |
| 
 | |
| 	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.
 | |
| 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";
 | |
| 		push@{ $rsp{data} }, "Usage: imgexport [-f] <image name> [directory] [[--extra <file:dir> ] ... ]";
 | |
| 		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,
 | |
| 		'extra=s' => \@extra,
 | |
| 		'v|verbose' => \$::VERBOSE
 | |
| 	);
 | |
| 
 | |
| 	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;
 | |
| 	my $cwd = $request->{cwd}; #getcwd;
 | |
| 	$cwd = $cwd->[0];
 | |
| 
 | |
| 	$callback->( {data => ["Exporting $img_name to $cwd..."]});
 | |
| 	# 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.
 | |
| 	make_bundle($img_name, $dest, $attrs, $callback,$cwd);
 | |
| 	
 | |
| }
 | |
| 
 | |
| 
 | |
| 
 | |
| 
 | |
| 
 | |
| # 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++;
 | |
| 	}
 | |
| 
 | |
| 	$attrs->{imagename} = $imagename;
 | |
| 
 | |
| 	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};
 | |
| 	my $osvers = $attrs->{osvers};
 | |
| 
 | |
| 	# 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");
 | |
| 		my $template = look_for_file('tmpl', $callback, $attrs, @arr);
 | |
| 		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
 | |
| 	if($osvers =~ /esx/){
 | |
| 		# don't do anything because these files don't exist for ESX stateless.
 | |
| 	}elsif($provmethod =~ /netboot/){
 | |
| 		@arr = ("$installroot/netboot");
 | |
| 
 | |
| 		# look for ramdisk
 | |
| 		my $ramdisk = look_for_file('initrd.gz', $callback, $attrs, @arr);
 | |
| 		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
 | |
| 		my $kernel = look_for_file('kernel', $callback, $attrs, @arr);
 | |
| 		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
 | |
| 		my $rootimg = look_for_file('rootimg.gz', $callback, $attrs, @arr);
 | |
| 		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;
 | |
| 	my $callback = shift;
 | |
| 	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, 
 | |
| 			my $dd = $distname; # dd is distro directory, or disco dave, whichever you prefer.
 | |
| 			if($dd =~ /win/){ $dd = 'windows' };
 | |
| 			until(-r "$d/$dd" or not $dd){
 | |
| 				$callback->({data=>["not in  $d/$dd..."]}) if $::VERBOSE;
 | |
| 				chop($dd);	
 | |
| 			}
 | |
| 			if($distname && $file eq 'tmpl'){		
 | |
| 				$callback->({data=>["looking in $d/$dd..."]}) if $::VERBOSE;
 | |
| 				# 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.
 | |
| 	my $dir = shift;
 | |
| 	#my $dir = getcwd;
 | |
| 
 | |
| 	# get rid of spaces and put in underlines.  
 | |
| 	$imagename =~ s/\s+/_/g;	
 | |
| 
 | |
| 
 | |
| 	# we may find that cwd doesn't work, so we use the request cwd.
 | |
| 	my $ttpath = mkdtemp("$dir/imgexport.$$.XXXXXX");
 | |
| 	$callback->({data=>["Creating $ttpath..."]}) if $::VERBOSE;
 | |
| 	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";
 | |
| 	print FILE  $xml->XMLout($attribs, noattr => 1, xmldecl => '<?xml version="1.0"?>');
 | |
| 	#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} }){
 | |
| 			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");
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	# now get right below all this stuff and tar it up.
 | |
| 	chdir($ttpath);
 | |
| 	$callback->( {data => ["Inside $ttpath."]});
 | |
| 	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."]});
 | |
| 	my $rc;
 | |
| 	if($::VERBOSE){
 | |
| 		 $callback->({data => ["tar czvf $dest . "]});	
 | |
| 		 $rc = system("tar czvf $dest . ");	
 | |
| 	}else{
 | |
| 		 $rc = system("tar czf $dest . ");	
 | |
| 	}
 | |
| 	$callback->( {data => ["Done!"]});
 | |
| 	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;
 | |
| 	}	
 | |
| }
 | |
| 
 | |
| sub extract_bundle {
 | |
| 	my $request = shift;
 | |
| 	#print Dumper($request);
 | |
| 	my $callback = shift;
 | |
| 	@ARGV = @{ $request->{arg} };
 | |
| 	my $xml;
 | |
| 	my $data;
 | |
| 	my $datas;
 | |
| 	my $error = 0;
 | |
| 	
 | |
| 	my $bundle = shift @ARGV;
 | |
| 	# extract the image in temp path in cwd
 | |
| 	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;
 | |
| 	}
 | |
| 
 | |
| 	my $tpath = mkdtemp("$dir/imgimport.$$.XXXXXX");
 | |
| 	
 | |
| 	$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");
 | |
| 	}
 | |
| 	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){
 | |
| 		#print "$imgdir \n";
 | |
| 		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($@){
 | |
| 			$callback->({error=>["invalid manifest.xml file inside the bundle.  Please verify the XML"],errorcode=>[1]});
 | |
| 			#my $foo = $@;
 | |
| 			#$foo =~ s/\n//;
 | |
| 			#$callback->({error=>[$foo],errorcode=>[1]});
 | |
| 			#foreach($@){
 | |
| 			#	last;
 | |
| 			#}
 | |
| 			return;
 | |
| 		}
 | |
| 		#print Dumper($data);
 | |
| 		#push @{$datas}, $data;
 | |
| 		
 | |
| 		# now we need to import the files...
 | |
| 		unless(verify_manifest($data, $callback)){
 | |
| 			$error++;
 | |
| 			next;		
 | |
| 		}
 | |
| 
 | |
| 		# check media first
 | |
| 		unless(check_media($data, $callback)){
 | |
| 			$error++;
 | |
| 			next;		
 | |
| 		}
 | |
| 
 | |
| 		#import manifest.xml into xCAT database
 | |
| 		unless(set_config($data, $callback)){
 | |
| 			$error++;
 | |
| 			next;
 | |
| 		}
 | |
| 		
 | |
| 		# now place files in appropriate directories.
 | |
| 		unless(make_files($data, $imgdir, $callback)){
 | |
| 			$error++;
 | |
| 			next;
 | |
| 		}
 | |
| 
 | |
| 		my $osimage = $data->{imagename};	
 | |
| 		$callback->({data=>["Successfully imported $osimage"]});
 | |
| 		
 | |
| 	}
 | |
| 
 | |
| 	# 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;
 | |
| }
 | |
| 
 | |
| 
 | |
| 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};
 | |
| 
 | |
| 	$callback->({data=>["Adding $osimage"]}) if $::VERBOSE;
 | |
| 
 | |
| 	# 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;
 | |
| 	return 1;
 | |
| }
 | |
| 
 | |
| 
 | |
| 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!
 | |
| 
 | |
| 	}elsif($data->{osvers} =~ /esx/){
 | |
| 		$callback->({info => ['this is an esx image']});
 | |
| 		# do nothing for ESX
 | |
| 		1;
 | |
| 	}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;
 | |
| 	my $imgdir = shift;
 | |
| 	my $callback = shift;
 | |
| 	my $os = $data->{osvers};
 | |
| 	my $arch = $data->{osarch};
 | |
| 	my $profile = $data->{profile};
 | |
| 	my $installroot = xCAT::Utils->getInstallDir();
 | |
| 	unless($installroot){
 | |
| 		$installroot = '/install';
 | |
| 	}
 | |
| 	
 | |
| 	if($data->{provmethod} =~ /install/){
 | |
| 		# 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});
 | |
| 	
 | |
| 		if($os =~ /win/){
 | |
| 			$os = 'windows';
 | |
| 		}elsif($os =~ /centos/){
 | |
| 			$os = 'centos';
 | |
| 		}elsif($os =~ /rh/){
 | |
| 			$os = 'rh';
 | |
| 		}
 | |
| 		
 | |
| 		my $instdir = "$installroot/custom/install/$os"; 
 | |
| 		#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.
 | |
| 		
 | |
| 		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);		
 | |
| 
 | |
| 	}elsif($data->{osvers} =~ /esx/){
 | |
| 		1;
 | |
| 	}elsif($data->{provmethod} =~/netboot|statelite/){
 | |
| 
 | |
| 		# 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.
 | |
| 	}
 | |
| 
 | |
| 	if($data->{extra}){
 | |
| 		# have to copy extras
 | |
| 		print "copying extras...\n" if $::VERBOSE;
 | |
| 		#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;
 | |
| 				}
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	# return 1 meant everything was successful!	
 | |
| 	return 1;
 | |
| }
 | |
| 
 | |
| 
 | |
| 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;
 | |
| 			}
 | |
| 		}
 | |
| 		# 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});
 | |
| 			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;
 | |
| }
 |