2013-10-18 16:13:52 -04:00
# IBM(c) 2007 EPL license http://www.eclipse.org/legal/epl-v10.html
package xCAT_plugin::configfpc ;
BEGIN
{
$ ::XCATROOT = $ ENV { 'XCATROOT' } ? $ ENV { 'XCATROOT' } : '/opt/xcat' ;
}
use strict ;
use lib "$::XCATROOT/lib/perl" ;
use xCAT::Utils ;
use xCAT::MsgUtils ;
use xCAT::SvrUtils ;
use Sys::Hostname ;
use xCAT::Table ;
use xCAT::TableUtils ;
use xCAT::NetworkUtils ;
use Data::Dumper ;
use xCAT::MacMap ;
use Socket ;
2013-10-23 17:01:35 -04:00
use Net::Ping ;
2013-10-18 16:13:52 -04:00
##########################################################################
## Command handler method from tables
###########################################################################
sub handled_commands {
return {
configfpc = > "configfpc" ,
} ;
}
sub process_request {
my $ request = shift ;
my $ callback = shift ;
my $ subreq = shift ;
$ ::CALLBACK = $ callback ;
2013-10-23 17:01:35 -04:00
if ( defined ( @ { $ ::args } ) ) {
@ ARGV = @ { $ ::args } ;
}
Getopt::Long:: Configure ( "bundling" , "no_ignore_case" , "no_pass_through" ) ;
my $ getopt_success = Getopt::Long:: GetOptions (
'help|h|?' = > \ $ ::opt_h ,
'i|I=s' = > \ $ ::opt_I ,
'verbose|V' = > \ $ ::opt_V ,
) ;
# Option -h for Help
if ( defined ( $ ::opt_h ) || ( ! $ getopt_success ) ) {
& configfpc_usage ;
return 0 ;
}
if ( ( ! $ ::opt_I ) ) { # missing required option - msg and return
my $ rsp ;
push @ { $ rsp - > { data } } , "Missing required option -i <adapter_interface> \n" ;
xCAT::MsgUtils - > message ( "I" , $ rsp , $ ::CALLBACK ) ;
& configfpc_usage ;
return 0 ;
}
# Option -V for verbose output
if ( defined ( $ ::opt_V ) ) {
$ ::VERBOSE = $ ::opt_V ;
}
# Option -i for kit component attributes
if ( defined ( $ ::opt_I ) ) {
$ ::interface = $ ::opt_I ;
}
2013-10-18 16:13:52 -04:00
my $ command = $ request - > { command } - > [ 0 ] ;
my $ localhostname = hostname ( ) ;
if ( $ command eq "configfpc" )
{
my $ rc ;
$ rc = configfpc ( $ request , $ callback , $ subreq ) ;
}
else
{
my % rsp ;
push @ { $ rsp { data } } , "$localhostname: Unsupported command: $command" ;
xCAT::MsgUtils - > message ( "I" , \ % rsp , $ callback ) ;
return 1 ;
}
return 0 ;
}
2013-10-23 17:01:35 -04:00
sub configfpc_usage {
my $ rsp ;
push @ { $ rsp - > { data } } ,
"\nUsage: configfpc - Configure the NeXtScale FPCs.i This command requires the -i option to give specify which network adapter to use to look for the FPCs.\n" ;
push @ { $ rsp - > { data } } ,
2013-10-28 16:14:27 -04:00
" configfpc -i interface \n " ;
2013-10-23 17:01:35 -04:00
push @ { $ rsp - > { data } } ,
2013-10-28 16:14:27 -04:00
" configfpc [-V|--verbose] -i interface \n " ;
2013-10-23 17:01:35 -04:00
push @ { $ rsp - > { data } } , " configfpc [-h|--help|-?] \n" ;
xCAT::MsgUtils - > message ( "I" , $ rsp , $ ::CALLBACK ) ;
return 0 ;
}
2013-10-18 16:13:52 -04:00
#
# Main process subroutine
#
###########################################################################
# This routine will look for NeXtWcale Fan Power Controllers (FPC) that have
# a default IP address of 192.169.0.100
#
# For each FPC found the code will
# 1 - ping the default IP address to get the default IP and MAC in the arp table
# 2 - use arp to get the MAC address
# 3 - lookup the MAC address
# ------ check for the MAC on the switch adn save the port number
# ------ lookup the node with that switch port number
# 4 - get the IP address for the node name found
# 5 - determine the assocaited netmask and gateway for this node IP
# 6 - use rspconfig (IPMI) to set the netmask, gateway, and IP address
# 7 - check to make sure that the new FPC IP address is responding
# 8 - remove the default FPC IP from the arp table
# 9 - use ping to determine if there is another default FPC IP and if so start this process again
###########################################################################
sub configfpc {
my $ request = shift ;
my $ callback = shift ;
my $ subreq = shift ;
# Use default userid and passwd
my $ ipmiuser = 'USERID' ;
my $ ipmipass = 'PASSW0RD' ;
my $ fpcip = '192.168.0.100' ;
2013-10-31 10:24:06 -04:00
my $ defnode = 'deffpc' ;
2013-10-18 16:13:52 -04:00
# This is the default FPC IP that we are looking for
my $ foundfpc = 0 ;
2013-10-23 17:01:35 -04:00
# Setup routing to 182.168.0.100 network
if ( $ ::VERBOSE ) {
my % rsp = { } ;
2013-10-31 10:24:06 -04:00
push @ { $ rsp { data } } , "Adding route definition for 192.168.0.101/16 to $::interface interface" ;
2013-10-23 17:01:35 -04:00
xCAT::MsgUtils - > message ( "I" , \ % rsp , $ callback ) ;
}
my $ setroute = `ip addr add dev $::interface 192.168.0.101/16` ;
2013-10-18 16:13:52 -04:00
#
# check for an FPC - this ping will also add the FPC IP and MAC to the ARP table
#
my $ res = `LANG=C ping -c 1 -w 5 $fpcip` ;
#my $res = system("LANG=C ping -c 1 -w 5 $fpcip 2>&1");
if ( $ res =~ /100% packet loss/g ) {
# xCAT::MsgUtils->message ("I", "There are no default $fpcip FPC IP addresses to process");
$ foundfpc = 0 ;
2013-10-23 17:01:35 -04:00
my % rsp = { } ;
2013-10-30 16:03:41 -04:00
push @ { $ rsp { data } } , "No default $fpcip IP addresses found" ;
2013-10-23 17:01:35 -04:00
xCAT::MsgUtils - > message ( "I" , \ % rsp , $ callback ) ;
2013-10-18 16:13:52 -04:00
exit ; # EXIT if we find no more default IP addresses on the network
}
else {
# xCAT::MsgUtils->message ("I", "Found $fpcip FPC IP addresses to process");
2013-10-23 17:01:35 -04:00
if ( $ ::VERBOSE ) {
my % rsp = { } ;
2013-10-30 16:03:41 -04:00
push @ { $ rsp { data } } , "Found $fpcip address" ;
2013-10-23 17:01:35 -04:00
xCAT::MsgUtils - > message ( "I" , \ % rsp , $ callback ) ;
}
2013-10-18 16:13:52 -04:00
$ foundfpc = 1 ;
}
2013-10-23 17:01:35 -04:00
2013-10-30 16:03:41 -04:00
my $ addnode = & add_node ( $ defnode , $ fpcip , $ callback ) ;
2013-10-23 17:01:35 -04:00
2013-10-18 16:13:52 -04:00
#
# Main loop - check to see if we found an FPC and continue to set the FPC infomration and look for the next one
#
while ( $ foundfpc ) {
# Process the default FPC IP to find the node associated with this FPC MAC
my ( $ node , $ fpcmac ) = & get_node ( $ callback ) ;
# Found the Node and MAC associated with this MAC - continue to setup this FPC
if ( $ node ) {
# get the network settings for this node
my ( $ netmask , $ gateway , $ newfpcip ) = & get_network_parms ( $ node , $ callback ) ;
# Change the FPC network netmask, gateway, and ip
& set_FPC_network_parms ( $ defnode , $ netmask , $ gateway , $ newfpcip , $ callback , $ subreq ) ;
2013-10-23 17:01:35 -04:00
# message changed network settings
my % rsp = { } ;
push @ { $ rsp { data } } , "Configured FPC with MAC $fpcmac as $node ($newfpcip)" ;
xCAT::MsgUtils - > message ( "I" , \ % rsp , $ callback ) ;
2013-10-18 16:13:52 -04:00
# Validate that new IP is working - Use ping to check if the new IP has been set
2013-10-23 17:01:35 -04:00
my $ p = Net::Ping - > new ( ) ;
my $ ping_success = 1 ;
while ( $ ping_success ) {
if ( $ p - > ping ( $ newfpcip ) ) {
my % rsp = { } ;
push @ { $ rsp { data } } , "Verified the FPC with MAC $fpcmac is responding to the new IP $newfpcip as node $node" ;
xCAT::MsgUtils - > message ( "I" , \ % rsp , $ callback ) ;
$ ping_success = 0 ;
}
else {
if ( $ ::VERBOSE ) {
my % rsp = { } ;
push @ { $ rsp { data } } , "ping to $newfpcip is unsuccessful. Retrying " ;
xCAT::MsgUtils - > message ( "I" , \ % rsp , $ callback ) ;
}
}
2013-10-18 16:13:52 -04:00
}
2013-10-23 17:01:35 -04:00
$ p - > close ( ) ;
2013-10-18 16:13:52 -04:00
# The Node associated with this MAC was not found - print an infomrational message and continue
} else {
my % rsp ;
2013-10-23 17:01:35 -04:00
push @ { $ rsp { data } } , "No FPC found that is associated with MAC address $fpcmac.\nCheck to see if the switch and switch table contain the information needed to locate this FPC MAC" ;
2013-10-28 16:14:27 -04:00
xCAT::MsgUtils - > message ( "E" , \ % rsp , $ callback ) ;
$ foundfpc = 0 ;
2013-10-18 16:13:52 -04:00
}
#
# Delete this FPC default IP Arp entry to get ready to look for another defautl FPC
#
2013-10-23 17:01:35 -04:00
if ( $ ::VERBOSE ) {
my % rsp = { } ;
push @ { $ rsp { data } } , "Removing default IP $fpcip from the arp table" ;
xCAT::MsgUtils - > message ( "I" , \ % rsp , $ callback ) ;
}
2013-10-18 16:13:52 -04:00
my $ arpout = `arp -d $fpcip` ;
2013-10-28 16:14:27 -04:00
if ( ( $ foundfpc == 1 ) ) { # if the last FPC was found and processed
# check for another FPC
$ res = `LANG=C ping -c 1 -w 5 $fpcip 2>&1` ;
if ( ( $ res =~ /100% packet loss/g ) && ( $ foundfpc == 1 ) ) {
my % rsp ;
2013-10-30 16:03:41 -04:00
push @ { $ rsp { data } } , "There are no more default IP address to process" ;
2013-10-28 16:14:27 -04:00
xCAT::MsgUtils - > message ( "I" , \ % rsp , $ callback ) ;
$ foundfpc = 0 ;
}
else {
$ foundfpc = 1 ;
}
2013-10-18 16:13:52 -04:00
}
}
#
2013-10-23 17:01:35 -04:00
# Cleanup on the way out - Delete route and remove the deffpc node definition
2013-10-18 16:13:52 -04:00
#
2013-10-23 17:01:35 -04:00
# Delete routing to 182.168.0.100 network
if ( $ ::VERBOSE ) {
my % rsp = { } ;
2013-10-30 16:03:41 -04:00
push @ { $ rsp { data } } , "Deleting route definition for 192.168.0.101/16 on interface $::interface" ;
2013-10-23 17:01:35 -04:00
xCAT::MsgUtils - > message ( "I" , \ % rsp , $ callback ) ;
}
my $ setroute = `ip addr del dev $::interface 192.168.0.101/16` ;
# Delete routing to 182.168.0.100 network
if ( $ ::VERBOSE ) {
my % rsp = { } ;
push @ { $ rsp { data } } , "Removing default FPC node definition $defnode" ;
xCAT::MsgUtils - > message ( "I" , \ % rsp , $ callback ) ;
}
2013-10-31 10:24:06 -04:00
# Remove the defnode node entry
my $ out = xCAT::Utils - > runxcmd ( { command = > [ "noderm" ] , node = > [ "$defnode" ] } , $ subreq , 0 , 2 ) ;
2013-10-18 16:13:52 -04:00
return 1 ;
}
#
# The get_network_parms subroutine
# takes the node name and gets the IP address for this node
# and collects the netmask and gateway and returns netmask, gateway, and IP address
#
sub get_network_parms {
my $ node = shift ;
my $ callback = shift ;
# Get the new ip address for this FPC
my $ newfpc = `getent hosts $node` ;
my ( $ newfpcip , $ junk ) = split ( /\s/ , $ newfpc ) ;
# collect gateway and netmask
my $ ip = $ newfpcip ;
my $ gateway ;
my $ netmask ;
if ( inet_aton ( $ ip ) ) {
$ ip = inet_ntoa ( inet_aton ( $ ip ) ) ;
} else {
my % rsp ;
push @ { $ rsp { data } } , "Unable to resolve $ip" ;
xCAT::MsgUtils - > message ( "I" , \ % rsp , $ callback ) ;
#xCAT::MsgUtils->message("S","Unable to resolve $ip");
return undef ;
}
my $ nettab = xCAT::Table - > new ( 'networks' ) ;
unless ( $ nettab ) { return undef } ;
my @ nets = $ nettab - > getAllAttribs ( 'net' , 'mask' , 'gateway' ) ;
foreach ( @ nets ) {
my $ net = $ _ - > { 'net' } ;
my $ mask = $ _ - > { 'mask' } ;
my $ gw = $ _ - > { 'gateway' } ;
$ ip =~ /([0-9]+)\.([0-9]+)\.([0-9]+)\.([0-9]+)/ ;
my $ ipnum = ( $ 1 << 24 ) + ( $ 2 << 16 ) + ( $ 3 << 8 ) + $ 4 ;
$ mask =~ /([0-9]+)\.([0-9]+)\.([0-9]+)\.([0-9]+)/ ;
my $ masknum = ( $ 1 << 24 ) + ( $ 2 << 16 ) + ( $ 3 << 8 ) + $ 4 ;
$ net =~ /([0-9]+)\.([0-9]+)\.([0-9]+)\.([0-9]+)/ or next ; #next if ipv6, TODO: IPv6 support
my $ netnum = ( $ 1 << 24 ) + ( $ 2 << 16 ) + ( $ 3 << 8 ) + $ 4 ;
if ( $ gw eq '<xcatmaster>' ) {
$ gw = xCAT::NetworkUtils - > my_ip_facing ( $ ip ) ;
}
if ( ( $ ipnum & $ masknum ) == $ netnum ) {
$ netmask = $ mask ;
$ gateway = $ gw ;
}
}
return ( $ netmask , $ gateway , $ newfpcip ) ;
}
#
# The set_FPC_network_parms subroutine
# uses rspconfig to set the netmask, gateway, and new ip address for this FPC
#
sub set_FPC_network_parms {
my $ defnode = shift ;
my $ netmask = shift ;
my $ gateway = shift ;
my $ newfpcip = shift ;
my $ callback = shift ;
my $ request = shift ;
my $ error ;
# Proceed with changing the FPC network parameters.
# Set FPC Netmask
2013-10-23 17:01:35 -04:00
if ( $ ::VERBOSE ) {
my % rsp = { } ;
2013-10-30 16:03:41 -04:00
push @ { $ rsp { data } } , "Use rspconfig to set the FPC netmask $netmask" ;
2013-10-23 17:01:35 -04:00
xCAT::MsgUtils - > message ( "I" , \ % rsp , $ callback ) ;
}
2013-10-18 16:13:52 -04:00
my $ netmaskout = xCAT::Utils - > runxcmd (
{
command = > [ "rspconfig" ] ,
2013-10-23 17:01:35 -04:00
node = > [ "$defnode" ] ,
2013-10-18 16:13:52 -04:00
arg = > [ "netmask=$netmask" ]
} ,
$ request , 0 , 1 ) ;
if ( $ ::RUNCMD_RC != 0 ) {
my % rsp ;
2013-10-30 16:03:41 -04:00
push @ { $ rsp { data } } , "Could not change nemask $netmask" ;
2013-10-18 16:13:52 -04:00
xCAT::MsgUtils - > message ( "I" , \ % rsp , $ callback ) ;
$ error + + ;
}
# Set FPC gateway
2013-10-23 17:01:35 -04:00
if ( $ ::VERBOSE ) {
my % rsp = { } ;
push @ { $ rsp { data } } , "Use rspconfig to set the FPC gateway $gateway" ;
xCAT::MsgUtils - > message ( "I" , \ % rsp , $ callback ) ;
}
2013-10-18 16:13:52 -04:00
my $ gatewayout = xCAT::Utils - > runxcmd (
{
command = > [ "rspconfig" ] ,
2013-10-23 17:01:35 -04:00
node = > [ "$defnode" ] ,
2013-10-18 16:13:52 -04:00
arg = > [ "gateway=$gateway" ]
} ,
$ request , 0 , 1 ) ;
if ( $ ::RUNCMD_RC != 0 ) {
my % rsp ;
2013-10-30 16:03:41 -04:00
push @ { $ rsp { data } } , "Could not change gateway $gateway" ;
2013-10-18 16:13:52 -04:00
xCAT::MsgUtils - > message ( "I" , \ % rsp , $ callback ) ;
$ error + + ;
}
# Set FPC Ip address
2013-10-23 17:01:35 -04:00
if ( $ ::VERBOSE ) {
my % rsp = { } ;
push @ { $ rsp { data } } , "Use rspconfig to set the FPC IP address $newfpcip" ;
xCAT::MsgUtils - > message ( "I" , \ % rsp , $ callback ) ;
}
2013-10-18 16:13:52 -04:00
my $ ipout = xCAT::Utils - > runxcmd (
{
command = > [ "rspconfig" ] ,
2013-10-23 17:01:35 -04:00
node = > [ "$defnode" ] ,
2013-10-18 16:13:52 -04:00
arg = > [ "ip=$newfpcip" ]
} ,
$ request , 0 , 1 ) ;
if ( $ ::RUNCMD_RC != 0 ) {
my % rsp ;
push @ { $ rsp { data } } , "Could not change ip address $newfpcip on default FPC" ;
xCAT::MsgUtils - > message ( "S" , \ % rsp , $ callback ) ;
$ error + + ;
}
return 1 ;
}
#
# This subroutine
# 1) gets the MAC from the arp table
# 2) uses Macmap to find the node associated with this MAC
# 3) returns the node and MAC
sub get_node {
my $ callback = shift ;
my $ fpcip = '192.168.0.100' ;
# get the FPC from the arp table
my $ arpout = `arp -a | grep $fpcip` ;
# format of arp command is: feihu-fpc (10.1.147.170) at 6c:ae:8b:08:20:35 [ether] on eth0
# extract the MAC address
my ( $ junk1 , $ junk2 , $ junk3 , $ fpcmac , $ junk4 , $ junk5 , $ junk6 ) = split ( " " , $ arpout ) ;
2013-10-30 16:03:41 -04:00
# set the FPC MAC as static for the arp table
my $ arpout = `arp -s $fpcip $fpcmac` ;
2013-10-18 16:13:52 -04:00
# Print a message that this MAC has been found
my % rsp ;
2013-10-30 16:03:41 -04:00
push @ { $ rsp { data } } , "Found IP $fpcip and MAC $fpcmac" ;
2013-10-18 16:13:52 -04:00
xCAT::MsgUtils - > message ( "I" , \ % rsp , $ callback ) ;
# Usee find_mac to 1) look for which switch port contains this MAC address
# and 2) look in the xcat DB to find the node associated with the switch port this MAC was found in
my $ macmap = xCAT::MacMap - > new ( ) ;
my $ node = '' ;
$ node = $ macmap - > find_mac ( $ fpcmac , 0 ) ;
2013-10-23 17:01:35 -04:00
# verbose
if ( $ ::VERBOSE ) {
my % rsp = { } ;
2013-10-31 10:24:06 -04:00
push @ { $ rsp { data } } , "Mapped MAC $fpcmac to node $node" ;
2013-10-23 17:01:35 -04:00
xCAT::MsgUtils - > message ( "I" , \ % rsp , $ callback ) ;
}
2013-10-18 16:13:52 -04:00
return ( $ node , $ fpcmac ) ;
}
2013-10-23 17:01:35 -04:00
#
# This subroutine adds the deffpc node entry for use by rspconfig
#
sub add_node {
my $ defnode = shift ;
2013-10-30 16:03:41 -04:00
my $ fpcip = shift ;
2013-10-23 17:01:35 -04:00
my $ callback = shift ;
# add this node entry
# Object name: feihu-fpc
# bmc=feihu-fpc (Table:ipmi - Key:node - Column:bmc)
# bmcpassword=PASSW0RD (Table:ipmi - Key:node - Column:password)
# bmcusername=USERID (Table:ipmi - Key:node - Column:username)
# cons=ipmi (Table:nodehm - Key:node - Column:cons)
# groups=fpc (Table:nodelist - Key:node - Column:groups)
# mgt=ipmi (Table:nodehm - Key:node - Column:mgt)
#
if ( $ ::VERBOSE ) {
my % rsp = { } ;
push @ { $ rsp { data } } , "Creating default FPC node deffpc with IP 192.168.0.100 for later use with rspconfig" ;
xCAT::MsgUtils - > message ( "I" , \ % rsp , $ callback ) ;
}
my $ nodelisttab = xCAT::Table - > new ( 'nodelist' , - create = > 1 ) ;
$ nodelisttab - > setNodeAttribs ( $ defnode , { groups = > 'defaultfpc' } ) ;
my $ nodehmtab = xCAT::Table - > new ( 'nodehm' , - create = > 1 ) ;
$ nodehmtab - > setNodeAttribs ( $ defnode , { mgt = > 'ipmi' } ) ;
my $ ipmitab = xCAT::Table - > new ( 'ipmi' , - create = > 1 ) ;
2013-10-30 16:03:41 -04:00
$ ipmitab - > setNodeAttribs ( $ defnode , { bmc = > $ fpcip , username = > 'USERID' , password = > 'PASSW0RD' } ) ;
2013-10-23 17:01:35 -04:00
return 0 ;
}
2013-10-18 16:13:52 -04:00
1 ;