mirror of
				https://github.com/xcat2/xcat-core.git
				synced 2025-10-30 19:02:27 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			249 lines
		
	
	
		
			9.2 KiB
		
	
	
	
		
			Perl
		
	
	
		
			Executable File
		
	
	
	
	
			
		
		
	
	
			249 lines
		
	
	
		
			9.2 KiB
		
	
	
	
		
			Perl
		
	
	
		
			Executable File
		
	
	
	
	
| #!/usr/bin/perl
 | |
| 
 | |
| # xCAT postscript for configuring bonding of nics.
 | |
| #  Usage:  configbond bond1 eth1 [eth3]
 | |
| #
 | |
| # Note: this postscript currently has some assumptions that are specific to the softlayer environment.
 | |
| #       We only use this to configure bond1, because bond0 gets configured by the node provisioning process.
 | |
| #       (altho this script would work for bond0)
 | |
| 
 | |
| use strict;
 | |
| # Check number of args
 | |
| 
 | |
| my $nargs = $#ARGV + 1;
 | |
| if (scalar(@ARGV) < 2 || scalar(@ARGV) > 3) {
 | |
|    system("logger -t xcat -p local4.err  'Usage: configbond <bond> <dev0> [<dev1>]'");
 | |
|    exit 1;
 | |
| }
 | |
| 
 | |
| my $bond = shift(@ARGV);
 | |
| my $nic = $ARGV[0];  
 | |
| my @devs;
 | |
| push(@devs,@ARGV);
 | |
| my $nicips = $ENV{NICIPS};
 | |
| my $nicnetworks = $ENV{NICNETWORKS};
 | |
| my $net_cnt = $ENV{NETWORKS_LINES};
 | |
| 
 | |
| #todo: change this script so they dont need to specify nicnetworks
 | |
| if (!$nicips || !$nicnetworks) { system("logger -t xcat -p local4.err 'configbond: must specify attributes nicips and nicnetworks in the xcat db for this node.'"); exit 1; }
 | |
| 
 | |
| #todo: these are specific to softlayer.  They should be another attribute or argument
 | |
| my $bondingopts = 'mode=4 miimon=100 downdelay=0 updelay=0 lacp_rate=fast xmit_hash_policy=1';
 | |
| 
 | |
| my $netmask ='';
 | |
| my $ipaddr = '';
 | |
| my $nic_num = '';
 | |
| my $subnet = '';
 | |
| my $nic_net = '';
 | |
| my $net_name = '';
 | |
| my @nic_nets = (); # array of networks for this nic
 | |
| my @nic_ips =();   # array of ipaddresses for this nic
 | |
| my @networks = (); # array of all networks from networks table.
 | |
|                    # { network_name, subnet, netmask }
 | |
| 
 | |
| system("logger -t xcat -p local4.err  'configbond: Master: $bond'");
 | |
| system("logger -t xcat -p local4.err  'configbond: Slaves: @devs'");
 | |
| #system("logger -t xcat -p local4.err  'configbond: NIC:           $nic'");
 | |
| system("logger -t xcat -p local4.err  'configbond: NICNETWORKS:   $nicnetworks'");
 | |
| system("logger -t xcat -p local4.err  'configbond: NICIPS:        $nicips'");
 | |
| 
 | |
| # Update modprobe
 | |
| my $file = "/etc/modprobe.d/$bond.conf";
 | |
| if (!open(FILE, ">$file")) { system("logger -t xcat -p local4.err 'configbond: cannot open $file.'"); exit 1; }
 | |
| 
 | |
| print FILE "alias $bond bonding\n";
 | |
| # the bonding options are put in the ifcfg file instead
 | |
| #print FILE "options $bond mode=balance-rr miimon=100\n";
 | |
| close FILE;
 | |
| 
 | |
| # create array of network info. Needed in case where there are 
 | |
| # more than one ip address per nic and shouldn't be many networks.
 | |
| my $net_info;
 | |
| my $cnt = 1;
 | |
| 
 | |
| while ( $cnt <= $net_cnt ) {
 | |
|     $net_info = $ENV{"NETWORKS_LINE$cnt"};
 | |
|     $net_info =~ /^netname=([^\|]*)\|\|/; 
 | |
|     $net_name = $1;
 | |
|     $net_info =~ /net=([^\|]*)\|\|/;
 | |
|     $subnet = $1;
 | |
|     $net_info =~ /mask=([^\|]*)\|\|/;
 | |
|     $netmask = $1;
 | |
|     push @{ $networks[$cnt-1] }, ($net_name, $subnet, $netmask);
 | |
|     $cnt +=1; 
 | |
| }
 | |
| 
 | |
| # get network or networks for this nic from NICNETWORKS: 
 | |
