Squashed commit of the following: commit3fe8f383a7Author: Oliver Gorwits <oliver@cpan.org> Date: Mon Mar 11 17:07:42 2019 +0000 add debug lines and tested commit3249739e42Author: Oliver Gorwits <oliver@cpan.org> Date: Mon Mar 11 16:54:11 2019 +0000 change config name to get_credentials commite78558397aAuthor: Oliver Gorwits <oliver@cpan.org> Date: Mon Mar 11 16:51:11 2019 +0000 separate out generic device auth to DeviceAuth module commit249f05165fAuthor: Oliver Gorwits <oliver@cpan.org> Date: Wed Mar 6 18:43:31 2019 +0000 release 2.040007 commite3af64df77Author: Oliver Gorwits <oliver@cpan.org> Date: Wed Mar 6 18:42:47 2019 +0000 #521-redux fix wifi date search commit48857ae300Author: Oliver Gorwits <oliver@cpan.org> Date: Mon Mar 4 12:03:31 2019 +0000 release 2.040006 commite09dab5362Author: Oliver Gorwits <oliver@cpan.org> Date: Mon Mar 4 11:39:12 2019 +0000 #527 update List::MoreUtils version requirement commit6e7de3fff3Author: Oliver Gorwits <oliver@cpan.org> Date: Mon Mar 4 09:59:41 2019 +0000 release 2.040005 commit0c98318a45Author: Oliver Gorwits <oliver@spike.local> Date: Mon Mar 4 09:57:18 2019 +0000 #526 fix discover syntax bug commite9efc45182Author: Oliver Gorwits <oliver@cpan.org> Date: Sun Mar 3 14:56:48 2019 +0000 release 2.040004 commit6cdfd80d10Author: Oliver Gorwits <oliver@cpan.org> Date: Sun Mar 3 14:34:00 2019 +0000 allow undiscovered neighbors report to use discover_{waps,phones} setting commitac381e0802Author: Oliver Gorwits <oliver@cpan.org> Date: Sun Mar 3 14:13:20 2019 +0000 #506 was a red herring commitb83e614c85Author: Oliver Gorwits <oliver@cpan.org> Date: Sun Mar 3 13:00:36 2019 +0000 make discover_{phones,waps} work with LLDP capabilities as well commit189d234b55Author: Oliver Gorwits <oliver@cpan.org> Date: Sun Mar 3 12:47:38 2019 +0000 check discover_no_type and friends earlier on in neighbors list build commit9c956466f3Author: Oliver Gorwits <oliver@cpan.org> Date: Sun Mar 3 12:32:07 2019 +0000 also update default config for new discover_phones and discover_waps settings commit09d29954d2Author: Oliver Gorwits <oliver@cpan.org> Date: Sun Mar 3 12:26:50 2019 +0000 #512 fix regression in phone/wap discovery exclusion commit2bae91f1b6Author: Oliver Gorwits <oliver@cpan.org> Date: Sun Mar 3 12:01:34 2019 +0000 rename match_devicetype() to match_to_setting() commit57cb6ddb70Author: Oliver Gorwits <oliver@cpan.org> Date: Sun Mar 3 09:19:39 2019 +0000 fix for over-eager fix to #506 commitef560fb59aAuthor: Oliver Gorwits <oliver@cpan.org> Date: Sat Mar 2 22:41:40 2019 +0000 #506 relax device renumber so it works for an alias commit7a8bcb094eAuthor: Oliver Gorwits <oliver@cpan.org> Date: Sat Mar 2 22:23:39 2019 +0000 #521 Search Node Date Range not working commita643820a62Author: Oliver Gorwits <oliver@cpan.org> Date: Sat Mar 2 21:54:27 2019 +0000 #428 Port-Channels not showing in netmap commit5ba5bcd295Merge:e7aacddba1f95028Author: Oliver Gorwits <oliver@cpan.org> Date: Sat Mar 2 20:04:11 2019 +0000 Merge branch 'master' of github.com:netdisco/netdisco commite7aacddbc6Author: Oliver Gorwits <oliver@cpan.org> Date: Sat Mar 2 20:01:05 2019 +0000 #498 Map with VLAN filter omits unconnected devices commita1f95028caAuthor: nick n <39005454+inphobia@users.noreply.github.com> Date: Sat Mar 2 19:54:22 2019 +0100 catch up with changes noticed that rc-sshcollector-core received updates to changes, add them here as well. didn't mention #499 & #522 commitce1b847ceaAuthor: Oliver Gorwits <oliver@cpan.org> Date: Sat Mar 2 18:47:44 2019 +0000 fix bug showing no nodes when only one matches in netmap commit78e30a7926Author: Oliver Gorwits <oliver@cpan.org> Date: Sat Mar 2 16:28:15 2019 +0000 #500 filtering in device/ports on native vlan duplicates entries commit9952f0c6c7Author: Oliver Gorwits <oliver@cpan.org> Date: Sat Mar 2 15:02:12 2019 +0000 #499 netdisco-do renumber reports wrong ip (inphobia) commitca3fd8f466Author: Oliver Gorwits <oliver@cpan.org> Date: Sat Mar 2 15:00:18 2019 +0000 #505 device renumber should update device port properties and device skips commit1265bc8470Author: Oliver Gorwits <oliver@cpan.org> Date: Sat Mar 2 14:52:21 2019 +0000 #520 catch slave ports defined without a master commitd4c7579c10Author: Oliver Gorwits <oliver@cpan.org> Date: Sat Mar 2 14:47:49 2019 +0000 #522 TypeAhead.pm can reference empty data (inphobia) commit77decc23b7Author: Oliver Gorwits <oliver@cpan.org> Date: Sat Mar 2 14:45:37 2019 +0000 #514 inconsistent results in ip inventory (inphobia) commit3f211650b8Author: nick n <39005454+inphobia@users.noreply.github.com> Date: Fri Mar 1 12:34:42 2019 +0100 last pieces for db schema upgrade last piece of #510
		
			
				
	
	
		
			277 lines
		
	
	
		
			9.7 KiB
		
	
	
	
		
			Perl
		
	
	
	
	
	
			
		
		
	
	
			277 lines
		
	
	
		
			9.7 KiB
		
	
	
	
		
			Perl
		
	
	
	
	
	
