509 lines
		
	
	
		
			13 KiB
		
	
	
	
		
			Perl
		
	
	
	
	
	
			
		
		
	
	
			509 lines
		
	
	
		
			13 KiB
		
	
	
	
		
			Perl
		
	
	
	
	
	
| # SNMP::Info::LLDP
 | |
| #
 | |
| # Eric Miller
 | |
| #
 | |
| # Copyright (c) 2007 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.
 | |
| #
 | |
| # 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::LLDP;
 | |
| $VERSION = '1.05';
 | |
| # $Id$
 | |
| 
 | |
| use strict;
 | |
| 
 | |
| use Exporter;
 | |
| use SNMP::Info;
 | |
|  
 | |
| @SNMP::Info::LLDP::ISA = qw/SNMP::Info Exporter/;
 | |
| @SNMP::Info::LLDP::EXPORT_OK = qw//;
 | |
| 
 | |
| use vars qw/$VERSION $DEBUG %FUNCS %GLOBALS %MIBS %MUNGE $INIT/;
 | |
| 
 | |
| %MIBS    = (
 | |
|             'LLDP-MIB'          => 'lldpLocSysCapEnabled',
 | |
|             'LLDP-EXT-DOT1-MIB' => 'lldpXdot1MIB',
 | |
|             'LLDP-EXT-DOT3-MIB' => 'lldpXdot3MIB',            
 | |
|             );
 | |
| 
 | |
| %GLOBALS = (
 | |
|             'lldp_sysname'   => 'lldpLocSysName',
 | |
|             'lldp_sysdesc'   => 'lldpLocSysDesc',
 | |
|             'lldp_sys_cap'   => 'lldpLocSysCapEnabled',
 | |
|            );
 | |
| 
 | |
| %FUNCS  = (
 | |
|            # LLDP-MIB::lldpLocManAddrTable
 | |
|            'lldp_lman_addr'     => 'lldpLocManAddrIfId',
 | |
|            # LLDP-MIB::lldpRemTable
 | |
|            'lldp_rem_id_type'   => 'lldpRemChassisIdSubtype',
 | |
|            'lldp_rem_id'        => 'lldpRemChassisId',
 | |
|            'lldp_rem_pid_type'  => 'lldpRemPortIdSubtype',
 | |
|            'lldp_rem_pid'       => 'lldpRemPortId',
 | |
|            'lldp_rem_desc'      => 'lldpRemPortDesc',
 | |
|            'lldp_rem_sysname'   => 'lldpRemSysName',
 | |
|            'lldp_rem_sysdesc'   => 'lldpRemSysDesc',
 | |
|            'lldp_rem_sys_cap'   => 'lldpRemSysCapEnabled',
 | |
|            # LLDP-MIB::lldpRemManAddrTable
 | |
|            'lldp_rman_addr'     => 'lldpRemManAddrIfSubtype',
 | |
|           );
 | |
| 
 | |
| 
 | |
| %MUNGE = (
 | |
|           'lldp_sysdesc'        => \&munge_null,
 | |
|           'lldp_sysname'        => \&munge_null,
 | |
|           'lldp_rem_sysname'    => \&munge_null,
 | |
|           'lldp_rem_sysdesc'    => \&munge_null,
 | |
|           'lldp_rem_port_desc'  => \&munge_null,
 | |
|           'lldp_sys_cap'        => \&munge_caps,
 | |
|           'lldp_rem_sys_cap'    => \&munge_caps,
 | |
|          );
 | |
| 
 | |
| sub munge_null {
 | |
|     my $text = shift || return;
 | |
|     
 | |
|     $text =~ s/\0//g;
 | |
|     return $text;
 | |
| }
 | |
| 
 | |
| sub munge_caps {
 | |
|     my $caps = shift;
 | |
|     return undef unless defined $caps;
 | |
| 
 | |
|     my $bits = unpack("b*",$caps);
 | |
|     return $bits;
 | |
| }
 | |
| 
 | |