| # eth0:1_0_0_0-255_255_0_0|network2,eth1:1_1_0_0
 | |
| # create array of networks for this nic
 | |
| foreach my $nic_networks (split(/,/,$nicnetworks)) {
 | |
|     my @net = ();
 | |
|     if ( $nic_networks =~ /!/ ) {
 | |
|         @net = split(/!/,$nic_networks);
 | |
|     } else {
 | |
|         @net = split(/:/,$nic_networks);
 | |
|     }
 | |
|     if ($net[0] eq $nic) {
 | |
|         @nic_nets = split(/\|/,$net[1]);
 | |
|         last;
 | |
|     }
 | |
| }
 | |
| 
 | |
| # get all nic ipaddress from $nicips: i.e. eth0:1.0.0.1|2.0.0.1,eth1:1.1.1.1
 | |
| # Then get all ips for this specific nic, i.e. eth0.
 | |
| foreach my $ips (split(/,/,$nicips)) {
 | |
|     my @ip = ();
 | |
|     if ( $ips =~ /!/ ) {
 | |
|         @ip = split(/!/,$ips);
 | |
|     } else {
 | |
|         @ip = split(/:/,$ips);
 | |
|     }
 | |
|     if ($ip[0] eq $nic ) {
 | |
|         @nic_ips = split(/\|/,$ip[1]);
 | |
|     }
 | |
| }
 | |
| 
 | |
| 
 | |
| my $i;
 | |
