459 lines
		
	
	
		
			11 KiB
		
	
	
	
		
			Perl
		
	
	
	
	
	
			
		
		
	
	
			459 lines
		
	
	
		
			11 KiB
		
	
	
	
		
			Perl
		
	
	
	
	
	
| # SNMP::Info::Layer3::F5
 | |
| #
 | |
| # Copyright (c) 2012 Eric Miller
 | |
| # 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 University of California, Santa Cruz 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::Layer3::F5;
 | |
| 
 | |
| use strict;
 | |
| use Exporter;
 | |
| use SNMP::Info::Layer3;
 | |
| 
 | |
| @SNMP::Info::Layer3::F5::ISA       = qw/SNMP::Info::Layer3 Exporter/;
 | |
| @SNMP::Info::Layer3::F5::EXPORT_OK = qw//;
 | |
| 
 | |
| use vars qw/$VERSION %GLOBALS %FUNCS %MIBS %MUNGE/;
 | |
| 
 | |
| $VERSION = '3.50';
 | |
| 
 | |
| %MIBS = (
 | |
|     %SNMP::Info::Layer3::MIBS,
 | |
|     'F5-BIGIP-SYSTEM-MIB' => 'sysAttrArpMaxEntries',
 | |
| );
 | |
| 
 | |
| %GLOBALS = (
 | |
|     %SNMP::Info::Layer3::GLOBALS,
 | |
|     'os_ver'     => 'sysProductVersion',
 | |
|     'mkt_name'   => 'sysPlatformInfoMarketingName',
 | |
|     'ps1_status' => 'sysChassisPowerSupplyStatus.1',
 | |
|     'ps2_status' => 'sysChassisPowerSupplyStatus.2',
 | |
| 
 | |
|     # Named serial1 to override serial1 in L3 serial method
 | |
|     'serial1'  => 'sysGeneralChassisSerialNum',
 | |
|     'qb_vlans' => 'sysVlanNumber',
 | |
|     'ports'    => 'sysInterfaceNumber',
 | |
| 
 | |
| );
 | |
| 
 | |
| %FUNCS = (
 | |
|     %SNMP::Info::Layer3::FUNCS,
 | |
| 
 | |
|     # sysInterfaceTable
 | |
|     'i_index'       => 'sysInterfaceName',
 | |
|     'i_description' => 'sysInterfaceName',
 | |
|     'i_mtu'         => 'sysInterfaceMtu',
 | |
|     'i_speed'       => 'sysInterfaceMediaActiveSpeed',
 | |
|     'i_mac'         => 'sysInterfaceMacAddr',
 | |
|     'i_up_admin'    => 'sysInterfaceEnabled',
 | |
|     'i_up'          => 'sysInterfaceStatus',
 | |
| 
 | |
|     # sysIfxStatTable
 | |
|     'i_octet_in64'       => 'sysIfxStatHcInOctets',
 | |
|     'i_octet_out64'      => 'sysIfxStatHcOutOctets',
 | |
|     'i_pkts_ucast_in64'  => 'sysIfxStatHcInUcastPkts',
 | |
|     'i_pkts_ucast_out64' => 'sysIfxStatHcOutUcastPkts',
 | |
|     'i_pkts_mutli_in64'  => 'sysIfxStatInMulticastPkts',
 | |
|     'i_pkts_multi_out64' => 'sysIfxStatOutMulticastPkts',
 | |
|     'i_pkts_bcast_in64'  => 'sysIfxStatInBroadcastPkts',
 | |
|     'i_pkts_bcast_out64' => 'sysIfxStatOutBroadcastPkts',
 | |
| 
 | |
|     # sysInterfaceStatTable
 | |
|     'i_discards_in'  => 'sysInterfaceStatDropsIn',
 | |
|     'i_discards_out' => 'sysInterfaceStatDropsOut',
 | |
|     'i_errors_in'    => 'sysInterfaceStatErrorsIn',
 | |
|     'i_errors_out'   => 'sysInterfaceStatErrorsOut',
 | |
| 
 | |
|     # sysInterfaceTable
 | |
|     'sys_i_duplex' => 'sysInterfaceMediaActiveDuplex',
 | |
| 
 | |
|     # sysChassisFanTable
 | |
|     'fan_state' => 'sysChassisFanStatus',
 | |
| 
 | |
|     # sysVlanTable
 | |
|     'sys_v_id' => 'sysVlanId',
 | |
|     'v_name'   => 'sysVlanVname',
 | |
| 
 | |
|     # sysVlanMemberTable
 | |
|     'sys_vm_tagged' => 'sysVlanMemberTagged',
 | |
|     'sys_vm_name'   => 'sysVlanMemberVmname',
 | |
|     'sys_vmp_name'  => 'sysVlanMemberParentVname',
 | |
| );
 | |
