541 lines
13 KiB
Plaintext
Raw Permalink Normal View History

#!/usr/bin/perl
use strict;
use Socket;
use Data::Dumper;
use Getopt::Long;
# each 0 is four bits: 0000 0000
# thus its broken down:
# - ports 1-8 are in the first hex number
# - ports 9-16 are in the second hex number
# - ports 17-24 are in the third hex number
# - ports 25-32 are in the fourth hex number
# - ports 33-40 are in the fifth hex number
# - ports 41-48 are in the sixth hex number
# - ports 49-56 are in the seventh hex number
my @bitmap = (0x01, 0x80, 0x40, 0x20, 0x10, 0x08, 0x04, 0x02);
sub getVlanMask{
my $session = shift;
my $vlan = shift;
my @xs;
my @hs;
my $oid = ".1.3.6.1.2.1.17.7.1.4.3.1.4.$vlan";
my $hs = $session->get($oid);
if($session->{ErrorNum}){
die "Couldn't get OID!" . $session->{ErrorStr} . "\n";
}
$hs =~ s/\"//g; # get rid of quotes!
@hs = split(" ",$hs);
#foreach(@hs){ print "$_\n"; }
@xs = map(hex, @hs);
#foreach (@xs){
# print $_ . "\n";
#}
unless(scalar(@xs) eq 7){
print "Could not get vlan mask for $vlan\n";
}
#print "Switch mask:\n";
#printf("%02x %02x %02x %02x %02x %02x %02x\n", @xs);
return(@xs);
}
# add port 1
# returns a 7 digit hex string to logically or with existing
# to add this port to the subnet
sub getPortMask {
my $port = shift;
my @portMask;
my $hex;
my $xIndex;
if($port < 9){ $xIndex = 0;
}elsif($port < 17){ $xIndex = 1;
}elsif($port < 25){ $xIndex = 2;
}elsif($port < 33){ $xIndex = 3;
}elsif($port < 41){ $xIndex = 4;
}elsif($port < 49){ $xIndex = 5;
}elsif($port < 57){ $xIndex = 6;
}else{
print "I don't know how to handle this port...\n";
exit;
}
$port = $port % 8;
#print "$port\n";
$hex = $bitmap[$port];
#printf("%02x\n", $hex);
for(0 .. 6){
if($_ eq $xIndex){
push @portMask, $hex;
}else{
push @portMask, hex(0);
}
}
#print "Port mask:\n";
#printf("%02x %02x %02x %02x %02x %02x %02x\n", @portMask);
return @portMask;
}
sub getSwitchInfo{
my $switch = shift;
my ($version,$community,@junk);
# bunch of xCAT tables stuff here to connect to switch
my $info = `tabdump switches | grep smc001 | sed 's/"//g'`;
if($?){
# if not in here, we try the defaults.
$version = 1;
$community = "public";
}else{
(undef,$version,undef,$community,@junk) = split(/,/,$info);
}
if($::DEBUG){
print "switch parameters for $switch:\n";
print "\tSNMP Version: $version\n";
print "\tSNMP Community: $community\n";
}
return($community, $version);
}
sub connectToSwitch{
my $switch = shift;
my $session;
my ($community, $snmpver) = getSwitchInfo($switch);
$session = new SNMP::Session(
DestHost => $switch,
Version => $snmpver,
Community => $community,
UseSprintValue => 1,
);
unless($session) {
#ERROR:
print "Failed to communicate with $switch\n";
print $SNMP::Session::ErrorStr . "\n";
#xCAT::MsgUtils->message("S","Failed to communicate with $switch");
}
return $session;
}
sub xorMasks{
my $sm = shift;
my $pm = shift;
my @nm;
foreach(0 .. 6){
my $foo = @$sm[$_] ^ @$pm[$_];
##printf("%02x\n", $foo);
$nm[$_] = $foo;
}
return @nm;
}
sub orMasks{
my $sm = shift;
my $pm = shift;
my @nm;
foreach(0 .. 6){
my $foo = @$sm[$_] | @$pm[$_];
##printf("%02x\n", $foo);
$nm[$_] = $foo;
}
return @nm;
}
sub andMasks{
my $sm = shift;
my $pm = shift;
my @nm;
foreach(0 .. 6){
my $foo = @$sm[$_] & @$pm[$_];
##printf("%02x\n", $foo);
$nm[$_] = $foo;
}
return @nm;
}
# KLUDGE function because I can't figure out how to do this with
# SNMP.pm and I give up after a week of trying.
sub snmpset {
my $sess = shift;
my $oid = shift;
my $type = shift;
my $val = shift;
my $snmpset = "/usr/bin/snmpset";
my ($cmd, $comm, $vers, $switch);
unless(-r "/usr/bin/snmpset"){
print "/usr/bin/snmpset command not found! Please install net-snmp-utils\n";
exit 1;
}
$comm = $sess->{Community};
$vers = $sess->{Version};
$switch = $sess->{DestHost};
$cmd = "$snmpset -c $comm -v $vers $switch $oid $type $val";
#print "$cmd\n";
system("$cmd >/dev/null");
#print "ec: $?\n";
if($? > 0){
print "Failed to execute command $cmd\n";
exit 1;
}
return $?;
}
sub addNodeToVlan{
# to run: switchport allowed vlan add $port (use bitmap)
# snmpset .. $oid1.$vlan x 00 00 00 00 00 00 00
my $oid1 = '.1.3.6.1.2.1.17.7.1.4.3.1.4';
# to run: switchport native vlan $vlan
# snmpset .. $oid2.$port u $vlan
my $oid2 = '.1.3.6.1.2.1.17.7.1.4.5.1.1';
my $session = shift;
my $vlan = shift;
my $port = shift;
#$oid1 .= ".$vlan";
#print "$oid1 \n";
my @xs; # netmask for current switch
my @pm; # netmask for port
my @jm; # the joined netmask
@xs = getVlanMask($session, $vlan);
@pm = getPortMask($port);
@jm = orMasks(\@xs, \@pm);
#print "Join mask:\n";
printf("%02x %02x %02x %02x %02x %02x %02x\n", @jm) if $::DEBUG;
my $mask = sprintf("%02x %02x %02x %02x %02x %02x %02x ", @jm);
#################################
# PART 1: add the switchport allowed capability
#################################
# TODO: This part I can't get working so I'm just going to do an
# snmpset command here instead.
#my $v1 = new SNMP::Varbind([$oid1,$vlan,$mask, 'OCTETSTR']);
#print Dumper($v1);
#$session->set($v1);
#if($session->{ErrorStr}) {
# print "Error! " . $session->{ErrorStr} . "\n";
#}
snmpset($session, "$oid1.$vlan", "x", "\'$mask\'");
#######################################
# PART 2: add the switchport native
#######################################
# first get the current one for part 3:
my $currNativeVlan = $session->get("$oid2.$port");
if($session->{ErrorNum}){
die "Couldn't get OID!" . $session->{ErrorStr} . "\n";
}
print "currNativeVLAN: $currNativeVlan\n" if $::DEBUG;
print "$currNativeVlan -> ";
# set the new switchport native vlan
my $v2 = new SNMP::Varbind([$oid2,$port,$vlan,'GAUGE32']);
#print Dumper($v2);
$session->set($v2);
if($session->{ErrorStr}) {
print "Error! " . $session->{ErrorStr} . "\n";
}
#######################################
# PART 3: take it off the other one it is on
#######################################
delPortFromVlan($session, $port, $currNativeVlan);
}
# returns the vlan number
sub getVlans{
my $session = shift;
my %vlans;
# IF-MIB:ifName: .1.3.6.1.2.1.31.1.1.1.1
my $ifName = '.1.3.6.1.2.1.31.1.1.1.1';
my $varbind = new SNMP::Varbind([$ifName, '']);
$session->getnext($varbind);
if($session->{ErrorStr}) {
print "Error! " . $session->{ErrorStr} . "\n";
}
# varbind: name: ifName, 1, Port1, OCTETSTR
while($varbind->[2]){
#print $varbind->[2] . "\n";
my $name = $varbind->[2];
if($name =~ /VLAN/){
#print "Found $name on " . $session->{DestHost} . "\n";
$name =~ s/VLAN//g;
# we subtract 1000 off the name to give us the actual VLAN.
$vlans{$name} = $varbind->[1] - 1000;
#foreach(@$varbind){
# print "\t$_\n";
#}
}
$session->getnext($varbind);
}
#print Dumper(%vlans);
#foreach(keys %vlans){
# print "VLAN: $_ has value: ". $vlans{$_} ."\n";
#}
return \%vlans;
}
# return 1 if port is on vlan return 0 if not on vlan
sub checkPortVlan{
my $rc = 0;
my $session = shift;
my $p = shift;
my $v = shift;
my @xs; # netmask for current switch
my @pm; # netmask for port
my @jm; # the joined netmask
@xs = getVlanMask($session, $v);
@pm = getPortMask($p);
@jm = andMasks(\@xs, \@pm);
#print "And Mask:\n";
#printf("%02x %02x %02x %02x %02x %02x %02x\n", @jm);
my $m = sprintf("%x%x%x%x%x%x%x", @jm);
#print "m: $m\n";
if($m > 0){
#print "port $p is on VLAN $v on switch " . $session->{DestHost} . "\n";
$rc = 1;
}
return $rc;
}
sub delPortFromVlan {
my $session = shift;
my $port = shift;
my $vlan = shift;
print "removing port $port from vlan $vlan\n" if $::DEBUG;
# first check and see if its on there:
unless(checkPortVlan($session, $port, $vlan)){
print "Port $port is not on VLAN $vlan\n";
}
my @vm = getVlanMask($session, $vlan);
my @pm = getPortMask($port);
my @xm = xorMasks(\@vm, \@pm);
#print "portmask:\n";
#printf("%02x %02x %02x %02x %02x %02x %02x\n", @pm);
#print "vlanmask:\n";
#printf("%02x %02x %02x %02x %02x %02x %02x\n", @vm);
#print "xormask:\n";
#printf("%02x %02x %02x %02x %02x %02x %02x\n", @xm);
# this is the untagged remove
my $oid1 = '.1.3.6.1.2.1.17.7.1.4.3.1.4';
my $mask = sprintf("%02x %02x %02x %02x %02x %02x %02x ", @xm);
snmpset($session, "$oid1.$vlan", "x", "\'$mask\'");
# this is the tagged remove
my $oid2 = '.1.3.6.1.2.1.17.7.1.4.3.1.2';
#my $mask = sprintf("%02x %02x %02x %02x %02x %02x %02x ", @xm);
#print "tagged mask: $mask\n";
snmpset($session, "$oid2.$vlan", "x", "\'$mask\'");
}
sub displayHelp{
my $ec = shift;
print "nodesw changes the vlan of a node to a specified vlan\n";
print "requires xCAT 2.0, Switch configured with SNMP sets, and only tested on SMC8648T\n";
print "nodesw -h|--help\n";
print "nodesw [-v] <noderange> vlan <vlan number>\n";
print "nodesw [-v] <noderange> show\n\n";
print "Author: Vallard Benincosa\n";
exit $ec;
}
sub getNodeRange{
my $nr = shift;
my @nr = `/opt/xcat/bin/nodels $nr switch.switch switch.port`;
my $nh;
chomp(@nr);
if($?){
print $nr[0];
exit 1;
}
foreach(@nr){
my($n,$char,$val) = split(/:/, $_);
$char = (split(/\./,$char))[1];
$val =~ s/ //g;
$nh->{$n}{$char} = $val;
}
if($::DEBUG){
foreach(keys %$nh){
print $_ .":";
print " switch:" . $nh->{$_}{'switch'} ;
print " port:" . $nh->{$_}{'port'};
print "\n";
}
}
# make sure all fields are defined
my $e = 0;
foreach my $node (keys %$nh){
unless($nh->{$node}{'switch'}){
print "$node does not have a defined switch in xCAT! (nodels $node switch.switch)\n";
$e++;
}
unless($nh->{$node}{'port'}){
print "$node does not have a defined port in xCAT! (nodels $node switch.port)\n";
$e++;
}
}
if($e > 0){
exit 1;
}
# return the node hash
return $nh;
}
sub show{
my $oid2 = '.1.3.6.1.2.1.17.7.1.4.5.1.1';
my $nh = shift;
foreach my $node (keys %$nh){
my $port = $nh->{$node}{'port'};
my $switch = $nh->{$node}{'switch'};
my $session = connectToSwitch($switch);
my $currNativeVlan = $session->get("$oid2.$port");
if($session->{ErrorNum}){
die "Couldn't get OID!" . $session->{ErrorStr} . "\n";
}
print "$node: $currNativeVlan\n";
}
}
##### commands:
# get VLANS
# check VLANs that contain port X
# we want to put this port on this new vlan, here is how we do it:
# connect to switch to see:
my $help =0;
$::DEBUG = 0;
GetOptions(
'h|help' => \$help,
'v|verbose' => \$::DEBUG
);
if($help){
displayHelp(0);
}
require SNMP;
$SNMP::debugging = 1;
$SNMP::verbose = 1;
$SNMP::best_guess = 1;
if($::DEBUG){
print "verbose is set to on!\n";
}
my $nr = "";
my $nodeRange = shift;
my $cmd = shift;
my $vlan = shift;
unless ($nodeRange){
print "missing noderange!\n\n";
displayHelp(1);
}
unless($cmd) {
print "missing operation! [show | vlan ]\n\n";
displayHelp(1);
}
if($cmd eq 'vlan'){
unless($vlan){
print "missing vlan number!\n\n";
displayHelp(1);
}
$nr = getNodeRange($nodeRange);
chVlan($nr, $vlan);
}elsif($cmd eq 'show'){
print "showing $nodeRange vlan settings\n" if $::DEBUG;
$nr = getNodeRange($nodeRange);
show($nr);
}else{
print "unrecognized operation requested: $cmd\n\n";
displayHelp(1);
}
################################################################################
# getVlans
# find all vlans of a switch
################################################################################
#my $vlans = getVlans($session);
################################################################################
# checkPort Vlan
################################################################################
#checkPortVlan($session, $port, $vlans->{$_};
################################################################################
# addPortToVLAN
################################################################################
# get all VLANs
sub chVlan{
my $nh = shift;
my $currSwitch = '';
my ($session, $vlans, $exists);
foreach my $node (keys %$nh){
my $port = $nh->{$node}{'port'};
my $switch = $nh->{$node}{'switch'};
unless($switch eq $currSwitch){
$session = connectToSwitch($switch);
$vlans = getVlans($session);
$exists = 0;
$currSwitch = $switch;
}
foreach(keys %$vlans){
# see if the requested VLAN actually exists
if( $_ eq $vlan){
print "VLAN $_ exists on $switch\n" if $::DEBUG;
# if it exists see if its already on it
$exists = 1;
if(checkPortVlan($session, $port, $vlans->{$_})){
print "$node: port $port already exists on $switch VLAN $vlan\n";
exit 1;
}else{
print "Adding Port: $port to VLAN: $vlan\n" if $::DEBUG;
print $node . ": ";
if(addNodeToVlan($session, $vlan, $port) eq 0){
print "Added port: $port to VLAN: $vlan\n" if $::DEBUG;
print "$vlan\n";
}
}
}
}
unless($exists){
print "VLAN $vlan does not exist on " . $session->{DestHost} . "\n";
exit 1;
}
}
}