package App::Netdisco::Web::Plugin::Device::Neighbors;
 | 
						||
 | 
						||
use Dancer ':syntax';
 | 
						||
use Dancer::Plugin::Ajax;
 | 
						||
use Dancer::Plugin::DBIC;
 | 
						||
use Dancer::Plugin::Auth::Extensible;
 | 
						||
 | 
						||
use List::Util 'first';
 | 
						||
use List::MoreUtils ();
 | 
						||
use App::Netdisco::Util::Permission 'check_acl_only';
 | 
						||
use App::Netdisco::Web::Plugin;
 | 
						||
 | 
						||
register_device_tab({ tag => 'netmap', label => 'Neighbors' });
 | 
						||
 | 
						||
ajax '/ajax/content/device/netmap' => require_login sub {
 | 
						||
    content_type('text/html');
 | 
						||
    template 'ajax/device/netmap.tt', {}, { layout => undef };
 | 
						||
};
 | 
						||
 | 
						||
ajax '/ajax/data/device/netmappositions' => require_login sub {
 | 
						||
    my $q = param('q');
 | 
						||
    my $qdev = schema('netdisco')->resultset('Device')
 | 
						||
      ->search_for_device($q) or send_error('Bad device', 400);
 | 
						||
 | 
						||
    my $p = param('positions') or send_error('Missing positions', 400);
 | 
						||
    my $positions = from_json($p) or send_error('Bad positions', 400);
 | 
						||
    send_error('Bad positions', 400) unless ref [] eq ref $positions;
 | 
						||
 | 
						||
    my $vlan = param('vlan');
 | 
						||
    undef $vlan if (defined $vlan and $vlan !~ m/^\d+$/);
 | 
						||
 | 
						||
    my $mapshow = param('mapshow');
 | 
						||
    return if !defined $mapshow or $mapshow !~ m/^(?:all|neighbors)$/;
 | 
						||
 | 
						||
    # list of groups selected by user and passed in param
 | 
						||
    my $hgroup = (ref [] eq ref param('hgroup') ? param('hgroup') : [param('hgroup')]);
 | 
						||
    # list of groups validated as real host groups and named host groups
 | 
						||
    my @hgrplist = List::MoreUtils::uniq
 | 
						||
                   grep { exists setting('host_group_displaynames')->{$_} }
 | 
						||
                   grep { exists setting('host_groups')->{$_} }
 | 
						||
                   grep { defined } @{ $hgroup };
 | 
						||
 | 
						||
    # list of locations selected by user and passed in param
 | 
						||
    my $lgroup = (ref [] eq ref param('lgroup') ? param('lgroup') : [param('lgroup')]);
 | 
						||
    my @lgrplist = List::MoreUtils::uniq grep { defined } @{ $lgroup };
 | 
						||
 | 
						||
    my %clean = ();
 | 
						||
    POSITION: foreach my $pos (@$positions) {
 | 
						||
      next unless ref {} eq ref $pos;
 | 
						||
      foreach my $k (qw/ID x y/) {
 | 
						||
        next POSITION unless exists $pos->{$k};
 | 
						||
        next POSITION unless $pos->{$k} =~ m/^[[:word:]\.-]+$/;
 | 
						||
      }
 | 
						||
      $clean{$pos->{ID}} = { x => $pos->{x}, y => $pos->{y} };
 | 
						||
    }
 | 
						||
    return unless scalar keys %clean;
 | 
						||
 | 
						||
    my $posrow = schema('netdisco')->resultset('NetmapPositions')->find({
 | 
						||
      device => (($mapshow eq 'neighbors') ? $qdev->ip : undef),
 | 
						||
      host_groups => \[ '= ?', [host_groups => [sort @hgrplist]] ],
 | 
						||
      locations   => \[ '= ?', [locations   => [sort @lgrplist]] ],
 | 
						||
      vlan => ($vlan || 0),
 | 
						||
    });
 | 
						||
 | 
						||
    if ($posrow) {
 | 
						||
      $posrow->update({ positions => to_json(\%clean) });
 | 
						||
    }
 | 
						||
    else {
 | 
						||
      schema('netdisco')->resultset('NetmapPositions')->create({
 | 
						||
        device => (($mapshow eq 'neighbors') ? $qdev->ip : undef),
 | 
						||
        host_groups => [sort @hgrplist],
 | 
						||
        locations   => [sort @lgrplist],
 | 
						||
        vlan => ($vlan || 0),
 | 
						||
        positions => to_json(\%clean),
 | 
						||
      });
 | 
						||
    }
 | 
						||
};
 | 
						||
 | 
						||
