SNMPv3 support (needs docs), set default snmpver to 3
This commit is contained in:
@@ -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
|
||||||
|
|||||||
@@ -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
|
||||||
|
|||||||
@@ -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;
|
||||||
|
|||||||
@@ -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");
|
||||||
|
|
||||||
|
|||||||
@@ -0,0 +1,6 @@
|
|||||||
|
|
||||||
|
BEGIN;
|
||||||
|
|
||||||
|
ALTER TABLE community ADD COLUMN snmp_auth_tag text;
|
||||||
|
|
||||||
|
COMMIT;
|
||||||
@@ -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? )
|
||||||
|
|||||||
@@ -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);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -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: []
|
||||||
|
|||||||
Reference in New Issue
Block a user