SNMPv3 support (needs docs), set default snmpver to 3

This commit is contained in:
Oliver Gorwits
2013-10-21 22:42:59 +01:00
parent b2c261050e
commit 9c9299bf01
8 changed files with 210 additions and 74 deletions

View File

@@ -151,7 +151,7 @@ sub store_device {
} }
my @properties = qw/ my @properties = qw/
snmp_ver snmp_comm snmp_ver
description uptime contact name location description uptime contact name location
layers ports mac serial model layers ports mac serial model
ps1_type ps2_type ps1_status ps2_status ps1_type ps2_type ps1_status ps2_status

View File

@@ -67,15 +67,16 @@ sub do_macsuck {
my $device_ports = {map {($_->port => $_)} my $device_ports = {map {($_->port => $_)}
$device->ports(undef, {prefetch => 'neighbor_alias'})->all}; $device->ports(undef, {prefetch => 'neighbor_alias'})->all};
my $port_macs = get_port_macs($device); my $port_macs = get_port_macs($device);
my $interfaces = $snmp->interfaces;
# get forwarding table data via basic snmp connection # get forwarding table data via basic snmp connection
my $fwtable = { 0 => _walk_fwtable($device, $snmp, $port_macs, $device_ports) }; my $fwtable = { 0 => _walk_fwtable($device, $snmp, $interfaces, $port_macs, $device_ports) };
# ...then per-vlan if supported # ...then per-vlan if supported
my @vlan_list = _get_vlan_list($device, $snmp); my @vlan_list = _get_vlan_list($device, $snmp);
foreach my $vlan (@vlan_list) { foreach my $vlan (@vlan_list) {
snmp_comm_reindex($snmp, $vlan); snmp_comm_reindex($snmp, $device, $vlan);
$fwtable->{$vlan} = _walk_fwtable($device, $snmp, $port_macs, $device_ports); $fwtable->{$vlan} = _walk_fwtable($device, $snmp, $interfaces, $port_macs, $device_ports);
} }
# now it's time to call store_node for every node discovered # now it's time to call store_node for every node discovered
@@ -285,14 +286,13 @@ sub _get_vlan_list {
# walks the forwarding table (BRIDGE-MIB) for the device and returns a # walks the forwarding table (BRIDGE-MIB) for the device and returns a
# table of node entries. # table of node entries.
sub _walk_fwtable { sub _walk_fwtable {
my ($device, $snmp, $port_macs, $device_ports) = @_; my ($device, $snmp, $interfaces, $port_macs, $device_ports) = @_;
my $cache = {}; my $cache = {};
my $fw_mac = $snmp->fw_mac; my $fw_mac = $snmp->fw_mac;
my $fw_port = $snmp->fw_port; my $fw_port = $snmp->fw_port;
my $fw_vlan = $snmp->qb_fw_vlan; my $fw_vlan = $snmp->qb_fw_vlan;
my $bp_index = $snmp->bp_index; my $bp_index = $snmp->bp_index;
my $interfaces = $snmp->interfaces;
# to map forwarding table port to device port we have # to map forwarding table port to device port we have
# fw_port -> bp_index -> interfaces # fw_port -> bp_index -> interfaces

View File

@@ -8,7 +8,7 @@ use base 'DBIx::Class::Schema';
__PACKAGE__->load_namespaces; __PACKAGE__->load_namespaces;
our $VERSION = 28; # schema version used for upgrades, keep as integer our $VERSION = 29; # schema version used for upgrades, keep as integer
use Path::Class; use Path::Class;
use File::Basename; use File::Basename;

View File

@@ -11,6 +11,8 @@ __PACKAGE__->add_columns(
{ data_type => "inet", is_nullable => 0 }, { data_type => "inet", is_nullable => 0 },
"snmp_comm_rw", "snmp_comm_rw",
{ data_type => "text", is_nullable => 1 }, { data_type => "text", is_nullable => 1 },
"snmp_auth_tag",
{ data_type => "text", is_nullable => 1 },
); );
__PACKAGE__->set_primary_key("ip"); __PACKAGE__->set_primary_key("ip");

View File

@@ -0,0 +1,6 @@
BEGIN;
ALTER TABLE community ADD COLUMN snmp_auth_tag text;
COMMIT;

View File

@@ -9,6 +9,7 @@ use base 'Exporter';
our @EXPORT = (); our @EXPORT = ();
our @EXPORT_OK = qw/ our @EXPORT_OK = qw/
get_device get_device
check_acl
check_no check_no
check_only check_only
is_discoverable is_discoverable
@@ -58,13 +59,24 @@ sub get_device {
->find_or_new({ip => $ip}); ->find_or_new({ip => $ip});
} }
sub _check_acl { =head2 check_acl( $ip, \@prefixes )
Given the IP address of a device, returns true if any of the IP prefixes in
C<< \@prefixes >> contains that device, otherwise returns false.
Normally you use C<check_no> and C<check_only>, passing the name of the
configuration setting to load. This helper instead requires not the name of
the setting, but its value.
=cut
sub check_acl {
my ($ip, $config) = @_; my ($ip, $config) = @_;
my $device = get_device($ip) or return 0; my $device = get_device($ip) or return 0;
my $addr = NetAddr::IP::Lite->new($device->ip); my $addr = NetAddr::IP::Lite->new($device->ip);
foreach my $item (@$config) { foreach my $item (@$config) {
if ($item =~ m/^(.*)\s*:\s*(.*)$/) { if ($item =~ m/^([^:]+)\s*:\s*([^:]+)$/) {
my $prop = $1; my $prop = $1;
my $match = $2; my $match = $2;
@@ -123,7 +135,7 @@ sub check_no {
my $config = setting($setting_name) || []; my $config = setting($setting_name) || [];
return 0 unless scalar @$config; return 0 unless scalar @$config;
return _check_acl($ip, $config); return check_acl($ip, $config);
} }
=head2 check_only( $ip, $setting_name ) =head2 check_only( $ip, $setting_name )
@@ -163,7 +175,7 @@ sub check_only {
my $config = setting($setting_name) || []; my $config = setting($setting_name) || [];
return 1 unless scalar @$config; return 1 unless scalar @$config;
return _check_acl($ip, $config); return check_acl($ip, $config);
} }
=head2 is_discoverable( $ip, $device_type? ) =head2 is_discoverable( $ip, $device_type? )

View File

@@ -1,7 +1,7 @@
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 qw/get_device check_no/; use App::Netdisco::Util::Device qw/get_device check_acl check_no/;
use SNMP::Info; use SNMP::Info;
use Try::Tiny; use Try::Tiny;
@@ -41,7 +41,7 @@ Returns C<undef> if the connection fails.
=cut =cut
sub snmp_connect { _snmp_connect_generic(@_, 'community') } sub snmp_connect { _snmp_connect_generic(@_, 'read') }
=head2 snmp_connect_rw( $ip ) =head2 snmp_connect_rw( $ip )
@@ -52,15 +52,15 @@ Returns C<undef> if the connection fails.
=cut =cut
sub snmp_connect_rw { _snmp_connect_generic(@_, 'community_rw') } sub snmp_connect_rw { _snmp_connect_generic(@_, 'write') }
sub _snmp_connect_generic { sub _snmp_connect_generic {
my $ip = shift; my ($ip, $mode) = @_;
$mode ||= 'read';
# get device details from db # get device details from db
my $device = get_device($ip); my $device = get_device($ip);
# TODO: only supporing v2c at the moment
my %snmp_args = ( my %snmp_args = (
DestHost => $device->ip, DestHost => $device->ip,
Retries => (setting('snmpretries') || 2), Retries => (setting('snmpretries') || 2),
@@ -87,11 +87,6 @@ sub _snmp_connect_generic {
$snmp_args{BulkWalk} = 0; $snmp_args{BulkWalk} = 0;
} }
# TODO: add version force support
# use existing SNMP version or try 2, 1
my @versions = (($device->snmp_ver || setting('snmpver') || 2));
push @versions, 1;
# use existing or new device class # use existing or new device class
my @classes = ('SNMP::Info'); my @classes = ('SNMP::Info');
if ($device->snmp_class) { if ($device->snmp_class) {
@@ -101,15 +96,12 @@ sub _snmp_connect_generic {
$snmp_args{AutoSpecity} = 1; $snmp_args{AutoSpecity} = 1;
} }
# TODO: add version force support
# use existing SNMP version or try 3, 2, 1
my @versions = reverse (1 .. ($device->snmp_ver || setting('snmpver') || 3));
# get the community string(s) # get the community string(s)
my $comm_type = pop; my @communities = _build_communities($device, $mode);
my @communities = @{ setting($comm_type) || []};
unshift @communities, $device->snmp_comm
if defined $device->snmp_comm
and defined $comm_type and $comm_type eq 'community';
unshift @communities, $device->community->snmp_comm_rw
if eval { $device->community->snmp_comm_rw }
and defined $comm_type and $comm_type eq 'community_rw';
my $info = undef; my $info = undef;
VERSION: foreach my $ver (@versions) { VERSION: foreach my $ver (@versions) {
@@ -119,14 +111,11 @@ sub _snmp_connect_generic {
next unless $class; next unless $class;
COMMUNITY: foreach my $comm (@communities) { COMMUNITY: foreach my $comm (@communities) {
next unless $comm; next if $ver eq 3 and exists $comm->{community};
next if $ver ne 3 and !exists $comm->{community};
$info = _try_connect($ver, $class, $comm, \%snmp_args);
if ($comm_type eq 'community_rw') {
_try_write($info, $comm, $device) or next COMMUNITY;
}
my %local_args = (%snmp_args, Version => $ver);
$info = _try_connect($device, $class, $comm, $mode, \%local_args);
last VERSION if $info; last VERSION if $info;
} }
} }
@@ -135,52 +124,32 @@ sub _snmp_connect_generic {
return $info; return $info;
} }
sub _try_write {
my ($info, $comm, $device) = @_;
my $happy = 0;
try {
debug sprintf '[%s] try_write with comm: %s', $device->ip, $comm;
$info->clear_cache;
my $rv = $info->set_location( $info->location );
$device->update_or_create_related('community', {snmp_comm_rw => $comm})
if $device->in_storage;
$happy = 1 if $rv;
}
catch {
debug $_;
};
return $happy;
}
sub _try_connect { sub _try_connect {
my ($ver, $class, $comm, $snmp_args) = @_; my ($device, $class, $comm, $mode, $snmp_args) = @_;
my %comm_args = _mk_info_commargs($comm);
my $info = undef; my $info = undef;
try { try {
debug debug
sprintf '[%s] try_connect with ver: %s, class: %s, comm: %s', sprintf '[%s] try_connect with ver: %s, class: %s, comm: %s',
$snmp_args->{DestHost}, $ver, $class, $comm; $snmp_args->{DestHost}, $snmp_args->{Version}, $class,
($comm->{community} || "v3user:$comm->{user}");
eval "require $class"; eval "require $class";
$info = $class->new(%$snmp_args, Version => $ver, Community => $comm); $info = $class->new(%$snmp_args, %comm_args);
undef $info unless ( $info = ($mode eq 'read' ? _try_read($info, $device, $comm)
(not defined $info->error) : _try_write($info, $device, $comm));
and defined $info->uptime
and ($info->layers or $info->description)
and $info->class
);
# first time a device is discovered, re-instantiate into specific class # first time a device is discovered, re-instantiate into specific class
if ($info and $info->device_type ne $class) { if ($info and $info->device_type ne $class) {
$class = $info->device_type; $class = $info->device_type;
debug debug
sprintf '[%s] try_connect with ver: %s, new class: %s, comm: %s', sprintf '[%s] try_connect with ver: %s, new class: %s, comm: %s',
$snmp_args->{DestHost}, $ver, $class, $comm; $snmp_args->{DestHost}, $snmp_args->{Version}, $class,
($comm->{community} || "v3user:$comm->{user}");
eval "require $class"; eval "require $class";
$info = $class->new(%$snmp_args, Version => $ver, Community => $comm); $info = $class->new(%$snmp_args, %comm_args);
} }
} }
catch { catch {
@@ -190,6 +159,75 @@ sub _try_connect {
return $info; return $info;
} }
sub _try_read {
my ($info, $device, $comm) = @_;
undef $info unless (
(not defined $info->error)
and defined $info->uptime
and ($info->layers or $info->description)
and $info->class
);
if ($device->in_storage) {
# read strings are tried before writes, so this should not accidentally
# store a write string if there's a good read string also in config.
$device->update({snmp_comm => $comm->{community}})
if exists $comm->{community};
$device->update_or_create_related('community',
{snmp_auth_tag => $comm->{tag}}) if $comm->{tag};
}
else {
$device->set_column(snmp_comm => $comm->{community})
if exists $comm->{community};
}
return $info;
}
sub _try_write {
my ($info, $device, $comm) = @_;
# SNMP v1/2 R/W must be able to read as well (?)
$info = _try_read($info, $device, $comm)
if exists $comm->{community};
return unless $info;
$info->set_location( $info->load_location )
or return undef;
if ($device->in_storage) {
# one of these two cols must be set
$device->update_or_create_related('community', {
($comm->{tag} ? (snmp_auth_tag => $comm->{tag}) : ()),
(exists $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,
AuthProto => uc (eval { $comm->{auth}->{proto} } || 'MD5'),
AuthPass => (eval { $comm->{auth}->{pass} } || ''),
PrivProto => uc (eval { $comm->{priv}->{proto} } || 'DES'),
PrivPass => (eval { $comm->{priv}->{pass} } || ''),
);
}
sub _build_mibdirs { sub _build_mibdirs {
my $home = (setting('mibhome') || dir(($ENV{NETDISCO_HOME} || $ENV{HOME}), 'netdisco-mibs')); my $home = (setting('mibhome') || dir(($ENV{NETDISCO_HOME} || $ENV{HOME}), 'netdisco-mibs'));
return map { dir($home, $_)->stringify } return map { dir($home, $_)->stringify }
@@ -203,7 +241,78 @@ sub _get_mibdirs_content {
return \@list; return \@list;
} }
=head2 snmp_comm_reindex( $snmp, $vlan ) sub _build_communities {
my ($device, $mode) = @_;
$mode ||= 'read';
my $config = (setting('snmp_auth') || []);
my $stored_tag = eval { $device->community->snmp_auth_tag };
my $snmp_comm_rw = eval { $device->community->snmp_comm_rw };
my @communities = ();
# first try last-known-good
push @communities, {read => 1, community => $device->snmp_comm}
if defined $device->snmp_comm and $mode eq 'read';
# first try last-known-good
push @communities, {read => 1, write => 1, community => $snmp_comm_rw}
if $snmp_comm_rw and $mode eq 'write';
# new style snmp config
foreach my $stanza (@$config) {
# user tagged
my $tag = '';
if (1 == scalar keys %$stanza) {
$tag = (keys %$stanza)[0];
$stanza = $stanza->{$tag};
# corner case: untagged lone community
if ($tag eq 'community') {
$tag = $stanza;
$stanza = {community => $tag};
}
}
# defaults
$stanza->{tag} ||= $tag;
$stanza->{read} = 1 if !exists $stanza->{read};
$stanza->{only} ||= ['any'];
$stanza->{only} = [$stanza->{only}] if ref '' eq ref $stanza->{only};
die "error: config: snmpv3 stanza in snmp_auth must have a tag\n"
if not $stanza->{tag}
and !exists $stanza->{community};
if ($stanza->{$mode} and check_acl($device, $stanza->{only})) {
if ($stored_tag and $stored_tag eq $stanza->{tag}) {
# last known-good by tag
unshift @communities, $stanza
}
else {
push @communities, $stanza
}
}
}
# legacy config (note: read strings tried before write)
if ($mode eq 'read') {
push @communities, map {{
read => 1,
community => $_,
}} @{setting('community') || []};
}
else {
push @communities, map {{
read => 1,
write => 1,
community => $_,
}} @{setting('community_rw') || []};
}
return @communities;
}
=head2 snmp_comm_reindex( $snmp, $device, $vlan )
Takes an established L<SNMP::Info> instance and makes a fresh connection using Takes an established L<SNMP::Info> instance and makes a fresh connection using
community indexing, with the given C<$vlan> ID. Works for all SNMP versions. community indexing, with the given C<$vlan> ID. Works for all SNMP versions.
@@ -211,15 +320,22 @@ community indexing, with the given C<$vlan> ID. Works for all SNMP versions.
=cut =cut
sub snmp_comm_reindex { sub snmp_comm_reindex {
my ($snmp, $vlan) = @_; my ($snmp, $device, $vlan) = @_;
my $ver = $snmp->snmp_ver; my $ver = $snmp->snmp_ver;
my $comm = $snmp->snmp_comm;
if ($ver == 3) { if ($ver == 3) {
$snmp->update(Context => "vlan-$vlan"); my $prefix = '';
my @comms = _build_communities($device, 'read');
foreach my $c (@comms) {
next unless $c->{tag}
and $c->{tag} eq (eval { $device->community->snmp_auth_tag } || '');
$prefix = $c->{context_prefix} and last;
}
my $prefix ||= 'vlan-';
$snmp->update(Context => ($prefix . $vlan));
} }
else { else {
my $comm = $snmp->snmp_comm;
$snmp->update(Community => $comm . '@' . $vlan); $snmp->update(Community => $comm . '@' . $vlan);
} }
} }

View File

@@ -68,7 +68,7 @@ bulkwalk_off: false
bulkwalk_no: [] bulkwalk_no: []
bulkwalk_repeaters: 20 bulkwalk_repeaters: 20
nonincreasing: false nonincreasing: false
snmpver: 2 snmpver: 3
snmptimeout: 1000000 snmptimeout: 1000000
snmpretries: 2 snmpretries: 2
discover_no: [] discover_no: []