2009-12-15 20:38:34 +00:00
|
|
|
package xCAT_plugin::litetree;
|
|
|
|
use xCAT::NodeRange;
|
|
|
|
use Data::Dumper;
|
|
|
|
use xCAT::Utils;
|
|
|
|
use Sys::Syslog;
|
|
|
|
use xCAT::GlobalDef;
|
|
|
|
use xCAT::Table;
|
|
|
|
use Getopt::Long;
|
2010-05-04 14:24:08 +00:00
|
|
|
use xCAT::SvrUtils;
|
2009-12-15 20:38:34 +00:00
|
|
|
Getopt::Long::Configure("bundling");
|
|
|
|
Getopt::Long::Configure("pass_through");
|
|
|
|
|
|
|
|
use strict;
|
|
|
|
# synchonize files and directories from mulitple sources.
|
|
|
|
|
|
|
|
# object is to return a list of files to be syncronized.
|
|
|
|
# requesting them. By default we will do read-write files.
|
|
|
|
|
|
|
|
|
|
|
|
my $syncdirTab = "litetree";
|
|
|
|
my $syncfileTab = "litefile";
|
|
|
|
my $errored = 0;
|
|
|
|
|
|
|
|
|
|
|
|
sub handled_commands {
|
|
|
|
# command is syncmount, syncdir is the perl module to use.
|
|
|
|
return {
|
|
|
|
litetree => "litetree",
|
|
|
|
litefile => "litetree",
|
|
|
|
ilitefile => "litetree"
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
sub usage {
|
|
|
|
my $command = shift;
|
|
|
|
my $callback = shift;
|
|
|
|
my $error = shift;
|
|
|
|
my $msg;
|
|
|
|
if($command eq "ilitefile"){
|
2010-04-02 10:07:35 +00:00
|
|
|
$msg = "Usage: ilitefile <imagename>
|
|
|
|
\texample:\n\tilitefile centos5.3-x86_64-statelite-compute"
|
|
|
|
} elsif($command eq "litefile") {
|
|
|
|
$msg = "Usage: litefile <noderange>\nexample:\n\tlitefile node1";
|
|
|
|
} elsif($command eq "litetree") {
|
|
|
|
$msg = "Usage: litetree <noderange>\nexample:\n\tlitetree node1";
|
|
|
|
} else{
|
2009-12-15 20:38:34 +00:00
|
|
|
$msg = "some general usage string";
|
|
|
|
}
|
|
|
|
|
|
|
|
if($error){
|
|
|
|
$callback->({error=>[$msg],errorcode=>[$error]});
|
|
|
|
}else{
|
|
|
|
$callback->({info=>[$msg]});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
sub process_request {
|
|
|
|
my $request = shift;
|
|
|
|
my $callback = shift;
|
|
|
|
my $noderange;
|
|
|
|
my $image;
|
|
|
|
# the request can come from node or some one running command
|
|
|
|
# on xCAT mgmt server
|
|
|
|
# argument could also be image...
|
|
|
|
|
|
|
|
# if request comes from user:
|
|
|
|
if($request->{node}){
|
|
|
|
$noderange = $request->{node};
|
|
|
|
|
|
|
|
# if request comes from node post script .awk file.
|
|
|
|
}elsif($request->{'_xcat_clienthost'}){
|
2010-03-04 22:22:46 +00:00
|
|
|
my @nodenames = noderange($request->{'_xcat_clienthost'}->[0].",".$request->{'_xcat_clientfqdn'}->[0]);
|
2010-05-06 17:38:28 +00:00
|
|
|
if (@nodenames) {
|
|
|
|
$noderange = \@nodenames;
|
|
|
|
$request->{node} = $noderange;
|
|
|
|
}
|
2009-12-15 20:38:34 +00:00
|
|
|
}else{
|
|
|
|
$callback->({error=>["Well Kemosabi, I can't figure out who you are."],errorcode=>[1]});
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if(!$noderange){
|
|
|
|
my $usage_string="Missing Noderange\n";
|
|
|
|
$callback->({error=>[$usage_string],errorcode=>[1]});
|
|
|
|
$request = {};
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
my $command = $request->{command}->[0];
|
|
|
|
if($command eq "litetree"){
|
2010-04-02 10:07:35 +00:00
|
|
|
unless($request->{node}) {
|
|
|
|
usage($command, $callback, 0);
|
|
|
|
return 1;
|
|
|
|
}
|
2009-12-15 20:38:34 +00:00
|
|
|
return syncmount("dir",$request,$callback,$noderange);
|
|
|
|
}elsif($command eq "litefile"){
|
2010-04-02 10:07:35 +00:00
|
|
|
unless($request->{node}) {
|
|
|
|
usage($command, $callback, 0);
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
return syncmount("file",$request, $callback,$noderange);
|
2009-12-15 20:38:34 +00:00
|
|
|
}elsif($command eq "ilitefile"){
|
|
|
|
#print Dumper($request);
|
|
|
|
unless($request->{arg}){
|
2010-04-02 10:07:35 +00:00
|
|
|
usage($command, $callback, 0);
|
2009-12-15 20:38:34 +00:00
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
return syncmount("image",$request, $callback,$request->{arg});
|
|
|
|
}else{
|
|
|
|
$callback->({error=>["error in code..."], errorcode=>[127]});
|
|
|
|
$request = {};
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
sub syncmount {
|
|
|
|
my $syncType = shift;
|
|
|
|
my $request = shift;
|
|
|
|
my $callback = shift;
|
|
|
|
# deterimine which node is calling this
|
|
|
|
# then find out what directories to use.
|
|
|
|
my $noderange = shift;
|
|
|
|
my @nodes = @{$noderange};
|
|
|
|
my $tab;
|
|
|
|
if($syncType eq 'dir'){
|
|
|
|
$tab = xCAT::Table->new($syncdirTab,-create=>1);
|
|
|
|
}elsif($syncType =~ /file|image/ ){
|
|
|
|
$tab = xCAT::Table->new($syncfileTab,-create=>1);
|
|
|
|
}else{
|
|
|
|
$callback->({error=>["error in code..."], errorcode=>[127]});
|
|
|
|
$request = {};
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
my $ostab;
|
|
|
|
my %osents;
|
|
|
|
unless($syncType =~ /image/){
|
|
|
|
$ostab = xCAT::Table->new('nodetype');
|
2010-02-19 18:39:56 +00:00
|
|
|
%osents = %{$ostab->getNodesAttribs(\@nodes,['profile','os','arch','provmethod'])};
|
2009-12-15 20:38:34 +00:00
|
|
|
}
|
|
|
|
foreach my $node (@nodes){
|
2010-02-19 18:39:56 +00:00
|
|
|
# node may be an image...
|
|
|
|
my $image;
|
|
|
|
my $ent;
|
|
|
|
if($syncType !~ /image/){
|
|
|
|
$ent = $osents{$node}->[0];
|
|
|
|
|
|
|
|
unless($ent->{os} && $ent->{arch} && $ent->{profile}){
|
|
|
|
$callback->({error=>["$node does not have os, arch, or profile defined in nodetype table"],errorcode=>[1]});
|
|
|
|
$request = {};
|
|
|
|
next;
|
2009-12-15 20:38:34 +00:00
|
|
|
}
|
2010-02-19 18:39:56 +00:00
|
|
|
if ((!$ent->{provmethod}) || ($ent->{provmethod} eq 'statelite') || ($ent->{provmethod} eq 'netboot') || ($ent->{provmethod} eq 'install')) {
|
|
|
|
$image = $ent->{os} . "-" . $ent->{arch} . "-statelite-" . $ent->{profile};
|
|
|
|
} elsif (($ent->{provmethod} ne 'netboot') && ($ent->{provmethod} ne 'install')) {
|
|
|
|
$image=$ent->{provmethod};
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
$image=$node;
|
|
|
|
}
|
|
|
|
my $fData = getNodeData($syncType,$node,$image,$tab,$callback);
|
|
|
|
# now we go through each directory and search for the file.
|
|
|
|
showSync($syncType,$callback, $node, $fData);
|
2009-12-15 20:38:34 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
# In most cases the syncdir will be on the management node so
|
|
|
|
# want to make sure its not us before we mount things.
|
|
|
|
sub showSync {
|
|
|
|
my $syncType = shift; # dir or file
|
|
|
|
my $callback = shift;
|
|
|
|
my $node = shift;
|
|
|
|
my $dirs = shift;
|
2010-01-22 22:39:57 +00:00
|
|
|
my $mnts;
|
|
|
|
my $first;
|
2009-12-15 20:38:34 +00:00
|
|
|
#print Dumper($dirs);
|
|
|
|
# go through each directory in priority order
|
|
|
|
#mkdir "/mnt/xcat";
|
|
|
|
if($syncType eq "dir"){
|
2010-01-22 22:39:57 +00:00
|
|
|
|
2009-12-15 20:38:34 +00:00
|
|
|
foreach my $priority (sort {$a <=> $b} keys %$dirs){
|
|
|
|
# split the nfs server up from the directory:
|
2010-01-22 22:39:57 +00:00
|
|
|
my $mntpnt;
|
2009-12-15 20:38:34 +00:00
|
|
|
my ($server, $dir) = split(/:/,$dirs->{$priority});
|
2010-01-22 22:39:57 +00:00
|
|
|
|
|
|
|
# if server is blank then its the directory:
|
|
|
|
unless($dir){
|
|
|
|
$dir = $server;
|
|
|
|
$server = '';
|
|
|
|
}
|
|
|
|
|
2009-12-17 21:41:28 +00:00
|
|
|
if(grep /\$|#CMD/, $dir){
|
2010-05-04 14:24:08 +00:00
|
|
|
$dir = xCAT::SvrUtils->subVars($dir,$node,'dir',$callback);
|
2009-12-15 20:38:34 +00:00
|
|
|
$dir =~ s/\/\//\//g;
|
|
|
|
}
|
2010-01-22 22:39:57 +00:00
|
|
|
$first = $dir;
|
|
|
|
$first =~ s!\/([^/]*)\/.*!$1!;
|
2009-12-15 20:38:34 +00:00
|
|
|
|
2010-01-22 22:39:57 +00:00
|
|
|
if($server){
|
|
|
|
if(grep /\$/, $server){
|
2010-05-04 14:24:08 +00:00
|
|
|
$server = xCAT::SvrUtils->subVars($server,$node,'server',$callback);
|
2010-01-22 22:39:57 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
$mntpnt = $server . ":";
|
|
|
|
# we have a server and need to make sure we can mount them under unique names
|
|
|
|
if($mnts->{$first} eq '' ){ # if the first mount point doesn't have a server then leave it.
|
|
|
|
$mnts->{$first} = $server;
|
|
|
|
}else{
|
|
|
|
# they may just have the name in twice:
|
|
|
|
unless($server eq $mnts->{$first}){
|
|
|
|
my $msg = "# " . $mnts->{$first} . " and $server both mount /$first. This not supported.";
|
|
|
|
$callback->({info => $msg});
|
|
|
|
return;
|
|
|
|
}else{
|
|
|
|
$mntpnt = ""; # only mount it once, so get rid of the directory
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
$mntpnt .= $dir;
|
2009-12-15 20:38:34 +00:00
|
|
|
# ok, now we have all the mount points. Let's go through them all?
|
|
|
|
$callback->({info => "$node: $mntpnt"});
|
|
|
|
}
|
2010-01-22 22:39:57 +00:00
|
|
|
|
2009-12-15 20:38:34 +00:00
|
|
|
}elsif($syncType =~ /file|image/){
|
|
|
|
foreach my $file (sort keys %$dirs){
|
|
|
|
my $options = $dirs->{$file};
|
|
|
|
# persistent,rw
|
|
|
|
my $out = sprintf("%s: %-13s %s", $node, $options, $file);
|
|
|
|
$callback->({info => $out});
|
|
|
|
}
|
|
|
|
}
|
2010-01-22 22:39:57 +00:00
|
|
|
|
2009-12-15 20:38:34 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
# get all the directories or files for given image related to this node.
|
|
|
|
sub getNodeData {
|
|
|
|
my $type = shift;
|
|
|
|
my $node = shift;
|
2010-02-19 18:39:56 +00:00
|
|
|
my $image = shift;
|
2009-12-15 20:38:34 +00:00
|
|
|
my $tab = shift;
|
|
|
|
my $cb = shift; # callback to print messages!!
|
|
|
|
# the image name will be something like rhels5.4-x86_64-nfsroot
|
2010-02-19 18:39:56 +00:00
|
|
|
#my $image;
|
|
|
|
#unless($type =~ /image/){
|
|
|
|
# $image = $ent->{os} . "-" . $ent->{arch} . "-statelite-" . $ent->{profile};
|
|
|
|
#}else{
|
|
|
|
# $image = $node;
|
|
|
|
#}
|
|
|
|
|
2009-12-15 20:38:34 +00:00
|
|
|
my @imageInfo;
|
|
|
|
my @attrs;
|
|
|
|
if($type eq "dir"){
|
|
|
|
@attrs = ['priority', 'directory'];
|
|
|
|
}elsif($type =~ /file|image/){
|
|
|
|
@attrs = ['file','options'];
|
|
|
|
}else{
|
|
|
|
print "Yikes! error in the code litefile;getNodeData!";
|
|
|
|
exit 1;
|
|
|
|
}
|
|
|
|
# get the directories with no names
|
|
|
|
push @imageInfo, $tab->getAttribs({image => ''}, @attrs);
|
|
|
|
# get the ALL directories
|
|
|
|
push @imageInfo, $tab->getAttribs({image => 'ALL'}, @attrs);
|
|
|
|
# get for the image specific directories
|
|
|
|
push @imageInfo, $tab->getAttribs({image => $image}, @attrs);
|
|
|
|
# pass back a reference to the directory
|
|
|
|
|
|
|
|
# now we need to sort them
|
|
|
|
return mergeArrays($type,\@imageInfo,$cb);
|
|
|
|
}
|
|
|
|
|
|
|
|
sub mergeArrays {
|
|
|
|
my $type = shift; # file or dir?
|
|
|
|
my $arr = shift; # array of info from the tables.
|
|
|
|
my $cb = shift; # callback routine
|
|
|
|
my $attrs;
|
|
|
|
if($type eq "dir"){
|
|
|
|
foreach(@$arr){
|
|
|
|
if($_->{directory} eq ''){ next; }
|
|
|
|
$attrs->{$_->{priority}} = $_->{directory};
|
|
|
|
}
|
|
|
|
}elsif($type =~ /file|image/){
|
|
|
|
foreach(@$arr){
|
|
|
|
if($_->{file} eq ''){ next; }
|
|
|
|
my $o = $_->{options};
|
|
|
|
if(!$o){
|
|
|
|
$o = "tmpfs,rw";
|
|
|
|
}
|
|
|
|
# TODO: put some logic in here to make sure that ro is alone.
|
|
|
|
# if type is ro and con, then this is wrong silly!
|
|
|
|
#if($p eq "ro" and $t eq "con"){
|
|
|
|
# my $f = $_->{file};
|
|
|
|
# $cb->({info => "#skipping: $f. not allowed to be ro and con"});
|
|
|
|
# next;
|
|
|
|
#}
|
|
|
|
$attrs->{$_->{file}} = $o;
|
|
|
|
}
|
|
|
|
}else{
|
|
|
|
print "Yikes! Error in the code in mergeArrays!\n";
|
|
|
|
exit 1;
|
|
|
|
}
|
|
|
|
#print "mergeArrays...\n";
|
|
|
|
#print Dumper($attrs);
|
|
|
|
return $attrs;
|
|
|
|
}
|
|
|
|
|
|
|
|
1;
|
|
|
|
#vim: set ts=2
|
|
|
|
|