152 lines
		
	
	
		
			5.5 KiB
		
	
	
	
		
			Perl
		
	
	
	
	
	
			
		
		
	
	
			152 lines
		
	
	
		
			5.5 KiB
		
	
	
	
		
			Perl
		
	
	
	
	
	
| package App::Netdisco::Worker::Plugin::Discover::Interfaces;
 | |
| 
 | |
| use Dancer ':syntax';
 | |
| use App::Netdisco::Worker::Plugin;
 | |
| use aliased 'App::Netdisco::Worker::Status';
 | |
| 
 | |
| use App::Netdisco::Transport::SNMP ();
 | |
| use Dancer::Plugin::DBIC 'schema';
 | |
| use Encode;
 | |
| 
 | |
| register_worker({ stage => 'early', driver => 'snmp' }, sub {
 | |
|   my ($job, $workerconf) = @_;
 | |
| 
 | |
|   my $device = $job->device;
 | |
|   my $snmp = App::Netdisco::Transport::SNMP->reader_for($device)
 | |
|     or return Status->defer("discover failed: could not SNMP connect to $device");
 | |
| 
 | |
|   my $interfaces     = $snmp->interfaces;
 | |
|   my $i_type         = $snmp->i_type;
 | |
|   my $i_ignore       = $snmp->i_ignore;
 | |
|   my $i_descr        = $snmp->i_description;
 | |
|   my $i_mtu          = $snmp->i_mtu;
 | |
|   my $i_speed        = $snmp->i_speed;
 | |
|   my $i_mac          = $snmp->i_mac;
 | |
|   my $i_up           = $snmp->i_up;
 | |
|   my $i_up_admin     = $snmp->i_up_admin;
 | |
|   my $i_name         = $snmp->i_name;
 | |
|   my $i_duplex       = $snmp->i_duplex;
 | |
|   my $i_duplex_admin = $snmp->i_duplex_admin;
 | |
|   my $i_stp_state    = $snmp->i_stp_state;
 | |
|   my $i_vlan         = $snmp->i_vlan;
 | |
|   my $i_lastchange   = $snmp->i_lastchange;
 | |
|   my $agg_ports      = $snmp->agg_ports;
 | |
| 
 | |
|   # clear the cached uptime and get a new one
 | |
|   my $dev_uptime = $snmp->load_uptime;
 | |
|   if (!defined $dev_uptime) {
 | |
|       error sprintf ' [%s] interfaces - Error! Failed to get uptime from device!',
 | |
|         $device->ip;
 | |
|       return Status->error("discover failed: no uptime from device $device!");
 | |
|   }
 | |
| 
 | |
|   # used to track how many times the device uptime wrapped
 | |
|   my $dev_uptime_wrapped = 0;
 | |
| 
 | |
|   # use SNMP-FRAMEWORK-MIB::snmpEngineTime if available to
 | |
|   # fix device uptime if wrapped
 | |
|   if (defined $snmp->snmpEngineTime) {
 | |
|       $dev_uptime_wrapped = int( $snmp->snmpEngineTime * 100 / 2**32 );
 | |
|       if ($dev_uptime_wrapped > 0) {
 | |
|           info sprintf ' [%s] interface - device uptime wrapped %d times - correcting',
 | |
|             $device->ip, $dev_uptime_wrapped;
 | |
|           $device->uptime( $dev_uptime + $dev_uptime_wrapped * 2**32 );
 | |
|       }
 | |
|   }
 | |
| 
 | |
|   # build device interfaces suitable for DBIC
 | |
|   my %interfaces;
 | |
|   foreach my $entry (keys %$interfaces) {
 | |
|       my $port = $interfaces->{$entry};
 | |
| 
 | |
|       if (not $port) {
 | |
|           debug sprintf ' [%s] interfaces - ignoring %s (no port mapping)',
 | |
|             $device->ip, $entry;
 | |
|           next;
 | |
|       }
 | |
| 
 | |
|       if (scalar grep {$port =~ m/^$_$/} @{setting('ignore_interfaces') || []}) {
 | |
|           debug sprintf
 | |
|             ' [%s] interfaces - ignoring %s (%s) (config:ignore_interfaces)',
 | |
|             $device->ip, $entry, $port;
 | |
|           next;
 | |
|       }
 | |
| 
 | |
|       if (exists $i_ignore->{$entry}) {
 | |
|           debug sprintf ' [%s] interfaces - ignoring %s (%s) (%s)',
 | |
|             $device->ip, $entry, $port, $i_type->{$entry};
 | |
|           next;
 | |
|       }
 | |
| 
 | |
|       my $lc = $i_lastchange->{$entry} || 0;
 | |
|       if (not $dev_uptime_wrapped and $lc > $dev_uptime) {
 | |
|           info sprintf ' [%s] interfaces - device uptime wrapped (%s) - correcting',
 | |
|             $device->ip, $port;
 | |
|           $device->uptime( $dev_uptime + 2**32 );
 | |
|           $dev_uptime_wrapped = 1;
 | |
|       }
 | |
| 
 | |
|       if ($device->is_column_changed('uptime') and $lc) {
 | |
|           if ($lc < $dev_uptime) {
 | |
|               # ambiguous: lastchange could be sysUptime before or after wrap
 | |
|               if ($dev_uptime > 30000 and $lc < 30000) {
 | |
|                   # uptime wrap more than 5min ago but lastchange within 5min
 | |
|                   # assume lastchange was directly after boot -> no action
 | |
|               }
 | |
|               else {
 | |
|                   # uptime wrap less than 5min ago or lastchange > 5min ago
 | |
|                   # to be on safe side, assume lastchange after counter wrap
 | |
|                   debug sprintf
 | |
|                     ' [%s] interfaces - correcting LastChange for %s, assuming sysUptime wrap',
 | |
|                     $device->ip, $port;
 | |
|                   $lc += $dev_uptime_wrapped * 2**32;
 | |
|               }
 | |
|           }
 | |
|       }
 | |
| 
 | |
|       $interfaces{$port} = {
 | |
|           port         => $port,
 | |
|           descr        => $i_descr->{$entry},
 | |
|           up           => $i_up->{$entry},
 | |
|           up_admin     => $i_up_admin->{$entry},
 | |
|           mac          => $i_mac->{$entry},
 | |
|           speed        => $i_speed->{$entry},
 | |
|           mtu          => $i_mtu->{$entry},
 | |
|           name         => Encode::decode('UTF-8', $i_name->{$entry}),
 | |
|           duplex       => $i_duplex->{$entry},
 | |
|           duplex_admin => $i_duplex_admin->{$entry},
 | |
|           stp          => $i_stp_state->{$entry},
 | |
|           type         => $i_type->{$entry},
 | |
|           vlan         => $i_vlan->{$entry},
 | |
|           pvid         => $i_vlan->{$entry},
 | |
|           is_master    => 'false',
 | |
|           slave_of     => undef,
 | |
|           lastchange   => $lc,
 | |
|       };
 | |
|   }
 | |
| 
 | |
|   # must do this after building %interfaces so that we can set is_master
 | |
|   foreach my $sidx (keys %$agg_ports) {
 | |
|       my $slave  = $interfaces->{$sidx} or next;
 | |
|       my $master = $interfaces->{ $agg_ports->{$sidx} } or next;
 | |
|       next unless exists $interfaces{$slave} and exists $interfaces{$master};
 | |
| 
 | |
|       $interfaces{$slave}->{slave_of} = $master;
 | |
|       $interfaces{$master}->{is_master} = 'true';
 | |
|   }
 | |
| 
 | |
|   schema('netdisco')->resultset('DevicePort')->txn_do_locked(sub {
 | |
|     my $gone = $device->ports->delete({keep_nodes => 1});
 | |
|     debug sprintf ' [%s] interfaces - removed %d interfaces',
 | |
|       $device->ip, $gone;
 | |
|     $device->update_or_insert(undef, {for => 'update'});
 | |
|     $device->ports->populate([values %interfaces]);
 | |
|     debug sprintf ' [%s] interfaces - added %d new interfaces',
 | |
|       $device->ip, scalar values %interfaces;
 | |
|   });
 | |
| 
 | |
|   return Status->done("Ended discover for $device");
 | |
| });
 | |
| 
 | |
| true;
 |