#!/usr/bin/env perl # IBM(c) 2007 EPL license http://www.eclipse.org/legal/epl-v10.html package xCAT::Zone; BEGIN { $::XCATROOT = $ENV{'XCATROOT'} ? $ENV{'XCATROOT'} : '/opt/xcat'; } # if AIX - make sure we include perl 5.8.2 in INC path. # Needed to find perl dependencies shipped in deps tarball. if ($^O =~ /^aix/i) { unshift(@INC, qw(/usr/opt/perl5/lib/5.8.2/aix-thread-multi /usr/opt/perl5/lib/5.8.2 /usr/opt/perl5/lib/site_perl/5.8.2/aix-thread-multi /usr/opt/perl5/lib/site_perl/5.8.2)); } use lib "$::XCATROOT/lib/perl"; # do not put a use or require for xCAT::Table here. Add to each new routine # needing it to avoid reprocessing of user tables ( ExtTab.pm) for each command call use POSIX qw(ceil); use File::Path; use Socket; use strict; use Symbol; use warnings "all"; #-------------------------------------------------------------------------------- =head1 xCAT::Zone =head2 Package Description This program module file, is a set of Zone utilities used by xCAT *zone commands. =cut #-------------------------------------------------------------------------------- =head3 genSSHRootKeys Arguments: callback for error messages directory in which to put the ssh RSA keys zonename rsa private key to use for generation ( optional) Returns: Error: 1 - key generation failure. Example: $rc =xCAT::Zone->genSSHRootKeys($callback,$keydir,$rsakey); =cut #-------------------------------------------------------------------------------- sub genSSHRootKeys { my ($class, $callback, $keydir,$zonename,$rsakey) = @_; # # create /keydir if needed # if (!-d $keydir) { my $cmd = "/bin/mkdir -m 700 -p $keydir"; my $output = xCAT::Utils->runcmd("$cmd", 0); if ($::RUNCMD_RC != 0) { my $rsp = {}; $rsp->{error}->[0] = "Could not create $keydir directory"; xCAT::MsgUtils->message("E", $rsp, $callback); return 1; } } #need to gen a new rsa key for root for the zone my $pubfile = "$keydir/id_rsa.pub"; my $pvtfile = "$keydir/id_rsa"; # if exists, remove the old files if (-r $pubfile) { my $cmd = "/bin/rm $keydir/id_rsa*"; my $output = xCAT::Utils->runcmd("$cmd", 0); if ($::RUNCMD_RC != 0) { my $rsp = {}; $rsp->{error}->[0] = "Could not remove id_rsa files from $keydir directory."; xCAT::MsgUtils->message("E", $rsp, $callback); return 1; } } # gen new RSA keys my $cmd; my $output; # if private key was input use it if (defined ($rsakey)) { $cmd="/usr/bin/ssh-keygen -y -f $rsakey > $pubfile"; $output = xCAT::Utils->runcmd("$cmd", 0); if ($::RUNCMD_RC != 0) { my $rsp = {}; $rsp->{error}->[0] = "Could not generate $pubfile from $rsakey"; xCAT::MsgUtils->message("E", $rsp, $callback); return 1; } # now copy the private key into the directory $cmd="cp $rsakey $keydir"; $output = xCAT::Utils->runcmd("$cmd", 0); if ($::RUNCMD_RC != 0) { my $rsp = {}; $rsp->{error}->[0] = "Could not run $cmd"; xCAT::MsgUtils->message("E", $rsp, $callback); return 1; } } else { # generate all new keys $cmd = "/usr/bin/ssh-keygen -t rsa -q -b 2048 -N '' -f $pvtfile"; $output = xCAT::Utils->runcmd("$cmd", 0); if ($::RUNCMD_RC != 0) { my $rsp = {}; $rsp->{error}->[0] = "Could not generate $pubfile"; xCAT::MsgUtils->message("E", $rsp, $callback); return 1; } } #make sure permissions are correct $cmd = "chmod 644 $pubfile;chown root $pubfile"; $output = xCAT::Utils->runcmd("$cmd", 0); if ($::RUNCMD_RC != 0) { my $rsp = {}; $rsp->{error}->[0] = "Could set permission and owner on $pubfile"; xCAT::MsgUtils->message("E", $rsp, $callback); return 1; } } #-------------------------------------------------------------------------------- =head3 getdefaultzone Arguments: None Returns: Name of the current default zone from the zone table Example: my $defaultzone =xCAT::Zone->getdefaultzone($callback); =cut #-------------------------------------------------------------------------------- sub getdefaultzone { my ($class, $callback) = @_; my $defaultzone; # read all the zone table and find the defaultzone, if it exists my $tab = xCAT::Table->new("zone"); if ($tab){ my @zones = $tab->getAllAttribs('zonename','defaultzone'); foreach my $zone (@zones) { # Look for the defaultzone=yes/1 entry if ((defined($zone->{defaultzone})) && (($zone->{defaultzone} =~ /^yes$/i ) || ($zone->{defaultzone} eq "1"))) { $defaultzone = $zone->{zonename}; } $tab->close(); } } else { my $rsp = {}; $rsp->{error}->[0] = "Error reading the zone table. "; xCAT::MsgUtils->message("E", $rsp, $callback); } return $defaultzone; } #-------------------------------------------------------------------------------- =head3 iszonedefined Arguments: zonename Returns: 1 if the zone is already in the zone table. Example: xCAT::Zone->iszonedefined($zonename); =cut #-------------------------------------------------------------------------------- sub iszonedefined { my ($class,$zonename) = @_; # checks the zone table to see if input zonename already in the table my $tab = xCAT::Table->new("zone"); $tab->close(); my $zonehash = $tab->getAttribs({zonename => $zonename},'sshkeydir'); if ( keys %$zonehash) { return 1; }else{ return 0; } } #-------------------------------------------------------------------------------- =head3 getzonekeydir Arguments: zonename Returns: path to the root ssh keys for the zone /etc/xcat/sshkeys//.ssh 1 - zone not defined Example: xCAT::Zone->getzonekeydir($zonename); =cut #-------------------------------------------------------------------------------- sub getzonekeydir { my ($class,$zonename) = @_; my $tab = xCAT::Table->new("zone"); $tab->close(); my $zonehash = $tab->getAttribs({zonename => $zonename},'sshkeydir'); if ( keys %$zonehash) { my $zonesshkeydir=$zonehash->{sshkeydir}; return $zonesshkeydir; }else{ return 1; # this is a bad error zone not defined } } #-------------------------------------------------------------------------------- =head3 getmyzonename Arguments: $node -one nodename Returns: $zonename Example: my $zonename=xCAT::Zone->getmyzonename($node); =cut #-------------------------------------------------------------------------------- sub getmyzonename { my ($class,$node,$callback) = @_; my @node; push @node,$node; my $zonename; my $nodelisttab = xCAT::Table->new("nodelist"); my $nodehash = $nodelisttab->getNodesAttribs(\@node, ['zonename']); $nodelisttab->close(); if ( defined ($nodehash->{$node}->[0]->{zonename})) { # it was defined in the nodelist table $zonename=$nodehash->{$node}->[0]->{zonename}; } else { # get the default zone $zonename =xCAT::Zone->getdefaultzone($callback); } return $zonename; } #-------------------------------------------------------------------------------- =head3 enableSSHbetweennodes Arguments: zonename Returns: 1 if the sshbetweennodes attribute is yes/1 or undefined 0 if the sshbetweennodes attribute is no/0 Example: xCAT::Zone->enableSSHbetweennodes($zonename); =cut #-------------------------------------------------------------------------------- sub enableSSHbetweennodes { my ($class,$node,$callback) = @_; # finds the zone of the node my $enablessh = 1; # default my $zonename=xCAT::Zone->getmyzonename($node); # reads the zone table my $tab = xCAT::Table->new("zone"); $tab->close(); # read both keys, want to know zone is in the zone table. If sshkeydir is not there # it is either missing or invalid anyway my $zonehash = $tab->getAttribs({zonename => $zonename},'sshbetweennodes','sshkeydir'); if (! ( keys %$zonehash)) { my $rsp = {}; $rsp->{error}->[0] = "$node has a zonename: $zonename that is not define in the zone table. Remove the zonename from the node, or create the zone using mkzone. The generated mypostscript may not reflect the correct setting for ENABLESSHBETWEENNODES"; xCAT::MsgUtils->message("E", $rsp, $callback); return $enablessh; } my $sshbetweennodes=$zonehash->{sshbetweennodes}; if (defined ($sshbetweennodes)) { if (($sshbetweennodes =~ /^no$/i) || ($sshbetweennodes eq "0")) { $enablessh = 0; } else { $enablessh = 1; } } else { # not defined default yes $enablessh = 1 ; # default } return $enablessh; } #-------------------------------------------------------------------------------- =head3 usingzones Arguments: none Returns: 1 if the zone table is not empty 0 if empty Example: xCAT::Zone->usingzones; =cut #-------------------------------------------------------------------------------- sub usingzones { my ($class) = @_; # reads the zonetable my $tab = xCAT::Table->new("zone"); my @zone = $tab->getAllAttribs('zonename'); $tab->close(); if (@zone) { return 1; }else{ return 0; } } #-------------------------------------------------------------------------------- =head3 getzoneinfo Arguments: callback An array of nodes Returns: Hash array by zonename point to the nodes in that zonename and sshkeydir -> {nodelist} -> array of nodes in the zone -> {sshkeydir} -> directory containing ssh RSA keys -> {defaultzone} -> is it the default zone Example: my %zonehash =xCAT::Zone->getzoneinfo($callback,@nodearray); Rules: If the nodes nodelist.zonename attribute is a zonename, it is assigned to that zone If the nodes nodelist.zonename attribute is undefined: If there is a defaultzone in the zone table, the node is assigned to that zone If there is no defaultzone in the zone table, the node is assigned to the ~.ssh keydir $::GETZONEINFO_RC 0 = good return 1 = error occured =cut #-------------------------------------------------------------------------------- sub getzoneinfo { my ($class, $callback,$nodes) = @_; $::GETZONEINFO_RC=0; my $zonehash; my $defaultzone; # read all the zone table my $zonetab = xCAT::Table->new("zone"); my @zones; if ($zonetab){ @zones = $zonetab->getAllAttribs('zonename','sshkeydir','sshbetweennodes','defaultzone'); $zonetab->close(); if (@zones) { foreach my $zone (@zones) { my $zonename=$zone->{zonename}; $zonehash->{$zonename}->{sshkeydir}= $zone->{sshkeydir}; $zonehash->{$zonename}->{defaultzone}= $zone->{defaultzone}; $zonehash->{$zonename}->{sshbetweennodes}= $zone->{sshbetweennodes}; # find the defaultzone if ((defined($zone->{defaultzone})) && (($zone->{defaultzone} =~ /^yes$/i ) || ($zone->{defaultzone} eq "1"))) { $defaultzone = $zone->{zonename}; } } } } else { my $rsp = {}; $rsp->{error}->[0] = "Error reading the zone table. "; xCAT::MsgUtils->message("E", $rsp, $callback); $::GETZONEINFO_RC =1; return; } my $nodelisttab = xCAT::Table->new("nodelist"); my $nodehash = $nodelisttab->getNodesAttribs(\@$nodes, ['zonename']); # for each of the nodes, look up it's zone name and assign to the zonehash # If the nodes nodelist.zonename attribute is a zonename, it is assigned to that zone # If the nodes nodelist.zonename attribute is undefined: # If there is a defaultzone in the zone table, the node is assigned to that zone # If there is no defaultzone error out foreach my $node (@$nodes) { my $zonename; $zonename=$nodehash->{$node}->[0]->{zonename}; if (defined($zonename)) { # zonename explicitly defined in nodelist.zonename # check to see if defined in the zone table unless ( xCAT::Zone->iszonedefined($zonename)) { my $rsp = {}; $rsp->{error}->[0] = "$node has a zonename: $zonename that is not define in the zone table. Remove the zonename from the node, or create the zone using mkzone."; xCAT::MsgUtils->message("E", $rsp, $callback); $::GETZONEINFO_RC =1; return; } push @{$zonehash->{$zonename}->{nodes}},$node; } else { # no explict zonename if (defined ($defaultzone)) { # there is a default zone in the zone table, use it push @{$zonehash->{$defaultzone}->{nodes}},$node; } else { # if no default, this is an error my $rsp = {}; $rsp->{error}->[0] = "There is no default zone defined in the zone table. There must be exactly one default zone. "; xCAT::MsgUtils->message("E", $rsp, $callback); $::GETZONEINFO_RC =1; return; } } } return $zonehash; } #-------------------------------------------------------------------------------- =head3 getnodesinzone Arguments: callback zonename Returns: Array of nodes Example: my @nodes =xCAT::Zone->getnodesinzone($callback,$zonename); =cut #-------------------------------------------------------------------------------- sub getnodesinzone { my ($class, $callback,$zonename) = @_; my @nodes; my $nodelisttab = xCAT::Table->new("nodelist"); my @nodelist=$nodelisttab->getAllAttribs('node','zonename'); # build the array of nodes in this zone foreach my $nodename (@nodelist) { if ((defined($nodename->{'zonename'})) && ($nodename->{'zonename'} eq $zonename)) { push @nodes,$nodename->{'node'}; } } return @nodes; } 1;