2007-10-26 22:44:33 +00:00
#!/usr/bin/env perl
# IBM(c) 2007 EPL license http://www.eclipse.org/legal/epl-v10.html
package xCAT_plugin::nodediscover ;
use xCAT::Table ;
use IO::Socket ;
use strict ;
use XML::Simple ;
2009-07-30 19:42:29 +00:00
$ XML:: Simple:: PREFERRED_PARSER = 'XML::Parser' ;
2007-10-26 22:44:33 +00:00
use Data::Dumper ;
use POSIX "WNOHANG" ;
use Storable qw( freeze thaw ) ;
use IO::Select ;
use IO::Handle ;
2008-03-21 20:00:07 +00:00
use xCAT::Utils ;
2007-10-26 22:44:33 +00:00
use Sys::Syslog ;
2010-03-17 23:26:14 +00:00
use Text::Balanced qw( extract_bracketed ) ;
2007-10-26 22:44:33 +00:00
2013-04-01 10:32:07 +00:00
use xCAT::DiscoveryUtils ;
2007-10-26 22:44:33 +00:00
2008-01-14 19:13:43 +00:00
sub gethosttag {
2013-06-08 23:39:06 +00:00
#This function tries to return a good hostname for a node based on the
#network to which it is connected (by $netn or maybe $ifname)
#heuristic:
#if the client had a valid IP address from a dhcp server, that is used as key
#once the matching network is found, and an explicit mapping defined, try that
#next, try to see if the ip for the case where hostname==nodename is on this net, if so, return that
#next, try to do nodename-ifname, return that if successful
#next, repeat process for all networks that have the common mgtifname field
#return undef for now if none of the above worked
my $ node = shift ;
my $ netn = shift ;
my $ ifname = shift ;
my $ usednames = shift ;
my % netmap = % { xCAT::NetworkUtils:: my_if_netmap ( ) } ;
my $ mgtifname = $ netmap { $ netn } ;
my $ secondpass = 0 ;
my $ name = "" ;
my $ defhost = inet_aton ( $ node ) ;
my $ nettab = xCAT::Table - > new ( 'networks' ) ;
my $ defn = "" ;
my @ netents = @ { $ nettab - > getAllEntries ( ) } ;
my $ pass ;
#TODO: mgtifname field will get trounced in hierarchical setup, use a live check to match accurately
foreach ( @ netents ) {
if ( $ _ - > { net } eq $ netn or ( $ mgtifname and $ mgtifname eq $ netmap { $ _ - > { net } } ) ) { #either is the network or shares physical interface
if ( $ _ - > { nodehostname } ) { #Check for a nodehostname rule in the table
$ name = $ node ;
if ( $ _ - > { nodehostname } =~ /^\/[^\/]*\/[^\/]*\/$/ ) {
my $ exp = substr ( $ _ - > { nodehostname } , 1 ) ;
chop $ exp ;
my @ parts = split ( '/' , $ exp , 2 ) ;
$ name =~ s/$parts[0]/$parts[1]/ ;
}
elsif ( $ _ - > { nodehostname } =~ /^\|.*\|.*\|$/ ) {
#Perform arithmetic and only arithmetic operations in bracketed issues on the right.
#Tricky part: don't allow potentially dangerous code, only eval if
#to-be-evaled expression is only made up of ()\d+-/%$
#Futher paranoia? use Safe module to make sure I'm good
my $ exp = substr ( $ _ - > { nodehostname } , 1 ) ;
chop $ exp ;
my @ parts = split ( '\|' , $ exp , 2 ) ;
my $ curr ;
my $ next ;
my $ prev ;
my $ retval = $ parts [ 1 ] ;
( $ curr , $ next , $ prev ) =
extract_bracketed ( $ retval , '()' , qr/[^()]*/ ) ;
unless ( $ curr ) { #If there were no paramaters to save, treat this one like a plain regex
$ name =~ s/$parts[0]/$parts[1]/ ;
}
while ( $ curr ) {
#my $next = $comps[0];
if ( $ curr =~ /^[\{\}()\-\+\/\%\*\$\d]+$/ or $ curr =~ /^\(sprintf\(["'%\dcsduoxefg]+,\s*[\{\}()\-\+\/\%\*\$\d]+\)\)$/ )
{
use integer ;
#We only allow integer operations, they are the ones that make sense for the application
my $ value = $ name ;
$ value =~ s/$parts[0]/$curr/ee ;
$ retval = $ prev . $ value . $ next ;
}
else {
print "$curr is bad\n" ;
}
( $ curr , $ next , $ prev ) =
extract_bracketed ( $ retval , '()' , qr/[^()]*/ ) ;
}
#At this point, $retval is the expression after being arithmetically contemplated, a generated regex, and therefore
#must be applied in total
$ name =~ s/$parts[0]/$retval/ ;
#print Data::Dumper::Dumper(extract_bracketed($parts[1],'()',qr/[^()]*/));
#use text::balanced extract_bracketed to parse earch atom, make sure nothing but arith operators, parans, and numbers are in it to guard against code execution
}
print "Name: $name\n" ;
#$name =~ s/$left/$right/;
if ( $ name and inet_aton ( $ name ) ) {
if ( $ netn eq $ _ - > { net } and not $ usednames - > { $ name } ) { return $ name ; }
#At this point, it could still be valid if block was entered due to mgtifname
my $ nnetn = inet_ntoa ( pack ( "N" , unpack ( "N" , inet_aton ( $ name ) ) & unpack ( "N" , inet_aton ( $ _ - > { mask } ) ) ) ) ;
if ( $ nnetn eq $ _ - > { net } and not $ usednames - > { $ name } ) { return $ name ; }
}
$ name = "" ; #Still here, this branch failed
}
$ defn = "" ;
if ( $ defhost ) {
$ defn = inet_ntoa ( pack ( "N" , unpack ( "N" , $ defhost ) & unpack ( "N" , inet_aton ( $ _ - > { mask } ) ) ) ) ;
2008-03-21 20:00:07 +00:00
}
2013-06-08 23:39:06 +00:00
if ( $ defn eq $ _ - > { net } and not $ usednames - > { $ node } ) { #the default nodename is on this network
return $ node ;
2008-03-21 20:00:07 +00:00
}
2013-06-08 23:39:06 +00:00
my $ tentativehost = $ node . "-" . $ ifname ;
my $ tnh = inet_aton ( $ tentativehost ) ;
if ( $ tnh ) {
my $ nnetn = inet_ntoa ( pack ( "N" , unpack ( "N" , $ tnh ) & unpack ( "N" , inet_aton ( $ _ - > { mask } ) ) ) ) ;
if ( $ nnetn eq $ _ - > { net } and not $ usednames - > { $ tentativehost } ) {
return $ tentativehost ;
}
}
}
}
2008-01-14 19:13:43 +00:00
}
2007-10-26 22:44:33 +00:00
sub handled_commands {
2013-06-08 23:39:06 +00:00
return {
#discovered => 'chain:ondiscover',
discovered = > 'nodediscover' ,
} ;
2007-10-26 22:44:33 +00:00
}
sub process_request {
2013-06-08 23:39:06 +00:00
my $ request = shift ;
my $ callback = shift ;
my $ doreq = shift ;
my $ node = $ request - > { node } - > [ 0 ] ;
my $ clientip = $ request - > { '_xcat_clientip' } ;
openlog ( "xCAT node discovery" , '' , 'local0' ) ;
2009-12-09 19:15:16 +00:00
2013-06-08 23:39:06 +00:00
#First, fill in tables with data fields..
if ( defined ( $ request - > { mtm } ) or defined ( $ request - > { serial } ) ) {
my $ vpdtab = xCAT::Table - > new ( "vpd" , - create = > 1 ) ;
if ( $ request - > { uuid } - > [ 0 ] ) {
$ vpdtab - > setNodeAttribs ( $ node , { uuid = > $ request - > { uuid } - > [ 0 ] } ) ;
}
if ( $ request - > { mtm } - > [ 0 ] ) {
$ vpdtab - > setNodeAttribs ( $ node , { mtm = > $ request - > { mtm } - > [ 0 ] } ) ;
}
if ( $ request - > { serial } ) {
$ vpdtab - > setNodeAttribs ( $ node , { serial = > $ request - > { serial } - > [ 0 ] } ) ;
2009-09-14 11:02:38 +00:00
}
2008-05-07 13:24:50 +00:00
}
2013-12-19 08:25:41 -05:00
2013-12-19 22:39:50 -05:00
# save inventory info into the hwinv table
2013-12-19 08:25:41 -05:00
if ( defined ( $ request - > { cpucount } ) or defined ( $ request - > { cputype } ) or defined ( $ request - > { memory } ) or defined ( $ request - > { disksize } ) ) {
my $ basicdata ;
2013-12-19 22:39:50 -05:00
my $ hwinv_tab = xCAT::Table - > new ( "hwinv" , - create = > 1 ) ;
2013-12-19 08:25:41 -05:00
if ( $ request - > { memory } - > [ 0 ] ) {
$ basicdata - > { memory } = $ request - > { memory } - > [ 0 ] ;
}
if ( $ request - > { disksize } - > [ 0 ] ) {
$ basicdata - > { disksize } = $ request - > { disksize } - > [ 0 ] ;
}
if ( $ request - > { cpucount } - > [ 0 ] ) {
$ basicdata - > { cpucount } = $ request - > { cpucount } - > [ 0 ] ;
}
if ( $ request - > { cputype } - > [ 0 ] ) {
$ basicdata - > { cputype } = $ request - > { cputype } - > [ 0 ] ;
}
2013-12-19 22:39:50 -05:00
$ hwinv_tab - > setNodeAttribs ( $ node , $ basicdata ) ;
2013-12-19 08:25:41 -05:00
}
2013-06-08 23:39:06 +00:00
my $ nrtab ;
my @ discoverynics ;
my @ forcenics ; #list of 'eth' style interface names to require to come up on post-discovery client dhcp restart
if ( defined ( $ request - > { arch } ) ) {
#Set the architecture in nodetype. If 32-bit only x86 or ppc detected, overwrite. If x86_64, only set if either not set or not an x86 family
my $ typetab = xCAT::Table - > new ( "nodetype" , - create = > 1 ) ;
( my $ nent ) = $ typetab - > getNodeAttribs ( $ node , [ 'arch' , 'supportedarchs' ] ) ;
if ( $ request - > { arch } - > [ 0 ] =~ /x86_64/ ) {
if ( $ nent and ( $ nent - > { arch } =~ /x86/ ) ) { #If already an x86 variant, do not change
unless ( $ nent and $ nent - > { supportedarchs } =~ /x86_64/ ) {
$ typetab - > setNodeAttribs ( $ node , { supportedarchs = > "x86,x86_64" } ) ;
}
} else {
$ typetab - > setNodeAttribs ( $ node , { arch = > $ request - > { arch } - > [ 0 ] , supportedarchs = > "x86,x86_64" } ) ;
#this check is so that if an admin explicitly declares a node 'x86', the 64 bit capability is ignored
}
} else {
unless ( $ nent and $ nent - > { supportedarchs } eq $ request - > { arch } - > [ 0 ] and $ nent - > { arch } eq $ request - > { arch } - > [ 0 ] ) {
$ typetab - > setNodeAttribs ( $ node , { arch = > $ request - > { arch } - > [ 0 ] , supportedarchs = > $ request - > { arch } - > [ 0 ] } ) ;
}
}
my $ currboot = '' ;
$ nrtab = xCAT::Table - > new ( 'noderes' ) ; #Attempt to check and set if wrong the netboot method on discovery, if admin omitted
( my $ rent ) = $ nrtab - > getNodeAttribs ( $ node , [ 'netboot' , 'discoverynics' ] ) ;
if ( $ rent and defined $ rent - > { discoverynics } ) {
@ discoverynics = split /,/ , $ rent - > { discoverynics } ;
}
if ( $ rent and $ rent - > { 'netboot' } ) {
$ currboot = $ rent - > { 'netboot' } ;
}
if ( $ request - > { arch } - > [ 0 ] =~ /x86/ and $ currboot !~ /pxe/ and $ currboot !~ /xnba/ ) {
$ nrtab - > setNodeAttribs ( $ node , { netboot = > 'xnba' } ) ;
2014-09-11 02:38:47 -07:00
} elsif ( $ request - > { arch } - > [ 0 ] =~ /ppc/ and $ request - > { platform } - > [ 0 ] =~ /PowerNV/ ) {
$ nrtab - > setNodeAttribs ( $ node , { netboot = > 'petitboot' } ) ;
2013-06-08 23:39:06 +00:00
} elsif ( $ request - > { arch } - > [ 0 ] =~ /ppc/ and $ currboot !~ /yaboot/ ) {
$ nrtab - > setNodeAttribs ( $ node , { netboot = > 'yaboot' } ) ;
}
2009-12-09 19:15:16 +00:00
}
2013-06-08 23:39:06 +00:00
my $ macstring = "" ;
if ( defined ( $ request - > { mac } ) ) {
my $ mactab = xCAT::Table - > new ( "mac" , - create = > 1 ) ;
my @ ifinfo ;
my % usednames ;
my % bydriverindex ;
my $ forcenic = 0 ; #-1 is force skip, 0 is use default behavior, 1 is force to be declared even if hosttag is skipped to do so
foreach ( @ { $ request - > { mac } } ) {
@ ifinfo = split /\|/ ;
if ( $ ifinfo [ 1 ] eq 'usb0' ) { #skip usb nic
next ;
}
$ bydriverindex { $ ifinfo [ 0 ] } += 1 ;
if ( scalar @ discoverynics ) {
$ forcenic = - 1 ; #$forcenic defaults to explicitly skip nic
foreach my $ nic ( @ discoverynics ) {
if ( $ nic =~ /:/ ) { #syntax like 'bnx2:0' to say the first bnx2 managed interface
( my $ driver , my $ index ) = split /:/ , $ nic ;
if ( $ driver eq $ ifinfo [ 0 ] and $ index == ( $ bydriverindex { $ driver } - 1 ) ) {
$ forcenic = 1 ; #force nic to be put into database
push @ forcenics , $ ifinfo [ 1 ] ;
last ;
}
} else { #simple 'eth2' sort of argument
if ( $ nic eq $ ifinfo [ 1 ] ) {
push @ forcenics , $ ifinfo [ 1 ] ;
$ forcenic = 1 ;
last ;
}
}
}
}
if ( $ forcenic == - 1 ) { #if force to skip, go to next nic
next ;
}
my $ currmac = lc ( $ ifinfo [ 2 ] ) ;
if ( $ ifinfo [ 3 ] ) {
( my $ ip , my $ netbits ) = split /\// , $ ifinfo [ 3 ] ;
if ( $ ip =~ /\d+\.\d+\.\d+\.\d+/ ) {
my $ ipn = unpack ( "N" , inet_aton ( $ ip ) ) ;
my $ mask = 2 ** $ netbits - 1 << ( 32 - $ netbits ) ;
my $ netn = inet_ntoa ( pack ( "N" , $ ipn & $ mask ) ) ;
my $ hosttag = gethosttag ( $ node , $ netn , @ ifinfo [ 1 ] , \ % usednames ) ;
print Dumper ( $ hosttag ) . "\n" ;
if ( $ hosttag ) {
$ usednames { $ hosttag } = 1 ;
if ( $ hosttag eq $ node ) {
$ macstring . = $ currmac . "|" ;
} else {
$ macstring . = $ currmac . "!" . $ hosttag . "|" ;
}
# following is for manual discovery by nodediscoverdef to define a undef node to predefine node
# this this case, the $clientip is null
unless ( $ clientip ) {
$ clientip = $ ip ;
}
} else {
if ( $ forcenic == 1 ) { $ macstring . = $ currmac . "|" ; } else { $ macstring . = $ currmac . "!*NOIP*|" ; }
}
}
} else {
if ( $ forcenic == 1 ) { $ macstring . = $ currmac . "|" ; }
}
}
$ macstring =~ s/\|\z// ;
$ mactab - > setNodeAttribs ( $ node , { mac = > $ macstring } ) ;
my % request = (
command = > [ 'makedhcp' ] ,
node = > [ $ node ]
) ;
$ doreq - > ( \ % request ) ;
2008-05-07 20:30:05 +00:00
}
2013-06-08 23:39:06 +00:00
#TODO: mac table? on the one hand, 'the' definitive interface was determined earlier...
#Delete the state it was in to make it traverse destiny once agoin
my $ chaintab = xCAT::Table - > new ( 'chain' ) ;
if ( $ chaintab ) {
$ chaintab - > setNodeAttribs ( $ node , { currstate = > '' , currchain = > '' } ) ;
$ chaintab - > close ( ) ;
2008-05-07 20:30:05 +00:00
}
2013-04-01 10:32:07 +00:00
2013-06-08 23:39:06 +00:00
# Update the switch port information if the 'updateswitch' flag is added in the request.
# 'updateswitch' is default added for sequential discovery
if ( $ request - > { 'updateswitch' } && $ macstring ) {
my $ firstmac ;
# Get the mac which defined as the management nic
my @ macents = split ( /\|/ , $ macstring ) ;
foreach my $ macent ( @ macents ) {
my ( $ mac , $ host ) = split ( /!/ , $ macent ) ;
unless ( $ firstmac ) {
$ firstmac = $ mac ;
}
if ( ! $ host || $ host eq $ node ) {
$ firstmac = $ mac ;
last ;
}
}
# search the management nic and record the switch informaiton
foreach my $ nic ( @ { $ request - > { nic } } ) {
if ( defined ( $ nic - > { 'hwaddr' } ) && $ nic - > { 'hwaddr' } - > [ 0 ] =~ /$firstmac/i ) {
if ( defined ( $ nic - > { 'switchname' } ) && defined ( $ nic - > { 'switchaddr' } ) ) {
# update the switch to switches table
my $ switchestab = xCAT::Table - > new ( 'switches' ) ;
if ( $ switchestab ) {
$ switchestab - > setAttribs ( { switch = > $ nic - > { 'switchname' } - > [ 0 ] } , { comments = > $ nic - > { 'switchdesc' } - > [ 0 ] } ) ;
$ switchestab - > close ( ) ;
2009-12-09 19:15:16 +00:00
}
2013-06-08 23:39:06 +00:00
# update the ip of switch to hosts table
my $ hosttab = xCAT::Table - > new ( 'hosts' ) ;
if ( $ hosttab ) {
$ hosttab - > setNodeAttribs ( $ nic - > { 'switchname' } - > [ 0 ] , { ip = > $ nic - > { 'switchaddr' } - > [ 0 ] } ) ;
$ hosttab - > commit ( ) ;
}
# add the switch as a node to xcat db
my $ nltab = xCAT::Table - > new ( 'nodelist' ) ;
if ( $ nltab ) {
$ nltab - > setNodeAttribs ( $ nic - > { 'switchname' } - > [ 0 ] , { groups = > "all,switch" } ) ;
$ nltab - > commit ( ) ;
}
if ( defined ( $ nic - > { 'switchport' } ) ) {
# update the switch table
my $ switchtab = xCAT::Table - > new ( 'switch' ) ;
if ( $ switchtab ) {
$ switchtab - > setNodeAttribs ( $ node , { switch = > $ nic - > { 'switchname' } - > [ 0 ] , port = > $ nic - > { 'switchport' } - > [ 0 ] } ) ;
$ switchtab - > close ( ) ;
}
2009-12-09 19:15:16 +00:00
}
}
}
2013-06-08 23:39:06 +00:00
}
2008-01-14 19:13:43 +00:00
}
2013-07-01 09:11:24 +00:00
# make sure the node has the correct ip configured
unless ( $ clientip ) {
$ callback - > ( { error = > [ "The node [$node] should have a correct IP address which belongs to the management network." ] , errorcode = > [ "1" ] } ) ;
return ;
}
2014-09-22 20:35:18 -07:00
if ( defined ( $ request - > { pbmc_node } ) and defined ( $ request - > { pbmc_node } - > [ 0 ] ) ) {
2014-09-22 19:09:11 -07:00
my $ pbmc_node = $ request - > { pbmc_node } - > [ 0 ] ;
xCAT::Utils - > cleanup_for_powerLE_hardware_discovery ( $ node , $ pbmc_node , $ doreq ) ;
}
2013-06-08 23:39:06 +00:00
my $ restartstring = "restart" ;
if ( scalar @ forcenics > 0 ) {
$ restartstring . = " (" . join ( "|" , @ forcenics ) . ")" ;
}
#now, notify the node to continue life
my $ sock = new IO::Socket:: INET (
PeerAddr = > $ clientip ,
PeerPort = > '3001' ,
Timeout = > '1' ,
Proto = > 'tcp'
2008-01-14 22:19:17 +00:00
) ;
2013-10-18 07:50:22 -04:00
unless ( $ sock ) { syslog ( "local4|err" , "Failed to notify $clientip that it's actually $node." ) ; return ; } #Give up if the node won't hear of it.
2013-06-08 23:39:06 +00:00
print $ sock $ restartstring ;
close ( $ sock ) ;
2007-10-26 22:44:33 +00:00
2013-06-08 23:39:06 +00:00
# sleep 2 seconds for genesis to complete the disocvery process
sleep ( 2 ) ;
2013-04-08 06:36:07 +00:00
2013-06-08 23:39:06 +00:00
#Update the discoverydata table to indicate the successful discovery
xCAT::DiscoveryUtils - > update_discovery_data ( $ request ) ;
2013-04-08 06:36:07 +00:00
2013-10-18 07:50:22 -04:00
syslog ( "local4|info" , "$node has been discovered" ) ;
2007-10-26 22:44:33 +00:00
}
1 ;