| 
 | |
| %MUNGE = ( %SNMP::Info::Layer3::MUNGE, );
 | |
| 
 | |
| sub vendor {
 | |
|     return 'f5';
 | |
| }
 | |
| 
 | |
| sub os {
 | |
|     return 'f5';
 | |
| }
 | |
| 
 | |
| sub fan {
 | |
|     my $f5        = shift;
 | |
|     my $fan_state = $f5->fan_state();
 | |
|     my $ret       = "";
 | |
|     my $s         = "";
 | |
|     foreach my $i ( sort { $a <=> $b } keys %$fan_state ) {
 | |
|         $ret .= $s . $i . ': ' . $fan_state->{$i};
 | |
|         $s = ', ';
 | |
|     }
 | |
|     return if ( $s eq "" );
 | |
|     return $ret;
 | |
| }
 | |
| 
 | |
| sub model {
 | |
|     my $f5 = shift;
 | |
| 
 | |
|     my $name = $f5->mkt_name();
 | |
| 
 | |
|     if ( defined $name ) { return $name; }
 | |
| 
 | |
|     my $id    = $f5->id();
 | |
|     my $model = &SNMP::translateObj($id);
 | |
|     if ( !defined $model ) { return $id; }
 | |
| 
 | |
|     return $model;
 | |
| }
 | |
| 
 | |
| # Override L3 interfaces
 | |
| sub interfaces {
 | |
|     my $f5      = shift;
 | |
|     my $partial = shift;
 | |
| 
 | |
|     return $f5->i_index($partial);
 | |
| }
 | |
| 
 | |
| # Override L3 i_name
 | |
| sub i_name {
 | |
|     my $f5      = shift;
 | |
|     my $partial = shift;
 | |
| 
 | |
|     return $f5->i_index($partial);
 | |
| }
 | |
| 
 | |
| # We don't have this, so fake it
 | |
| sub i_type {
 | |
|     my $f5      = shift;
 | |
|     my $partial = shift;
 | |
| 
 | |
|     my $idx = $f5->i_index($partial);
 | |
| 
 | |
|     my %i_type;
 | |
|     foreach my $if ( keys %$idx ) {
 | |
|         $i_type{$if} =
 | |
|           ((exists $f5->{sess}->{UseEnums} and $f5->{sess}->{UseEnums})
 | |
|            ? 'ethernetCsmacd' : 6 );
 | |
|     }
 | |
|     return \%i_type;
 | |
| }
 | |
| 
 | |
| # Override L3 i_duplex
 | |
| sub i_duplex {
 | |
|     my $f5      = shift;
 | |
|     my $partial = shift;
 | |
| 
 | |
|     my $duplexes = $f5->sys_i_duplex() || {};
 | |
| 
 | |
|     my %i_duplex;
 | |
|     foreach my $if ( keys %$duplexes ) {
 | |
|         my $duplex = $duplexes->{$if};
 | |
|         next unless defined $duplex;
 | |
|         next if ( $duplex eq 'none' );
 | |
| 
 | |
|         $i_duplex{$if} = $duplex;
 | |
|     }
 | |
|     return \%i_duplex;
 | |
| }
 | |
| 
 | |
| # Override Bridge v_index
 | |
| sub v_index {
 | |
|     my $f5      = shift;
 | |
|     my $partial = shift;
 | |
| 
 | |
|     return $f5->sys_v_id($partial);
 | |
| }
 | |
| 
 | |
| sub i_vlan {
 | |
|     my $f5      = shift;
 | |
|     my $partial = shift;
 | |
| 
 | |
|     my $index  = $f5->i_index($partial) || {};
 | |
|     my $tagged = $f5->sys_vm_tagged()   || {};
 | |
|     my $vlans  = $f5->v_index()         || {};
 | |
| 
 | |
|     my $i_vlan = {};
 | |
|     foreach my $iid ( keys %$tagged ) {
 | |
|         my $tag = $tagged->{$iid};
 | |
|         next if ( $tag eq 'true' );
 | |
| 
 | |
|         # IID is length.vlan name index.length.interface index
 | |
|         # Split out and use as the IID to get the VLAN ID and ifIndex
 | |
|         my @iid_array = split /\./, $iid;
 | |
|         my $len       = $iid_array[0];
 | |
|         my $v_idx     = join '.', ( splice @iid_array, 0, $len + 1 );
 | |
|         my $idx       = join '.', @iid_array;
 | |
| 
 | |
|         # Check to make sure we can map to a port
 | |
|         my $p_idx = $index->{$idx};
 | |
|         next unless $p_idx;
 | |
| 
 | |
|         my $vlan = $vlans->{$v_idx};
 | |
|         next unless $vlan;
 | |
| 
 | |
|         $i_vlan->{$idx} = $vlan;
 | |
|     }
 | |
|     return $i_vlan;
 | |
| }
 | |