# copied from SNMP::Info to avoid introducing dependency to web frontend
 | 
						||
sub munge_highspeed {
 | 
						||
    my $speed = shift;
 | 
						||
    my $fmt   = "%d Mbps";
 | 
						||
 | 
						||
    if ( $speed > 9999999 ) {
 | 
						||
        $fmt = "%d Tbps";
 | 
						||
        $speed /= 1000000;
 | 
						||
    }
 | 
						||
    elsif ( $speed > 999999 ) {
 | 
						||
        $fmt = "%.1f Tbps";
 | 
						||
        $speed /= 1000000.0;
 | 
						||
    }
 | 
						||
    elsif ( $speed > 9999 ) {
 | 
						||
        $fmt = "%d Gbps";
 | 
						||
        $speed /= 1000;
 | 
						||
    }
 | 
						||
    elsif ( $speed > 999 ) {
 | 
						||
        $fmt = "%.1f Gbps";
 | 
						||
        $speed /= 1000.0;
 | 
						||
    }
 | 
						||
    return sprintf( $fmt, $speed );
 | 
						||
}
 | 
						||
 | 
						||
sub to_speed {
 | 
						||
  my $speed = shift or return '';
 | 
						||
  ($speed = munge_highspeed($speed / 1_000_000)) =~ s/(?:\.0 |bps$)//g;
 | 
						||
  return $speed;
 | 
						||
}
 | 
						||
 | 
						||
sub make_node_infostring {
 | 
						||
  my $node = shift or return '';
 | 
						||
  my $fmt = ('<b>%s</b> is %s <b>%s %s</b><br>running <b>%s %s</b><br>Serial: <b>%s</b><br>'
 | 
						||
    .'Uptime: <b>%s</b><br>Location: <b>%s</b><br>Contact: <b>%s</b>');
 | 
						||
  return sprintf $fmt, $node->ip,
 | 
						||
    ((($node->vendor || '') =~ m/^[aeiou]/i) ? 'an' : 'a'),
 | 
						||
    ucfirst($node->vendor || ''),
 | 
						||
    map {defined $_ ? $_ : ''}
 | 
						||
    map {$node->$_}
 | 
						||
        (qw/model os os_ver serial uptime_age location contact/);
 | 
						||
}
 | 
						||
 | 
						||
