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. */ namespace OSS\SNMP\MIBS\Cisco; /** * A class for performing SNMP V2 queries on Cisco devices * * @copyright Copyright (c) 2012, Open Source Solutions Limited, Dublin, Ireland * @author Barry O'Donovan */ class CDP extends \OSS\SNMP\MIBS\Cisco { const OID_CDP_INTERFACE_ENABLED = '.1.3.6.1.4.1.9.9.23.1.1.1.1.2'; const OID_CDP_INTERFACE_NAME = '.1.3.6.1.4.1.9.9.23.1.1.1.1.6'; const OID_CDP_CACHE_NEIGHBOUR_ADDRESS_TYPE = '.1.3.6.1.4.1.9.9.23.1.2.1.1.3'; const OID_CDP_CACHE_NEIGHBOUR_ADDRESS = '.1.3.6.1.4.1.9.9.23.1.2.1.1.4'; const OID_CDP_CACHE_NEIGHBOUR_VERSION = '.1.3.6.1.4.1.9.9.23.1.2.1.1.5'; const OID_CDP_CACHE_NEIGHBOUR_ID = '.1.3.6.1.4.1.9.9.23.1.2.1.1.6'; const OID_CDP_CACHE_NEIGHBOUR_PORT = '.1.3.6.1.4.1.9.9.23.1.2.1.1.7'; const OID_CDP_CACHE_NEIGHBOUR_PLATFORM = '.1.3.6.1.4.1.9.9.23.1.2.1.1.8'; const OID_CDP_CACHE_NEIGHBOUR_CAPABILITY = '.1.3.6.1.4.1.9.9.23.1.2.1.1.9'; const OID_CDP_CACHE_NEIGHBOUR_VTP_MGMT_DOMAIN = '.1.3.6.1.4.1.9.9.23.1.2.1.1.10'; const OID_CDP_CACHE_NEIGHBOUR_NATIVE_VLAN = '.1.3.6.1.4.1.9.9.23.1.2.1.1.11'; const OID_CDP_CACHE_NEIGHBOUR_DUPLEX = '.1.3.6.1.4.1.9.9.23.1.2.1.1.12'; const OID_CDP_CACHE_NEIGHBOUR_LAST_CHANGE = '.1.3.6.1.4.1.9.9.23.1.2.1.1.24'; const OID_CDP_GLOBAL_RUN = '.1.3.6.1.4.1.9.9.23.1.3.1.0'; const OID_CDP_GLOBAL_MESSAGE_INTERVAL = '.1.3.6.1.4.1.9.9.23.1.3.2.0'; const OID_CDP_GLOBAL_HOLDTIME = '.1.3.6.1.4.1.9.9.23.1.3.3.0'; const OID_CDP_GLOBAL_DEVICE_ID = '.1.3.6.1.4.1.9.9.23.1.3.4.0'; const OID_CDP_GLOBAL_LAST_CHANGE = '.1.3.6.1.4.1.9.9.23.1.3.5.0'; /** * Get the device's global CDP (Cisco Discovery Protocol) run status * * "An indication of whether the Cisco Discovery Protocol is currently running. Entries in cdpCacheTable are * deleted when CDP is disabled." * * @return boolean True if enabled globally, else false */ public function globalRun() { return $this->getSNMP()->ppTruthValue( $this->getSNMP()->get( self::OID_CDP_GLOBAL_RUN ) ); } /** * Get the interval at which CDP messages are to be generated * * "The interval at which CDP messages are to be generated. The default value is 60 seconds." * * @return int The interval at which CDP messages are to be generated */ public function globalMessageInterval() { return $this->getSNMP()->get( self::OID_CDP_GLOBAL_MESSAGE_INTERVAL ); } /** * Get the time for the receiving device holds CDP message * * "The time for the receiving device holds CDP message. The default value is 180 seconds." * * @return int The time for the receiving device holds CDP message */ public function globalHoldTime() { return $this->getSNMP()->get( self::OID_CDP_GLOBAL_HOLDTIME ); } /** * The time when the cache table was last changed * * "Indicates the time when the cache table was last changed. It * is the most recent time at which any row was last created, * modified or deleted." * * @return int The time (timeticks) when the cache table was last changed */ public function globalLastChange() { return $this->getSNMP()->get( self::OID_CDP_GLOBAL_LAST_CHANGE ); } /** * Get the device's CDP (Cisco Discovery Protocol) ID * * @return string The device's CDP (Cisco Discovery Protocol) ID */ public function id() { return $this->getSNMP()->get( self::OID_CDP_GLOBAL_DEVICE_ID ); } /** * Get the device's interfaces CDP enabled status * * Applies the TruthValue post processor (self::ppTruthValue()) to turn * SNMP values into true / false. * * @return array The device's interfaces CDP enabled status' (as boolean true / false) */ public function interfaceEnabled() { return $this->getSNMP()->ppTruthValue( $this->getSNMP()->walk1d( self::OID_CDP_INTERFACE_ENABLED ) ); } /** * Get the device's interface names as seen in CDP * * "The name of the local interface as advertised by CDP in the Port-ID TLV" * * @return array The device's interface names as seen in CDP */ public function interfaceNames() { return $this->getSNMP()->walk1d( self::OID_CDP_INTERFACE_NAME ); } /** * Constant for possible value of CDP neighbour address type * @see neighbourAddressTypes() */ const CDP_CACHE_NEIGHBOUR_ADDRESS_TYPE_IP = 1; /** * Text representation of CDP neighbour address type * * @see neighbourAddressTypes() * @var array Text representation of CDP neighbour address type */ public static $CDP_CACHE_NEIGHBOUR_ADDRESS_TYPES = array( self::CDP_CACHE_NEIGHBOUR_ADDRESS_TYPE_IP => 'ip' ); /** * Get the CDP neighbours' address type indexed by the current device's port ID * * "An indication of the type of address contained in the corresponding instance of cdpCacheAddress" * * @param boolean $translate If true, return the string representation via self::$VTP_VLAN_TYPES * @return array The CDP neighbours' address type indexed by the current device's port ID */ public function neighbourAddressTypes( $translate = false ) { $types = $this->getSNMP()->subOidWalk( self::OID_CDP_CACHE_NEIGHBOUR_ADDRESS_TYPE, 15 ); if( !$translate ) return $types; return $this->getSNMP()->translate( $types, self::$CDP_CACHE_NEIGHBOUR_ADDRESS_TYPES ); } /** * Get the device's CDP neighbour addresses indexed by the current device's port ID * * "The (first) network-layer address of the device * as reported in the Address TLV of the most recently received * CDP message. For example, if the corresponding instance of * cacheAddressType had the value 'ip(1)', then this object * would be an IPv4-address. If the neighbor device is * SNMP-manageable, it is supposed to generate its CDP messages * such that this address is one at which it will receive SNMP * messages. Use cdpCtAddressTable to extract the remaining * addresses from the Address TLV received most recently." * * @return array The device's CDP neighbour addresses indexed by the current device's port ID */ public function neighbourAddresses() { $addresses = $this->getSNMP()->subOidWalk( self::OID_CDP_CACHE_NEIGHBOUR_ADDRESS, 15 ); foreach( $addresses as $portId => $address ) { if( strlen( $address ) == 8 && $this->neighbourAddressTypes()[ $portId ] == self::CDP_CACHE_NEIGHBOUR_ADDRESS_TYPE_IP ) $addresses[ $portId ] = long2ip( hexdec( $address ) ); } return $addresses; } /** * Get the device's CDP neighbour version indexed by the current device's port ID * * "The Version string as reported in the most recent CDP * message. The zero-length string indicates no Version * field (TLV) was reported in the most recent CDP message." * * @return array The device's CDP neighbour version indexed by the current device's port ID */ public function neighbourVersions() { return $this->getSNMP()->subOidWalk( self::OID_CDP_CACHE_NEIGHBOUR_VERSION, 15 ); } /** * Get the device's CDP neighbours (by their CDP ID) indexed by the current device's port ID * * "The Device-ID string as reported in the most recent CDP * message. The zero-length string indicates no Device-ID * field (TLV) was reported in the most recent CDP message." * * @return array The device's CDP neighbours (by their CDP ID) indexed by the current device's port ID */ public function neighbourId() { return $this->getSNMP()->subOidWalk( self::OID_CDP_CACHE_NEIGHBOUR_ID, 15 ); } /** * Get the device's CDP neighbours connected port *description* indexed by the current device's port ID * * E.g. a sample call may return: * * Array * ( * [10101] => GigabitEthernet0/1 * [10102] => FastEthernet0/2 * [10103] => GigabitEthernet1/0/24 * [10105] => GigabitEthernet1/0/2 * ) * * meaning, for example, that our local port with ID 10101 is connected to port GigabitEthernet0/1 on the neighbour * connected to that local port. You can discover the neighbour ID via neighbourId(). * * "The Port-ID string as reported in the most recent CDP * message. This will typically be the value of the ifName * object (e.g., 'Ethernet0'). The zero-length string * indicates no Port-ID field (TLV) was reported in the * most recent CDP message." * * @see neighbourId() * @return array The device's CDP neighbours connected port *description* indexed by the current device's port ID */ public function neighbourPort() { return $this->getSNMP()->subOidWalk( self::OID_CDP_CACHE_NEIGHBOUR_PORT, 15 ); } /** * Get the device's CDP neighbour platforms indexed by the current device's port ID * * "The Device's Hardware Platform as reported in the most recent CDP message. The zero-length string indicates * that no Platform field (TLV) was reported in the most recent CDP message." * * @return array The device's CDP neighbour platforms indexed by the current device's port ID */ public function neighbourPlatforms() { return $this->getSNMP()->subOidWalk( self::OID_CDP_CACHE_NEIGHBOUR_PLATFORM, 15 ); } /** * Constant for possible value of CDP neighbour capability * @see neighbourCapability() */ const CDP_CACHE_NEIGHBOUR_CAPABILITY_ROUTER = 0b1; /** * Constant for possible value of CDP neighbour capability * @see neighbourCapability() */ const CDP_CACHE_NEIGHBOUR_CAPABILITY_TRANSPARENT_BRIDGE = 0b10; /** * Constant for possible value of CDP neighbour capability * @see neighbourCapability() */ const CDP_CACHE_NEIGHBOUR_CAPABILITY_SOURCE_ROUTE_BRIDGE = 0b100; /** * Constant for possible value of CDP neighbour capability * @see neighbourCapability() */ const CDP_CACHE_NEIGHBOUR_CAPABILITY_SWITCH = 0b1000; /** * Constant for possible value of CDP neighbour capability * @see neighbourCapability() */ const CDP_CACHE_NEIGHBOUR_CAPABILITY_HOST = 0b10000; /** * Constant for possible value of CDP neighbour capability * @see neighbourCapability() */ const CDP_CACHE_NEIGHBOUR_CAPABILITY_IGMP_CAPABLE = 0b100000; /** * Constant for possible value of CDP neighbour capability * @see neighbourCapability() */ const CDP_CACHE_NEIGHBOUR_CAPABILITY_REPEATER = 0b1000000; /** * Text representation of CDP capabilities * * @see neighbourCapability() * @var array Text representation of CDP neighbour capabilities */ public static $CDP_CACHE_NEIGHBOUR_CAPABILITIES = array( self::CDP_CACHE_NEIGHBOUR_CAPABILITY_ROUTER => 'Router', self::CDP_CACHE_NEIGHBOUR_CAPABILITY_TRANSPARENT_BRIDGE => 'Transparent Bridge', self::CDP_CACHE_NEIGHBOUR_CAPABILITY_SOURCE_ROUTE_BRIDGE => 'Source Route Bridge', self::CDP_CACHE_NEIGHBOUR_CAPABILITY_SWITCH => 'Switch', self::CDP_CACHE_NEIGHBOUR_CAPABILITY_HOST => 'Host', self::CDP_CACHE_NEIGHBOUR_CAPABILITY_IGMP_CAPABLE => 'IGMP Capable', self::CDP_CACHE_NEIGHBOUR_CAPABILITY_REPEATER => 'Repeater' ); /** * Get the device's CDP neighbour capabilities (as a decimal integer) indexed by the current device's port ID * * "The Device's Functional Capabilities as reported in the most recent CDP message. For latest set of specific * values, see the latest version of the CDP specification. The zero-length string indicates no Capabilities field * (TLV) was reported in the most recent CDP message." * * @see REFERENCE "Cisco Discovery Protocol Specification, 10/19/94." * @see http://www.cisco.com/univercd/cc/td/doc/product/lan/trsrb/frames.htm#xtocid12 * @see http://wiki.wireshark.org/CDP * * @return array The device's CDP neighbour capabilities (as a decimal integer) indexed by the current device's port ID */ public function neighbourCapability() { $rtn = $this->getSNMP()->subOidWalk( self::OID_CDP_CACHE_NEIGHBOUR_CAPABILITY, 15 ); foreach( $rtn as $k => $v ) $rtn[$k] = (int)hexdec( $v ); return $rtn; } /** * Query if a given neighbour (by connected port ID) has the given capability * * Example: * * if( $host->useCisco_CDP()->neighbourHasCapability( $portId, \OSS\SNMP\MIBS\Cisco\CDP::CDP_CACHE_NEIGHBOUR_CAPABILITY_SWITCH ) * echo "Host is a switch!!"; * * @param int $portId The CDP neighbour by connected local port ID * @param int $capability The capability to query for (defined by self::CDP_CACHE_NEIGHBOUR_CAPABILITY_XXX constants) * @return boolean True if the neighbour has the given capability */ public function neighbourHasCapability( $portId, $capability ) { if( $this->neighbourCapability()[ $portId ] & $capability ) return true; return false; } /** * Get an array of individual capabilities of a given neighbour (by connected port ID) * * Example: * * print_r( $host->useCisco_CDP()->neighbourCapabilities( 10111 ) ) * * [0] => 8 // self::CDP_CACHE_NEIGHBOUR_CAPABILITY_SWITCH * [1] => 32 // self::CDP_CACHE_NEIGHBOUR_CAPABILITY_IGMP_CAPABLE * * print_r( $host->useCisco_CDP()->neighbourCapabilities( 10111, true ) ) * * [0] => "Switch" // self::CDP_CACHE_NEIGHBOUR_CAPABILITY_SWITCH * [1] => "IGMP Capable" // self::CDP_CACHE_NEIGHBOUR_CAPABILITY_IGMP_CAPABLE * * * * @param int $portId The CDP neighbour by connected local port ID * @param int $translate Set to true to return descriptions rather than integers * @return array Individual capabilities of a given neighbour */ public function neighbourCapabilities( $portId, $translate = false ) { $capabilities = array(); foreach( self::$CDP_CACHE_NEIGHBOUR_CAPABILITIES as $mask => $description ) { if( $this->neighbourCapability()[ $portId ] & $mask ) $capabilities[] = $mask; } if( $translate ) return $this->getSNMP()->translate( $capabilities, self::$CDP_CACHE_NEIGHBOUR_CAPABILITIES ); return $capabilities; } /** * Get the device's CDP neighbours' VTP management domain indexed by the current device's port ID * * "The VTP Management Domain for the remote device's interface, as reported in the most recently received CDP message. * This object is not instantiated if no VTP Management Domain field (TLV) was reported in the most recently received CDP message." * * @see REFERENCE "managementDomainName in CISCO-VTP-MIB" * * @return array The device's CDP neighbours' VTP management domain indexed by the current device's port ID */ public function neighbourVTPMgmtDomain() { return $this->getSNMP()->subOidWalk( self::OID_CDP_CACHE_NEIGHBOUR_VTP_MGMT_DOMAIN, 15 ); } /** * Get the remote device's interface's native VLAN (indexed by local portId) * * "The remote device's interface's native VLAN, as reported in the * most recent CDP message. The value 0 indicates * no native VLAN field (TLV) was reported in the most * recent CDP message." * * @return array The remote device's interface's native VLAN (indexed by local portId) */ public function neighbourNativeVLAN() { return $this->getSNMP()->subOidWalk( self::OID_CDP_CACHE_NEIGHBOUR_NATIVE_VLAN, 15 ); } /** * Constant for possible value of CDP neighbour duplex * @see neighbourDuplexMode() */ const CDP_CACHE_NEIGHBOUR_DUPLEX_UNKNOWN = 1; /** * Constant for possible value of CDP neighbour duplex * @see neighbourDuplexMode() */ const CDP_CACHE_NEIGHBOUR_DUPLEX_HALF = 2; /** * Constant for possible value of CDP neighbour duplex * @see neighbourDuplexMode() */ const CDP_CACHE_NEIGHBOUR_DUPLEX_FULL = 3; /** * Text representation of CDP capabilities * * @see neighbourDuplexMode() * @var array Text representation of CDP neighbour duplex modes */ public static $CDP_CACHE_NEIGHBOUR_DUPLEXES = array( self::CDP_CACHE_NEIGHBOUR_DUPLEX_UNKNOWN => 'unknown', self::CDP_CACHE_NEIGHBOUR_DUPLEX_HALF => 'half-duplex', self::CDP_CACHE_NEIGHBOUR_DUPLEX_FULL => 'full-duplex' ); /** * Get the remote device's interface's duplex mode (indexed by local portId) * * "The remote device's interface's duplex mode, as reported in the * most recent CDP message. The value unknown(1) indicates * no duplex mode field (TLV) was reported in the most * recent CDP message." * * @param boolean $translate If true, return the string representation via self::$VTP_VLAN_TYPES * @return array The remote device's interface's duplex mode (indexed by local portId) */ public function neighbourDuplexMode( $translate = false ) { $modes = $this->getSNMP()->subOidWalk( self::OID_CDP_CACHE_NEIGHBOUR_DUPLEX, 15 ); if( !$translate ) return $modes; return $this->getSNMP()->translate( $modes, self::$CDP_CACHE_NEIGHBOUR_DUPLEXES ); } /** * Get the remote device's last change time (indexed by local portId) * * "Indicates the time when this cache entry was last changed. * This object is initialised to the current time when the entry * gets created and updated to the current time whenever the value * of any (other) object instance in the corresponding row is * modified." * * @return array The remote device's last change time(indexed by local portId) */ public function neighbourLastChange() { return $this->getSNMP()->subOidWalk( self::OID_CDP_CACHE_NEIGHBOUR_LAST_CHANGE, 15 ); } /** * CDP utility function to get all CDP neighbours and their connected ports. * * Returns an array of neighbours indexed by the neighbour CDP ID. For example: * * * Array * ( * [cr-sw03.ixdub1.opensolutions.ie] => Array * ( * [0] => Array * ( * [localPortId] => 10101 * [localPort] => GigabitEthernet1/0/1 * [remotePort] => GigabitEthernet0/1 * ) * * [1] => Array * ( * [localPortId] => 10102 * [localPort] => GigabitEthernet1/0/2 * [remotePort] => FastEthernet0/2 * ) * * ) * [ ... ] * ) * * @see neighbourId() * @see \OSS\SNMP\MIBS\Interface::descriptions() * @see neighbourPort() * @return array CDP neighbours and their connected ports */ public function neighbours() { $neighbours = array(); foreach( $this->neighbourId() as $localPortId => $neighbourCdpId ) { if( !isset( $neighbours[ $neighbourCdpId ] ) ) { $neighbours[ $neighbourCdpId ] = array(); $count = 0; } else $count = count( $neighbours[ $neighbourCdpId ] ); $neighbours[ $neighbourCdpId ][$count]['localPortId'] = $localPortId; $neighbours[ $neighbourCdpId ][$count]['localPortName'] = $this->getSNMP()->useIface()->names()[$localPortId]; $neighbours[ $neighbourCdpId ][$count]['localPort'] = $this->getSNMP()->useIface()->descriptions()[$localPortId]; try { if( $this->getSNMP()->useLAG()->isAggregatePorts()[$localPortId] ) { $neighbours[ $neighbourCdpId ][$count]['isLAG'] = true; $neighbours[ $neighbourCdpId ][$count]['lagPortId'] = $this->getSNMP()->useLAG()->portAttachedIds()[$localPortId]; $neighbours[ $neighbourCdpId ][$count]['lagPortName'] = $this->getSNMP()->useIface()->names()[ $neighbours[ $neighbourCdpId ][$count]['lagPortId'] ]; } else { $neighbours[ $neighbourCdpId ][$count]['isLAG'] = false; } } catch( \OSS\Exception $e ) { $neighbours[ $neighbourCdpId ][$count]['isLAG'] = false; } $neighbours[ $neighbourCdpId ][$count]['remotePort'] = $this->neighbourPort()[$localPortId]; } return $neighbours; } /** * Recursivily crawls all CDP neighbours to build up a flat array of all devices * indexed by the CDP device id. * * Array form is same as that returned by neighbours() * * @see neighbours() * @param array $devices Unless you're doing something funky, just pass an empty array. This is where the result will be found. * @param string $device CDP device ID of next host to crawl. On first pass, set to null - used internally when recursing * @param array $ignore An array of CDP device IDs to *ignore*. I.e. to not include in recursive crawling * @return array The resultant array of all crawled devices (same as that passed in the @param $devices parameter) */ public function crawl( &$devices = array(), $device = null, $ignore = array() ) { if( !count( $devices ) ) { $device = $this->id(); $devices[ $device ] = $this->neighbours(); } foreach( $devices[ $device ] as $feNeighbour => $feConnections ) { if( in_array( $feNeighbour, $ignore ) ) { if( isset( $devices[ $device ][$feNeighbour] ) ) unset( $devices[ $device ][$feNeighbour] ); continue; } if( !isset( $devices[ $feNeighbour ] ) ) { #echo "Crawling $feNeighbour
\n"; #flush();ob_end_flush(); $snmp = new \OSS\SNMP( $feNeighbour, $this->getSNMP()->getCommunity() ); try { $devices[ $feNeighbour ] = $snmp->useCisco_CDP()->neighbours(); unset( $snmp ); $this->crawl( $devices, $feNeighbour, $ignore ); } catch( \OSS\Exception $e ) { // this device did not respond / have CDP enabled / CDP available - skip unset( $devices[$feNeighbour] ); } } } // find LAGs and more foreach( $devices as $parent => $neighbours ) { foreach( $neighbours as $neighbour => $links ) { foreach( $links as $idx => $link ) { if( $link['isLAG'] and !isset( $link['remoteLagPortId'] ) ) { if( isset( $devices[ $neighbour ][ $parent ] ) ) { foreach( $devices[ $neighbour ][ $parent ] as $_idx => $_link ) { if( $_link['localPort'] == $link['remotePort'] ) { $devices[ $parent ][ $neighbour ][ $idx ][ 'remoteLagPortId' ] = $_link['lagPortId']; $devices[ $parent ][ $neighbour ][ $idx ][ 'remoteLagPortName' ] = $_link['lagPortName']; $devices[ $neighbour ][ $parent ][ $_idx ][ 'remoteLagPortId' ] = $link['lagPortId']; $devices[ $neighbour ][ $parent ][ $_idx ][ 'remoteLagPortName' ] = $link['lagPortName']; break; } } } } if( !isset( $link['remotePortId'] ) ) { if( isset( $devices[ $neighbour ][ $parent ] ) ) { foreach( $devices[ $neighbour ][ $parent ] as $_idx => $_link ) { if( $_link['localPort'] == $link['remotePort'] ) { $devices[ $parent ][ $neighbour ][ $idx ][ 'remotePortId' ] = $_link['localPortId']; $devices[ $parent ][ $neighbour ][ $idx ][ 'remotePortName' ] = $_link['localPortName']; break; } } } } } } } return $devices; } /** * Find the layer 2 topology as an array with no link mentioned more than once. * * Huh? This function: * * * takes the result of crawl() (or calls crawl()) to get the CDP topology * * foreach device, builds an array of device to device links * * SO LONG as that link has already not been accounted for in the other direction. * * I.e. if a link is found A -> B, then the same B -> A link will not be included * * The array returned is, for example: * * [cr-sw04.degkcp.example.ie] => Array * ( * [cd-sw02.degkcp.example.ie] => Array * ( * [GigabitEthernet1/0/3] => Array * ( * [remotePort] => FastEthernet0/1 * [isLAG] => false * ) * * [cr-sw03.degkcp.example.ie] => Array * ( * [GigabitEthernet1/0/23] => Array * ( * [remotePort] => GigabitEthernet1/0/23 * [isLAG] => false * ) * [GigabitEthernet1/0/24] => Array * ( * [remotePort] => GigabitEthernet1/0/24 * [isLAG] => false * ) * ) * ) * * This tells us that cr-sw04(GigabitEthernet1/0/3) is connected to cd-sw02(FastEthernet0/1). * * It also tells us that cr-sw04 has two connections to cr-sw03. * * You'll notice it also tells us if it's a LAG or not. More information can be added as needed. * * @see crawl() * @param array $devices The result of crawl() (if null, this function performs a crawl()) * @return array L2 topology as described above. */ public function linkTopology( $devices = null ) { if( $devices == null ) $devices = $this->crawl(); $links = array(); foreach( $devices as $feDevice => $feNeighbours ) { foreach( $feNeighbours as $fe2Device => $fe2Links ) { foreach( $fe2Links as $fe2Link ) { // have we already accounted for this link on the other side? if( isset( $links[ $fe2Device ][ $feDevice ][ $fe2Link['remotePort'] ] ) ) continue; if( !isset( $links[ $feDevice ] ) ) $links[ $feDevice ] = array(); if( !isset( $links[ $feDevice ][ $fe2Device ] ) ) $links[ $feDevice ][ $fe2Device ] = array(); $links[ $feDevice ][ $fe2Device ][ $fe2Link['localPort'] ] = array(); $links[ $feDevice ][ $fe2Device ][ $fe2Link['localPort'] ][ 'remotePort' ] = $fe2Link['remotePort']; $links[ $feDevice ][ $fe2Device ][ $fe2Link['localPort'] ][ 'isLAG' ] = $fe2Link['isLAG']; $links[ $feDevice ][ $fe2Device ][ $fe2Link['localPort'] ][ 'localPortId' ] = $fe2Link['localPortId']; $links[ $feDevice ][ $fe2Device ][ $fe2Link['localPort'] ][ 'remotePortId' ] = $fe2Link['remotePortId']; if( $fe2Link['isLAG'] ) { $links[ $feDevice ][ $fe2Device ][ $fe2Link['localPort'] ][ 'localLagPortId' ] = $fe2Link['lagPortId']; $links[ $feDevice ][ $fe2Device ][ $fe2Link['localPort'] ][ 'localLagPortName' ] = $fe2Link['lagPortName']; $links[ $feDevice ][ $fe2Device ][ $fe2Link['localPort'] ][ 'remoteLagPortId' ] = $fe2Link['remoteLagPortId']; $links[ $feDevice ][ $fe2Device ][ $fe2Link['localPort'] ][ 'remoteLagPortName' ] = $fe2Link['remoteLagPortName']; $links[ $feDevice ][ $fe2Device ][ $fe2Link['localPort'] ][ 'remotePortId' ] = $fe2Link['remoteLagPortId']; } } } } return $links; } }