| 
 | |
| sub i_vlan_membership {
 | |
|     my $f5      = shift;
 | |
|     my $partial = shift;
 | |
| 
 | |
|     my $index  = $f5->i_index($partial) || {};
 | |
|     my $tagged = $f5->sys_vm_tagged()   || {};
 | |
|     my $vlans  = $f5->v_index()         || {};
 | |
| 
 | |
|     my $i_vlan_membership = {};
 | |
|     foreach my $iid ( keys %$tagged ) {
 | |
| 
 | |
|         # IID is length.vlan name index.length.interface index
 | |
|         # Split out and use as the IID to get the VLAN ID and ifIndex
 | |
|         my @iid_array = split /\./, $iid;
 | |
|         my $len       = $iid_array[0];
 | |
|         my $v_idx     = join '.', ( splice @iid_array, 0, $len + 1 );
 | |
|         my $idx       = join '.', @iid_array;
 | |
| 
 | |
|         # Check to make sure we can map to a port
 | |
|         my $p_idx = $index->{$idx};
 | |
|         next unless $p_idx;
 | |
| 
 | |
|         my $vlan = $vlans->{$v_idx};
 | |
|         next unless $vlan;
 | |
| 
 | |
|         push( @{ $i_vlan_membership->{$idx} }, $vlan );
 | |
|     }
 | |
|     return $i_vlan_membership;
 | |
| }
 | |
| 
 | |
| sub i_vlan_membership_untagged {
 | |
|     my $f5      = shift;
 | |
|     my $partial = shift;
 | |
| 
 | |
|     my $index  = $f5->i_index($partial) || {};
 | |
|     my $tagged = $f5->sys_vm_tagged()   || {};
 | |
|     my $vlans  = $f5->v_index()         || {};
 | |
| 
 | |
|     my $i_vlan_membership = {};
 | |
|     foreach my $iid ( keys %$tagged ) {
 | |
|         
 | |
|         next unless $tagged->{$iid} eq 'false';
 | |
|         # IID is length.vlan name index.length.interface index
 | |
|         # Split out and use as the IID to get the VLAN ID and ifIndex
 | |
|         my @iid_array = split /\./, $iid;
 | |
|         my $len       = $iid_array[0];
 | |
|         my $v_idx     = join '.', ( splice @iid_array, 0, $len + 1 );
 | |
|         my $idx       = join '.', @iid_array;
 | |
| 
 | |
|         # Check to make sure we can map to a port
 | |
|         my $p_idx = $index->{$idx};
 | |
|         next unless $p_idx;
 | |
| 
 | |
|         my $vlan = $vlans->{$v_idx};
 | |
|         next unless $vlan;
 | |
| 
 | |
|         push( @{ $i_vlan_membership->{$idx} }, $vlan );
 | |
|     }
 | |
|     return $i_vlan_membership;
 | |
| }
 | |
| 
 | |
| 1;
 | |
| __END__
 | |
| 
 | |
| =head1 NAME
 | |
| 
 | |
| SNMP::Info::Layer3::F5 - SNMP Interface to F5 network devices.
 | |
| 
 | |
| =head1 AUTHOR
 | |
| 
 | |
| Eric Miller
 | |
| 
 | |
| =head1 SYNOPSIS
 | |
| 
 | |
|  # Let SNMP::Info determine the correct subclass for you. 
 | |
|  my $f5 = new SNMP::Info(
 | |
|                           AutoSpecify => 1,
 | |
|                           Debug       => 1,
 | |
|                           DestHost    => 'myswitch',
 | |
|                           Community   => 'public',
 | |
|                           Version     => 2
 | |
|                         ) 
 | |
|     or die "Can't connect to DestHost.\n";
 | |
| 
 | |
|  my $class      = $f5->class();
 | |
|  print "SNMP::Info determined this device to fall under subclass : $class\n";
 | |
| 
 | |
| =head1 DESCRIPTION
 | |
| 
 | |
| Abstraction subclass for F5 network devices.
 | |
| 
 | |
| For speed or debugging purposes you can call the subclass directly, but not
 | |
| after determining a more specific class using the method above. 
 | |
| 
 | |
|  my $f5 = new SNMP::Info::Layer3::F5(...);
 | |
| 
 | |
