415 lines
		
	
	
		
			12 KiB
		
	
	
	
		
			Perl
		
	
	
	
	
	
			
		
		
	
	
			415 lines
		
	
	
		
			12 KiB
		
	
	
	
		
			Perl
		
	
	
	
	
	
| # SNMP::Info::CiscoStpExtensions
 | |
| #
 | |
| # Copyright (c)2009 Carlos Vicente
 | |
| # All rights reserved.
 | |
| #
 | |
| # Redistribution and use in source and binary forms, with or without
 | |
| # modification, are permitted provided that the following conditions are met:
 | |
| #
 | |
| #     * Redistributions of source code must retain the above copyright notice,
 | |
| #       this list of conditions and the following disclaimer.
 | |
| #     * Redistributions in binary form must reproduce the above copyright notice,
 | |
| #       this list of conditions and the following disclaimer in the documentation
 | |
| #       and/or other materials provided with the distribution.
 | |
| #     * Neither the name of the author nor the
 | |
| #       names of its contributors may be used to endorse or promote products
 | |
| #       derived from this software without specific prior written permission.
 | |
| #
 | |
| # THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND
 | |
| # ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
 | |
| # WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
 | |
| # DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR
 | |
| # ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
 | |
| # (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
 | |
| # LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON
 | |
| # ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
 | |
| # (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
 | |
| # SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
 | |
| 
 | |
| package SNMP::Info::CiscoStpExtensions;
 | |
| 
 | |
| use strict;
 | |
| use warnings;
 | |
| use Exporter;
 | |
| use SNMP::Info;
 | |
| use SNMP::Info::Bridge;
 | |
| 
 | |
| our ($VERSION, $DEBUG, %MIBS, %FUNCS, %GLOBALS, %MUNGE, %PORTSTAT, $INIT);
 | |
| 
 | |
| $VERSION = '3.73';
 | |
| 
 | |
| @SNMP::Info::CiscoStpExtensions::ISA = qw/SNMP::Info::Bridge SNMP::Info Exporter/;
 | |
| @SNMP::Info::CiscoStpExtensions::EXPORT_OK = qw//;
 | |
| 
 | |
| %MIBS    = (
 | |
|             %SNMP::Info::Bridge::MIBS,
 | |
|             'CISCO-STP-EXTENSIONS-MIB' => 'stpxSpanningTreeType',
 | |
|            );
 | |
| 
 | |
| %GLOBALS = (
 | |
|             %SNMP::Info::Bridge::GLOBALS,
 | |
|             'stpx_mst_config_digest' => 'stpxSMSTConfigDigest',
 | |
|             'stpx_mst_region_name'   => 'stpxMSTRegionName',
 | |
|             'stpx_mst_region_rev'    => 'stpxSMSTRegionRevision',
 | |
|             'stpx_stp_type'          => 'stpxSpanningTreeType',
 | |
|             'stpx_bpduguard_enable'  => 'stpxFastStartBpduGuardEnable',
 | |
|             'stpx_bpdufilter_enable' => 'stpxFastStartBpduFilterEnable',
 | |
|             'stpx_faststart_default' => 'stpxFastStartGlobalDefaultMode',
 | |
|            );
 | |
| 
 | |
| %FUNCS   = (
 | |
|             %SNMP::Info::Bridge::FUNCS,
 | |
|             'stpx_rootguard_enabled'      => 'stpxRootGuardConfigEnabled',
 | |
|             'stpx_loopguard_enabled'      => 'stpxLoopGuardConfigEnabled',
 | |
|             'stpx_faststart_enabled'      => 'stpxFastStartPortEnable',
 | |
|             'stpx_faststart_operational'  => 'stpxFastStartPortMode',
 | |
|             'stpx_port_bpduguard_mode'    => 'stpxFastStartPortBpduGuardMode',
 | |
|             'stpx_port_bpdufilter_mode'   => 'stpxFastStartPortBpduFilterMode',
 | |
|             'stpx_smst_root'              => 'stpxSMSTInstanceCISTRegionalRoot',
 | |
|             'stpx_smst_vlans_mapped_1k2k' => 'stpxSMSTInstanceVlansMapped1k2k',
 | |
|             'stpx_smst_vlans_mapped_3k4k' => 'stpxSMSTInstanceVlansMapped3k4k',
 | |
|            );
 | |
| 
 | |
| %MUNGE   = (
 | |
|             %SNMP::Info::Bridge::MUNGE,
 | |
|            'stpx_mst_config_digest'      => \&SNMP::Info::CiscoStpExtensions::oct2str,
 | |
|            );
 | |
| 
 | |
| 
 | |
