| Net-SNMP-Mixin-Dot1dStp documentation | Contained in the Net-SNMP-Mixin-Dot1dStp distribution. |
Net::SNMP::Mixin::Dot1dStp - mixin class for 802.1D spanning tree information
Version 0.03
use Net::SNMP;
use Net::SNMP::Mixin;
my $session = Net::SNMP->session( -hostname => 'foo.bar.com' );
$session->mixer('Net::SNMP::Mixin::Dot1dStp');
$session->init_mixins;
snmp_dispatcher();
die $session->errors if $session->errors;
my $stp_group = $session->get_dot1d_stp_group;
printf "TopoChanges: %d\n", $stp_group->{dot1dStpTopChanges};
printf "ThisRootPort: %d\n", $stp_group->{dot1dStpRootPort};
printf "ThisRootCost: %d\n", $stp_group->{dot1dStpRootCost};
printf "ThisStpPrio: %d\n", $stp_group->{dot1dStpPriority};
printf "RootBridgeMAC: %s\n",
$stp_group->{dot1dStpDesignatedRootAddress};
printf "RootBridgePrio: %d\n",
$stp_group->{dot1dStpDesignatedRootPriority};
my $stp_ports = $session->get_dot1d_stp_port_table;
foreach my $port ( sort { $a <=> $b } keys %$stp_ports ) {
my $enabled = $stp_ports->{$port}{dot1dStpPortEnable};
next unless defined $enabled && $enabled == 1;
printf "----------- STP Port: %d ---------\n", $port;
printf "PState: %d\n", $stp_ports->{$port}{dot1dStpPortState};
printf "PStateStr: %d\n",
$stp_ports->{$port}{dot1dStpPortStateString};
printf "PPrio: %d\n",
$stp_ports->{$port}{dot1dStpPortPriority};
printf "PCost: %d\n",
$stp_ports->{$port}{dot1dStpPortPathCost};
printf "PDesigCost: %d\n",
$stp_ports->{$port}{dot1dStpPortDesignatedCost};
printf "DBridgePrio: %d\n",
$stp_ports->{$port}{dot1dStpPortDesignatedBridgePriority};
printf "DBridgeMAC: %d\n",
$stp_ports->{$port}{dot1dStpPortDesignatedBridgeAddress};
printf "DPortPrio: %d\n",
$stp_ports->{$port}{dot1dStpPortDesignatedPortPriority};
printf "DPortNr: %d\n",
$stp_ports->{$port}{dot1dStpPortDesignatedPortNumber};
}
This mixin reads data from the dot1dStp group out of the BRIDGE-MIB. Normally it's implemented by those bridges that support the Spanning Tree Protocol.
Returns the dot1dStp group as a hash reference:
{
dot1dStpProtocolSpecification => INTEGER,
dot1dStpPriority => INTEGER,
dot1dStpTimeSinceTopologyChange => TIME_TICKS,
dot1dStpTopChanges => COUNTER,
dot1dStpRootCost => INTEGER,
dot1dStpRootPort => INTEGER,
dot1dStpMaxAge => TIMEOUT,
dot1dStpHelloTime => TIMEOUT,
dot1dStpHoldTime => INTEGER,
dot1dStpForwardDelay => TIMEOUT,
dot1dStpBridgeMaxAge => TIMEOUT,
dot1dStpBridgeHelloTime => TIMEOUT,
dot1dStpBridgeForwardDelay => TIMEOUT,
dot1dStpDesignatedRoot => BridgeId,
dot1dStpDesignatedRootPriority => INTEGER,
dot1dStpDesignatedRootAddress => MacAddress,
}
The dot1dStpDesignatedRoot is a BridgeId struct of priority and MacAddress. The mixin method splits this already into dot1dStpDesignatedRootPriority and dot1dStpDesignatedRootAddress for your convenience.
Returns the dot1dStpPortTable as a hash reference. The keys are the dot1d STP port numbers for which this entry contains Spanning Tree Protocol management information:
{
INTEGER => { # dot1dStpPort
dot1dStpPortPriority => INTEGER,
dot1dStpPortState => INTEGER,
dot1dStpPortStateString => String,
dot1dStpPortEnable => INTEGER,
dot1dStpPortPathCost => INTEGER,
dot1dStpPortDesignatedRootId => BridgeId,
dot1dStpPortDesignatedCost => INTEGER,
dot1dStpPortDesignatedBridgeId => BridgeId,
dot1dStpPortDesignatedPort => PortId,
dot1dStpPortForwardTransitions => COUNTER,
# dot1dStpPortDesignatedRootId is a struct (BridgeId) of
# priority and MacAddress
#
dot1dStpPortDesignatedRootPriority => INTEGER,
dot1dStpPortDesignatedRootAddress => MacAddress,
# dot1dStpPortDesignatedBridgeId is a struct (BridgeId) of
# priority and MacAddress
#
dot1dStpPortDesignatedBridgePriority => INTEGER,
dot1dStpPortDesignatedBridgeAddress => MacAddress,
# dot1dStpPortDesignatedPort is a struct (PortId) of
# priority and bridge port number
#
dot1dStpPortDesignatedPortPriority => INTEGER,
dot1dStpPortDesignatedPortNumber => INTEGER,
},
... ,
}
The structs BridgeId and PortId are already splitted by this mixin method into the relevant values for your convenience.
The dot1dStpPort has the same value as the dot1dBasePort and isn't necessarily the ifIndex of the switch.
See also the Net::SNMP::Mixin::Dot1dBase for a mixin to get the mapping between the ifIndexes and the dot1dBasePorts if needed.
Fetch the dot1dSTP related snmp values from the host. Don't call this method direct!
Fetch the local system data from the dot1dStp tree once during object initialization.
The callback for _fetch_dot1d_stp_group.
Fetch the dot1dStpPortTable once during object initialization.
The callback for _fetch_dot1d_stp_port_tbl().
Split a bridge id in priority and MAC address. Returns a list of (bridgePrio, bridgeMac).
Split a bridge port id in priority and bridge port number. Returns a list of (portPrio, portNumber).
There are no known bugs at the time of this release. However, if you spot a bug or are experiencing difficulties that are not explained within the POD documentation, please submit a bug to the RT system (see link below). However, it would help greatly if you are able to pinpoint problems or even supply a patch.
Fixes are dependant upon their severity and my availablity. Should a fix not be forthcoming, please feel free to (politely) remind me by sending an email to gaissmai@cpan.org .
RT: http://rt.cpan.org/Public/Dist/Display.html?Name=Net-SNMP-Mixin-Dot1dStp
Karl Gaissmaier <karl.gaissmaier at uni-ulm.de>
Copyright 2008 Karl Gaissmaier, all rights reserved.
This program is free software; you can redistribute it and/or modify it under the same terms as Perl itself.
| Net-SNMP-Mixin-Dot1dStp documentation | Contained in the Net-SNMP-Mixin-Dot1dStp distribution. |
package Net::SNMP::Mixin::Dot1dStp; use strict; use warnings; # # store this package name in a handy variable, # used for unambiguous prefix of mixin attributes # storage in object hash # my $prefix = __PACKAGE__; # # this module import config # use Carp (); use Net::SNMP::Mixin::Util qw/idx2val hex2octet normalize_mac push_error/; # # this module export config # my @mixin_methods; BEGIN { @mixin_methods = ( qw/ get_dot1d_stp_group get_dot1d_stp_port_table / ); } use Sub::Exporter -setup => { exports => [@mixin_methods], groups => { default => [@mixin_methods], }, }; # # SNMP oid constants used in this module # use constant { DOT1D_STP_GROUP => '1.3.6.1.2.1.17.2', DOT1D_STP_PROTO => '1.3.6.1.2.1.17.2.1.0', DOT1D_STP_PRIO => '1.3.6.1.2.1.17.2.2.0', DOT1D_STP_TIME_SINCE_TOPO_CHANGE => '1.3.6.1.2.1.17.2.3.0', DOT1D_STP_TOPO_CHANGES => '1.3.6.1.2.1.17.2.4.0', DOT1D_STP_DESIGNATED_ROOT => '1.3.6.1.2.1.17.2.5.0', DOT1D_STP_ROOT_COST => '1.3.6.1.2.1.17.2.6.0', DOT1D_STP_ROOT_PORT => '1.3.6.1.2.1.17.2.7.0', DOT1D_STP_MAX_AGE => '1.3.6.1.2.1.17.2.8.0', DOT1D_STP_HELLO_TIME => '1.3.6.1.2.1.17.2.9.0', DOT1D_STP_HOLD_TIME => '1.3.6.1.2.1.17.2.10.0', DOT1D_STP_FWD_DELAY => '1.3.6.1.2.1.17.2.11.0', DOT1D_STP_BRIDGE_MAX_AGE => '1.3.6.1.2.1.17.2.12.0', DOT1D_STP_BRIDGE_HELLO_TIME => '1.3.6.1.2.1.17.2.13.0', DOT1D_STP_BRIDGE_FWD_DELAY => '1.3.6.1.2.1.17.2.14.0', DOT1D_STP_PORT_TABLE => '1.3.6.1.2.1.17.2.15', DOT1D_STP_PORT_PRIO => '1.3.6.1.2.1.17.2.15.1.2', DOT1D_STP_PORT_STATE => '1.3.6.1.2.1.17.2.15.1.3', DOT1D_STP_PORT_ENABLE => '1.3.6.1.2.1.17.2.15.1.4', DOT1D_STP_PORT_PATH_COST => '1.3.6.1.2.1.17.2.15.1.5', DOT1D_STP_PORT_DESIGNATED_ROOT => '1.3.6.1.2.1.17.2.15.1.6', DOT1D_STP_PORT_DESIGNATED_COST => '1.3.6.1.2.1.17.2.15.1.7', DOT1D_STP_PORT_DESIGNATED_BRIDGE => '1.3.6.1.2.1.17.2.15.1.8', DOT1D_STP_PORT_DESIGNATED_PORT => '1.3.6.1.2.1.17.2.15.1.9', DOT1D_STP_PORT_FORWARD_TRANSITIONS => '1.3.6.1.2.1.17.2.15.1.10', };
our $VERSION = '0.03';
sub get_dot1d_stp_group { my $session = shift; Carp::croak "'$prefix' not initialized," unless $session->{$prefix}{__initialized}; # just a shallow copy for shallow values my $result = { %{ $session->{$prefix}{dot1dStpGroup} } }; # split BridgeId in priority and address my ( $root_bridge_prio, $root_bridge_address ) = _unpack_bridge_id( $result->{dot1dStpDesignatedRoot} ); $result->{dot1dStpDesignatedRootPriority} = $root_bridge_prio; $result->{dot1dStpDesignatedRootAddress} = $root_bridge_address; return $result; }
sub get_dot1d_stp_port_table { my $session = shift; Carp::croak "'$prefix' not initialized," unless $session->{$prefix}{__initialized}; # stash for return values my $result = {}; # # the port's current state translation table # my %port_state_enum = ( 1 => 'disabled', 2 => 'blocking', 3 => 'listening', 4 => 'learning', 5 => 'forwarding', 6 => 'broken', ); # the MIB tables are stored in {column}{row}{value} order # but we return {row}{column}{value} # # grab all rows from one random choosen column my @rows = keys %{ $session->{$prefix}{dot1dStpPortTbl}{dot1dStpPortPriority} }; foreach my $row (@rows) { # loop over all columns foreach my $column ( keys %{ $session->{$prefix}{dot1dStpPortTbl} } ) { # rebuild in reverse order: result(row,column) = stash(column,row) # side effect: make a shallow copy for shallow values $result->{$row}{$column} = $session->{$prefix}{dot1dStpPortTbl}{$column}{$row}; } # # additonal calculated values from the structs # # # resolve enum # $result->{$row}{dot1dStpPortStateString} = $port_state_enum{ $result->{$row}{dot1dStpPortState} }; my ( $bridge_prio, $bridge_address); # # split dot1dStpPortDesignatedRoot # ( $bridge_prio, $bridge_address ) = _unpack_bridge_id( $result->{$row}{dot1dStpPortDesignatedRoot} ); $result->{$row}{dot1dStpPortDesignatedRootPriority} = $bridge_prio; $result->{$row}{dot1dStpPortDesignatedRootAddress} = $bridge_address; # # split dot1dStpPortDesignatedBridge # ( $bridge_prio, $bridge_address ) = _unpack_bridge_id( $result->{$row}{dot1dStpPortDesignatedBridge} ); $result->{$row}{dot1dStpPortDesignatedBridgePriority} = $bridge_prio; $result->{$row}{dot1dStpPortDesignatedBridgeAddress} = $bridge_address; my ( $portPrio, $portNumber ); # # split dot1dStpPortDesignatedPort # ( $portPrio, $portNumber ) = _unpack_bridge_port_id( $result->{$row}{dot1dStpPortDesignatedPort} ); $result->{$row}{dot1dStpPortDesignatedPortPriority} = $portPrio; $result->{$row}{dot1dStpPortDesignatedPortNumber} = $portNumber; } return $result; }
sub _init { my ( $session, $reload ) = @_; die "$prefix already initalized and reload not forced.\n" if $session->{$prefix}{__initialized} && not $reload; # populate the object with needed mib values # # initialize the object for STP infos _fetch_dot1d_stp_group($session); return if $session->error; _fetch_dot1d_stp_port_tbl($session); return if $session->error; return 1; }
sub _fetch_dot1d_stp_group { my $session = shift; my $result; $result = $session->get_entries( -columns => [ DOT1D_STP_GROUP, ], -endindex => '14.0', # define callback if in nonblocking mode $session->nonblocking ? ( -callback => \&_dot1d_stp_group_cb ) : (), ); unless (defined $result) { # Net::SNMP looses sometimes error messages in nonblocking # mode, so we save them in an extra buffer my $err_msg = $session->error; push_error($session, "$prefix: $err_msg") if $err_msg; return; } # in nonblocking mode the callback will be called asynchronously return 1 if $session->nonblocking; # ok we are in synchronous mode, call the result mangling function # by hand _dot1d_stp_group_cb($session); }
sub _dot1d_stp_group_cb { my $session = shift; my $vbl = $session->var_bind_list; unless (defined $vbl) { # Net::SNMP looses sometimes error messages in nonblocking # mode, so we save them in an extra buffer my $err_msg = $session->error; push_error($session, "$prefix: $err_msg") if $err_msg; return; } unless ( defined $vbl->{ DOT1D_STP_PROTO() } ) { my $err_msg = "No Spanning Tree Protocol running"; push_error( $session, "$prefix: $err_msg" ) if $err_msg; return; } my $stash_ptr = $session->{$prefix}{dot1dStpGroup} = {}; $stash_ptr->{dot1dStpProtocolSpecification} = $vbl->{ DOT1D_STP_PROTO() }; $stash_ptr->{dot1dStpPriority} = $vbl->{ DOT1D_STP_PRIO() }; $stash_ptr->{dot1dStpTimeSinceTopologyChange} = $vbl->{ DOT1D_STP_TIME_SINCE_TOPO_CHANGE() }; $stash_ptr->{dot1dStpTopChanges} = $vbl->{ DOT1D_STP_TOPO_CHANGES() }; $stash_ptr->{dot1dStpDesignatedRoot} = $vbl->{ DOT1D_STP_DESIGNATED_ROOT() }; $stash_ptr->{dot1dStpRootCost} = $vbl->{ DOT1D_STP_ROOT_COST() }; $stash_ptr->{dot1dStpRootPort} = $vbl->{ DOT1D_STP_ROOT_PORT() }; $stash_ptr->{dot1dStpMaxAge} = $vbl->{ DOT1D_STP_MAX_AGE() }; $stash_ptr->{dot1dStpHelloTime} = $vbl->{ DOT1D_STP_HELLO_TIME() }; $stash_ptr->{dot1dStpHoldTime} = $vbl->{ DOT1D_STP_HOLD_TIME() }; $stash_ptr->{dot1dStpForwardDelay} = $vbl->{ DOT1D_STP_FWD_DELAY() }; $stash_ptr->{dot1dStpBridgeMaxAge} = $vbl->{ DOT1D_STP_BRIDGE_MAX_AGE() }; $stash_ptr->{dot1dStpBridgeHelloTime} = $vbl->{ DOT1D_STP_BRIDGE_HELLO_TIME() }; $stash_ptr->{dot1dStpBridgeForwardDelay} = $vbl->{ DOT1D_STP_BRIDGE_FWD_DELAY() }; $session->{$prefix}{__initialized}++; }
sub _fetch_dot1d_stp_port_tbl { my $session = shift; my $result; # fetch the dot1dStpPortTable $result = $session->get_entries( -columns => [ DOT1D_STP_PORT_PRIO, DOT1D_STP_PORT_STATE, DOT1D_STP_PORT_ENABLE, DOT1D_STP_PORT_PATH_COST, DOT1D_STP_PORT_DESIGNATED_ROOT, DOT1D_STP_PORT_DESIGNATED_COST, DOT1D_STP_PORT_DESIGNATED_BRIDGE, DOT1D_STP_PORT_DESIGNATED_PORT, DOT1D_STP_PORT_FORWARD_TRANSITIONS, ], # define callback if in nonblocking mode $session->nonblocking ? ( -callback => \&_dot1d_stp_port_tbl_cb ) : (), ); unless (defined $result) { # Net::SNMP looses sometimes error messages in nonblocking # mode, so we save them in an extra buffer my $err_msg = $session->error; push_error($session, "$prefix: $err_msg") if $err_msg; return; } # in nonblocking mode the callback will be called asynchronously return 1 if $session->nonblocking; # ok we are in synchronous mode, call the result mangling function # by hand _dot1d_stp_port_tbl_cb($session); }
sub _dot1d_stp_port_tbl_cb { my $session = shift; my $vbl = $session->var_bind_list; unless (defined $vbl) { # Net::SNMP looses sometimes error messages in nonblocking # mode, so we save them in an extra buffer my $err_msg = $session->error; push_error($session, "$prefix: $err_msg") if $err_msg; return; } my $stash_ptr = $session->{$prefix}{dot1dStpPortTbl} = {}; $stash_ptr->{dot1dStpPortPriority} = idx2val( $vbl, DOT1D_STP_PORT_PRIO ); $stash_ptr->{dot1dStpPortState} = idx2val( $vbl, DOT1D_STP_PORT_STATE ); $stash_ptr->{dot1dStpPortEnable} = idx2val( $vbl, DOT1D_STP_PORT_ENABLE ); $stash_ptr->{dot1dStpPortPathCost} = idx2val( $vbl, DOT1D_STP_PORT_PATH_COST ); $stash_ptr->{dot1dStpPortDesignatedRoot} = idx2val( $vbl, DOT1D_STP_PORT_DESIGNATED_ROOT ); $stash_ptr->{dot1dStpPortDesignatedCost} = idx2val( $vbl, DOT1D_STP_PORT_DESIGNATED_COST ); $stash_ptr->{dot1dStpPortDesignatedBridge} = idx2val( $vbl, DOT1D_STP_PORT_DESIGNATED_BRIDGE ); $stash_ptr->{dot1dStpPortDesignatedPort} = idx2val( $vbl, DOT1D_STP_PORT_DESIGNATED_PORT ); $stash_ptr->{dot1dStpPortForwardTransitions} = idx2val( $vbl, DOT1D_STP_PORT_FORWARD_TRANSITIONS ); $session->{$prefix}{__initialized}++; }
sub _unpack_bridge_id { my $bridgeId = shift; return unless $bridgeId; # convert to untranslated OCTET_STRING in case it's # already translated by Net::SNMP $bridgeId = hex2octet($bridgeId); # the bridgeId struct is the concatenation of: # dot1dStpPriority and dot1dStpBridgeMAC # # unpack the struct my ( $bridgePrio, $bridgeMac ) = unpack 'nH*', $bridgeId; # convert to a normalized adress format $bridgeMac = normalize_mac($bridgeMac); return ( $bridgePrio, $bridgeMac ); }
sub _unpack_bridge_port_id { my $portId = shift; return unless $portId; # convert to untranslated OCTET_STRING in case it's # already translated by Net::SNMP $portId = hex2octet($portId); # the portId is the concatenation of: # portPriority(4bit) and dot1dBasePort(12bit) # my $portPrio = ( unpack 'n', $portId ) >> 12; my $dot1dBasePort = ( unpack 'n', $portId ) & 0x0FFF; # priority <0-15> (default: 8 ) - The range of 0-240 is # divided into 16 steps. These steps are numbered from # 0 to 15. It is multiplied by 16 to calculate the # priority value used by the STP protocol. $portPrio *= 16; return ( $portPrio, $dot1dBasePort ); }
unless ( caller() ) { print "$prefix compiles and initializes successful.\n"; } 1; # vim: sw=2