2755 lines
79 KiB
Perl
Executable File

#-------------------------------------------------------
=head1
xCAT plugin package to handle xdsh
Supported command:
nodenetconn
ipforward (internal command)
=cut
#-------------------------------------------------------
package xCAT_plugin::vlan;
BEGIN
{
$::XCATROOT = $ENV{'XCATROOT'} ? $ENV{'XCATROOT'} : '/opt/xcat';
}
use lib "$::XCATROOT/lib/perl";
use strict;
use xCAT::Table;
use xCAT::Utils;
use xCAT::NetworkUtils;
use xCAT::MsgUtils;
use Getopt::Long;
use xCAT::NodeRange;
use Data::Dumper;
use Text::Balanced qw(extract_bracketed);
use xCAT::SwitchHandler;
use Safe;
my $evalcpt = new Safe;
$evalcpt->share('&mknum');
$evalcpt->permit('require');
my %Switches=();
1;
#-------------------------------------------------------
=head3 handled_commands
Return list of commands handled by this plugin
=cut
#-------------------------------------------------------
sub handled_commands
{
return {
mkvlan => "vlan",
chvlan => "vlan",
rmvlan => "vlan",
lsvlan => "vlan",
chvlanports => "vlan",
};
}
#-------------------------------------------------------
=head3 preprocess_request
Preprocess the command
=cut
#-------------------------------------------------------
sub preprocess_request
{
my $request = shift;
my $callback = shift;
my $sub_req = shift;
my $command = $request->{command}->[0];
$::CALLBACK = $callback;
#if already preprocessed, go straight to request
if ((defined($request->{_xcatpreprocessed}))
&& ($request->{_xcatpreprocessed}->[0] == 1))
{
return [$request];
}
if ($command eq "mkvlan") {
return preprocess_mkvlan($request,$callback,$sub_req);
} elsif($command eq "chvlanports"){
return preprocess_chvlanports($request,$callback,$sub_req);
} elsif ($command eq "chvlan") {
return preprocess_chvlan($request,$callback,$sub_req);
} elsif ($command eq "rmvlan") {
return preprocess_rmvlan($request,$callback,$sub_req);
} elsif ($command eq "lsvlan") {
return preprocess_lsvlan($request,$callback,$sub_req);
} else {
my $rsp={};
$rsp->{error}->[0]= "$command: unsupported command.";
$callback->($rsp);
return 1;
}
}
sub preprocess_mkvlan {
my $request = shift;
my $callback = shift;
my $sub_req = shift;
my $command = $request->{command}->[0];
my $args = $request->{arg};
# parse the options
@ARGV=();
if ($args) {
@ARGV=@{$args};
}
Getopt::Long::Configure("bundling");
Getopt::Long::Configure("no_pass_through");
my $nr;
my $net;
my $netmask;
my $prefix;
my $nic;
if(!GetOptions(
'h|help' => \$::HELP,
'v|version' => \$::VERSION,
'n|nodes=s' => \$nr,
't|net=s' => \$net,
'm|mask=s' => \$netmask,
'p|prefix=s' => \$prefix,
'i|interface=s' => \$nic,
))
{
&mkvlan_usage($callback);
return 1;
}
# display the usage if -h or --help is specified
if ($::HELP) {
&mkvlan_usage($callback);
return 0;
}
# display the version statement if -v or --verison is specified
if ($::VERSION)
{
my $rsp={};
$rsp->{data}->[0]= xCAT::Utils->Version();
$callback->($rsp);
return 0;
}
my $vlan_id=0;
if (@ARGV>0) {
$vlan_id=$ARGV[0];
}
my @nodes=();
if ($nr) {
#get nodes
@nodes = noderange($nr);
if (nodesmissed) {
my $rsp={};
$rsp->{data}->[0]= "Invalid nodes in noderange:".join(',',nodesmissed);
$callback->($rsp);
return 1;
}
} else {
my $rsp={};
$rsp->{data}->[0]= "Please specify a list of nodes to be added to the new vlan.";
$callback->($rsp);
return 1;
}
if ($net && (!$netmask)) {
my $rsp={};
$rsp->{data}->[0]= "Please specify a netmask for the vlan.";
$callback->($rsp);
return 1;
}
if ($netmask && (!$net)) {
my $rsp={};
$rsp->{data}->[0]= "Please specify a network address for the vlan.";
$callback->($rsp);
return 1;
}
#let process _request to handle it
my $reqcopy = {%$request};
$reqcopy->{_xcatpreprocessed}->[0] = 1;
$reqcopy->{vlanid}->[0] = $vlan_id;
$reqcopy->{node} = \@nodes;
if ($net) { $reqcopy->{net}->[0]=$net; }
if ($netmask) { $reqcopy->{netmask}->[0]=$netmask; }
if ($prefix) { $reqcopy->{prefix}->[0]=$prefix; }
if ($nic) { $reqcopy->{nic}->[0]=$nic; }
return [$reqcopy];
}
# TODO: finish this
sub preprocess_chvlanports {
my $request = shift;
my $callback = shift;
my $sub_req = shift;
my $command = $request->{command}->[0];
my $args = $request->{arg};
# parse the options
@ARGV=();
if ($args) {
@ARGV=@{$args};
}
Getopt::Long::Configure("bundling");
Getopt::Long::Configure("no_pass_through");
my $nr;
my $delete;
my $nic;
if(!GetOptions(
'h|help' => \$::HELP,
'v|version' => \$::VERSION,
'n|nodes=s' => \$nr,
'd|delete' => \$delete,
'i|interface=s' => \$nic,
))
{
&chvlanports_usage($callback);
return 1;
}
# display the usage if -h or --help is specified
if ($::HELP) {
&chvlanports_usage($callback);
return 0;
}
# display the version statement if -v or --verison is specified
if ($::VERSION)
{
my $rsp={};
$rsp->{data}->[0]= xCAT::Utils->Version();
$callback->($rsp);
return 0;
}
if (@ARGV==0) {
my $rsp={};
$rsp->{data}->[0]= "Please specify a vlan id.";
$callback->($rsp);
return 1;
}
my $vlan_id=$ARGV[0];
my @nodes=();
if (!$nr) {
my $rsp={};
$rsp->{data}->[0]= "Please specify a range of nodes to be added to vlan $vlan_id.";
$callback->($rsp);
return 1;
} else {
#get nodes
@nodes = noderange($nr);
if (nodesmissed) {
my $rsp={};
$rsp->{data}->[0]= "Invalid nodes in noderange:".join(',',nodesmissed);
$callback->($rsp);
return 1;
}
}
if (!$nic) {
my $rsp={};
$rsp->{data}->[0]= "Please specify a network interface for nodes";
$callback->($rsp);
return 1;
}
#let process _request to handle it
my $reqcopy = {%$request};
$reqcopy->{_xcatpreprocessed}->[0] = 1;
$reqcopy->{vlanid}->[0]=$vlan_id;
$reqcopy->{node} = \@nodes;
$reqcopy->{delete}->[0] = $delete;
$reqcopy->{nic}->[0]=$nic;
return [$reqcopy];
}
sub preprocess_chvlan {
my $request = shift;
my $callback = shift;
my $sub_req = shift;
my $command = $request->{command}->[0];
my $args = $request->{arg};
# parse the options
@ARGV=();
if ($args) {
@ARGV=@{$args};
}
Getopt::Long::Configure("bundling");
Getopt::Long::Configure("no_pass_through");
my $nr;
my $delete;
my $nic;
if(!GetOptions(
'h|help' => \$::HELP,
'v|version' => \$::VERSION,
'n|nodes=s' => \$nr,
'd|delete' => \$delete,
'i|interface=s' => \$nic,
))
{
&chvlan_usage($callback);
return 1;
}
# display the usage if -h or --help is specified
if ($::HELP) {
&chvlan_usage($callback);
return 0;
}
# display the version statement if -v or --verison is specified
if ($::VERSION)
{
my $rsp={};
$rsp->{data}->[0]= xCAT::Utils->Version();
$callback->($rsp);
return 0;
}
if (@ARGV==0) {
my $rsp={};
$rsp->{data}->[0]= "Please specify a vlan id.";
$callback->($rsp);
return 1;
}
my $vlan_id=$ARGV[0];
my @nodes=();
if (!$nr) {
my $rsp={};
$rsp->{data}->[0]= "Please specify a range of nodes to be added to vlan $vlan_id.";
$callback->($rsp);
return 1;
} else {
#get nodes
@nodes = noderange($nr);
if (nodesmissed) {
my $rsp={};
$rsp->{data}->[0]= "Invalid nodes in noderange:".join(',',nodesmissed);
$callback->($rsp);
return 1;
}
}
#let process _request to handle it
my $reqcopy = {%$request};
$reqcopy->{_xcatpreprocessed}->[0] = 1;
$reqcopy->{vlanid}->[0]=$vlan_id;
$reqcopy->{node} = \@nodes;
$reqcopy->{delete}->[0] = $delete;
if ($nic) { $reqcopy->{nic}->[0]=$nic; }
return [$reqcopy];
}
sub preprocess_rmvlan {
my $request = shift;
my $callback = shift;
my $sub_req = shift;
my $command = $request->{command}->[0];
my $args = $request->{arg};
# parse the options
@ARGV=();
if ($args) {
@ARGV=@{$args};
}
Getopt::Long::Configure("bundling");
Getopt::Long::Configure("no_pass_through");
my $nodes;
if(!GetOptions(
'h|help' => \$::HELP,
'v|version' => \$::VERSION,
))
{
&rmvlan_usage($callback);
return 1;
}
# display the usage if -h or --help is specified
if ($::HELP) {
&rmvlan_usage($callback);
return 0;
}
# display the version statement if -v or --verison is specified
if ($::VERSION)
{
my $rsp={};
$rsp->{data}->[0]= xCAT::Utils->Version();
$callback->($rsp);
return 0;
}
if (@ARGV==0) {
my $rsp={};
$rsp->{data}->[0]= "Please specify a vlan id.";
$callback->($rsp);
return 1;
}
my $vlan_id=$ARGV[0];
#let process _request to handle it
my $reqcopy = {%$request};
$reqcopy->{_xcatpreprocessed}->[0] = 1;
$reqcopy->{vlanid}->[0]=$vlan_id;
return [$reqcopy];
}
sub preprocess_lsvlan {
my $request = shift;
my $callback = shift;
my $sub_req = shift;
my $command = $request->{command}->[0];
my $args = $request->{arg};
# parse the options
@ARGV=();
if ($args) {
@ARGV=@{$args};
}
Getopt::Long::Configure("bundling");
Getopt::Long::Configure("no_pass_through");
my $nodes;
if(!GetOptions(
'h|help' => \$::HELP,
'v|version' => \$::VERSION,
))
{
&lsvlan_usage($callback);
return 1;
}
# display the usage if -h or --help is specified
if ($::HELP) {
&lsvlan_usage($callback);
return 0;
}
# display the version statement if -v or --verison is specified
if ($::VERSION)
{
my $rsp={};
$rsp->{data}->[0]= xCAT::Utils->Version();
$callback->($rsp);
return 0;
}
my $vlan_id=0;
if (@ARGV>0) {
$vlan_id=$ARGV[0];
}
#let process _request to handle it
my $reqcopy = {%$request};
$reqcopy->{_xcatpreprocessed}->[0] = 1;
$reqcopy->{vlanid}->[0]=$vlan_id;
return [$reqcopy];
}
#-------------------------------------------------------
=head3 process_request
Process the command
=cut
#-------------------------------------------------------
sub process_request
{
my $request = shift;
my $callback = shift;
my $sub_req = shift;
my $command = $request->{command}->[0];
my $args = $request->{arg};
$::CALLBACK = $callback;
if ($command eq "mkvlan") {
return process_mkvlan($request,$callback,$sub_req);
} elsif ($command eq "chvlanports"){
return process_chvlanports($request,$callback,$sub_req);
} elsif ($command eq "chvlan") {
return process_chvlan($request,$callback,$sub_req);
} elsif ($command eq "rmvlan") {
return process_rmvlan($request,$callback,$sub_req);
} elsif ($command eq "lsvlan") {
return process_lsvlan($request,$callback,$sub_req);
} else {
my $rsp={};
$rsp->{error}->[0]= "$command: unsupported command.";
$callback->($rsp);
return 1;
}
return 0;
}
sub process_mkvlan {
my $request = shift;
my $callback = shift;
my $sub_req = shift;
my $vlan_id=0;
if (exists($request->{vlanid})) {
$vlan_id=$request->{vlanid}->[0];
}
my @nodes=();
if (exists($request->{node})) {
@nodes=@{$request->{node}};
}
my $net;
if (exists($request->{net})) {
$net=$request->{net}->[0];
}
my $netmask;
if (exists($request->{netmask})) {
$netmask=$request->{netmask}->[0];
}
my $prefix;
if (exists($request->{prefix})) {
$prefix=$request->{prefix}->[0];
}
my $nic;
if (exists($request->{nic})) {
$nic=$request->{nic}->[0];
}
#check if the vlan is already defined on the table
my $nwtab=xCAT::Table->new("networks", -create =>1);
if ($vlan_id > 0) {
if ($nwtab) {
my @tmp1=$nwtab->getAllAttribs(('vlanid', 'net', 'mask'));
if ((@tmp1) && (@tmp1 > 0)) {
foreach(@tmp1) {
if ($vlan_id eq $_->{vlanid}) {
my $rsp={};
$rsp->{error}->[0]= "The vlan $vlan_id is already used. Please choose another vlan id.";
$callback->($rsp);
return 1;
}
}
}
}
}
#make sure the vlan id is not currently used by another vlan
#if no vlan id is supplied, automatically generate a new vlan id
if ($vlan_id > 0) {
if (!verify_vlanid($vlan_id, $callback)) { return 1; }
} else {
$vlan_id=get_next_vlanid($callback);
if (!$vlan_id) { return 1;}
}
if (!$prefix) {
$prefix="v". $vlan_id . "n"; #default
}
####now get the switch ports for the nodes
my %swinfo=(); # example: %swinfo=( switch1=>{ 23=>{ nodes: [node1]
# interface: eth0
# },
# 24=>{ nodes: [node2,node3],
# vmhost: kvm1
# interface: eth1
# }
# },
# switch12=>{ 3=>{ nodes: [node4]},
# 7=>{ nodes: [node4]
# interface: eth0 },
# 9=>{ nodes: [node6]}
# }
# )
my %vminfo=(); # example: %vminfo=( kvm1=>{clients:[node1,node2...]},
# kvm2=>{clients:[node2,node3...]},
# )
#
my @vmnodes=(); # vm clients
my @snodes=(); # stand alone nodes
#get the vm hosts
my $vmtab=xCAT::Table->new("vm", -create =>1);
my $vmtmphash = $vmtab->getNodesAttribs(\@nodes, ['host', 'nics']) ;
foreach my $node (@nodes) {
my $host;
my $ent=$vmtmphash->{$node}->[0];
if (ref($ent) and defined $ent->{host}) {
$host = $ent->{host};
if (exists($vminfo{$host})) {
my $pa=$vminfo{$host}->{clients};
push(@$pa, $node);
} else {
$vminfo{$host}->{clients}=[$node];
}
push(@vmnodes, $node);
}
}
if (@vmnodes > 0) {
foreach my $node (@nodes) {
if (! grep /^$node$/, @vmnodes) {
push(@snodes, $node);
}
}
} else {
@snodes=@nodes;
}
#get the switch and port numbers for each node
my @vmhosts=keys(%vminfo);
my @anodes=(@snodes, @vmhosts); #nodes that connects to the switch
my %swsetup=();
my $swtab=xCAT::Table->new("switch", -create =>1);
my $swtmphash = $swtab->getNodesAttribs(\@anodes, ['switch', 'port', 'vlan', 'interface']) ;
my @missed_nodes=();
foreach my $node (@anodes) {
my $node_enties=$swtmphash->{$node};
if ($node_enties) {
my $i=-1;
my $use_this=0;
foreach my $ent (@$node_enties) {
$i++;
if (ref($ent) and defined $ent->{switch} and defined $ent->{port}) {
my $switch;
my $port;
$switch = $ent->{switch};
$port = $ent->{port};
my $interface="primary";
if (defined $ent->{interface}) { $interface=$ent->{interface};}
# for primary nic, the interface can be empty, "primary" or "primary:eth0"
#print "***nic=$nic, interface=$interface\n";
if ($nic) {
if ($interface =~ /primary/) {
$interface =~ s/primary(:)?//g;
}
if ($interface && ($interface eq $nic)) { $use_this=1; }
} else {
if ($interface =~ /primary/) { $use_this=1; }
}
if (! $use_this) {
next;
}
else {
$swsetup{$node}->{port}=$port;
$swsetup{$node}->{switch}=$switch;
if (defined $ent->{vlan}) {
$swsetup{$node}->{vlan}=$ent->{vlan};
} else {
$swsetup{$node}->{vlan}="";
}
}
if ($interface) {
$swinfo{$switch}->{$port}->{interface}=$interface;
}
if (exists($vminfo{$node})) {
$swinfo{$switch}->{$port}->{vmhost}=$node;
$swinfo{$switch}->{$port}->{nodes}=$vminfo{$node}->{clients};
} else {
$swinfo{$switch}->{$port}->{nodes}=[$node];
}
last;
}
}
if ( $use_this != 1 ) {
push (@missed_nodes, $node);
}
}
}
if (@missed_nodes > 0) {
my $rsp={};
$rsp->{error}->[0]= "Cannot proceed, please define switch and port info on the switch table for the following nodes:\n @missed_nodes\n";
$callback->($rsp);
return 1;
}
#print "vminfo=" . Dumper(%vminfo) . "\n";
#print "swinfo=" . Dumper(%swinfo) . "\n";
#print "snodes=" . Dumper(@snodes) . "\n";
#print "anodes=" . Dumper(@anodes) . "\n";
#print "vmnodes=" . Dumper(@vmnodes) . "\n";
#print "swtmphash" . Dumper($swtmphash). "\n";
### verify the ports are not used by other vlans
#if (!verify_switch_ports($vlan_id, \%swinfo, $callback)) { return 1;}
### now pick a network address for the vlan
if (!$net) {
($net, $netmask)=get_subnet($vlan_id, $callback);
}
### save the vlan on the networks table
my %key_col = (netname=>"vlan$vlan_id");
my %tb_cols=(vlanid=>$vlan_id, net=>$net, mask=>$netmask);
$nwtab->setAttribs(\%key_col, \%tb_cols);
### configure vlan on the switch
if (!create_vlan($vlan_id, \%swinfo, $callback)) { return 1;}
if (!add_ports($vlan_id, \%swinfo, $callback)) { return 1;}
my @sws=keys(%swinfo);
if (!add_crossover_ports($vlan_id, \@sws, $callback)) { return 1;}
### add the vlanid for the standalone nodes on the switch table
### append the vlan id for the vmhosts on the switch table
add_vlan_to_switch_table(\%swsetup, $vlan_id);
### get node ip and vlan hostname from the hosts table.
#If it is not defined, put the default into the host table
my @allnodes=(@anodes, @vmnodes);
if (!add_vlan_ip_host($net, $netmask, $prefix, 1, \@allnodes, $callback)) { return 1;}
### for vm nodes, add an additional nic on the vm.nics
if (@vmnodes > 0) {
my %setupnics=();
my $new_nic="vl$vlan_id";
foreach my $node (@vmnodes) {
my $ent=$vmtmphash->{$node}->[0];
my $nics;
if (ref($ent) and defined $ent->{nics}) {
$nics=$ent->{nics};
my @a=split(",", $nics);
if (!grep(/^$new_nic$/, @a)) { $nics="$nics,$new_nic"; }
} else {
$nics=$new_nic;
}
$setupnics{$node}={nics=>"$nics"};
}
$vmtab->setNodesAttribs(\%setupnics);
}
### populate the /etc/hosts and make the DNS server on the mn aware this change
$::CALLBACK = $callback;
my $res = xCAT::Utils->runxcmd( {
command => ['makehosts'],
}, $sub_req, 0, 1);
my $rsp = {};
$rsp->{data}->[0] = "Running makehosts...";
$callback->($rsp);
if ($res && (@$res > 0)) {
$rsp = {};
$rsp->{data} = $res;
$callback->($rsp);
}
$::CALLBACK = $callback;
my $res = xCAT::Utils->runxcmd( {
command => ['makedns'],
}, $sub_req, 0, 1);
my $rsp = {};
$rsp->{data}->[0] = "Running makedns...";
$callback->($rsp);
if ($res && (@$res > 0)) {
$rsp->{data} = $res;
$callback->($rsp);
}
my $cmd = "service named restart";
my $rc=system $cmd;
### now go to the nodes to configure the vlan interface
$::CALLBACK = $callback;
my $args = ["-P", "configvlan $vlan_id --keephostname"];
my $res = xCAT::Utils->runxcmd( {
command => ['updatenode'],
node => \@snodes,
arg => $args
}, $sub_req, 0, 1);
my $rsp = {};
$rsp->{data}->[0] = "Running updatenode...";
$callback->($rsp);
if ($res && (@$res > 0)) {
$rsp->{data} = $res;
$callback->($rsp);
}
### add configvlan postscripts to the postscripts table for the node
# so that next time the node bootup, the vlan can get configured
my @pnodes=(@snodes, @vmnodes);
add_postscript($callback, \@pnodes);
my $rsp={};
$rsp->{data}->[0]= "The vlan is successfully configured. ";
$rsp->{data}->[1]= " vlan id: $vlan_id";
$rsp->{data}->[2]= " vlan subnet: $net";
$rsp->{data}->[3]= " vlan netmask: $netmask";
#$rsp->{data}->[4]= " vlan dhcp server:";
#$rsp->{data}->[5]= " vlan dns server:";
#$rsp->{data}->[6]= " vlan gateway:";
$callback->($rsp);
return 0;
}
#-------------------------------------------------------
=head3 add_vlan_to_switch_table
It adds the vlan id to the switch.vlan for the given nodes.
=cut
#-------------------------------------------------------
sub add_vlan_to_switch_table {
my $swsetup=shift;
my $vlan_id=shift;
my $swtab1 = xCAT::Table->new( 'switch', -create=>1, -autocommit=>0 );
foreach my $node (keys(%$swsetup)) {
my %keyhash=();
my %updates=();
$keyhash{'node'} = $node;
$keyhash{'switch'}= $swsetup->{$node}->{switch};
$keyhash{'port'} = $swsetup->{$node}->{port};
$updates{'vlan'}=$vlan_id;
my $vlan;
if($swsetup->{$node}->{vlan}) {
$vlan=$swsetup->{$node}->{vlan};
my @a=split(",", $vlan);
if (!grep(/^$vlan_id$/, @a)) { $vlan="$vlan,$vlan_id";}
$updates{'vlan'}=$vlan;
}
$swtab1->setAttribs( \%keyhash,\%updates );
}
$swtab1->commit;
}
sub remove_vlan_from_switch_table {
my $swsetup=shift;
my $vlan_id=shift;
#remove the vlan id from the switch table for standalone nodes
my $swtab1 = xCAT::Table->new('switch', -create=>1, -autocommit=>0 );
foreach my $node (keys(%$swsetup)) {
my %keyhash=();
my %updates=();
$keyhash{'node'} = $node;
$keyhash{'switch'}= $swsetup->{$node}->{switch};
$keyhash{'port'} = $swsetup->{$node}->{port};
$updates{'vlan'} = "";
if($swsetup->{$node}->{vlan}) {
my @a=split(',', $swsetup->{$node}->{vlan});
my @b=grep(!/^$vlan_id$/,@a);
if (@b>0) {
$updates{'vlan'}=join(',', @b);
}
}
$swtab1->setAttribs( \%keyhash,\%updates );
}
$swtab1->commit;
}
#-------------------------------------------------------
=head3 add_postscript
It adds the 'configvlan' postscript to the postscript table
for the given nodes.
=cut
#-------------------------------------------------------
sub add_postscript {
my $callback=shift;
my $anodes=shift;
my $posttab=xCAT::Table->new("postscripts", -create =>1);
if ($posttab) {
(my $ref1) = $posttab->getAttribs({node => 'xcatdefaults'}, ('postscripts', 'postbootscripts'));
#if configvlan is in xcadefaults, then do nothing
if ($ref1) {
if ($ref1->{postscripts}) {
my @a = split(/,/, $ref1->{postscripts});
if (grep(/^configvlan$/, @a)) { next; }
}
if ($ref1->{postbootscripts}) {
my @a = split(/,/, $ref1->{postbootscripts});
if (grep(/^configvlan$/, @a)) { next; }
}
}
#now check for each node
my %setup_hash;
my $postcache = $posttab->getNodesAttribs($anodes,[qw(postscripts postbootscripts)]);
foreach my $node (@$anodes) {
my $ref = $postcache->{$node}->[0];
if ($ref) {
if (exists($ref->{postscripts})) {
my @a = split(/,/, $ref->{postscripts});
if (grep(/^configvlan$/, @a)) { next; }
}
if (exists($ref->{postbootscripts})) {
my $post=$ref->{postbootscripts};
my @old_a=split(',', $post);
if (grep(/^configvlan$/, @old_a)) {
next;
} else {
$setup_hash{$node}={postbootscripts=>"$post,configvlan"};
}
} else {
$setup_hash{$node}={postbootscripts=>"configvlan"};
}
} else {
$setup_hash{$node}={postbootscripts=>"configvlan"};
}
}
if (keys(%setup_hash) > 0) {
$posttab->setNodesAttribs(\%setup_hash);
}
}
return 0;
}
#-------------------------------------------------------
=head3 add_vlan_ip_host
It goes to the hosts.otherinterfaces to see if the vlan ip and hostname
is defined. If not, it will add the default in the table.
The default is v<vlanid>n<node#>
=cut
#-------------------------------------------------------
sub add_vlan_ip_host {
my $subnet=shift;
my $netmask=shift;
my $prefix=shift;
my $node_number=shift;
my $nodes=shift;
my $callback=shift;
my $hoststab = xCAT::Table->new('hosts');
my $hostscache = $hoststab->getNodesAttribs($nodes,[qw(node otherinterfaces)]);
my %setup_hash;
foreach my $node (@$nodes) {
my $ref = $hostscache->{$node}->[0];
my $found=0;
my $otherinterfaces;
if ($ref && exists($ref->{otherinterfaces})){
$otherinterfaces = $ref->{otherinterfaces};
my @itf_pairs=split(/,/, $otherinterfaces);
foreach (@itf_pairs) {
my ($name,$ip)=split(/:/, $_);
if ($name =~ /^-/ ) {
$name = $node.$name;
}
if(xCAT::NetworkUtils->ishostinsubnet($ip, $netmask, $subnet)) {
$found=1;
}
}
}
if (!$found) {
my $hostname=$prefix . "$node_number";
my $ip="";
if ($subnet =~ /\d+\.\d+\.\d+\.\d+/) {# ipv4 address
$subnet =~ /([0-9]+)\.([0-9]+)\.([0-9]+)\.([0-9]+)/;
my $netnum = ($1<<24)+($2<<16)+($3<<8)+$4;
my $ipnum=$netnum + $node_number;
my @a=();
for (my $i=3; $i>-1; $i--) {
$a[$i]=$ipnum % 256;
$ipnum = int($ipnum / 256);
}
$ip= "$a[0].$a[1].$a[2].$a[3]";
#print "ip=$ip\n";
} else {
my $rsp={};
$rsp->{error}->[0]= "Does not support IPV6 address yet.";
$callback->($rsp);
return 0;
}
$node_number++;
if ($otherinterfaces) {
$setup_hash{$node}={otherinterfaces=>"$hostname:$ip,$otherinterfaces"};
} else {
$setup_hash{$node}={otherinterfaces=>"$hostname:$ip"};
}
}
} #foreach node
if (keys(%setup_hash) > 0) {
$hoststab->setNodesAttribs(\%setup_hash);
}
return 1;
}
#-------------------------------------------------------
=head3 get_prefix_and_nodenumber
It gets the prefix and max node number from the current nodes
in the given vlan.
=cut
#-------------------------------------------------------
sub get_prefix_and_nodenumber {
my $vlan_id = shift;
my $subnet = shift;
my $netmask = shift;
#get all the nodes that are in the vlan
my $swtab=xCAT::Table->new("switch", -create =>0);
my @nodes=();
my @tmp1=$swtab->getAllAttribs(('node', 'vlan'));
if ((@tmp1) && (@tmp1 > 0)) {
foreach my $ent (@tmp1) {
my @nodes_tmp=noderange($ent->{node});
foreach my $node (@nodes_tmp) {
if ($ent->{vlan}) {
my @a=split(",", $ent->{vlan});
if (grep(/^$vlan_id$/,@a)) {
push(@nodes, $node);
}
}
}
}
}
#get all the vm clients if the node is a vm host
my $vmtab=xCAT::Table->new("vm", -create =>0);
my @vmnodes=();
if ($vmtab) {
my @tmp1=$vmtab->getAllAttribs('node', 'host');
if ((@tmp1) && (@tmp1 > 0)) {
foreach(@tmp1) {
my $host = $_->{host};
if (grep(/^$host$/, @nodes)) {
my @nodes_tmp=noderange($_->{node});
foreach my $node (@nodes_tmp) {
push(@vmnodes, $node);
}
}
}
}
}
@nodes=(@nodes, @vmnodes);
#print "nodes=@nodes\n";
#now go to hosts table to get the prefix and max node number
my $hoststab = xCAT::Table->new('hosts');
my $hostscache = $hoststab->getNodesAttribs(\@nodes,[qw(node otherinterfaces)]);
my $max=0;
my $prefix;
foreach my $node (@nodes) {
my $ref = $hostscache->{$node}->[0];
my $otherinterfaces;
if ($ref && exists($ref->{otherinterfaces})){
$otherinterfaces = $ref->{otherinterfaces};
my @itf_pairs=split(/,/, $otherinterfaces);
my @itf_pairs2=();
foreach (@itf_pairs) {
my ($name,$ip)=split(/:/, $_);
if(xCAT::NetworkUtils->ishostinsubnet($ip, $netmask, $subnet)) {
$name =~ /^(.*)([\d]+)$/;
if ($2 > $max) { $max=$2;}
if (!$prefix) { $prefix=$1;}
#print "name=$name\n, 1=$1, 2=$2\n";
}
}
}
} #foreach node
return ($prefix, $max);
}
#-------------------------------------------------------
=head3 remove_vlan_ip_host
It goes to the hosts.otherinterfaces to see if the vlan ip and hostname
is defined. If it is, it will remove it. It also remove the entried in
the /etc/hosts file
=cut
#-------------------------------------------------------
sub remove_vlan_ip_host {
my $subnet=shift;
my $netmask=shift;
my $nodes=shift;
my $callback=shift;
my $hoststab = xCAT::Table->new('hosts');
my $hostscache = $hoststab->getNodesAttribs($nodes,[qw(node otherinterfaces)]);
my %setup_hash;
foreach my $node (@$nodes) {
my $ref = $hostscache->{$node}->[0];
my $otherinterfaces;
if ($ref && exists($ref->{otherinterfaces})){
$otherinterfaces = $ref->{otherinterfaces};
my @itf_pairs=split(/,/, $otherinterfaces);
my @itf_pairs2=();
my $index=0;
foreach (@itf_pairs) {
my ($name,$ip)=split(/:/, $_);
if(!xCAT::NetworkUtils->ishostinsubnet($ip, $netmask, $subnet)) {
$itf_pairs2[$index]=$_;
$index++;
} else {
my $cmd="sed -i /$ip/d /etc/hosts";
my $rc=system $cmd;
}
}
if (@itf_pairs2 > 0) {
my $new_intf=join(",", @itf_pairs2);
$setup_hash{$node}={otherinterfaces=>$new_intf}
} else {
$setup_hash{$node}={otherinterfaces=>""};
}
}
} #foreach node
if (keys(%setup_hash) > 0) {
$hoststab->setNodesAttribs(\%setup_hash);
}
return 1;
}
#-------------------------------------------------------
=head3 verify_vlanid
It goes to all the switches to make sure that the vlan
id is not used by other vlans.
=cut
#-------------------------------------------------------
sub verify_vlanid {
my $vlan_id=shift;
my $callback=shift;
my $switchestab=xCAT::Table->new('switches',-create=>0);
my @tmp1=$switchestab->getAllAttribs(('switch'));
if ((@tmp1) && (@tmp1 > 0)) {
foreach(@tmp1) {
my @switches_tmp=noderange($_->{switch});
if (@switches_tmp==0) { push @switches_tmp, $_->{switch}; } #sometimes the switch name is not on the node list table.
foreach my $switch (@switches_tmp) {
my $swh;
if (exists($Switches{$switch})) { $swh=$Switches{$switch};}
else {
$swh=new xCAT::SwitchHandler->new($switch);
$Switches{$switch}=$swh;
}
my @ids=$swh->get_vlan_ids();
print "ids=@ids\n";
foreach my $id (@ids) {
if ($id == $vlan_id) {
my $rsp={};
$rsp->{error}->[0]= "The vlan id $vlan_id already exists on switch $switch. Please choose another vlan id.\n";
$callback->($rsp);
return 0;
}
}
}
}
}
return 1;
}
#-------------------------------------------------------
=head3 get_next_vlanid
It automatically generates the vlan ID. It goes to all the
switches, get the smallest common integer that is not used
by any existing vlans.
=cut
#-------------------------------------------------------
sub get_next_vlanid {
my $callback=shift;
my $switchestab=xCAT::Table->new('switches',-create=>0);
my @tmp1=$switchestab->getAllAttribs(('switch'));
my %vlanids=();
if ((@tmp1) && (@tmp1 > 0)) {
foreach(@tmp1) {
my @switches_tmp=noderange($_->{switch});
if (@switches_tmp==0) { push @switches_tmp, $_->{switch}; } #sometimes the switch name is not on the node list table.
foreach my $switch (@switches_tmp) {
my $swh;
if (exists($Switches{$switch})) { $swh=$Switches{$switch};}
else {
$swh=new xCAT::SwitchHandler->new($switch);
$Switches{$switch}=$swh;
}
my @ids=$swh->get_vlan_ids();
foreach my $id (@ids) {
$vlanids{$id}=1;
}
}
}
}
for (my $index=2; $index<255; $index++) {
if (! exists($vlanids{$index})) { return $index; }
}
my $rsp={};
$rsp->{data}->[0]= "No valid vlan ID can be used any more, Please remove unused vlans.\n";
$callback->($rsp);
return 0;
}
#-------------------------------------------------------
=head3 verify_switch_ports
It checks if the switch ports to be configured are used by other vlans.
=cut
#-------------------------------------------------------
sub verify_switch_ports {
my $vlan_id=shift;
my $swinfo=shift;
my $callback=shift;
my $ret=1;
foreach my $switch (keys %$swinfo) {
my $porthash=$swinfo->{$switch};
my $swh;
if (exists($Switches{$switch})) { $swh=$Switches{$switch};}
else {
$swh=new xCAT::SwitchHandler->new($switch);
$Switches{$switch}=$swh;
}
my $port_vlan_hash=$swh->get_vlanids_for_ports(keys(%$porthash));
print "port_vlan_hash=" . Dumper($port_vlan_hash) . "\n";
my @error_ports=();
foreach my $port (keys(%$port_vlan_hash)) {
my $val=$port_vlan_hash->{$port};
foreach my $tmp_vid (@$val) {
if (($tmp_vid != $vlan_id) && ($tmp_vid != 1) && ($tmp_vid ne 'NA')) {
if (exists($porthash->{$port}->{vmhost})) { next; } #skip the vmhost, vmhost can have more than one vlans
push(@error_ports, $port);
last;
}
}
}
if (@error_ports >0) {
$ret=0;
my $error_str;
foreach(@error_ports) {
my @tmp=@{$porthash->{$_}->{nodes}};
my $ids_tmp=$port_vlan_hash->{$_};
$error_str .= "$_: vlan-ids=@$ids_tmp nodes=@tmp\n";
}
my $rsp={};
$rsp->{error}->[0]= "The following ports on switch $switch are used by other vlans.\n$error_str";
$callback->($rsp);
}
}
return $ret;
}
#-------------------------------------------------------
=head3 create_vlan
It goes to the switches and create a new vlan.
Returns: 1 -- suggessful
0 -- fail
=cut
#-------------------------------------------------------
sub create_vlan {
my $vlan_id=shift;
my $swinfo=shift;
my $callback=shift;
my $ret=1;
foreach my $switch (keys %$swinfo) {
my $swh;
if (exists($Switches{$switch})) { $swh=$Switches{$switch};}
else {
$swh=new xCAT::SwitchHandler->new($switch);
$Switches{$switch}=$swh;
}
#check if the vlan already exists on the switch
my @ids=$swh->get_vlan_ids();
my $vlan_exists=0;
foreach my $id (@ids) {
if ($id == $vlan_id) {
$vlan_exists=1;
last;
}
}
if (!$vlan_exists) {
#create the vlan
print "create vlan $vlan_id on switch $switch\n";
my @ret=$swh->create_vlan($vlan_id);
if ($ret[0] != 0) {
my $rsp={};
$rsp->{error}->[0]= "create_vlan: $ret[1]";
$callback->($rsp);
}
}
}
return 1;
}
#-------------------------------------------------------
=head3 add_ports
It adds the ports to the vlan.
Returns: 1 -- suggessful
0 -- fail
=cut
#-------------------------------------------------------
sub add_ports {
my $vlan_id=shift;
my $swinfo=shift;
my $callback=shift;
my $portmode=shift;
my $ret=1;
foreach my $switch (keys %$swinfo) {
my $porthash=$swinfo->{$switch};
my $swh;
if (exists($Switches{$switch})) { $swh=$Switches{$switch};}
else {
$swh=new xCAT::SwitchHandler->new($switch);
$Switches{$switch}=$swh;
}
my @ret=$swh->add_ports_to_vlan($vlan_id, $portmode, keys(%$porthash));
if ($ret[0] != 0) {
my $rsp={};
$rsp->{error}->[0]= "add_ports_to_vlan: $ret[1]";
$callback->($rsp);
}
}
return 1;
}
#-------------------------------------------------------
=head3 add_crossover_ports
It enables the vlan on the cross-over links.
Returns: 1 -- suggessful
0 -- fail
=cut
#-------------------------------------------------------
sub add_crossover_ports {
my $vlan_id=shift;
my $psws=shift;
my $callback=shift;
#now make sure the links between the switches allows this vlan to go through
my @sws=@$psws;
print "sws=@sws\n";
if (@sws > 1) {
foreach my $switch (@sws) {
my $swh;
if (exists($Switches{$switch})) { $swh=$Switches{$switch};}
else {
$swh=new xCAT::SwitchHandler->new($switch);
$Switches{$switch}=$swh;
}
my @sws_b=grep(!/^$switch$/, @sws);
my @ret=$swh->add_crossover_ports_to_vlan($vlan_id, @sws_b);
if ($ret[0] != 0) {
my $rsp={};
$rsp->{error}->[0]= "add_crossover_ports: $ret[1]";
$callback->($rsp);
} else {
if ($ret[1]) {
my $rsp={};
$rsp->{data}->[0]= "add_crossover_ports: $ret[1]";
$callback->($rsp);
}
}
}
}
return 1;
}
#-------------------------------------------------------
=head3 remove_ports
It goes to the switches and create a new vlan.
Returns: 1 -- suggessful
0 -- fail
=cut
#-------------------------------------------------------
sub remove_ports {
my $vlan_id=shift;
my $swinfo=shift;
my $callback=shift;
my $novmhost=shift;
my $ret=1;
foreach my $switch (keys %$swinfo) {
my $porthash=$swinfo->{$switch};
my $swh;
if (exists($Switches{$switch})) { $swh=$Switches{$switch};}
else {
$swh=new xCAT::SwitchHandler->new($switch);
$Switches{$switch}=$swh;
}
my @ports=();
if ($novmhost) { #skip the vm hosts for chvlan
foreach my $port (keys(%$porthash)) {
if (!exists($porthash->{$port}->{vmhost})) {
push(@ports, $port);
}
}
}else {
@ports=keys(%$porthash);
}
my @ret=$swh->remove_ports_from_vlan($vlan_id, @ports);
if ($ret[0] != 0) {
my $rsp={};
$rsp->{error}->[0]= "remove_ports_from_vlan: $ret[1]";
$callback->($rsp);
}
}
return 1;
}
#-------------------------------------------------------
=head3 get_subnet
It gets the subnet address and netmask for the given
vlan ID. The pattern is defined by "vlannets" and "vlanmask"
on the site table. The default is "10.<$vlanid>.0.0"/"255.255.0.0".
=cut
#-------------------------------------------------------
sub get_subnet {
my $vlan_id=shift;
my $callback = shift;
my $net;
my $mask;
#get vlannets and vlanidmask from the site table
my $vlannets="|(\\d+)|10.(\$1+0).0.0|";
my $vlanmask="255.255.0.0";
my $sitetab = xCAT::Table->new('site');
my $sent = $sitetab->getAttribs({key=>'vlannets'},'value');
if ($sent and ($sent->{value})) {
$vlannets=$sent->{value};
}
$sent = $sitetab->getAttribs({key=>'vlanmask'},'value');
if ($sent and ($sent->{value})) {
$vlanmask=$sent->{value};
}
$mask = $vlanmask;
if ($vlannets =~ /^\/[^\/]*\/[^\/]*\/$/)
{
my $exp = substr($vlannets, 1);
chop $exp;
my @parts = split('/', $exp, 2);
$net=$vlan_id;
$net =~ s/$parts[0]/$parts[1]/;
}
elsif ($vlannets =~ /^\|.*\|.*\|$/)
{
my $exp = substr($vlannets, 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
undef $@; #extract_bracketed would have set $@ if it didn't return, undef $@
$retval = $vlan_id;
$retval =~ s/$parts[0]/$parts[1]/;
}
while ($curr)
{
my $value = $vlan_id;
$value =~ s/$parts[0]/$curr/;
$value = $evalcpt->reval('use integer;'.$value);
$retval = $prev . $value . $next;
($curr, $next, $prev) =
extract_bracketed($retval, '()', qr/[^()]*/);
}
undef $@;
$net = $vlan_id;
$net =~ s/$parts[0]/$retval/;
}
return ($net, $mask);
}
# process_chvlanports only support physical nodes
# bond not supported and multi nics are not supported
sub process_chvlanports {
my $request = shift;
my $callback = shift;
my $sub_req = shift;
my $vlan_id=0;
my $nic = "";
my @nodes=();
my $delete=0;
# validate vlan id value.
$vlan_id=$request->{vlanid}->[0];
#debug message.
xCAT::MsgUtils->message('S',"vlanid: $vlan_id");
if ($vlan_id <= 0) {
my $rsp={};
$rsp->{error}->[0]= "Invalid vlan id: $vlan_id";
$callback->($rsp);
return;
}
#Check if the vlan is defined in networks table.
my $net="";
my $netmask="";
my $nwtab=xCAT::Table->new("networks", -create =>1);
if ($nwtab) {
my @nwentires=$nwtab->getAllAttribs(('vlanid', 'net', 'mask'));
foreach(@nwentires) {
if ($vlan_id eq $_->{vlanid}) {
$net=$_->{net};
$netmask=$_->{mask};
}
}
}
if ((!$net) || (!$netmask)) {
my $rsp = {};
$rsp->{data}->[0] = "Can not find valid network/netmask definition from table networks for vlan $vlan_id.";
$callback->($rsp);
return 1;
}
$nic=$request->{nic}->[0];
#debug message.
xCAT::MsgUtils->message('S',"nic is: $nic");
@nodes=@{$request->{node}};
#debug message.
my $nodesstr = Dumper(@nodes);
xCAT::MsgUtils->message('S',"nodes are: $nodesstr");
$delete=$request->{delete}->[0];
#debug message.
xCAT::MsgUtils->message('S',"delete flag: $delete");
####now get the switch ports for the nodes
my %swinfo=(); # example: %swinfo=( switch1=>{ 23=>{ nodes: [node1]},
# 24=>{ nodes: [node2]},
# },
# switch12=>{ 3=>{ nodes: [node4],
# }
# )
my %nodeswinfo=(); # example: %nodeswinfo=(
# node1=>{eth0=>{switch => swith1,
# port => 1,
# vlanids => [2,3]},
# eth1=>{switch => switch1,
# port => 2,
# vlanids => [5]}}
# )
my %swsetup=();
#get the switch and port numbers for each node
my $swtab=xCAT::Table->new("switch", -create =>1);
my $swtmphash = $swtab->getNodesAttribs(\@nodes, ['switch', 'port', 'vlan', 'interface']);
foreach my $node (keys (%$swtmphash)) {
my $node_enties=$swtmphash->{$node};
foreach my $ent (@$node_enties) {
if (ref($ent) and defined $ent->{switch} and defined $ent->{port}){
my $switch = $ent->{switch};
my $port = $ent->{port};
my $interface = "";
if( defined $ent->{interface} ){ $interface = $ent->{interface}};
my $vlan = "";
if( defined $ent->{vlan}) {$vlan = $ent->{vlan}};
if ($interface){
$nodeswinfo{$node}->{$interface}->{switch}=$switch;
$nodeswinfo{$node}->{$interface}->{port}=$port;
$nodeswinfo{$node}->{$interface}->{vlan} = $vlan;
}
}
}
}
#debug message.
my $nodesinfostr = Dumper(%nodeswinfo);
xCAT::MsgUtils->message('S',"nodeswinfo: $nodesinfostr");
# Validate node's switch info %nodeswinfo and build up %swinfo, %swsetup.
my @missed_switch_nodes = ();
my @missed_vlanid_nodes = ();
foreach my $node (@nodes){
# Check whether the switch, port and interface info defined for all nodes.
if(defined $nodeswinfo{$node} && defined $nodeswinfo{$node}->{$nic}){
} else{
push(@missed_switch_nodes, $node);
next;
}
if ($delete){
# For delete mode, must make sure all node's has such a vlan ID defined for the interface.
my @vlanids = split(",",$nodeswinfo{$node}->{$nic}->{vlan});
if (@vlanids && (grep /^$vlan_id$/, @vlanids)){
my $switch = $nodeswinfo{$node}->{$nic}->{switch};
my $port = $nodeswinfo{$node}->{$nic}->{port};
#setup swinfo and swsetup .
$swinfo{$switch}->{$port}->{hosts} = [$node];
$swsetup{$node}->{switch} = $switch;
$swsetup{$node}->{port} = $port;
$swsetup{$node}->{vlan} = $nodeswinfo{$node}->{$nic}->{vlan};
}else{
push(@missed_vlanid_nodes, $node);
next;
}
} else{
# non-delete mode, just setup swinfo directly.
my $switch = $nodeswinfo{$node}->{$nic}->{switch};
my $port = $nodeswinfo{$node}->{$nic}->{port};
#setup swinfo and swsetup for add_ports, add_crossover_ports and add_vlan_to_switch_table call later.
$swinfo{$switch}->{$port}->{hosts} = [$node];
$swsetup{$node}->{switch} = $switch;
$swsetup{$node}->{port} = $port;
$swsetup{$node}->{vlan} = $nodeswinfo{$node}->{$nic}->{vlan};
}
}
if (@missed_switch_nodes > 0) {
my $rsp={};
$rsp->{error}->[0]= "Cannot proceed, please define switch, port and interface info on the switch table for the following nodes:\n @missed_switch_nodes\n";
$callback->($rsp);
return 1;
}
if (@missed_vlanid_nodes > 0) {
my $rsp={};
$rsp->{error}->[0]= "Cannot proceed, no such vlan ID $vlan_id defined for following nodes:\n @missed_vlanid_nodes\n";
$callback->($rsp);
return 1;
}
#debug message.
my $swinfostr = Dumper(%swinfo);
xCAT::MsgUtils->message('S',"swinfo: $swinfostr");
my $swsetupstr = Dumper(%swsetup);
xCAT::MsgUtils->message('S',"swsetup: $swsetupstr");
# Do actual configurations on switches
if (!$delete) {
### add ports to the vlan
if (!add_ports($vlan_id, \%swinfo, $callback, 1)) { return 1;}
xCAT::MsgUtils->message('S',"Adding ports to switch success!");
### add the cross-over ports to the vlan
my @sws=keys(%swinfo);;
#get all the switches that are in the vlan
my $swtab=xCAT::Table->new("switch", -create =>0);
if ($swtab) {
my @tmp1=$swtab->getAllAttribs('switch', 'vlan');
if ((@tmp1) && (@tmp1 > 0)) {
foreach my $item (@tmp1) {
my $vlan=$item->{vlan};
my $sw=$item->{switch};
if ($vlan) {
my @a=split(",",$vlan);
if (grep(/^$vlan_id$/, @a)) {
if (!grep(/^$sw$/, @sws)) {
push(@sws, $sw);
}
}
}
}
}
}
if (!add_crossover_ports($vlan_id, \@sws, $callback)) { return 1;}
xCAT::MsgUtils->message('S',"Configuring cross over ports success!");
#add the vlanid for the standalone nodes on the switch table
#append the vlan id for the vmhosts on the switch table
add_vlan_to_switch_table(\%swsetup, $vlan_id);
xCAT::MsgUtils->message('S',"Adding vlan to switch table success!");
# done
my $rsp={};
$rsp->{data}->[0]= "The interface $nic of following nodes are added to the vlan $vlan_id:\n@nodes";
$callback->($rsp);
} else{
### remove ports from the vlan
my $novmhost=1;
if (!remove_ports($vlan_id, \%swinfo, $callback, $novmhost)) { return 1;}
xCAT::MsgUtils->message('S',"Removing ports from vlan success!");
#remove the vlan id from the switch table.
remove_vlan_from_switch_table(\%swsetup,$vlan_id);
xCAT::MsgUtils->message('S',"Removing ports from switch table success!");
# done
my $rsp={};
$rsp->{data}->[0]= "The interface $nic of following nodes are removed from the vlan $vlan_id:\n@nodes";
$callback->($rsp);
}
return 0;
}
sub process_chvlan {
my $request = shift;
my $callback = shift;
my $sub_req = shift;
my $vlan_id=0;
if (exists($request->{vlanid})) {
$vlan_id=$request->{vlanid}->[0];
}
if ($vlan_id == 0) {
my $rsp={};
$rsp->{error}->[0]= "Invalid vlan id: $vlan_id";
$callback->($rsp);
return;
}
my $nic;
if (exists($request->{nic})) {
$nic=$request->{nic}->[0];
}
my @nodes=();
if (exists($request->{node})) {
@nodes=@{$request->{node}};
}
my $delete=0;
if (exists($request->{delete})) {
$delete=$request->{delete}->[0];
}
my $net;
my $netmask;
#check if the vlan is already defined on the table
my $found=0;
my $nwtab=xCAT::Table->new("networks", -create =>1);
if ($vlan_id > 0) {
if ($nwtab) {
my @tmp1=$nwtab->getAllAttribs(('vlanid', 'net', 'mask'));
if ((@tmp1) && (@tmp1 > 0)) {
foreach(@tmp1) {
if ($vlan_id eq $_->{vlanid}) {
$found=1;
$net=$_->{net};
$netmask=$_->{mask};
}
}
}
}
}
if (!$found) {
my $rsp = {};
$rsp->{data}->[0] = "The vlan $vlan_id does not exist.";
$callback->($rsp);
return 1;
}
if ((!$net) || (!$netmask)) {
my $rsp = {};
$rsp->{data}->[0] = "Please make sure subnet and netmask are specified on the networks table for vlan $vlan_id.";
$callback->($rsp);
return 1;
}
####now get the switch ports for the nodes
my %swinfo=(); # example: %swinfo=( switch1=>{ 23=>{ nodes: [node1]},
# 24=>{ nodes: [node2,node3],
# vmhost: kvm1
# }
# },
# switch12=>{ 3=>{ nodes: [node4]},
# 7=>{ nodes: [node5]},
# 9=>{ nodes: [node6]}
# }
# )
my %vminfo=(); # example: %vminfo=( kvm1=>{clients:[node1,node2...]},
# kvm2=>{clients:[node2,node3...]},
# )
#
my @vmnodes=(); # vm clients
my @snodes=(); # stand alone nodes
#get the vm hosts
my $vmtab=xCAT::Table->new("vm", -create =>1);
my $vmtmphash = $vmtab->getNodesAttribs(\@nodes, ['host','nics']) ;
foreach(@nodes) {
my $node=$_;
my $host;
my $ent=$vmtmphash->{$node}->[0];
if (ref($ent) and defined $ent->{host}) {
$host = $ent->{host};
if (exists($vminfo{$host})) {
my $pa=$vminfo{$host}->{clients};
push(@$pa, $node);
} else {
$vminfo{$host}->{clients}=[$node];
}
push(@vmnodes, $node);
}
}
if (@vmnodes > 0) {
foreach my $node (@nodes) {
if (! grep /^$node$/, @vmnodes) {
push(@snodes, $node);
}
}
} else {
@snodes=@nodes;
}
#get the switch and port numbers for each node
my @vmhosts=keys(%vminfo);
my @anodes=(@snodes, @vmhosts); #nodes that connects to the switch
my %swsetup=();
my $swtab=xCAT::Table->new("switch", -create =>1);
my $swtmphash = $swtab->getNodesAttribs(\@anodes, ['switch', 'port', 'vlan', 'interface']) ;
my @missed_nodes=();
foreach my $node (@anodes) {
my $switch;
my $port;
my $node_enties=$swtmphash->{$node};
if ($node_enties) {
my $i=-1;
my $use_this=0;
foreach my $ent (@$node_enties) {
$i++;
if (ref($ent) and defined $ent->{switch} and defined $ent->{port}) {
$switch = $ent->{switch};
$port = $ent->{port};
my $interface="primary";
if (defined $ent->{interface}) { $interface=$ent->{interface};}
# for primary nic, the interface can be empty, "primary" or "primary:eth0"
if ($delete) {
if (defined($ent->{vlan})) {
my @a=split(',', $ent->{vlan});
if (grep(/^$vlan_id$/, @a)) { $use_this=1; }
}
} else {
if ($nic) {
if ($interface =~ /primary/) {
$interface =~ s/primary(:)?//g;
}
if ($interface && ($interface eq $nic)) { $use_this=1; }
} else {
if ($interface =~ /primary/) { $use_this=1; }
}
}
if (! $use_this) {
next;
} else {
$swsetup{$node}->{port}=$port;
$swsetup{$node}->{switch}=$switch;
if (defined $ent->{vlan}) {
$swsetup{$node}->{vlan}=$ent->{vlan};
} else {
$swsetup{$node}->{vlan}="";
}
}
if ($interface) {
$swinfo{$switch}->{$port}->{interface}=$interface;
}
if (exists($vminfo{$node})) {
$swinfo{$switch}->{$port}->{vmhost}=$node;
$swinfo{$switch}->{$port}->{nodes}=$vminfo{$node}->{clients};
} else {
$swinfo{$switch}->{$port}->{nodes}=[$node];
}
last;
}
}
if ( $use_this != 1 ) {
push (@missed_nodes, $node);
}
}
}
if (@missed_nodes > 0) {
my $rsp={};
$rsp->{error}->[0]= "Cannot proceed, please define switch and port info on the switch table for the following nodes:\n @missed_nodes\n";
$callback->($rsp);
return 1;
}
#print "vminfo=" . Dumper(%vminfo) . "\n";
#print "swinfo=" . Dumper(%swinfo) . "\n";
#print "anodes=" . Dumper(@anodes) . "\n";
#print "vmnodes=" . Dumper(@vmnodes) . "\n";
if (!$delete) {
### verify the ports are not used by other vlans
#if (!verify_switch_ports($vlan_id, \%swinfo, $callback)) { return 1;}
###create the vlan if it does not exist
if (!create_vlan($vlan_id, \%swinfo, $callback)) { return 1;}
### add ports to the vlan
if (!add_ports($vlan_id, \%swinfo, $callback)) { return 1;}
### add the cross-over ports to the vlan
my @sws=keys(%swinfo);;
#get all the switches that are in the vlan
my $swtab=xCAT::Table->new("switch", -create =>0);
if ($swtab) {
my @tmp1=$swtab->getAllAttribs('switch', 'vlan');
if ((@tmp1) && (@tmp1 > 0)) {
foreach my $item (@tmp1) {
my $vlan=$item->{vlan};
my $sw=$item->{switch};
if ($vlan) {
my @a=split(",",$vlan);
if (grep(/^$vlan_id$/, @a)) {
if (!grep(/^$sw$/, @sws)) {
push(@sws, $sw);
}
}
}
}
}
}
if (!add_crossover_ports($vlan_id, \@sws, $callback)) { return 1;}
#add the vlanid for the standalone nodes on the switch table
#append the vlan id for the vmhosts on the switch table
add_vlan_to_switch_table(\%swsetup, $vlan_id);
#we'll derive the prefix and the node numbers from the existing
#nodes on the vlan
my ($prefix, $start_number)=get_prefix_and_nodenumber($vlan_id, $net, $netmask);
### get node ip and vlan hostname from the hosts table.
#If it is not defined, put the default into the host table
my @allnodes=(@anodes, @vmnodes);
if (!add_vlan_ip_host($net, $netmask, $prefix, $start_number+1, \@allnodes, $callback)) { return 1;}
### for vm nodes, add an additional nic on the vm.nics
if (@vmnodes > 0) {
my %setupnics=();
my $new_nic="vl$vlan_id";
foreach my $node (@vmnodes) {
my $ent=$vmtmphash->{$node}->[0];
my $nics;
if (ref($ent) and defined $ent->{nics}) {
$nics=$ent->{nics};
my @a=split(",", $nics);
if (!grep(/^$new_nic$/, @a)) { $nics="$nics,$new_nic"; }
} else {
$nics=$new_nic;
}
$setupnics{$node}={nics=>"$nics"};
}
$vmtab->setNodesAttribs(\%setupnics);
}
### populate the /etc/hosts and make the DNS server on the mn aware this change
$::CALLBACK = $callback;
my $res = xCAT::Utils->runxcmd({
command => ['makehosts'],
}, $sub_req, 0, 1);
my $rsp = {};
$rsp->{data}->[0] = "Running makehosts...";
$callback->($rsp);
if ($res && (@$res > 0)) {
$rsp = {};
$rsp->{data} = $res;
$callback->($rsp);
}
$callback->($rsp);
$::CALLBACK = $callback;
my $res = xCAT::Utils->runxcmd({
command => ['makedns'],
}, $sub_req, 0, 1);
my $rsp = {};
$rsp->{data}->[0] = "Running makedns...";
if ($res && (@$res > 0)) {
$callback->($rsp);
$rsp->{data} = $res;
$callback->($rsp);
}
my $cmd = "service named restart";
my $rc=system $cmd;
### now go to the nodes to configure the vlan interface
$::CALLBACK = $callback;
my $args = ["-P", "configvlan $vlan_id --keephostname"];
my $res = xCAT::Utils->runxcmd( {
command => ['updatenode'],
node => \@snodes,
arg => $args
}, $sub_req, 0, 1);
my $rsp = {};
$rsp->{data}->[0] = "Running updatenode...";
if ($res && (@$res > 0)) {
$callback->($rsp);
$rsp->{data} = $res;
$callback->($rsp);
}
### add configvlan postscripts to the postscripts table for the node
my @pnodes=(@snodes, @vmnodes);
add_postscript($callback, \@pnodes);
# done
my $rsp={};
$rsp->{data}->[0]= "The following nodes are added to the vlan $vlan_id:\n@nodes";
$callback->($rsp);
} else {
### go to the nodes to de-configure the vlan interface
if (@snodes > 0) {
my $args = ["-P", "deconfigvlan $vlan_id"];
my $res = xCAT::Utils->runxcmd( {
command => ['updatenode'],
node => \@snodes,
arg => $args
}, $sub_req, 0, 1);
my $rsp = {};
$rsp->{data}->[0] = "Running updatenode...";
if ($res && (@$res > 0)) {
$callback->($rsp);
$rsp->{data} = $res;
$callback->($rsp);
}
}
### remove ports from the vlan
my $novmhost=1;
if (!remove_ports($vlan_id, \%swinfo, $callback, $novmhost)) { return 1;}
#remove the vlan id from the switch table for standalone nodes
#cannot call this function because %swsetup contains vmhosts
#remove_vlan_from_switch_table(\%swsetup,$vlan_id);
#print "swsetup=". Dumper(%swsetup);
my $swtab1 = xCAT::Table->new('switch', -create=>1, -autocommit=>0 );
foreach my $node (@snodes) {
if (exists($swsetup{$node})) {
my %keyhash=();
my %updates=();
$keyhash{'node'} = $node;
$keyhash{'switch'}= $swsetup{$node}->{switch};
$keyhash{'port'} = $swsetup{$node}->{port};
$updates{'vlan'} = "";
if($swsetup{$node}->{vlan}) {
my @a=split(',', $swsetup{$node}->{vlan});
my @b=grep(!/^$vlan_id$/,@a);
if (@b>0) {
$updates{'vlan'}=join(',', @b);
}
}
$swtab1->setAttribs( \%keyhash,\%updates );
}
}
$swtab1->commit;
#remove the vlan from the vm.nic for vm clients
if (@vmnodes > 0) {
my %setupnics=();
my $new_nic="vl$vlan_id";
foreach my $node (@vmnodes) {
my $ent=$vmtmphash->{$node}->[0];
my $nics='';
if (ref($ent) and defined $ent->{nics}) {
$nics=$ent->{nics};
my @a=split(",", $nics);
my @b=grep(!/^$new_nic$/, @a);
if (@b>0) { $nics=join(',', @b); }
}
$setupnics{$node}={nics=>"$nics"};
}
$vmtab->setNodesAttribs(\%setupnics);
}
#remove the node's vlan hostname and the ip from the host table and /etc/hosts
my @pnodes=(@snodes, @vmnodes);
if (!remove_vlan_ip_host($net, $netmask, \@pnodes, $callback)) { return 1;}
#refresh the DNS server
my $res = xCAT::Utils->runxcmd( {
command => ['makedns'],
}, $sub_req, 0, 1);
my $rsp = {};
$rsp->{data}->[0] = "Running makedns...";
if ($res && (@$res > 0)) {
$callback->($rsp);
$rsp->{data} = $res;
$callback->($rsp);
}
my $cmd = "service named restart";
my $rc=system $cmd;
# remove configvlan postscripts from the postscripts table for the node
remove_postscript($callback, \@pnodes);
# done
my $rsp={};
$rsp->{data}->[0]= "The following nodes are removed from the vlan $vlan_id:\n@nodes";
$callback->($rsp);
}
return 0;
}
sub process_rmvlan {
my $request = shift;
my $callback = shift;
my $sub_req = shift;
my $vlan_id=0;
if (exists($request->{vlanid})) {
$vlan_id=$request->{vlanid}->[0];
}
my @anodes=();
my %swportinfo=();
my %swsetup=();
my $swtab=xCAT::Table->new("switch", -create =>0);
if ($swtab) {
my @tmp1=$swtab->getAllAttribs(('node', 'switch', 'port', 'vlan'));
if ((@tmp1) && (@tmp1 > 0)) {
foreach my $ent (@tmp1) {
my @nodes_tmp=noderange($ent->{node});
foreach my $node (@nodes_tmp) {
my $switch=$ent->{switch};
my $port=$ent->{port};
if ($ent->{vlan}) {
my @a=split(",", $ent->{vlan});
if (grep(/^$vlan_id$/,@a)) {
push(@anodes, $node);
if (exists($swportinfo{$switch})) {
my $pa=$swportinfo{$switch};
push(@$pa, $port);
} else {
$swportinfo{$switch}=[$port];
}
$swsetup{$node}->{port}=$port;
$swsetup{$node}->{switch}=$switch;
$swsetup{$node}->{vlan}=$ent->{vlan};
}
}
}
}
}
}
my $switchestab=xCAT::Table->new('switches',-create=>0);
if ($switchestab) {
my @tmp1=$switchestab->getAllAttribs(('switch'));
if ((@tmp1) && (@tmp1 > 0)) {
foreach(@tmp1) {
my @switches_tmp=noderange($_->{switch});
if (@switches_tmp==0) { push @switches_tmp, $_->{switch}; } #sometimes the switch name is not on the node list table.
foreach my $switch (@switches_tmp) {
my $ports=[];
if (exists($swportinfo{$switch})) {
$ports = $swportinfo{$switch};
}
my $swh;
if (exists($Switches{$switch})) { $swh=$Switches{$switch};}
else {
$swh=new xCAT::SwitchHandler->new($switch);
$Switches{$switch}=$swh;
}
print "switch=$switch, ports=@$ports\n";
if (@$ports > 0) {
my @ret=$swh->remove_ports_from_vlan($vlan_id, @$ports);
if ($ret[0] != 0) {
my $rsp={};
$rsp->{error}->[0]= "remove_ports_from_vlan: $ret[1]";
$callback->($rsp);
}
my @ret=$swh->remove_vlan($vlan_id);
if ($ret[0] != 0) {
my $rsp={};
$rsp->{error}->[0]= "remove_vlan: $ret[1]";
$callback->($rsp);
}
} else {
#check if the vlan exists on the switch
my @ids=$swh->get_vlan_ids();
foreach my $id (@ids) {
if ($id == $vlan_id) {
#remove it if exists
my @ret=$swh->remove_vlan($vlan_id);
if ($ret[0] != 0) {
my $rsp={};
$rsp->{error}->[0]= "remove_vlan: $ret[1]";
$callback->($rsp);
}
last;
}
}
}
}
}
}
}
### now go to the nodes to de-configure the vlan interface
my $args = ["-P", "deconfigvlan $vlan_id"];
my $res = xCAT::Utils->runxcmd( {
command => ['updatenode'],
node => \@anodes,
arg => $args
}, $sub_req, 0, 1);
my $rsp = {};
$rsp->{data}->[0] = "Running updatenode...";
if ($res && (@$res > 0)) {
$callback->($rsp);
$rsp->{data} = $res;
$callback->($rsp);
}
#remove the vlan from the networks table
my $nwtab=xCAT::Table->new("networks", -create =>1);
my $sent = $nwtab->getAttribs({vlanid=>"$vlan_id"},'net','mask');
my $net;
my $netmask;
if ($sent and ($sent->{net})) {
$net=$sent->{net};
$netmask=$sent->{mask};
}
my %key_col = (vlanid=>$vlan_id);
$nwtab->delEntries(\%key_col);
#remove the vlan from the switch table for standalone nodes and vm hosts
remove_vlan_from_switch_table(\%swsetup,$vlan_id);
#remove the vlan nic from vm.nics for the vm clients
my @vmnodes=();
my %vmsetup=();
my $vmtab=xCAT::Table->new("vm", -create =>0);
if ($vmtab) {
my @tmp1=$vmtab->getAllAttribs(('node','host', 'nics'));
if ((@tmp1) && (@tmp1 > 0)) {
foreach(@tmp1) {
my @nodes_tmp=noderange($_->{node});
my $nics=$_->{nics};
my $new_nic="vl$vlan_id";
if ($nics) {
foreach my $node (@nodes_tmp) {
my @a=split(",", $nics);
if (grep(/^$new_nic$/,@a)) {
push(@vmnodes, $node);
my @b=grep(!/^$new_nic$/,@a);
if (@b>0) {
$vmsetup{$node}={nics=>join(',', @b)};
} else {
$vmsetup{$node}={nics=>''};
}
}
}
}
}
}
if (keys(%vmsetup) > 0) {
$vmtab->setNodesAttribs(\%vmsetup);
}
}
#remove the node's vlan hostname and the ip from the host table and /etc/hosts
my @allnodes=(@anodes, @vmnodes);
if (!remove_vlan_ip_host($net, $netmask, \@allnodes, $callback)) { return 1;}
#refresh the DNS server
my $res = xCAT::Utils->runxcmd( {
command => ['makedns'],
}, $sub_req, 0, 1);
my $rsp = {};
$rsp->{data}->[0] = "Running makedns...";
if ($res && (@$res > 0)) {
$callback->($rsp);
$rsp->{data} = $res;
$callback->($rsp);
}
my $cmd = "service named restart";
my $rc=system $cmd;
### remove configvlan postscripts from the postscripts table for the node
# note: if configvlan is in xcatdefaults, it will not get removed becase
# it may affect other vlans
#remove_postscript($callback, \@allnodes); --- will not remove for multi-vlan support
}
#-------------------------------------------------------
=head3 remove_postscript
It removes configvlan postscripts from the postscripts table for the node
Note: if configvlan is in xcatdefaults, it will not get removed becase
it may affect other vlans
=cut
#-------------------------------------------------------
sub remove_postscript {
my $callback=shift;
my $anodes=shift;
my $posttab=xCAT::Table->new("postscripts", -create =>0);
if ($posttab) {
my %setup_hash;
my $postcache = $posttab->getNodesAttribs($anodes,[qw(postscripts postbootscripts)]);
foreach my $node (@$anodes) {
my $ref = $postcache->{$node}->[0];
if ($ref) {
if (exists($ref->{postbootscripts})) {
my $post=$ref->{postbootscripts};
my @old_a=split(',', $post);
my @new_a = grep (!/^configvlan$/, @old_a);
if (scalar(@new_a) != scalar(@old_a)) {
#print "newa =@new_a\n";
$setup_hash{$node}={postbootscripts=>join(',', @new_a)};
}
}
if (exists($ref->{postscripts})) {
my $post=$ref->{postscripts};
my @old_a=split(',', $post);
my @new_a = grep (!/^configvlan$/, @old_a);
if (scalar(@new_a) != scalar(@old_a)) {
$setup_hash{$node}={postscripts=>join(',', @new_a)};
}
}
}
}
if (keys(%setup_hash) > 0) {
$posttab->setNodesAttribs(\%setup_hash);
}
}
}
sub process_lsvlan {
my $request = shift;
my $callback = shift;
my $sub_req = shift;
my $vlan_id=0;
if (exists($request->{vlanid})) {
$vlan_id=$request->{vlanid}->[0];
}
my %vlans=();
#get all the vm clients if the node is a vm host
my $nwtab=xCAT::Table->new("networks", -create =>0);
if ($nwtab) {
my @tmp1=$nwtab->getAllAttribs('net', 'mask', 'vlanid');
if ((@tmp1) && (@tmp1 > 0)) {
foreach(@tmp1) {
if (exists($_->{vlanid})) {
$vlans{$_->{vlanid}}->{net}=$_->{net};
$vlans{$_->{vlanid}}->{mask}=$_->{mask};
}
}
}
}
if($vlan_id !=0 && !exists($vlans{$vlan_id})) {
my $rsp={};
$rsp->{data}->[0] = "the vlan $vlan_id is not defined for the cluster nodes.";
$rsp->{errorcode} = -1;
$callback->($rsp);
return;
}
if ($vlan_id == 0) { #just show the existing vlan ids
my $rsp={};
my $index=0;
if (keys(%vlans) > 0) {
foreach my $id (sort keys(%vlans)) {
$rsp->{data}->[$index] = "vlan $id:\n subnet " . $vlans{$id}->{net}. "\n netmask " . $vlans{$id}->{mask} . "\n";
$index++;
}
} else {
$rsp->{data}->[0] = "No vlans defined for the cluster nodes."
}
$callback->($rsp);
} else { #shows the details
#get all the nodes that are in the vlan
my $swtab=xCAT::Table->new("switch", -create =>0);
my @nodes=();
if ($swtab) {
my @tmp1=$swtab->getAllAttribs('node', 'vlan', 'interface');
if ((@tmp1) && (@tmp1 > 0)) {
foreach my $grp (@tmp1) {
my $vlan=$grp->{vlan};
my $nic="primary";
if ($grp->{interface}) { $nic=$grp->{interface};}
my @nodes_tmp=noderange($grp->{node});
if ($vlan) {
my @a=split(",",$vlan);
if (grep(/^$vlan_id$/, @a)) {
foreach my $node (@nodes_tmp) {
push(@nodes, $node);
$vlans{$vlan_id}->{node}->{$node}->{name} = $node;
$vlans{$vlan_id}->{node}->{$node}->{interface} = $nic;
}
}
}
}
}
}
#get all the vm clients if the node is a vm host
my $vmtab=xCAT::Table->new("vm", -create =>0);
my @vmnodes=();
if ($vmtab) {
my @tmp1=$vmtab->getAllAttribs('node', 'host', 'nics');
if ((@tmp1) && (@tmp1 > 0)) {
my $new_nic="vl$vlan_id";
foreach(@tmp1) {
my $host = $_->{host};
my $nics = $_->{nics};
if ($nics) {
my @a=split(",", $nics);
if (grep(/^$new_nic$/, @a)) {
my @nodes_tmp=noderange($_->{node});
foreach my $node (@nodes_tmp) {
push(@vmnodes, $node);
$vlans{$vlan_id}->{node}->{$node}->{name} = $node;
$vlans{$vlan_id}->{node}->{$node}->{vmhost} = $host;
}
}
}
}
}
}
@nodes=(@nodes, @vmnodes);
#now go to hosts table to get the host name and ip on the vlan
my $hoststab = xCAT::Table->new('hosts');
my $hostscache = $hoststab->getNodesAttribs(\@nodes,[qw(node otherinterfaces)]);
my $max=0;
my $prefix;
foreach my $node (@nodes) {
my $ref = $hostscache->{$node}->[0];
my $otherinterfaces;
if ($ref && exists($ref->{otherinterfaces})){
$otherinterfaces = $ref->{otherinterfaces};
my @itf_pairs=split(/,/, $otherinterfaces);
my @itf_pairs2=();
foreach (@itf_pairs) {
my ($name,$ip)=split(/:/, $_);
if(xCAT::NetworkUtils->ishostinsubnet($ip, $vlans{$vlan_id}->{mask}, $vlans{$vlan_id}->{net})) {
$vlans{$vlan_id}->{node}->{$node}->{ip}=$ip;
$vlans{$vlan_id}->{node}->{$node}->{vname}=$name;
}
}
}
} #foreach node
my $rsp={};
$rsp->{data}->[0]="vlan $vlan_id";
$rsp->{data}->[1]=" subnet " . $vlans{$vlan_id}->{net};
$rsp->{data}->[2]=" netmask " . $vlans{$vlan_id}->{mask} . "\n";
my $node_hash=$vlans{$vlan_id}->{node};
#print Dumper($node_hash);
if ($node_hash && keys(%$node_hash) > 0) {
$rsp->{data}->[3]=" hostname\tip address\tnode \tvm host \tinterface";
my $index=4;
foreach (sort keys(%$node_hash)) {
my $vname=$node_hash->{$_}->{vname};
if (!$vname) { $vname=" ";}
my $ip=$node_hash->{$_}->{ip};
if (!$ip) { $ip=" "; }
my $name=$node_hash->{$_}->{name};
if (!$name) { $name=" ";}
my $host=$node_hash->{$_}->{vmhost};
if (!$host) { $host=" ";}
my $nic=$node_hash->{$_}->{interface};
$rsp->{data}->[$index] = " $vname\t$ip\t$name\t$host\t$nic";
$index++;
}
}
$callback->($rsp);
}
}
sub mkvlan_usage {
my $cb=shift;
my $rsp={};
$rsp->{data}->[0]= "Usage: mkvlan -h";
$rsp->{data}->[1]= " mkvlan -v";
$rsp->{data}->[2]= " mkvlan [vlanid] -n noderange [-t net -m mask] [-p node_prefix] [-i nic]";
$cb->($rsp);
}
sub rmvlan_usage {
my $cb=shift;
my $rsp={};
$rsp->{data}->[0]= "Usage: rmvlan -h";
$rsp->{data}->[1]= " rmvlan -v";
$rsp->{data}->[2]= " rmvlan vlanid";
$cb->($rsp);
}
sub chvlanports_usage{
my $cb=shift;
my $rsp={};
$rsp->{data}->[0]= "Usage: chvlanports -h";
$rsp->{data}->[1]= " chvlanports -v";
$rsp->{data}->[2]= " chvlanports vlanid -n noderange -i nic";
$rsp->{data}->[3]= " chvlanports vlanid -n noderange -i nic -d";
$cb->($rsp);
}
sub chvlan_usage {
my $cb=shift;
my $rsp={};
$rsp->{data}->[0]= "Usage: chvlan -h";
$rsp->{data}->[1]= " chvlan -v";
$rsp->{data}->[2]= " chvlan vlanid -n noderange [-i nic]";
$rsp->{data}->[3]= " chvlan vlanid -n noderange -d";
$cb->($rsp);
}
sub lsvlan_usage {
my $cb=shift;
my $rsp={};
$rsp->{data}->[0]= "Usage: lsvlan -h";
$rsp->{data}->[1]= " lsvlan -v";
$rsp->{data}->[2]= " lsvlan";
$rsp->{data}->[3]= " lsvlan vlanid";
$cb->($rsp);
}
#-------------------------------------------------------
=head3 getNodeVlanConfData
This function is called by Postage.pm to collect all the
environmental variables for setting up a vlan for a given
node.
=cut
#-------------------------------------------------------
sub getNodeVlanConfData {
my $node=shift;
if ($node =~ /xCAT_plugin::vlan/) {
$node=shift;
}
my @scriptd=();
my $swtab = xCAT::Table->new("switch", -create => 0);
if ($swtab) {
my $tmp_switch = $swtab->getNodesAttribs([$node], ['vlan','interface'],prefetchcache=>1);
#print Dumper($tmp_switch);
if (defined($tmp_switch) && (exists($tmp_switch->{$node})) && (defined($tmp_switch->{$node}->[0]))) {
my $tmp_node_array=$tmp_switch->{$node};
my $index=0;
foreach my $tmp (@$tmp_node_array) {
if (exists($tmp->{vlan})) {
my $nic="primary";
if (exists($tmp->{interface})) { $nic=$tmp->{interface};}
my @vlanid_array = split(',', $tmp->{vlan});
foreach my $vlan (@vlanid_array) {
$index++;
push @scriptd, "VLANID_$index='" . $vlan . "'\n";
push @scriptd, "export VLANID_$index\n";
push @scriptd, "VLANNIC_$index='" . $nic . "'\n";
push @scriptd, "export VLANNIC_$index\n";
my @temp_data=getNodeVlanOtherConfData($node, $vlan, $index);
@scriptd = (@scriptd,@temp_data);
}
}
}
if ($index > 0) {
push @scriptd, "VLANMAXINDEX='" . $index . "'\n";
push @scriptd, "export VLANMAXINDEX\n";
}
} else {
my $vmtab = xCAT::Table->new("vm", -create => 0);
if ($vmtab) {
my $tmp1 = $vmtab->getNodeAttribs($node, ['nics','host'],prefetchcache=>1);
my $vlan;
my $index=0;
if (defined($tmp1) && ($tmp1) && $tmp1->{nics})
{
push @scriptd, "VMNODE='YES'\n";
push @scriptd, "export VMNODE\n";
push @scriptd, "VMNICS='" . $tmp1->{nics} . "'\n";
push @scriptd, "export VMNICS\n";
my @nics=split(',', $tmp1->{nics});
#get the vlan id and interface from the host
my $host=$tmp1->{host};
#my $host_vlan_info=get_vm_host_vlan_info($host);
my $nic_position=0;
foreach my $nic (@nics) {
$nic_position++;
if ($nic =~ /^vl([\d]+)$/) {
$vlan = $1;
$index++;
push @scriptd, "VLANID_$index='" . $vlan . "'\n";
push @scriptd, "export VLANID_$index\n";
push @scriptd, "VLAN_VMNICPOS_$index='" . $nic_position . "'\n";
push @scriptd, "export VLAN_VMNICPOS_$index\n";
#if ($host_vlan_info && (exists($host_vlan_info->{$vlan}))) {
# push @scriptd, "HOST_VLANNIC_$index='" . $host_vlan_info->{$vlan} . "'\n";
# push @scriptd, "export HOST_VLANNIC_$index\n";
#}
my @temp_data=getNodeVlanOtherConfData($node, $vlan, $index);
@scriptd = (@scriptd,@temp_data);
}
} #end foreach
}
if ($index > 0) {
push @scriptd, "VLANMAXINDEX='" . $index . "'\n";
push @scriptd, "export VLANMAXINDEX\n";
}
}
}
}
return @scriptd;
}
sub getNodeVlanOtherConfData {
my $node=shift;
if ($node =~ /xCAT_plugin::vlan/) {
$node=shift;
}
my $vlan=shift;
my $index=shift;
my @scriptd=();
my $nwtab=xCAT::Table->new("networks", -create =>0);
if ($nwtab) {
my $sent = $nwtab->getAttribs({vlanid=>"$vlan"},'net','mask');
my $subnet;
my $netmask;
if ($sent and ($sent->{net})) {
$subnet=$sent->{net};
$netmask=$sent->{mask};
}
if (($subnet) && ($netmask)) {
my $hoststab = xCAT::Table->new("hosts", -create => 0);
if ($hoststab) {
my $tmp = $hoststab->getNodeAttribs($node, ['otherinterfaces'],prefetchcache=>1);
if (defined($tmp) && ($tmp) && $tmp->{otherinterfaces})
{
my $otherinterfaces = $tmp->{otherinterfaces};
my @itf_pairs=split(/,/, $otherinterfaces);
foreach (@itf_pairs) {
my ($name,$ip)=split(/:/, $_);
if(xCAT::NetworkUtils->ishostinsubnet($ip, $netmask, $subnet)) {
if ($name =~ /^-/ ) {
$name = $node.$name;
}
push @scriptd, "VLANHOSTNAME_$index='" . $name . "'\n";
push @scriptd, "export VLANHOSTNAME_$index\n";
push @scriptd, "VLANIP_$index='" . $ip . "'\n";
push @scriptd, "export VLANIP_$index\n";
push @scriptd, "VLANSUBNET_$index='" . $subnet . "'\n";
push @scriptd, "export VLANSUBNET_$index\n";
push @scriptd, "VLANNETMASK_$index='" . $netmask . "'\n";
push @scriptd, "export VLANNETMASK_$index\n";
last;
}
}
}
}
}
}
return @scriptd;
}
#-------------------------------------------------------
=head3 get_vm_host_vlan_info
This function returns a hash pointer that has the vlan id
and the interface info for the given KVM host. A host can
support more than one vlans for a interface. For example:
{1=>"eth0", 2=>"eth0", 3=>"eth1" ...}
=cut
#-------------------------------------------------------
sub get_vm_host_vlan_info {
my $host=shift;
my $host_vlan_info={};
my $swtab = xCAT::Table->new("switch", -create => 0);
if ($swtab) {
my $tmp_switch = $swtab->getNodesAttribs([$host], ['vlan','interface'],prefetchcache=>1);
if (defined($tmp_switch) && (exists($tmp_switch->{$host}))) {
my $tmp_node_array=$tmp_switch->{$host};
foreach my $tmp (@$tmp_node_array) {
if (exists($tmp->{vlan})) {
my $vlans = $tmp->{vlan};
my $nic="primary";
if (exists($tmp->{interface})) {
$nic=$tmp->{interface};
}
foreach my $vlan (split(',',$vlans)) {
$host_vlan_info->{$vlan}=$nic;
}
}
}
}
return $host_vlan_info;
}
}