| # Report version of STP via standard method
 | |
| sub stp_ver {
 | |
|      my $self = shift;
 | |
|      my $stp_ver = $self->SUPER::stp_ver();
 | |
|      if ( !defined($stp_ver) || $stp_ver eq 'unknown' ){
 | |
|          if ( defined $self->stpx_stp_type() ){
 | |
|              $stp_ver = $self->stpx_stp_type();
 | |
|          }
 | |
|      }
 | |
|      return $stp_ver;
 | |
| }
 | |
| 
 | |
| sub mst_config_digest {
 | |
|     my $self = shift;
 | |
|     return $self->stpx_mst_config_digest;
 | |
| }
 | |
| 
 | |
| sub mst_region_name {
 | |
|     my $self = shift;
 | |
|     return $self->stpx_mst_region_name;
 | |
| }
 | |
| 
 | |
| sub mst_region_rev {
 | |
|     my $self = shift;
 | |
|     return $self->stpx_mst_region_rev;
 | |
| }
 | |
| 
 | |
| 
 | |
| sub mst_vlan2instance {
 | |
|     my $self = shift;
 | |
| 
 | |
|     # Get MST vlan-to-instance mapping
 | |
|     my $m1k2k = $self->stpx_smst_vlans_mapped_1k2k;
 | |
|     my $m3k4k = $self->stpx_smst_vlans_mapped_3k4k;
 | |
| 
 | |
|     # Get list of VLANs
 | |
|     my $vlan_membership = $self->i_vlan_membership;
 | |
|     my @vlans;
 | |
|     foreach my $iid ( keys %$vlan_membership ){
 | |
|         if ( my $vm = $vlan_membership->{$iid} ){
 | |
|             foreach my $vid ( @$vm ){
 | |
|                 push @vlans, $vid;
 | |
|             }
 | |
|         }
 | |
|     }
 | |
|     my %res;
 | |
|     foreach my $vlan ( @vlans ){
 | |
|         if ( $vlan < 2048 ){
 | |
|             foreach my $inst ( keys %$m1k2k ){
 | |
|                 my $list = $m1k2k->{$inst};
 | |
|                 my $vlanlist = [split(//, unpack("B*", $list))];
 | |
|                 if ( @$vlanlist[$vlan] ){
 | |
|                     $res{$vlan} = $inst;
 | |
|                     last;
 | |
|                 }
 | |
|             }
 | |
|         }else{
 | |
|             foreach my $inst ( keys %$m3k4k ){
 | |
|                 my $list = $m3k4k->{$inst};
 | |
|                 my $vlanlist = [split(//, unpack("B*", $list))];
 | |
|                 if ( @$vlanlist[$vlan-2048] ){
 | |
|                     $res{$vlan} = $inst;
 | |
|                     last;
 | |
|                 }
 | |
|             }
 | |
|         }
 | |
|     }
 | |
|     return \%res;
 | |
| }
 | |
| 
 | |
| sub i_rootguard_enabled {
 | |
|     my $self    = shift;
 | |
|     my $partial = shift;
 | |
| 
 | |
|     my $rg_enabled = $self->stpx_rootguard_enabled();
 | |
|     my $bp_index   = $self->bp_index($partial);
 | |
| 
 | |
|     my %res;
 | |
|     foreach my $index ( keys %$rg_enabled ){
 | |
|         my $enabled = $rg_enabled->{$index};
 | |
|         my $iid     = $bp_index->{$index};
 | |
|         next unless defined $iid;
 | |
|         next unless defined $enabled;
 | |
|         $res{$iid} = $enabled;
 | |
|     }
 | |
|     return \%res;
 | |
| }
 | |
| 
 | |
| sub i_loopguard_enabled {
 | |
|     my $self    = shift;
 | |
|     my $partial = shift;
 | |
| 
 | |
|     my $lg_enabled = $self->stpx_loopguard_enabled();
 | |
|     my $bp_index   = $self->bp_index($partial);
 | |
| 
 | |
|     my %res;
 | |
|     foreach my $index ( keys %$lg_enabled ){
 | |
|         my $enabled = $lg_enabled->{$index};
 | |
|         my $iid     = $bp_index->{$index};
 | |
|         next unless defined $iid;
 | |
|         next unless defined $enabled;
 | |
|         $res{$iid} = $enabled;
 | |
|     }
 | |
|     return \%res;
 | |
| }
 | |
| 
 | |
| sub i_bpduguard_enabled {
 | |
|     my $self    = shift;
 | |
|     my $partial = shift;
 | |
| 
 | |
|     my $bpdugm_default = $self->stpx_bpduguard_enable();
 | |
|     my $bp_index       = $self->bp_index($partial);
 | |
|     my $bpdugm         = $self->stpx_port_bpduguard_mode();
 | |
| 
 | |
|     my %res;
 | |
|     foreach my $index ( keys %$bpdugm ){
 | |
|         my $mode = $bpdugm->{$index};
 | |
|         my $iid  = $bp_index->{$index};
 | |
|         next unless defined $iid;
 | |
|         next unless defined $mode;
 | |
|         if ( $mode eq 'default' ){
 | |
|             $res{$iid} =  $bpdugm_default;
 | |
|         }else{
 | |
|             $res{$iid} = $mode;
 | |
|         }
 | |
|     }
 | |
|     return \%res;
 | |
| }
 | |
| 
 | |
| sub i_bpdufilter_enabled {
 | |
|     my $self    = shift;
 | |
|     my $partial = shift;
 | |
| 
 | |
|     my $bpdufm_default = $self->stpx_bpdufilter_enable();
 | |
|     my $bp_index       = $self->bp_index($partial);
 | |
|     my $bpdufm         = $self->stpx_port_bpdufilter_mode();
 | |
| 
 | |
|     my %res;
 | |
|     foreach my $index ( keys %$bpdufm ){
 | |
|         my $mode = $bpdufm->{$index};
 | |
|         my $iid  = $bp_index->{$index};
 | |
|         next unless defined $iid;
 | |
|         next unless defined $mode;
 | |
|         if ( $mode eq 'default' ){
 | |
|             $res{$iid} =  $bpdufm_default;
 | |
|         }else{
 | |
|             $res{$iid} = $mode;
 | |
|         }
 | |
|     }
 | |
|     return \%res;
 | |
| }
 | |
| 
 | |
| sub i_faststart_enabled {
 | |
|     my $self    = shift;
 | |
|     my $partial = shift;
 | |
| 
 | |
|     my $faststart_default = $self->stpx_faststart_default();
 | |
|     my $bp_index          = $self->bp_index($partial);
 | |
|     my $faststart         = $self->stpx_faststart_enabled();
 | |
|     my $faststart_oper    = $self->stpx_faststart_operational();
 | |
| 
 | |
|     my %res;
 | |
|     # stpxFastStartPortEnable is deprecated in favour of stpxFastStartPortMode
 | |
|     # see https://github.com/netdisco/netdisco/issues/12
 | |
|     foreach my $index ( keys %$faststart, keys %$faststart_oper ){
 | |
|         my $mode = $faststart_oper->{$index} || $faststart->{$index};
 | |
|         my $iid  = $bp_index->{$index};
 | |
|         next unless defined $iid;
 | |
|         next unless defined $mode;
 | |
|         if ( $mode eq 'default' ){
 | |
|             $res{$iid} =  $faststart_default;
 | |
|         }else{
 | |
|             $res{$iid} = $mode;
 | |
|         }
 | |
|         $res{$iid} = 'enable'  if $res{$iid} eq 'true';
 | |
|         $res{$iid} = 'disable' if $res{$iid} eq 'false';
 | |
|         $res{$iid} = 1 if $res{$iid} =~ m/enable/i; # enableForTrunk
 | |
|         $res{$iid} = 0 if $res{$iid} eq 'disable';
 | |
|     }
 | |
|     return \%res;
 | |
| }
 | |
| 
 | |
| 
 | |
| sub oct2str {
 | |
|     my ($v) = @_;
 | |
|     return sprintf('%s', unpack('H*', $v));
 | |
| }
 | |
| 
 | |
| 1;
 | |
| __END__
 | |
| 
 | |
| =head1 NAME
 | |
| 
 | |
| SNMP::Info::CiscoStpExtensions - SNMP Interface to C<CISCO-STP-EXTENSIONS-MIB>
 | |
| 
 | |
| =head1 AUTHOR
 | |
| 
 | |
| Carlos Vicente
 | |
| 
 | |
| =head1 SYNOPSIS
 | |
| 
 | |
|    my $stpx = new SNMP::Info(
 | |
|                          AutoSpecify => 1,
 | |
|                          Debug       => 1,
 | |
|                          DestHost    => 'myswitch',
 | |
|                          Community   => 'public',
 | |
|                          Version     => 2
 | |
|                        )
 | |
| 
 | |
|    or die "Can't connect to DestHost.\n";
 | |
| 
 | |
|    my $class = $stpx->class();
 | |
|    print " Using device sub class : $class\n";
 | |
| 
 | |
| =head1 DESCRIPTION
 | |
| 
 | |
| Create or use a subclass of SNMP::Info that inherits this class.  Do not use
 | |
| directly.
 | |
| 
 | |
| For debugging you can call new() directly as you would in SNMP::Info
 | |
| 
 | |
|  my $stpx = new SNMP::Info::CiscoStpExtensions(...);
 | |
| 
 | |
| =head2 Inherited Classes
 | |
| 
 | |
| =over
 | |
| 
 | |
| =item SNMP::Info
 | |
| 
 | |
| =item SNMP::Info::Bridge
 | |
| 
 | |
| =back
 | |
| 
 | |
| =head2 Required MIBs
 | |
| 
 | |
| =over
 | |
| 
 | |
| =item F<CISCO-STP-EXTENSIONS-MIB>
 | |
| 
 | |
| =back
 | |
| 
 | |
| =head1 GLOBALS
 | |
| 
 | |
| These are methods that return scalar values from SNMP
 | |
| 
 | |
| =over
 | |
| 
 | |
| =item $stpx->stp_ver()
 | |
| 
 | |
| Returns the particular STP version running on this device.
 | |
| Meant to override SNMP::Info::Brigde::stp_ver()
 | |
| 
 | |
| Values: C<pvstPlus>, C<mistp>, C<mistpPvstPlus>, C<mst>, C<rapidPvstPlus>
 | |
| 
 | |
| (C<stpxSpanningTreeType>)
 | |
| 
 | |
| =back
 | |
| 
 | |
| =head1 TABLE METHODS
 | |
| 
 | |
| These are methods that return tables of information in the form of a reference
 | |
| to a hash.
 | |
| 
 | |
| =over
 | |
| 
 | |
| =item $stpx->mst_config_digest()
 | |
| 
 | |
| Returns the Multiple Spanning Tree (MST) configuration digest
 | |
| 
 | |
| (C<stpxSMSTConfigDigest>)
 | |
| 
 | |
| =item $stpx->mst_region_name()
 | |
| 
 | |
| Returns the Multiple Spanning Tree (MST) region name
 | |
| 
 | |
| (C<stpxMSTRegionName>)
 | |
| 
 | |
| =item $stpx->mst_region_rev()
 | |
| 
 | |
| Returns the Multiple Spanning Tree (MST) region name
 | |
| 
 | |
| (C<stpxSMSTRegionRevision>)
 | |
| 
 | |
| =item $stpx->mst_vlan2instance()
 | |
| 
 | |
| Returns the mapping of vlan to MST instance in the form of a hash reference
 | |
| with key = VLAN id, value = STP instance
 | |
| 
 | |
| =item $stpx->i_rootguard_enabled()
 | |
| 
 | |
| Returns 1 or 0 depending on whether C<RootGuard> is enabled on a given port.
 | |
| Format is a hash reference with key = C<ifIndex>, value = [1|0]
 | |
| 
 | |
| (C<stpxRootGuardConfigEnabled>)
 | |
| 
 | |
| =item $stpx->i_loopguard_enabled()
 | |
| 
 | |
| Returns 1 or 0 depending on whether C<LoopGuard> is enabled on a given port.
 | |
| Format is a hash reference with key = C<ifIndex>, value = [1|0]
 | |
| 
 | |
| (C<stpxLoopGuardConfigEnabled>)
 | |
| 
 | |
| =item $stpx->i_bpduguard_enabled()
 | |
| 
 | |
| Returns 1 or 0 depending on whether C<BpduGuard> is enabled on a given port.
 | |
| Format is a hash reference with key = C<ifIndex>, value = [1|0]
 | |
| 
 | |
| (C<stpxFastStartPortBpduGuardMode>)
 | |
| 
 | |
| =item $stpx->i_bpdufilter_enabled()
 | |
| 
 | |
| Returns 1 or 0 depending on whether C<BpduFilter> is enabled on a given port.
 | |
| Format is a hash reference with key = C<ifIndex>, value = [1|0]
 | |
| 
 | |
| (C<stpxFastStartBpduFilterEnable>)
 | |
| 
 | |
| =item $stpx->i_faststart_enabled()
 | |
| 
 | |
| Returns 1 or 0 depending on whether FastStart (aka PortFast) is enabled on a
 | |
| given port.  Format is a hash reference with key = C<ifIndex>, value = [1|0]
 | |
| 
 | |
| (C<stpxFastStartPortEnable> and C<stpxFastStartPortMode>)
 | |
| 
 | |
| =back
 | |
| 
 | |
| =head1 MUNGES
 | |
| 
 | |
| =over
 | |
| 
 | |
| =item oct2str()
 | |
| 
 | |
| Unpacks H* into a string
 | |
| 
 | |
| =back
 | |
| 
 | |
| =cut
 |