| sub hasLLDP {
 | |
|     my $lldp = shift;
 | |
| 
 | |
|     # We may be have LLDP, but nothing in lldpRemoteSystemsData Tables
 | |
|     # so we could be running LLDP but not return any useful information
 | |
|     my $lldp_cap = $lldp->lldp_sys_cap();
 | |
| 
 | |
|     return 1 if defined $lldp_cap;
 | |
|     return undef;
 | |
| }
 | |
| 
 | |
| sub lldp_if {
 | |
|     my $lldp    = shift;
 | |
|     my $partial = shift;
 | |
| 
 | |
|     my $addr = $lldp->lldp_rem_pid($partial) || {};
 | |
|     
 | |
|     my %lldp_if;
 | |
|     foreach my $key (keys %$addr) {
 | |
|         my @aOID = split ('\.',$key);
 | |
| 	my $port = $aOID[1];
 | |
| 	$lldp_if{$key} = $port;
 | |
|     }
 | |
|     return \%lldp_if;
 | |
| }
 | |
| 
 | |
| sub lldp_ip {
 | |
|     my $lldp = shift;
 | |
|     my $partial = shift;
 | |
| 
 | |
|     my $rman_addr = $lldp->lldp_rman_addr($partial) || {};
 | |
|     
 | |
|     my %lldp_ip;    
 | |
|     foreach my $key (keys %$rman_addr) {
 | |
|         my($index, $proto, $addr) = _lldp_addr_index($key);
 | |
|         next unless defined $index;
 | |
|         next unless $proto == 1;
 | |
|         $lldp_ip{$index} = $addr;
 | |
|     }
 | |
|     return \%lldp_ip;
 | |
| }
 | |
| 
 | |
| sub lldp_addr {
 | |
|     my $lldp = shift;
 | |
|     my $partial = shift;
 | |
| 
 | |
|     my $rman_addr = $lldp->lldp_rman_addr($partial) || {};
 | |
|     
 | |
|     my %lldp_ip;    
 | |
|     foreach my $key (keys %$rman_addr) {
 | |
|         my($index, $proto, $addr) = _lldp_addr_index($key);
 | |
|         next unless defined $index;
 | |
|         $lldp_ip{$index} = $addr;
 | |
|     }
 | |
|     return \%lldp_ip;
 | |
| }
 | |
| 
 | |