sub make_link_infostring {
 | 
						||
  my $link = shift or return '';
 | 
						||
 | 
						||
  my $domain = quotemeta( setting('domain_suffix') || '' );
 | 
						||
  (my $left_name = lc($link->{left_dns} || $link->{left_name} || $link->{left_ip})) =~ s/$domain$//;
 | 
						||
  (my $right_name = lc($link->{right_dns} || $link->{right_name} || $link->{right_ip})) =~ s/$domain$//;
 | 
						||
 | 
						||
  my @zipped = List::MoreUtils::zip6
 | 
						||
    @{$link->{left_port}}, @{$link->{left_descr}},
 | 
						||
    @{$link->{right_port}}, @{$link->{right_descr}};
 | 
						||
 | 
						||
  return join '<br><br>', map { sprintf '<b>%s:%s</b> (%s)<br><b>%s:%s</b> (%s)',
 | 
						||
    $left_name, $_->[0], ($_->[1] || 'no description'),
 | 
						||
    $right_name, $_->[2], ($_->[3] || 'no description') } @zipped;
 | 
						||
}
 | 
						||
 | 
						||
ajax '/ajax/data/device/netmap' => require_login sub {
 | 
						||
    my $q = param('q');
 | 
						||
    my $qdev = schema('netdisco')->resultset('Device')
 | 
						||
      ->search_for_device($q) or send_error('Bad device', 400);
 | 
						||
 | 
						||
    my $vlan = param('vlan');
 | 
						||
    undef $vlan if (defined $vlan and $vlan !~ m/^\d+$/);
 | 
						||
 | 
						||
    my $colorby = (param('colorby') || 'speed');
 | 
						||
    my $mapshow = (param('mapshow') || 'neighbors');
 | 
						||
    $mapshow = 'neighbors' if $mapshow !~ m/^(?:all|neighbors)$/;
 | 
						||
    $mapshow = 'all' unless $qdev->in_storage;
 | 
						||
 | 
						||
    # list of groups selected by user and passed in param
 | 
						||
    my $hgroup = (ref [] eq ref param('hgroup') ? param('hgroup') : [param('hgroup')]);
 | 
						||
    # list of groups validated as real host groups and named host groups
 | 
						||
    my @hgrplist = List::MoreUtils::uniq
 | 
						||
                   grep { exists setting('host_group_displaynames')->{$_} }
 | 
						||
                   grep { exists setting('host_groups')->{$_} }
 | 
						||
                   grep { defined } @{ $hgroup };
 | 
						||
 | 
						||
    # list of locations selected by user and passed in param
 | 
						||
    my $lgroup = (ref [] eq ref param('lgroup') ? param('lgroup') : [param('lgroup')]);
 | 
						||
    my @lgrplist = List::MoreUtils::uniq grep { defined } @{ $lgroup };
 | 
						||
 | 
						||
    my %ok_dev = ();
 | 
						||
    my %logvals = ();
 | 
						||
    my %metadata = ();
 | 
						||
    my %data = ( nodes => [], links => [] );
 | 
						||
    my $domain = quotemeta( setting('domain_suffix') || '' );
 | 
						||
 | 
						||
    # LINKS
 | 
						||
 | 
						||
    my $links = schema('netdisco')->resultset('Virtual::DeviceLinks')->search({
 | 
						||
      ($mapshow eq 'neighbors' ? ( -or => [
 | 
						||
          { left_ip  => $qdev->ip },
 | 
						||
          { right_ip => $qdev->ip },
 | 
						||
      ]) : ())
 | 
						||
    }, { result_class => 'DBIx::Class::ResultClass::HashRefInflator' });
 | 
						||
 | 
						||
    while (my $link = $links->next) {
 | 
						||
      push @{$data{'links'}}, {
 | 
						||
        FROMID => $link->{left_ip},
 | 
						||
        TOID   => $link->{right_ip},
 | 
						||
        INFOSTRING => make_link_infostring($link),
 | 
						||
        SPEED  => to_speed($link->{aggspeed}),
 | 
						||
      };
 | 
						||
 | 
						||
      ++$ok_dev{$link->{left_ip}};
 | 
						||
      ++$ok_dev{$link->{right_ip}};
 | 
						||
    }
 | 
						||
 | 
						||
    # DEVICES (NODES)
 | 
						||
 | 
						||
    my $posrow = schema('netdisco')->resultset('NetmapPositions')->find({
 | 
						||
      device => (($mapshow eq 'neighbors') ? $qdev->ip : undef),
 | 
						||
      host_groups => \[ '= ?', [host_groups => [sort @hgrplist]] ],
 | 
						||
      locations   => \[ '= ?', [locations   => [sort @lgrplist]] ],
 | 
						||
      vlan => ($vlan || 0),
 | 
						||
    });
 | 
						||
    my $pos_for = from_json( $posrow ? $posrow->positions : '{}' );
 | 
						||
 | 
						||
    my $devices = schema('netdisco')->resultset('Device')->search({}, {
 | 
						||
      '+select' => [\'floor(log(throughput.total))'], '+as' => ['log'],
 | 
						||
      join => 'throughput',
 | 
						||
    })->with_times;
 | 
						||
 | 
						||
    # filter by vlan for all or neighbors only
 | 
						||
    if ($vlan) {
 | 
						||
      $devices = $devices->search(
 | 
						||
        { 'vlans.vlan' => $vlan },
 | 
						||
        { join => 'vlans' }
 | 
						||
      );
 | 
						||
    }
 | 
						||
 | 
						||
    DEVICE: while (my $device = $devices->next) {
 | 
						||
      # if in neighbors mode then use %ok_dev to filter
 | 
						||
      next DEVICE if ($device->ip ne $qdev->ip)
 | 
						||
        and ($mapshow eq 'neighbors')
 | 
						||
        and (not $ok_dev{$device->ip}); # showing only neighbors but no link
 | 
						||
 | 
						||
      # if location picked then filter
 | 
						||
      next DEVICE if ((scalar @lgrplist) and ((!defined $device->location)
 | 
						||
        or (0 == scalar grep {$_ eq $device->location} @lgrplist)));
 | 
						||
 | 
						||
      # if host groups picked then use ACLs to filter
 | 
						||
      my $first_hgrp =
 | 
						||
        first { check_acl_only($device, setting('host_groups')->{$_}) } @hgrplist;
 | 
						||
      next DEVICE if ((scalar @hgrplist) and (not $first_hgrp));
 | 
						||
 | 
						||
      # now reset first_hgroup to be the group matching the device, if any
 | 
						||
      $first_hgrp = first { check_acl_only($device, setting('host_groups')->{$_}) }
 | 
						||
                          keys %{ setting('host_group_displaynames') || {} };
 | 
						||
 | 
						||
      ++$logvals{ $device->get_column('log') || 1 };
 | 
						||
      (my $name = lc($device->dns || $device->name || $device->ip)) =~ s/$domain$//;
 | 
						||
 | 
						||
      my %color_lkp = (
 | 
						||
        speed => (($device->get_column('log') || 1) * 1000),
 | 
						||
        hgroup => ($first_hgrp ?
 | 
						||
          setting('host_group_displaynames')->{$first_hgrp} : 'Other'),
 | 
						||
        lgroup => ($device->location || 'Other'),
 | 
						||
      );
 | 
						||
 | 
						||
      my $node = {
 | 
						||
        ID => $device->ip,
 | 
						||
        SIZEVALUE => (param('dynamicsize') ? $color_lkp{speed} : 3000),
 | 
						||
        ((exists $color_lkp{$colorby}) ? (COLORVALUE => $color_lkp{$colorby}) : ()),
 | 
						||
        LABEL => (param('showips') ? ($device->ip .' '. $name) : $name),
 | 
						||
        ORIG_LABEL => $name,
 | 
						||
        INFOSTRING => make_node_infostring($device),
 | 
						||
        LINK => uri_for('/device', {
 | 
						||
          tab => 'netmap',
 | 
						||
          q => $device->ip,
 | 
						||
          firstsearch => 'on',
 | 
						||
        })->path_query,
 | 
						||
      };
 | 
						||
 | 
						||
      if (exists $pos_for->{$device->ip}) {
 | 
						||
        $node->{'fixed'} = 1;
 | 
						||
        $node->{'x'} = $pos_for->{$device->ip}->{'x'};
 | 
						||
        $node->{'y'} = $pos_for->{$device->ip}->{'y'};
 | 
						||
      }
 | 
						||
      else {
 | 
						||
        ++$metadata{'newnodes'};
 | 
						||
      }
 | 
						||
 | 
						||
      push @{$data{'nodes'}}, $node;
 | 
						||
      $metadata{'centernode'} = $device->ip
 | 
						||
        if $qdev and $qdev->in_storage and $device->ip eq $qdev->ip;
 | 
						||
    }
 | 
						||
 | 
						||
    # to help get a sensible range of node sizes
 | 
						||
    $metadata{'numsizes'} = scalar keys %logvals;
 | 
						||
 | 
						||
    content_type('application/json');
 | 
						||
    to_json({ data => \%data, %metadata });
 | 
						||
};
 | 
						||
 | 
						||
true;
 |