From 2ae86b9fcdb3f88a5c975d5bb91fbd0f0c35ee5d Mon Sep 17 00:00:00 2001 From: Oliver Gorwits Date: Thu, 4 Aug 2022 22:56:23 +0100 Subject: [PATCH] make the SNMP Browser faster and colour branches with data --- lib/App/Netdisco/DB.pm | 2 +- lib/App/Netdisco/DB/Result/SNMPObject.pm | 2 + .../DB/Result/Virtual/FilteredSNMPObject.pm | 21 ++++-- .../Netdisco/DB/Result/Virtual/OidChildren.pm | 4 ++ lib/App/Netdisco/Web/Plugin/Device/SNMP.pm | 65 ++++++------------- lib/App/Netdisco/Worker/Plugin/LoadMIBs.pm | 13 +++- .../App-Netdisco-DB-72-73-PostgreSQL.sql | 5 ++ 7 files changed, 59 insertions(+), 53 deletions(-) create mode 100644 share/schema_versions/App-Netdisco-DB-72-73-PostgreSQL.sql diff --git a/lib/App/Netdisco/DB.pm b/lib/App/Netdisco/DB.pm index 84ebd55f..183e0f0d 100644 --- a/lib/App/Netdisco/DB.pm +++ b/lib/App/Netdisco/DB.pm @@ -11,7 +11,7 @@ __PACKAGE__->load_namespaces( ); our # try to hide from kwalitee - $VERSION = 72; # schema version used for upgrades, keep as integer + $VERSION = 73; # schema version used for upgrades, keep as integer use Path::Class; use File::ShareDir 'dist_dir'; diff --git a/lib/App/Netdisco/DB/Result/SNMPObject.pm b/lib/App/Netdisco/DB/Result/SNMPObject.pm index f5aefd05..2360ebcc 100644 --- a/lib/App/Netdisco/DB/Result/SNMPObject.pm +++ b/lib/App/Netdisco/DB/Result/SNMPObject.pm @@ -21,6 +21,8 @@ __PACKAGE__->add_columns( { data_type => "text", is_nullable => 1 }, "index", { data_type => "text[]", is_nullable => 1, default_value => \"'{}'::text[]" }, + "num_children", + { data_type => "integer", is_nullable => 0, default_value => \'0' }, ); __PACKAGE__->set_primary_key("oid"); diff --git a/lib/App/Netdisco/DB/Result/Virtual/FilteredSNMPObject.pm b/lib/App/Netdisco/DB/Result/Virtual/FilteredSNMPObject.pm index 56cf8e5f..927c0dba 100644 --- a/lib/App/Netdisco/DB/Result/Virtual/FilteredSNMPObject.pm +++ b/lib/App/Netdisco/DB/Result/Virtual/FilteredSNMPObject.pm @@ -12,11 +12,20 @@ __PACKAGE__->table("filtered_snmp_object"); __PACKAGE__->result_source_instance->is_virtual(1); __PACKAGE__->result_source_instance->view_definition(< ? + AND db.oid LIKE so.oid || '.%'))) + + WHERE array_length(so.oid_parts,1) = ? + AND so.oid LIKE ?::text || '.%' + + GROUP BY so.oid, so.oid_parts, so.mib, so.leaf, so.type, so.access, so.index, so.num_children ENDSQL ); @@ -29,6 +38,8 @@ __PACKAGE__->add_columns( 'type' => { data_type => 'text' }, 'access' => { data_type => 'text' }, 'index' => { data_type => 'text[]' }, + 'num_children' => { data_type => 'integer' }, + 'browser' => { data_type => 'integer' }, ); 1; diff --git a/lib/App/Netdisco/DB/Result/Virtual/OidChildren.pm b/lib/App/Netdisco/DB/Result/Virtual/OidChildren.pm index 97575393..0e3903d4 100644 --- a/lib/App/Netdisco/DB/Result/Virtual/OidChildren.pm +++ b/lib/App/Netdisco/DB/Result/Virtual/OidChildren.pm @@ -8,6 +8,10 @@ use base 'DBIx::Class::Core'; __PACKAGE__->table_class('DBIx::Class::ResultSource::View'); +# NO LONGER USED BY NETDISCO +# (PostgreSQL cannot handle GROUP BY using ARRAY element where +# element ID is a bind value) + __PACKAGE__->table("oid_children"); __PACKAGE__->result_source_instance->is_virtual(1); __PACKAGE__->result_source_instance->view_definition(<resultset('Virtual::GenericReport')->result_source; - $rs->view_definition($query); - $rs->remove_columns($rs->columns); - $rs->add_columns(qw/part children/); - - my %kids = map { ($base .'.'. $_->{part}) => $_ } - schema('netdisco')->resultset('Virtual::GenericReport') - ->search(undef, { - result_class => 'DBIx::Class::ResultClass::HashRefInflator', - bind => [$ip, $base], - })->hri->all; - - return [{ - text => 'No SNMP data for this device.', - children => \0, - state => { disabled => \1 }, - icon => 'icon-search', - }] unless scalar keys %kids; - my %meta = map { ('.'. join '.', @{$_->{oid_parts}}) => $_ } schema('netdisco')->resultset('Virtual::FilteredSNMPObject') ->search({}, { bind => [ + $ip, + (scalar @parts + 1), + (scalar @parts + 1), $base, - (scalar @parts + 1), - [[ map {$_->{part}} values %kids ]], - (scalar @parts + 1), ] })->hri->all; my @items = map {{ id => $_, - text => ($meta{$_}->{leaf} .' ('. $kids{$_}->{part} .')'), + text => ($meta{$_}->{leaf} .' ('. $meta{$_}->{oid_parts}->[-1] .')'), - # for nodes with only one child, recurse to prefetch... - children => (($kids{$_}->{children} == 1) - ? _get_snmp_data($ip, ("${base}.". $kids{$_}->{part}), 1) - : ($kids{$_}->{children} ? \1 : \0)), + ($meta{$_}->{browser} ? (icon => 'icon-folder-close text-info') + : (icon => 'icon-folder-close-alt muted')), + (scalar @{$meta{$_}->{index}} + ? (icon => 'icon-th'.($meta{$_}->{browser} ? ' text-info' : ' muted')) : ()), + + (($meta{$_}->{num_children} == 0 and ($meta{$_}->{type} or $meta{$_}->{oid_parts}->[-1] == 0)) + ? (icon => 'icon-leaf'.($meta{$_}->{browser} ? ' text-info' : ' muted')) : ()), + + # jstree will async call to expand these, and while it's possible + # for us to prefetch by calling _get_snmp_data() and passing to + # children, it's much slower UX. async is better for search especially + children => ($meta{$_}->{num_children} ? \1 : \0), + # and set the display to open to show the single child - state => { opened => ( ($recurse or $kids{$_}->{children} == 1) - ? \1 - : \0 ) }, + state => { opened => (($meta{$_}->{num_children} == 1) ? \1 : \0 ) }, - ($kids{$_}->{children} ? () : (icon => 'icon-leaf')), - (scalar @{$meta{$_}->{index}} ? (icon => 'icon-th') : ()), - }} sort {$kids{$a}->{part} <=> $kids{$b}->{part}} keys %kids; + }} sort {$meta{$a}->{oid_parts}->[-1] <=> $meta{$b}->{oid_parts}->[-1]} keys %meta; return \@items; } diff --git a/lib/App/Netdisco/Worker/Plugin/LoadMIBs.pm b/lib/App/Netdisco/Worker/Plugin/LoadMIBs.pm index 7094e049..fc75d816 100644 --- a/lib/App/Netdisco/Worker/Plugin/LoadMIBs.pm +++ b/lib/App/Netdisco/Worker/Plugin/LoadMIBs.pm @@ -19,13 +19,20 @@ register_worker({ phase => 'main' }, sub { my @report = read_lines(catfile($home, qw(EXTRAS reports all_oids)), 'latin-1'); my @browser = (); + my %children = (); + foreach my $line (@report) { my ($oid, $qual_leaf, $type, $access, $index) = split m/,/, $line; next unless defined $oid and defined $qual_leaf; + my ($mib, $leaf) = split m/::/, $qual_leaf; + my @oid_parts = grep {length} (split m/\./, $oid); + ++$children{ join '.', '', @oid_parts[0 .. (@oid_parts - 2)] } + if scalar @oid_parts > 1; + push @browser, { oid => $oid, - oid_parts => [ grep {length} (split m/\./, $oid) ], + oid_parts => [ @oid_parts ], mib => $mib, leaf => $leaf, type => $type, @@ -34,6 +41,10 @@ register_worker({ phase => 'main' }, sub { }; } + foreach my $row (@browser) { + $row->{num_children} = $children{ $row->{oid} } || 0; + } + debug sprintf "loadmibs - loaded %d objects from netdisco-mibs", scalar @browser; diff --git a/share/schema_versions/App-Netdisco-DB-72-73-PostgreSQL.sql b/share/schema_versions/App-Netdisco-DB-72-73-PostgreSQL.sql new file mode 100644 index 00000000..6f7eb52d --- /dev/null +++ b/share/schema_versions/App-Netdisco-DB-72-73-PostgreSQL.sql @@ -0,0 +1,5 @@ +BEGIN; + +ALTER TABLE snmp_object ADD COLUMN "num_children" integer DEFAULT 0 NOT NULL; + +COMMIT;