| sub lldp_port {
 | |
|     my $lldp = shift;
 | |
|     my $partial = shift;
 | |
| 
 | |
|     my $pdesc  = $lldp->lldp_rem_desc($partial) || {};
 | |
|     my $pid    = $lldp->lldp_rem_pid($partial) || {};
 | |
|     my $ptype  = $lldp->lldp_rem_pid_type($partial) || {};
 | |
| 
 | |
|     my %lldp_port;    
 | |
|     foreach my $key (sort keys %$pid) {
 | |
|         my $port = $pdesc->{$key};
 | |
| 	unless ($port) {
 | |
|             $port = $pid->{$key};
 | |
|             next unless $port;
 | |
| 	    my $type = $ptype->{$key};
 | |
|             next unless $type;
 | |
|             # May need to format other types in the future, i.e. Network address
 | |
|             if ($type =~ /mac/) {
 | |
|                 $port = join(':',map { sprintf "%02x",$_ } unpack('C*',$port));
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         # Nortel lldpRemPortDesc doesn't match ifDescr, but we can still
 | |
|         # figure out slot.port based upon lldpRemPortDesc 
 | |
|         if ( $port =~ /^(Unit\s+(\d+)\s+)?Port\s+(\d+)$/ ) {
 | |
|             $port = defined $1 ? "$2.$3" : "$3";
 | |
|         }
 | |
| 
 | |
|         $lldp_port{$key} = $port;
 | |
|     }
 | |
|     return \%lldp_port;
 | |
| }
 | |
| 
 | |
| sub lldp_id {
 | |
|     my $lldp = shift;
 | |
|     my $partial = shift;
 | |
|     
 | |
|     my $ch_type = $lldp->lldp_rem_id_type($partial) || {};
 | |
|     my $ch = $lldp->lldp_rem_id($partial) || {};
 | |
| 
 | |
|     my %lldp_id;    
 | |
|     foreach my $key (keys %$ch) {
 | |
| 	my $id = $ch->{$key};
 | |
|         next unless $id;
 | |
| 	my $type = $ch_type->{$key};
 | |
|         next unless $type;
 | |
|         # May need to format other types in the future
 | |
|         if ($type =~ /mac/) {
 | |
|             $id = join(':',map { sprintf "%02x",$_ } unpack('C*',$id));
 | |
|         }
 | |
|         $lldp_id{$key} = $id;
 | |
|     }
 | |
|     return \%lldp_id;
 | |
| }
 | |
| 
 | |
| #sub root_ip {
 | |
| #    my $lldp = shift;
 | |
| #
 | |
| #    my $man_addr = $lldp->lldp_lman_addr() || {};
 | |
| #
 | |
| #    foreach my $key (keys %$man_addr) {
 | |
| #        my @oids   = split(/\./, $key);
 | |
| #        my $proto  = shift(@oids);
 | |
| #        my $length = shift(@oids);
 | |
| #        # IPv4
 | |
| #        if ($proto == 1) {
 | |
| #            my $addr = join('.',@oids);
 | |
| #            return $addr if (defined $addr and $lldp->snmp_connect_ip($addr));
 | |
| #        }
 | |
| #    }
 | |
| #    return undef;
 | |
| #}
 | |
| 
 | |
| # Break up the lldpRemManAddrTable INDEX into common index, protocol,
 | |
| # and address.
 | |
| sub _lldp_addr_index {
 | |
|     my $idx = shift;
 | |
|     my @oids   = split(/\./, $idx);
 | |
|     my $index  = join('.', splice(@oids, 0, 3));
 | |
|     my $proto  = shift(@oids);
 | |
|     my $length = shift(@oids);
 | |
|     # IPv4
 | |
|     if ($proto == 1) {
 | |
|         return ($index, $proto, join('.',@oids));
 | |
|     }
 | |
|     # MAC
 | |
|     elsif ($proto == 6) {
 | |
|         return ($index, $proto, join(':',map { sprintf "%02x",$_ } @oids));
 | |
|     }
 | |
|     # TODO - Need to handle other protocols, i.e. IPv6
 | |
|     else {
 | |
|         return undef;
 | |
|     }
 | |
| }
 | |
| 
 | |
| 1;
 | |
| __END__
 | |
| 
 | |
| =head1 NAME
 | |
| 
 | |
| SNMP::Info::LLDP - SNMP Interface to the Link Layer Discovery Protocol (LLDP)
 | |
| 
 | |
| =head1 AUTHOR
 | |
| 
 | |
| Eric Miller
 | |
| 
 | |
| =head1 SYNOPSIS
 | |
| 
 | |
|  my $lldp = new SNMP::Info ( 
 | |
|                              AutoSpecify => 1,
 | |
|                              Debug       => 1,
 | |
|                              DestHost    => 'router', 
 | |
|                              Community   => 'public',
 | |
|                              Version     => 2
 | |
|                            );
 | |
| 
 | |
|  my $class = $lldp->class();
 | |
|  print " Using device sub class : $class\n";
 | |
| 
 | |
|  $haslldp   = $lldp->hasLLDP() ? 'yes' : 'no';
 | |
| 
 | |
|  # Print out a map of device ports with LLDP neighbors:
 | |
|  my $interfaces    = $lldp->interfaces();
 | |
|  my $lldp_if       = $lldp->lldp_if();
 | |
|  my $lldp_ip       = $lldp->lldp_ip();
 | |
|  my $lldp_port     = $lldp->lldp_port();
 | |
| 
 | |
|  foreach my $lldp_key (keys %$lldp_ip){
 | |
|     my $iid           = $lldp_if->{$lldp_key};
 | |
|     my $port          = $interfaces->{$iid};
 | |
|     my $neighbor      = $lldp_ip->{$lldp_key};
 | |
|     my $neighbor_port = $lldp_port->{$lldp_key};
 | |
|     print "Port : $port connected to $neighbor / $neighbor_port\n";
 | |
|  }
 | |
| 
 | |
| =head1 DESCRIPTION
 | |
| 
 | |
| SNMP::Info::LLDP is a subclass of SNMP::Info that provides an object oriented 
 | |
| interface to LLDP information through SNMP.
 | |
| 
 | |
| LLDP is a Layer 2 protocol that allows a network device to advertise its
 | |
| identity and capabilities on the local network providing topology information.
 | |
| The protocol is defined in the IEEE standard 802.1AB.
 | |
| 
 | |
| Create or use a device subclass that inherits this class.  Do not use directly.
 | |
| 
 | |
| =head2 Inherited Classes
 | |
| 
 | |
| None.
 | |
| 
 | |
| =head2 Required MIBs
 | |
| 
 | |
| =over
 | |
| 
 | |
| =item LLDP-MIB
 | |
| 
 | |
| =item LLDP-EXT-DOT1-MIB
 | |
| 
 | |
| =item LLDP-EXT-DOT3-MIB  
 | |
| 
 | |
| =back
 | |
| 
 | |
| =head1 GLOBAL METHODS
 | |
| 
 | |
| These are methods that return scalar values from SNMP
 | |
| 
 | |
| =over
 | |
| 
 | |
| =item $lldp->hasLLDP()
 | |
| 
 | |
| Is LLDP is active in this device?  
 | |
| 
 | |
| Note:  LLDP may be active, but nothing in B<lldpRemoteSystemsData> Tables so
 | |
| the device would not return any useful topology information.
 | |
| 
 | |
| =item $lldp->lldp_sysname()
 | |
| 
 | |
| The string value used to identify the system name of the local system.  If the
 | |
| local agent supports IETF RFC 3418, B<lldpLocSysName> object should have the
 | |
| same value of B<sysName> object.
 | |
| 
 | |
| Nulls are removed before the value is returned. 
 | |
| 
 | |
| (B<lldpLocSysName>)
 | |
| 
 | |
| =item $lldp->lldp_sysdesc()
 | |
| 
 | |
| The string value used to identify the system description of the local system.
 | |
| If the local agent supports IETF RFC 3418, B<lldpLocSysDesc> object should have
 | |
| the same value of B<sysDesc> object.
 | |
|  
 | |
| Nulls are removed before the value is returned.
 | |
| 
 | |
| (B<lldpLocSysDesc>)
 | |
| 
 | |
| =item  $lldp->lldp_sys_cap() 
 | |
| 
 | |
| Returns which system capabilities are enabled on the local system.  Results
 | |
| are munged into an ascii binary string, LSB.  Each digit represents a bit
 | |
| from the table below:
 | |
| 
 | |
| =over
 | |
| 
 | |
| =item Bit 'other(0)' indicates that the system has capabilities other than
 | |
| those listed below.
 | |
| 
 | |
| =item Bit 'repeater(1)' indicates that the system has repeater capability.
 | |
| 
 | |
| =item Bit 'bridge(2)' indicates that the system has bridge capability.
 | |
| 
 | |
| =item Bit 'wlanAccessPoint(3)' indicates that the system has WLAN access
 | |
| point capability.
 | |
| 
 | |
| =item Bit 'router(4)' indicates that the system has router capability.
 | |
| 
 | |
| =item Bit 'telephone(5)' indicates that the system has telephone capability.
 | |
| 
 | |
| =item Bit 'docsisCableDevice(6)' indicates that the system has DOCSIS Cable
 | |
| Device capability (IETF RFC 2669 & 2670).
 | |
| 
 | |
| =item Bit 'stationOnly(7)' indicates that the system has only station
 | |
| capability and nothing else."
 | |
| 
 | |
| =back
 | |
| 
 | |
| (B<lldpLocSysCapEnabled>)
 | |
| 
 | |
| =back
 | |
| 
 | |
| =head1 TABLE METHODS
 | |
| 
 | |
| These are methods that return tables of information in the form of a reference
 | |
| to a hash.
 | |
| 
 | |
| =over
 | |
| 
 | |
| =item $lldp->lldp_id()
 | |
| 
 | |
| Returns the string value used to identify the chassis component	associated
 | |
| with the remote system.
 | |
| 
 | |
| (B<lldpRemChassisId>)
 | |
| 
 | |
| =item $lldp->lldp_if()
 | |
| 
 | |
| Returns the mapping to the SNMP Interface Table.
 | |
| 
 | |
| =item  $lldp->lldp_ip()
 | |
| 
 | |
| Returns remote IPv4 address.  Returns undef for all other address types, use
 | |
| lldp_addr if you want any return address type.
 | |
| 
 | |
| =item  $lldp->lldp_addr()
 | |
| 
 | |
| Returns remote address.  Type may be any IANA Address Family Number.  Currently
 | |
| only returns IPv4 or MAC addresses.
 | |
| 
 | |
| =item $lldp->lldp_port()
 | |
| 
 | |
| Returns remote port ID
 | |
| 
 | |
| =back
 | |
| 
 | |
| =head2 LLDP Remote Table (B<lldpRemTable>)
 | |
| 
 | |
| =over
 | |
| 
 | |
| =item $lldp->lldp_rem_id_type()
 | |
| 
 | |
| Returns the type of encoding used to identify the chassis associated with
 | |
| the remote system.
 | |
| 
 | |
| (B<lldpRemChassisIdSubtype>)
 | |
| 
 | |
| =item $lldp->lldp_rem_id()
 | |
| 
 | |
| Returns the string value used to identify the chassis component	associated
 | |
| with the remote system.
 | |
| 
 | |
| (B<lldpRemChassisId>)
 | |
| 
 | |
| =item $lldp->lldp_rem_pid_type()
 | |
| 
 | |
| Returns the type of port identifier encoding used in the associated
 | |
| B<lldpRemPortId> object.
 | |
| 
 | |
| (B<lldpRemPortIdSubtype>)
 | |
| 
 | |
| =item $lldp->lldp_rem_pid()
 | |
| 
 | |
| Returns the string value used to identify the port component associated with
 | |
| the remote system.
 | |
| 
 | |
| (B<lldpRemPortId>)
 | |
| 
 | |
| =item $lldp->lldp_rem_desc()
 | |
| 
 | |
| Returns the string value used to identify the description of the given port
 | |
| associated with the remote system.
 | |
| 
 | |
| Nulls are removed before the value is returned. 
 | |
| 
 | |
| (B<lldpRemPortDesc>)
 | |
| 
 | |
| =item $lldp->lldp_rem_sysname()
 | |
| 
 | |
| Returns the string value used to identify the system name of the remote system.
 | |
| 
 | |
| Nulls are removed before the value is returned. 
 | |
| 
 | |
| (B<lldpRemSysName>)
 | |
| 
 | |
| =item $lldp->lldp_rem_sysdesc()
 | |
| 
 | |
| Returns the string value used to identify the system description of the
 | |
| remote system.
 | |
| 
 | |
| Nulls are removed before the value is returned. 
 | |
| 
 | |
| (B<lldpRemSysDesc>)
 | |
| 
 | |
| =item  $lldp->lldp_rem_sys_cap() 
 | |
| 
 | |
| Returns which system capabilities are enabled on the local system.  Results
 | |
| are munged into an ascii binary string, LSB.  Each digit
 | |
| represents a bit from the table below:
 | |
| 
 | |
| =over
 | |
| 
 | |
| =item Bit 'other(0)' indicates that the system has capabilities other than
 | |
| those listed below.
 | |
| 
 | |
| =item Bit 'repeater(1)' indicates that the system has repeater capability.
 | |
| 
 | |
| =item Bit 'bridge(2)' indicates that the system has bridge capability.
 | |
| 
 | |
| =item Bit 'wlanAccessPoint(3)' indicates that the system has WLAN access
 | |
| point capability.
 | |
| 
 | |
| =item Bit 'router(4)' indicates that the system has router capability.
 | |
| 
 | |
| =item Bit 'telephone(5)' indicates that the system has telephone capability.
 | |
| 
 | |
| =item Bit 'docsisCableDevice(6)' indicates that the system has DOCSIS Cable
 | |
| Device capability (IETF RFC 2669 & 2670).
 | |
| 
 | |
| =item Bit 'stationOnly(7)' indicates that the system has only station
 | |
| capability and nothing else."
 | |
| 
 | |
| =back
 | |
| 
 | |
| (B<lldpRemSysCapEnabled>)
 | |
| 
 | |
| =back
 | |
| 
 | |
| =cut
 | 
