417 lines
		
	
	
		
			9.9 KiB
		
	
	
	
		
			Perl
		
	
	
	
	
	
			
		
		
	
	
			417 lines
		
	
	
		
			9.9 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.09';
 | 
						|
 | 
						|
%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_in'       => 'sysIfxStatHcInOctets',
 | 
						|
    'i_octet_out'      => 'sysIfxStatHcOutOctets',
 | 
						|
    'i_pkts_ucast_in'  => 'sysIfxStatHcInUcastPkts',
 | 
						|
    'i_pkts_ucast_out' => 'sysIfxStatHcOutUcastPkts',
 | 
						|
 | 
						|
    # 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} = 'ethernetCsmacd';
 | 
						|
    }
 | 
						|
    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;
 | 
						|
}
 | 
						|
 | 
						|
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->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
 |