| for ($i=0; $i < (scalar @nic_ips) ; $i++ ) {
 | |
| 
 | |
|     # Time to create the interfaces.
 | |
|     # loop through the nic networks, find the matching networks to get the
 | |
|     # subnet and netmask and then create the appropriate ifcfg file for linux
 | |
|     my $specific_nic = $nic;
 | |
|     if ($i > 0) {
 | |
|        $specific_nic = $nic . ":" . ($i);
 | |
|     }
 | |
| 
 | |
|     #todo: support case in which nicnetworks is not specified, find the correct network by calculation
 | |
|     $cnt = 0;
 | |
|     $subnet = "";
 | |
|     $netmask = "";
 | |
|     $net_name = "";
 | |
|     while ( $cnt < $net_cnt ) {
 | |
|         if ( $networks[$cnt][0] eq $nic_nets[$i] ) {
 | |
|         
 | |
|             $subnet = $networks[$cnt][1];
 | |
|             $netmask = $networks[$cnt][2];
 | |
|             $cnt = $net_cnt; # found match - get out.
 | |
|         } 
 | |
|         else {
 | |
|             $cnt++; 
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     # check that there is a subnet and netmask set
 | |
|     if ( !(length($subnet) > 0) || !(length($netmask) > 0) ) {
 | |
|         system("logger -t xcat -p local4.err  'configbond: network subnet or netmask not set.'");
 | |
|         exit 1;
 | |
|     }
 | |
|         system("logger -t xcat -p local4.err  'configbond: network subnet and netmask: $subnet, $netmask'");
 | |
|         system("logger -t xcat -p local4.err  'configbond: $specific_nic, $nic_ips[$i]'");
 | |
| 
 | |
|     # Write the master info to the ifcfg file
 | |
|     if (-d "/etc/sysconfig/network-scripts") {
 | |
|         # rhel/centos/fedora
 | |
|         my $dir = "/etc/sysconfig/network-scripts";
 | |
|         if (!open(FILE, ">$dir/ifcfg-$bond")) { system("logger -t xcat -p local4.err 'configbond: cannot open $dir/ifcfg-$bond.'"); exit 1; }
 | |
| 
 | |
|         print FILE "DEVICE=$bond\n";
 | |
|         print FILE "BOOTPROTO=none\n";
 | |
|         print FILE "IPADDR=$nic_ips[$i]\n";
 | |
|         print FILE "NETMASK=$netmask\n";
 | |
|         print FILE "ONBOOT=yes\n";
 | |
|         print FILE "USERCTL=no\n";
 | |
|         print FILE qq(BONDING_OPTS="$bondingopts"\n);
 | |
|         close FILE;
 | |
| 
 | |
|         # Configure slaves
 | |
|         my @output = `ip addr show 2>&1`;       # to check for existance of the device later
 | |
|         foreach my $dev (@devs) {
 | |
|             # as a convenience, make sure the device exists before adding it to the bond
 | |
|             if (!grep(m/^\d+:\s+$dev:/, @output)) {
 | |
|                 system("logger -t xcat -p local4.err  'configbond: not configuring $dev because it does not exist.'");
 | |
|                 unlink("$dir/ifcfg-$dev");          # in case it was left over in the image we are cloning
 | |
|                 next;
 | |
|             }
 | |
| 
 | |
|             system("logger -t xcat -p local4.err  'configbond: slave dev: $dev'");
 | |
|             if (!open(FILE, ">$dir/ifcfg-$dev")) { system("logger -t xcat -p local4.err 'configbond: cannot open $dir/ifcfg-$dev'"); exit 1; }
 | |
|             print FILE "DEVICE=$dev\n";
 | |
|             print FILE "BOOTPROTO=none\n";
 | |
|             print FILE "MASTER=$bond\n";
 | |
|             print FILE "ONBOOT=yes\n";
 | |
|             print FILE "SLAVE=yes\n";
 | |
|             print FILE "USERCTL=no\n";
 | |
|             close FILE;
 | |
|         }
 | |
|         }
 | |
|     elsif (-d "/etc/sysconfig/network") {
 | |
|         # sles
 | |
|         my $dir = "/etc/sysconfig/network";
 | |
|         if (!open(FILE, ">$dir/ifcfg-$bond")) { system("logger -t xcat -p local4.err 'configbond: cannot open $dir/ifcfg-$bond.'"); exit 1; }
 | |
| 
 | |
|         print FILE "BOOTPROTO=static\n";
 | |
|         print FILE "BONDING_MASTER=yes\n";
 | |
|         print FILE "BONDING_MODULE_OPTS='$bondingopts'\n";
 | |
|         print FILE "NAME='Bonded Interface'\n";
 | |
|         print FILE "IPADDR=$nic_ips[$i]\n";
 | |
|         print FILE "NETMASK=$netmask\n";
 | |
|         print FILE "STARTMODE=onboot\n";
 | |
|         print FILE "USERCONTROL=no\n";
 | |
|         my $devnum = 0;
 | |
|         my @output = `ip addr show 2>&1`;       # to check for existance of the device later
 | |
|         foreach my $dev (@devs) {
 | |
|             if (!grep(m/^\d+:\s+$dev:/, @output)) { next; }
 | |
|             print FILE "BONDING_SLAVE_$devnum=$dev\n";
 | |
|             $devnum++;
 | |
|         }
 | |
|         close FILE;
 | |
| 
 | |
|         # Configure slaves
 | |
|         foreach my $dev (@devs) {
 | |
|             # as a convenience, make sure the device exists before adding it to the bond
 | |
|             if (!grep(m/^\d+:\s+$dev:/, @output)) {
 | |
|                 system("logger -t xcat -p local4.err  'configbond: not configuring $dev because it does not exist.'");
 | |
|                 unlink("$dir/ifcfg-$dev");          # in case it was left over in the image we are cloning
 | |
|                 next;
 | |
|             }
 | |
|             
 | |
|             system("logger -t xcat -p local4.err  'configbond: slave dev: $dev'");
 | |
|             if (!open(FILE, ">$dir/ifcfg-$dev")) { system("logger -t xcat -p local4.err 'configbond: cannot open $dir/ifcfg-$dev'"); exit 1; }
 | |
|             print FILE "BOOTPROTO=none\n";
 | |
|             print FILE "STARTMODE=hotplug\n";
 | |
|             close FILE;
 | |
|         }
 | |
|         }
 | |
|     else {
 | |
|         # do not recognize this distro
 | |
|         system("logger -t xcat -p local4.err 'configbond: network directory is not either the Red Hat or SuSE format.'");
 | |
|         exit 1;
 | |
|     }
 | |
| 
 | |
|     # Apply the changes.  Since we are only doing bond1 right now, lets not restart the whole network
 | |
|     # so we dont disrupt the installnic connection.  Instead we just need to bring down the slave nics,
 | |
|     # and then bring up the bond nic.
 | |
|     #runcmd("service network restart");
 | |
|     foreach my $dev (@devs) {
 | |
|         runcmd("ifdown $dev");
 | |
|     }
 | |
|     runcmd("ifdown $bond");     # in case it was already up
 | |
|     runcmd("ifup $bond");       # note: this wont reload the bonding kernel module, so we are depending on the provisioning process to already have set the correct bonding options
 | |
|     system("logger -t xcat -p local4.info 'configbond: successfully configured $specific_nic.'");
 | |
| 
 | |
| }
 | |
| exit 0;
 | |
| 
 | |
| sub runcmd {
 | |
| 	my $cmd = shift @_;
 | |
| 	$cmd .= ' 2>&1';
 | |
| 	my @output = `$cmd`;
 | |
| 	my $rc = $? >> 8;
 | |
| 	if ($rc) {
 | |
| 		system("logger -t xcat -p local4.err 'configeth: command $cmd failed with rc $rc: " . join('',@output) . "'");
 | |
| 		my $errout= "configeth: command $cmd failed with rc $rc.";
 | |
|                 `echo $errout`;
 | |
| 		exit $rc;
 | |
| 	}
 | |
| }
 |