change SNMP to be a cached transport singleton
This commit is contained in:
@@ -100,7 +100,7 @@ unless ($action) {
|
|||||||
use NetAddr::IP qw/:rfc3021 :lower/;
|
use NetAddr::IP qw/:rfc3021 :lower/;
|
||||||
use Dancer ':script';
|
use Dancer ':script';
|
||||||
|
|
||||||
use App::Netdisco::Util::SNMP ();
|
use App::Netdisco::Core::Transport::SNMP;
|
||||||
use App::Netdisco::Util::Device
|
use App::Netdisco::Util::Device
|
||||||
qw/get_device delete_device renumber_device/;
|
qw/get_device delete_device renumber_device/;
|
||||||
|
|
||||||
@@ -138,7 +138,7 @@ unless ($action) {
|
|||||||
$extra = $class;
|
$extra = $class;
|
||||||
undef $class;
|
undef $class;
|
||||||
}
|
}
|
||||||
my $i = App::Netdisco::Util::SNMP::snmp_connect($device, $class);
|
my $i = App::Netdisco::Core::Transport::SNMP->instance->reader_for($device, $class);
|
||||||
Data::Printer::p($i->$extra);
|
Data::Printer::p($i->$extra);
|
||||||
return ('done', sprintf "Showed %s response from %s.", $extra, $device->ip);
|
return ('done', sprintf "Showed %s response from %s.", $extra, $device->ip);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
package App::Netdisco::Backend::Worker::Interactive::DeviceActions;
|
package App::Netdisco::Backend::Worker::Interactive::DeviceActions;
|
||||||
|
|
||||||
use App::Netdisco::Util::SNMP 'snmp_connect_rw';
|
use App::Netdisco::Core::Transport::SNMP;
|
||||||
use App::Netdisco::Util::Device 'get_device';
|
use App::Netdisco::Util::Device 'get_device';
|
||||||
use App::Netdisco::Backend::Util ':all';
|
use App::Netdisco::Backend::Util ':all';
|
||||||
|
|
||||||
@@ -22,7 +22,7 @@ sub _set_device_generic {
|
|||||||
$data ||= '';
|
$data ||= '';
|
||||||
|
|
||||||
# snmp connect using rw community
|
# snmp connect using rw community
|
||||||
my $info = snmp_connect_rw($ip)
|
my $info = App::Netdisco::Core::Transport::SNMP->instance->writer_for($ip)
|
||||||
or return job_defer("Failed to connect to device [$ip] to update $slot");
|
or return job_defer("Failed to connect to device [$ip] to update $slot");
|
||||||
|
|
||||||
my $method = 'set_'. $slot;
|
my $method = 'set_'. $slot;
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
package App::Netdisco::Backend::Worker::Interactive::PortActions;
|
package App::Netdisco::Backend::Worker::Interactive::PortActions;
|
||||||
|
|
||||||
use App::Netdisco::Util::Port ':all';
|
use App::Netdisco::Util::Port ':all';
|
||||||
use App::Netdisco::Util::SNMP 'snmp_connect_rw';
|
use App::Netdisco::Core::Transport::SNMP;
|
||||||
use App::Netdisco::Util::Device 'get_device';
|
use App::Netdisco::Util::Device 'get_device';
|
||||||
use App::Netdisco::Backend::Util ':all';
|
use App::Netdisco::Backend::Util ':all';
|
||||||
|
|
||||||
@@ -74,7 +74,7 @@ sub _set_port_generic {
|
|||||||
|
|
||||||
if ($device->vendor ne 'netdisco') {
|
if ($device->vendor ne 'netdisco') {
|
||||||
# snmp connect using rw community
|
# snmp connect using rw community
|
||||||
my $info = snmp_connect_rw($ip)
|
my $info = App::Netdisco::Core::Transport::SNMP->instance->writer_for($ip)
|
||||||
or return job_defer("Failed to connect to device [$ip] to control port");
|
or return job_defer("Failed to connect to device [$ip] to control port");
|
||||||
|
|
||||||
my $iid = get_iid($info, $port)
|
my $iid = get_iid($info, $port)
|
||||||
@@ -127,7 +127,7 @@ sub power {
|
|||||||
$data = 'false' if $data =~ m/^(off|no|down)$/;
|
$data = 'false' if $data =~ m/^(off|no|down)$/;
|
||||||
|
|
||||||
# snmp connect using rw community
|
# snmp connect using rw community
|
||||||
my $info = snmp_connect_rw($ip)
|
my $info = App::Netdisco::Core::Transport::SNMP->instance->writer_for($ip)
|
||||||
or return job_defer("Failed to connect to device [$ip] to control power");
|
or return job_defer("Failed to connect to device [$ip] to control power");
|
||||||
|
|
||||||
my $powerid = get_powerid($info, $port)
|
my $powerid = get_powerid($info, $port)
|
||||||
|
|||||||
@@ -2,7 +2,7 @@ package App::Netdisco::Backend::Worker::Poller::Common;
|
|||||||
|
|
||||||
use Dancer qw/:moose :syntax :script/;
|
use Dancer qw/:moose :syntax :script/;
|
||||||
|
|
||||||
use App::Netdisco::Util::SNMP 'snmp_connect';
|
use App::Netdisco::Core::Transport::SNMP;
|
||||||
use App::Netdisco::Util::Device 'get_device';
|
use App::Netdisco::Util::Device 'get_device';
|
||||||
use App::Netdisco::Backend::Util ':all';
|
use App::Netdisco::Backend::Util ':all';
|
||||||
use App::Netdisco::JobQueue qw/jq_queued jq_insert/;
|
use App::Netdisco::JobQueue qw/jq_queued jq_insert/;
|
||||||
@@ -63,7 +63,7 @@ sub _single_body {
|
|||||||
return job_defer("$job_type deferred: $host is not ${job_type}able");
|
return job_defer("$job_type deferred: $host is not ${job_type}able");
|
||||||
}
|
}
|
||||||
|
|
||||||
my $snmp = snmp_connect($device);
|
my $snmp = App::Netdisco::Core::Transport::SNMP->instance->reader_for($device);
|
||||||
if (!defined $snmp) {
|
if (!defined $snmp) {
|
||||||
return job_defer("$job_type failed: could not SNMP connect to $host");
|
return job_defer("$job_type failed: could not SNMP connect to $host");
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -2,7 +2,7 @@ package App::Netdisco::Backend::Worker::Poller::Device;
|
|||||||
|
|
||||||
use Dancer qw/:moose :syntax :script/;
|
use Dancer qw/:moose :syntax :script/;
|
||||||
|
|
||||||
use App::Netdisco::Util::SNMP 'snmp_connect';
|
use App::Netdisco::Core::Transport::SNMP;
|
||||||
use App::Netdisco::Util::Device qw/get_device is_discoverable_now/;
|
use App::Netdisco::Util::Device qw/get_device is_discoverable_now/;
|
||||||
use App::Netdisco::Core::Discover ':all';
|
use App::Netdisco::Core::Discover ':all';
|
||||||
use App::Netdisco::Backend::Util ':all';
|
use App::Netdisco::Backend::Util ':all';
|
||||||
@@ -59,7 +59,7 @@ sub discover {
|
|||||||
return job_defer("discover deferred: $host is not discoverable");
|
return job_defer("discover deferred: $host is not discoverable");
|
||||||
}
|
}
|
||||||
|
|
||||||
my $snmp = snmp_connect($device);
|
my $snmp = App::Netdisco::Core::Transport::SNMP->instance->reader_for($device);
|
||||||
if (!defined $snmp) {
|
if (!defined $snmp) {
|
||||||
return job_defer("discover failed: could not SNMP connect to $host");
|
return job_defer("discover failed: could not SNMP connect to $host");
|
||||||
}
|
}
|
||||||
|
|||||||
283
lib/App/Netdisco/Core/Transport/SNMP.pm
Normal file
283
lib/App/Netdisco/Core/Transport/SNMP.pm
Normal file
@@ -0,0 +1,283 @@
|
|||||||
|
package App::Netdisco::Core::Transport::SNMP;
|
||||||
|
|
||||||
|
use Dancer qw/:syntax :script/;
|
||||||
|
use App::Netdisco::Util::SNMP 'build_communities';
|
||||||
|
use App::Netdisco::Util::Device 'get_device';
|
||||||
|
use App::Netdisco::Util::Permission ':all';
|
||||||
|
|
||||||
|
use SNMP::Info;
|
||||||
|
use Try::Tiny;
|
||||||
|
use Module::Load ();
|
||||||
|
use Path::Class 'dir';
|
||||||
|
|
||||||
|
use base 'Dancer::Object::Singleton';
|
||||||
|
|
||||||
|
=head1 NAME
|
||||||
|
|
||||||
|
App::Netdisco::Core::Transport::SNMP
|
||||||
|
|
||||||
|
=head1 DESCRIPTION
|
||||||
|
|
||||||
|
Singleton for SNMP connections. Returns cached L<SNMP::Info> instance for a
|
||||||
|
given device IP, or else undef. Prefix calls to this class with:
|
||||||
|
|
||||||
|
App::Netdisco::Core::Transport::SNMP->instance()
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
__PACKAGE__->attributes(qw/ readers writers /);
|
||||||
|
|
||||||
|
sub init {
|
||||||
|
my ( $class, $self ) = @_;
|
||||||
|
$self->readers( {} );
|
||||||
|
$self->writers( {} );
|
||||||
|
return $self;
|
||||||
|
}
|
||||||
|
|
||||||
|
=head1 reader_for( $ip, $useclass? )
|
||||||
|
|
||||||
|
Given an IP address, returns an L<SNMP::Info> instance configured for and
|
||||||
|
connected to that device. The IP can be any on the device, and the management
|
||||||
|
interface will be connected to.
|
||||||
|
|
||||||
|
If the device is known to Netdisco and there is a cached SNMP community
|
||||||
|
string, this will be tried first, and then other community string(s) from the
|
||||||
|
application configuration will be tried.
|
||||||
|
|
||||||
|
If C<$useclass> is provided, it will be used as the L<SNMP::Info> device
|
||||||
|
class instead of the class in the Netdisco database.
|
||||||
|
|
||||||
|
Returns C<undef> if the connection fails.
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub reader_for {
|
||||||
|
my ($self, $ip, $useclass) = @_;
|
||||||
|
my $device = get_device($ip) or return undef;
|
||||||
|
return $self->readers->{$device->ip}
|
||||||
|
if exists $self->readers->{$device->ip};
|
||||||
|
debug sprintf 'snmp reader cache warm: [%s]', $device->ip;
|
||||||
|
return ($self->readers->{$device->ip}
|
||||||
|
= _snmp_connect_generic('read', $device, $useclass));
|
||||||
|
}
|
||||||
|
|
||||||
|
=head2 writer_for( $ip, $useclass? )
|
||||||
|
|
||||||
|
Same as C<reader_for> but uses the read-write community string(s) from the
|
||||||
|
application configuration file.
|
||||||
|
|
||||||
|
Returns C<undef> if the connection fails.
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub writer_for {
|
||||||
|
my ($self, $ip, $useclass) = @_;
|
||||||
|
my $device = get_device($ip) or return undef;
|
||||||
|
return $self->writers->{$device->ip}
|
||||||
|
if exists $self->writers->{$device->ip};
|
||||||
|
debug sprintf 'snmp writer cache warm: [%s]', $device->ip;
|
||||||
|
return ($self->writers->{$device->ip}
|
||||||
|
= _snmp_connect_generic('write', $device, $useclass));
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
sub _snmp_connect_generic {
|
||||||
|
my ($mode, $device, $useclass) = @_;
|
||||||
|
$mode ||= 'read';
|
||||||
|
|
||||||
|
my %snmp_args = (
|
||||||
|
AutoSpecify => 0,
|
||||||
|
DestHost => $device->ip,
|
||||||
|
# 0 is falsy. Using || with snmpretries equal to 0 will set retries to 2.
|
||||||
|
# check if the setting is 0. If not, use the default value of 2.
|
||||||
|
Retries => (setting('snmpretries') || setting('snmpretries') == 0 ? 0 : 2),
|
||||||
|
Timeout => (setting('snmptimeout') || 1000000),
|
||||||
|
NonIncreasing => (setting('nonincreasing') || 0),
|
||||||
|
BulkWalk => ((defined setting('bulkwalk_off') && setting('bulkwalk_off'))
|
||||||
|
? 0 : 1),
|
||||||
|
BulkRepeaters => (setting('bulkwalk_repeaters') || 20),
|
||||||
|
MibDirs => [ _build_mibdirs() ],
|
||||||
|
IgnoreNetSNMPConf => 1,
|
||||||
|
Debug => ($ENV{INFO_TRACE} || 0),
|
||||||
|
DebugSNMP => ($ENV{SNMP_TRACE} || 0),
|
||||||
|
);
|
||||||
|
|
||||||
|
# an override for bulkwalk
|
||||||
|
$snmp_args{BulkWalk} = 0 if check_acl_no($device, 'bulkwalk_no');
|
||||||
|
|
||||||
|
# further protect against buggy Net-SNMP, and disable bulkwalk
|
||||||
|
if ($snmp_args{BulkWalk}
|
||||||
|
and ($SNMP::VERSION eq '5.0203' || $SNMP::VERSION eq '5.0301')) {
|
||||||
|
|
||||||
|
warning sprintf
|
||||||
|
"[%s] turning off BulkWalk due to buggy Net-SNMP - please upgrade!",
|
||||||
|
$device->ip;
|
||||||
|
$snmp_args{BulkWalk} = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
# get the community string(s)
|
||||||
|
my @communities = build_communities($device, $mode);
|
||||||
|
|
||||||
|
# which SNMP versions to try and in what order
|
||||||
|
my @versions =
|
||||||
|
( check_acl_no($device->ip, 'snmpforce_v3') ? (3)
|
||||||
|
: check_acl_no($device->ip, 'snmpforce_v2') ? (2)
|
||||||
|
: check_acl_no($device->ip, 'snmpforce_v1') ? (1)
|
||||||
|
: (reverse (1 .. (setting('snmpver') || 3))) );
|
||||||
|
|
||||||
|
# use existing or new device class
|
||||||
|
my @classes = ($useclass || 'SNMP::Info');
|
||||||
|
if ($device->snmp_class and not $useclass) {
|
||||||
|
unshift @classes, $device->snmp_class;
|
||||||
|
}
|
||||||
|
|
||||||
|
my $info = undef;
|
||||||
|
COMMUNITY: foreach my $comm (@communities) {
|
||||||
|
next unless $comm;
|
||||||
|
|
||||||
|
VERSION: foreach my $ver (@versions) {
|
||||||
|
next unless $ver;
|
||||||
|
|
||||||
|
next if $ver eq 3 and exists $comm->{community};
|
||||||
|
next if $ver ne 3 and !exists $comm->{community};
|
||||||
|
|
||||||
|
CLASS: foreach my $class (@classes) {
|
||||||
|
next unless $class;
|
||||||
|
|
||||||
|
my %local_args = (%snmp_args, Version => $ver);
|
||||||
|
$info = _try_connect($device, $class, $comm, $mode, \%local_args,
|
||||||
|
($useclass ? 0 : 1) );
|
||||||
|
last COMMUNITY if $info;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return $info;
|
||||||
|
}
|
||||||
|
|
||||||
|
sub _try_connect {
|
||||||
|
my ($device, $class, $comm, $mode, $snmp_args, $reclass) = @_;
|
||||||
|
my %comm_args = _mk_info_commargs($comm);
|
||||||
|
my $debug_comm = '<hidden>';
|
||||||
|
if ($ENV{SHOW_COMMUNITY}) {
|
||||||
|
$debug_comm = ($comm->{community} ||
|
||||||
|
(sprintf 'v3:%s:%s/%s', ($comm->{user},
|
||||||
|
($comm->{auth}->{proto} || 'noAuth'),
|
||||||
|
($comm->{priv}->{proto} || 'noPriv'))) );
|
||||||
|
}
|
||||||
|
my $info = undef;
|
||||||
|
|
||||||
|
try {
|
||||||
|
debug
|
||||||
|
sprintf '[%s] try_connect with ver: %s, class: %s, comm: %s',
|
||||||
|
$snmp_args->{DestHost}, $snmp_args->{Version}, $class, $debug_comm;
|
||||||
|
Module::Load::load $class;
|
||||||
|
|
||||||
|
$info = $class->new(%$snmp_args, %comm_args) or return;
|
||||||
|
$info = ($mode eq 'read' ? _try_read($info, $device, $comm)
|
||||||
|
: _try_write($info, $device, $comm));
|
||||||
|
|
||||||
|
# first time a device is discovered, re-instantiate into specific class
|
||||||
|
if ($reclass and $info and $info->device_type ne $class) {
|
||||||
|
$class = $info->device_type;
|
||||||
|
debug
|
||||||
|
sprintf '[%s] try_connect with ver: %s, new class: %s, comm: %s',
|
||||||
|
$snmp_args->{DestHost}, $snmp_args->{Version}, $class, $debug_comm;
|
||||||
|
|
||||||
|
Module::Load::load $class;
|
||||||
|
$info = $class->new(%$snmp_args, %comm_args);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
catch {
|
||||||
|
debug $_;
|
||||||
|
};
|
||||||
|
|
||||||
|
return $info;
|
||||||
|
}
|
||||||
|
|
||||||
|
sub _try_read {
|
||||||
|
my ($info, $device, $comm) = @_;
|
||||||
|
|
||||||
|
return undef unless (
|
||||||
|
(not defined $info->error)
|
||||||
|
and defined $info->uptime
|
||||||
|
and ($info->layers or $info->description)
|
||||||
|
and $info->class
|
||||||
|
);
|
||||||
|
|
||||||
|
$device->in_storage
|
||||||
|
? $device->update({snmp_ver => $info->snmp_ver})
|
||||||
|
: $device->set_column(snmp_ver => $info->snmp_ver);
|
||||||
|
|
||||||
|
if ($comm->{community}) {
|
||||||
|
$device->in_storage
|
||||||
|
? $device->update({snmp_comm => $comm->{community}})
|
||||||
|
: $device->set_column(snmp_comm => $comm->{community});
|
||||||
|
}
|
||||||
|
|
||||||
|
# regardless of device in storage, save the hint
|
||||||
|
$device->update_or_create_related('community',
|
||||||
|
{snmp_auth_tag_read => $comm->{tag}}) if $comm->{tag};
|
||||||
|
|
||||||
|
return $info;
|
||||||
|
}
|
||||||
|
|
||||||
|
sub _try_write {
|
||||||
|
my ($info, $device, $comm) = @_;
|
||||||
|
|
||||||
|
my $loc = $info->load_location;
|
||||||
|
$info->set_location($loc) or return undef;
|
||||||
|
return undef unless ($loc eq $info->load_location);
|
||||||
|
|
||||||
|
$device->in_storage
|
||||||
|
? $device->update({snmp_ver => $info->snmp_ver})
|
||||||
|
: $device->set_column(snmp_ver => $info->snmp_ver);
|
||||||
|
|
||||||
|
# one of these two cols must be set
|
||||||
|
$device->update_or_create_related('community', {
|
||||||
|
($comm->{tag} ? (snmp_auth_tag_write => $comm->{tag}) : ()),
|
||||||
|
($comm->{community} ? (snmp_comm_rw => $comm->{community}) : ()),
|
||||||
|
});
|
||||||
|
|
||||||
|
return $info;
|
||||||
|
}
|
||||||
|
|
||||||
|
sub _mk_info_commargs {
|
||||||
|
my $comm = shift;
|
||||||
|
return () unless ref {} eq ref $comm and scalar keys %$comm;
|
||||||
|
|
||||||
|
return (Community => $comm->{community})
|
||||||
|
if exists $comm->{community};
|
||||||
|
|
||||||
|
my $seclevel =
|
||||||
|
(exists $comm->{auth} ?
|
||||||
|
(exists $comm->{priv} ? 'authPriv' : 'authNoPriv' )
|
||||||
|
: 'noAuthNoPriv');
|
||||||
|
|
||||||
|
return (
|
||||||
|
SecName => $comm->{user},
|
||||||
|
SecLevel => $seclevel,
|
||||||
|
( exists $comm->{auth} ? (
|
||||||
|
AuthProto => uc ($comm->{auth}->{proto} || 'MD5'),
|
||||||
|
AuthPass => ($comm->{auth}->{pass} || ''),
|
||||||
|
( exists $comm->{priv} ? (
|
||||||
|
PrivProto => uc ($comm->{priv}->{proto} || 'DES'),
|
||||||
|
PrivPass => ($comm->{priv}->{pass} || ''),
|
||||||
|
) : ()),
|
||||||
|
) : ()),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
sub _build_mibdirs {
|
||||||
|
my $home = (setting('mibhome') || dir(($ENV{NETDISCO_HOME} || $ENV{HOME}), 'netdisco-mibs'));
|
||||||
|
return map { dir($home, $_)->stringify }
|
||||||
|
@{ setting('mibdirs') || _get_mibdirs_content($home) };
|
||||||
|
}
|
||||||
|
|
||||||
|
sub _get_mibdirs_content {
|
||||||
|
my $home = shift;
|
||||||
|
my @list = map {s|$home/||; $_} grep {m/[a-z0-9]/} grep {-d} glob("$home/*");
|
||||||
|
return \@list;
|
||||||
|
}
|
||||||
|
|
||||||
|
true;
|
||||||
@@ -1,18 +1,12 @@
|
|||||||
package App::Netdisco::Util::SNMP;
|
package App::Netdisco::Util::SNMP;
|
||||||
|
|
||||||
use Dancer qw/:syntax :script/;
|
use Dancer qw/:syntax :script/;
|
||||||
use App::Netdisco::Util::Device 'get_device';
|
|
||||||
use App::Netdisco::Util::Permission ':all';
|
use App::Netdisco::Util::Permission ':all';
|
||||||
|
|
||||||
use SNMP::Info;
|
|
||||||
use Try::Tiny;
|
|
||||||
use Module::Load ();
|
|
||||||
use Path::Class 'dir';
|
|
||||||
|
|
||||||
use base 'Exporter';
|
use base 'Exporter';
|
||||||
our @EXPORT = ();
|
our @EXPORT = ();
|
||||||
our @EXPORT_OK = qw/
|
our @EXPORT_OK = qw/
|
||||||
snmp_connect snmp_connect_rw snmp_comm_reindex
|
build_communities snmp_comm_reindex
|
||||||
/;
|
/;
|
||||||
our %EXPORT_TAGS = (all => \@EXPORT_OK);
|
our %EXPORT_TAGS = (all => \@EXPORT_OK);
|
||||||
|
|
||||||
@@ -22,244 +16,22 @@ App::Netdisco::Util::SNMP
|
|||||||
|
|
||||||
=head1 DESCRIPTION
|
=head1 DESCRIPTION
|
||||||
|
|
||||||
A set of helper subroutines to support parts of the Netdisco application.
|
Helper functions for L<SNMP::Info> instances.
|
||||||
|
|
||||||
There are no default exports, however the C<:all> tag will export all
|
There are no default exports, however the C<:all> tag will export all
|
||||||
subroutines.
|
subroutines.
|
||||||
|
|
||||||
=head1 EXPORT_OK
|
=head1 EXPORT_OK
|
||||||
|
|
||||||
=head2 snmp_connect( $ip )
|
=head2 build_communities( $device, $mode )
|
||||||
|
|
||||||
Given an IP address, returns an L<SNMP::Info> instance configured for and
|
Takes an established L<SNMP::Info> instance and returns a set of potential
|
||||||
connected to that device. The IP can be any on the device, and the management
|
SNMP community authentication settings that are configured in Netdisco, for
|
||||||
interface will be connected to.
|
the given mode (C<read> or C<write>).
|
||||||
|
|
||||||
If the device is known to Netdisco and there is a cached SNMP community
|
|
||||||
string, this will be tried first, and then other community string(s) from the
|
|
||||||
application configuration will be tried.
|
|
||||||
|
|
||||||
Returns C<undef> if the connection fails.
|
|
||||||
|
|
||||||
=cut
|
=cut
|
||||||
|
|
||||||
sub snmp_connect { _snmp_connect_generic('read', @_) }
|
sub build_communities {
|
||||||
|
|
||||||
=head2 snmp_connect_rw( $ip )
|
|
||||||
|
|
||||||
Same as C<snmp_connect> but uses the read-write community string(s) from the
|
|
||||||
application configuration file.
|
|
||||||
|
|
||||||
Returns C<undef> if the connection fails.
|
|
||||||
|
|
||||||
=cut
|
|
||||||
|
|
||||||
sub snmp_connect_rw { _snmp_connect_generic('write', @_) }
|
|
||||||
|
|
||||||
sub _snmp_connect_generic {
|
|
||||||
my ($mode, $ip, $useclass) = @_;
|
|
||||||
$mode ||= 'read';
|
|
||||||
|
|
||||||
# get device details from db
|
|
||||||
my $device = get_device($ip);
|
|
||||||
|
|
||||||
my %snmp_args = (
|
|
||||||
AutoSpecify => 0,
|
|
||||||
DestHost => $device->ip,
|
|
||||||
# 0 is falsy. Using || with snmpretries equal to 0 will set retries to 2.
|
|
||||||
# check if the setting is 0. If not, use the default value of 2.
|
|
||||||
Retries => (setting('snmpretries') || setting('snmpretries') == 0 ? 0 : 2),
|
|
||||||
Timeout => (setting('snmptimeout') || 1000000),
|
|
||||||
NonIncreasing => (setting('nonincreasing') || 0),
|
|
||||||
BulkWalk => ((defined setting('bulkwalk_off') && setting('bulkwalk_off'))
|
|
||||||
? 0 : 1),
|
|
||||||
BulkRepeaters => (setting('bulkwalk_repeaters') || 20),
|
|
||||||
MibDirs => [ _build_mibdirs() ],
|
|
||||||
IgnoreNetSNMPConf => 1,
|
|
||||||
Debug => ($ENV{INFO_TRACE} || 0),
|
|
||||||
DebugSNMP => ($ENV{SNMP_TRACE} || 0),
|
|
||||||
);
|
|
||||||
|
|
||||||
# an override for bulkwalk
|
|
||||||
$snmp_args{BulkWalk} = 0 if check_acl_no($device, 'bulkwalk_no');
|
|
||||||
|
|
||||||
# further protect against buggy Net-SNMP, and disable bulkwalk
|
|
||||||
if ($snmp_args{BulkWalk}
|
|
||||||
and ($SNMP::VERSION eq '5.0203' || $SNMP::VERSION eq '5.0301')) {
|
|
||||||
|
|
||||||
warning sprintf
|
|
||||||
"[%s] turning off BulkWalk due to buggy Net-SNMP - please upgrade!",
|
|
||||||
$device->ip;
|
|
||||||
$snmp_args{BulkWalk} = 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
# get the community string(s)
|
|
||||||
my @communities = _build_communities($device, $mode);
|
|
||||||
|
|
||||||
# which SNMP versions to try and in what order
|
|
||||||
my @versions =
|
|
||||||
( check_acl_no($device->ip, 'snmpforce_v3') ? (3)
|
|
||||||
: check_acl_no($device->ip, 'snmpforce_v2') ? (2)
|
|
||||||
: check_acl_no($device->ip, 'snmpforce_v1') ? (1)
|
|
||||||
: (reverse (1 .. (setting('snmpver') || 3))) );
|
|
||||||
|
|
||||||
# use existing or new device class
|
|
||||||
my @classes = ($useclass || 'SNMP::Info');
|
|
||||||
if ($device->snmp_class and not $useclass) {
|
|
||||||
unshift @classes, $device->snmp_class;
|
|
||||||
}
|
|
||||||
|
|
||||||
my $info = undef;
|
|
||||||
COMMUNITY: foreach my $comm (@communities) {
|
|
||||||
next unless $comm;
|
|
||||||
|
|
||||||
VERSION: foreach my $ver (@versions) {
|
|
||||||
next unless $ver;
|
|
||||||
|
|
||||||
next if $ver eq 3 and exists $comm->{community};
|
|
||||||
next if $ver ne 3 and !exists $comm->{community};
|
|
||||||
|
|
||||||
CLASS: foreach my $class (@classes) {
|
|
||||||
next unless $class;
|
|
||||||
|
|
||||||
my %local_args = (%snmp_args, Version => $ver);
|
|
||||||
$info = _try_connect($device, $class, $comm, $mode, \%local_args,
|
|
||||||
($useclass ? 0 : 1) );
|
|
||||||
last COMMUNITY if $info;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return $info;
|
|
||||||
}
|
|
||||||
|
|
||||||
sub _try_connect {
|
|
||||||
my ($device, $class, $comm, $mode, $snmp_args, $reclass) = @_;
|
|
||||||
my %comm_args = _mk_info_commargs($comm);
|
|
||||||
my $debug_comm = '<hidden>';
|
|
||||||
if ($ENV{SHOW_COMMUNITY}) {
|
|
||||||
$debug_comm = ($comm->{community} ||
|
|
||||||
(sprintf 'v3:%s:%s/%s', ($comm->{user},
|
|
||||||
($comm->{auth}->{proto} || 'noAuth'),
|
|
||||||
($comm->{priv}->{proto} || 'noPriv'))) );
|
|
||||||
}
|
|
||||||
my $info = undef;
|
|
||||||
|
|
||||||
try {
|
|
||||||
debug
|
|
||||||
sprintf '[%s] try_connect with ver: %s, class: %s, comm: %s',
|
|
||||||
$snmp_args->{DestHost}, $snmp_args->{Version}, $class, $debug_comm;
|
|
||||||
Module::Load::load $class;
|
|
||||||
|
|
||||||
$info = $class->new(%$snmp_args, %comm_args) or return;
|
|
||||||
$info = ($mode eq 'read' ? _try_read($info, $device, $comm)
|
|
||||||
: _try_write($info, $device, $comm));
|
|
||||||
|
|
||||||
# first time a device is discovered, re-instantiate into specific class
|
|
||||||
if ($reclass and $info and $info->device_type ne $class) {
|
|
||||||
$class = $info->device_type;
|
|
||||||
debug
|
|
||||||
sprintf '[%s] try_connect with ver: %s, new class: %s, comm: %s',
|
|
||||||
$snmp_args->{DestHost}, $snmp_args->{Version}, $class, $debug_comm;
|
|
||||||
|
|
||||||
Module::Load::load $class;
|
|
||||||
$info = $class->new(%$snmp_args, %comm_args);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
catch {
|
|
||||||
debug $_;
|
|
||||||
};
|
|
||||||
|
|
||||||
return $info;
|
|
||||||
}
|
|
||||||
|
|
||||||
sub _try_read {
|
|
||||||
my ($info, $device, $comm) = @_;
|
|
||||||
|
|
||||||
return undef unless (
|
|
||||||
(not defined $info->error)
|
|
||||||
and defined $info->uptime
|
|
||||||
and ($info->layers or $info->description)
|
|
||||||
and $info->class
|
|
||||||
);
|
|
||||||
|
|
||||||
$device->in_storage
|
|
||||||
? $device->update({snmp_ver => $info->snmp_ver})
|
|
||||||
: $device->set_column(snmp_ver => $info->snmp_ver);
|
|
||||||
|
|
||||||
if ($comm->{community}) {
|
|
||||||
$device->in_storage
|
|
||||||
? $device->update({snmp_comm => $comm->{community}})
|
|
||||||
: $device->set_column(snmp_comm => $comm->{community});
|
|
||||||
}
|
|
||||||
|
|
||||||
# regardless of device in storage, save the hint
|
|
||||||
$device->update_or_create_related('community',
|
|
||||||
{snmp_auth_tag_read => $comm->{tag}}) if $comm->{tag};
|
|
||||||
|
|
||||||
return $info;
|
|
||||||
}
|
|
||||||
|
|
||||||
sub _try_write {
|
|
||||||
my ($info, $device, $comm) = @_;
|
|
||||||
|
|
||||||
my $loc = $info->load_location;
|
|
||||||
$info->set_location($loc) or return undef;
|
|
||||||
return undef unless ($loc eq $info->load_location);
|
|
||||||
|
|
||||||
$device->in_storage
|
|
||||||
? $device->update({snmp_ver => $info->snmp_ver})
|
|
||||||
: $device->set_column(snmp_ver => $info->snmp_ver);
|
|
||||||
|
|
||||||
# one of these two cols must be set
|
|
||||||
$device->update_or_create_related('community', {
|
|
||||||
($comm->{tag} ? (snmp_auth_tag_write => $comm->{tag}) : ()),
|
|
||||||
($comm->{community} ? (snmp_comm_rw => $comm->{community}) : ()),
|
|
||||||
});
|
|
||||||
|
|
||||||
return $info;
|
|
||||||
}
|
|
||||||
|
|
||||||
sub _mk_info_commargs {
|
|
||||||
my $comm = shift;
|
|
||||||
return () unless ref {} eq ref $comm and scalar keys %$comm;
|
|
||||||
|
|
||||||
return (Community => $comm->{community})
|
|
||||||
if exists $comm->{community};
|
|
||||||
|
|
||||||
my $seclevel =
|
|
||||||
(exists $comm->{auth} ?
|
|
||||||
(exists $comm->{priv} ? 'authPriv' : 'authNoPriv' )
|
|
||||||
: 'noAuthNoPriv');
|
|
||||||
|
|
||||||
return (
|
|
||||||
SecName => $comm->{user},
|
|
||||||
SecLevel => $seclevel,
|
|
||||||
( exists $comm->{auth} ? (
|
|
||||||
AuthProto => uc ($comm->{auth}->{proto} || 'MD5'),
|
|
||||||
AuthPass => ($comm->{auth}->{pass} || ''),
|
|
||||||
( exists $comm->{priv} ? (
|
|
||||||
PrivProto => uc ($comm->{priv}->{proto} || 'DES'),
|
|
||||||
PrivPass => ($comm->{priv}->{pass} || ''),
|
|
||||||
) : ()),
|
|
||||||
) : ()),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
sub _build_mibdirs {
|
|
||||||
my $home = (setting('mibhome') || dir(($ENV{NETDISCO_HOME} || $ENV{HOME}), 'netdisco-mibs'));
|
|
||||||
return map { dir($home, $_)->stringify }
|
|
||||||
@{ setting('mibdirs') || _get_mibdirs_content($home) };
|
|
||||||
}
|
|
||||||
|
|
||||||
sub _get_mibdirs_content {
|
|
||||||
my $home = shift;
|
|
||||||
# warning 'Netdisco SNMP work will be slow - loading ALL MIBs. Consider setting mibdirs.';
|
|
||||||
my @list = map {s|$home/||; $_} grep {-d} glob("$home/*");
|
|
||||||
return \@list;
|
|
||||||
}
|
|
||||||
|
|
||||||
sub _build_communities {
|
|
||||||
my ($device, $mode) = @_;
|
my ($device, $mode) = @_;
|
||||||
$mode ||= 'read';
|
$mode ||= 'read';
|
||||||
my $seen_tags = {}; # for cleaning community table
|
my $seen_tags = {}; # for cleaning community table
|
||||||
@@ -421,4 +193,4 @@ sub snmp_comm_reindex {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
1;
|
true;
|
||||||
|
|||||||
Reference in New Issue
Block a user