#!/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;
		}
	}
}