| =head2 Inherited Classes
 | |
| 
 | |
| =over
 | |
| 
 | |
| =item SNMP::Info::Layer3
 | |
| 
 | |
| =back
 | |
| 
 | |
| =head2 Required MIBs
 | |
| 
 | |
| =over
 | |
| 
 | |
| =item F<F5-BIGIP-COMMON-MIB>
 | |
| 
 | |
| =item F<F5-BIGIP-SYSTEM-MIB>
 | |
| 
 | |
| =item Inherited Classes' MIBs
 | |
| 
 | |
| See L<SNMP::Info::Layer3/"Required MIBs"> for its own MIB requirements.
 | |
| 
 | |
| =back
 | |
| 
 | |
| =head1 GLOBALS
 | |
| 
 | |
| These are methods that return scalar value from SNMP
 | |
| 
 | |
| =over
 | |
| 
 | |
| =item $f5->model()
 | |
| 
 | |
| Return (C<sysPlatformInfoMarketingName>), otherwise tries to reference
 | |
| $f5->id() to F<F5-BIGIP-COMMON-MIB>. 
 | |
| 
 | |
| =item $f5->vendor()
 | |
| 
 | |
| Returns 'f5'
 | |
| 
 | |
| =item $f5->os()
 | |
| 
 | |
| Returns 'f5'
 | |
| 
 | |
| =item $f5->os_ver()
 | |
| 
 | |
| Returns the software version reported by C<sysProductVersion>
 | |
| 
 | |
| =item $f5->fan()
 | |
| 
 | |
| Combines (C<sysChassisFanStatus>) into a single string.
 | |
| 
 | |
| =item $f5->ps1_status()
 | |
| 
 | |
| Returns status of primary power supply
 | |
| 
 | |
| =item $f5->ps2_status()
 | |
| 
 | |
| Returns status of redundant power supply
 | |
| 
 | |
| =back
 | |
| 
 | |
| =head2 Globals imported from SNMP::Info::Layer3
 | |
| 
 | |
| See documentation in L<SNMP::Info::Layer3/"GLOBALS"> for details.
 | |
| 
 | |
| =head1 TABLE METHODS
 | |
| 
 | |
| These are methods that return tables of information in the form of a
 | |
| reference to a hash.
 | |
| 
 | |
| =head2 Overrides
 | |
| 
 | |
| =over
 | |
| 
 | |
| =item $f5->interfaces()
 | |
| 
 | |
| Returns reference to the map between IID and physical port.
 | |
| 
 | |
| (C<sysInterfaceName>).
 | |
| 
 | |
| =item $f5->i_duplex()
 | |
| 
 | |
| Returns reference to hash.  Maps port operational duplexes to IIDs.
 | |
| 
 | |
| =item $f5->i_vlan()
 | |
| 
 | |
| Returns a mapping between C<ifIndex> and the default VLAN.
 | |
| 
 | |
| =item $f5->i_vlan_membership()
 | |
| 
 | |
| Returns reference to hash of arrays: key = C<ifIndex>, value = array of VLAN
 | |
| IDs.
 | |
| 
 | |
|   Example:
 | |
|   my $interfaces = $f5->interfaces();
 | |
|   my $vlans      = $f5->i_vlan_membership();
 | |
|   
 | |
|   foreach my $iid (sort keys %$interfaces) {
 | |
|     my $port = $interfaces->{$iid};
 | |
|     my $vlan = join(',', sort(@{$vlans->{$iid}}));
 | |
|     print "Port: $port VLAN: $vlan\n";
 | |
|   }
 | |
| 
 | |
| =item $f5->i_vlan_membership_untagged()
 | |
| 
 | |
| Returns reference to hash of arrays: key = C<ifIndex>, value = array of VLAN
 | |
| IDs.  These are the VLANs which are members of the untagged egress list for
 | |
| the port.
 | |
| 
 | |
| =item $f5->v_index()
 | |
| 
 | |
| Returns VLAN IDs
 | |
| 
 | |
| =item $f5->v_name()
 | |
| 
 | |
| Human-entered name for vlans.
 | |
| 
 | |
| =item $f5->i_name()
 | |
| 
 | |
| Returns the human set port name if exists.
 | |
| 
 | |
| =item $f5->i_type()
 | |
| 
 | |
| Returns C<'ethernetCsmacd'> for each C<ifIndex>.
 | |
| 
 | |
| =back
 | |
| 
 | |
| =head2 Table Methods imported from SNMP::Info::Layer3
 | |
| 
 | |
| See documentation in L<SNMP::Info::Layer3/"TABLE METHODS"> for details.
 | |
| 
 | |
| =cut
 |