256 lines
		
	
	
		
			8.4 KiB
		
	
	
	
		
			Perl
		
	
	
	
	
	
			
		
		
	
	
			256 lines
		
	
	
		
			8.4 KiB
		
	
	
	
		
			Perl
		
	
	
	
	
	
| package App::Netdisco::Web::Plugin::Search::Node;
 | |
| 
 | |
| use Dancer ':syntax';
 | |
| use Dancer::Plugin::DBIC;
 | |
| use Dancer::Plugin::Auth::Extensible;
 | |
| 
 | |
| use NetAddr::IP::Lite ':lower';
 | |
| use Regexp::Common 'net';
 | |
| use NetAddr::MAC ();
 | |
| use POSIX qw/strftime/;
 | |
| 
 | |
| use App::Netdisco::Web::Plugin;
 | |
| use App::Netdisco::Util::Web 'sql_match';
 | |
| 
 | |
| register_search_tab({
 | |
|     tag => 'node',
 | |
|     label => 'Node',
 | |
|     api_endpoint => 1,
 | |
|     api_parameters => [
 | |
|       q => {
 | |
|         description => 'MAC Address or IP Address or Hostname (without Domain Suffix) of a Node (supports SQL or "*" wildcards)',
 | |
|         required => 1,
 | |
|       },
 | |
|       partial => {
 | |
|         description => 'Partially match the "q" parameter (wildcard characters not required)',
 | |
|         type => 'boolean',
 | |
|         default => 'false',
 | |
|       },
 | |
|       deviceports => {
 | |
|         description => 'MAC Address search will include Device Port MACs',
 | |
|         type => 'boolean',
 | |
|         default => 'true',
 | |
|       },
 | |
|       show_vendor => {
 | |
|         description => 'Include interface Vendor in results',
 | |
|         type => 'boolean',
 | |
|         default => 'false',
 | |
|       },
 | |
|       archived => {
 | |
|         description => 'Include archived records in results',
 | |
|         type => 'boolean',
 | |
|         default => 'false',
 | |
|       },
 | |
|       daterange => {
 | |
|         description => 'Date Range in format "YYYY-MM-DD to YYYY-MM-DD"',
 | |
|         default => ('1970-01-01 to '. strftime('%Y-%m-%d', gmtime)),
 | |
|       },
 | |
|       age_invert => {
 | |
|         description => 'Results should NOT be within daterange',
 | |
|         type => 'boolean',
 | |
|         default => 'false',
 | |
|       },
 | |
|       # mac_format is used only in the template (will be IEEE) in results
 | |
|       #mac_format => {
 | |
|       #},
 | |
|       # stamps param is used only in the template (they will be included)
 | |
|       #stamps => {
 | |
|       #},
 | |
|     ],
 | |
| });
 | |
| 
 | |
| # nodes matching the param as an IP or DNS hostname or MAC
 | |
| get '/ajax/content/search/node' => require_login sub {
 | |
|     my $node = param('q');
 | |
|     send_error('Missing node', 400) unless $node;
 | |
|     return unless ($node =~ m/\w/); # need some alphanum at least
 | |
|     content_type('text/html');
 | |
| 
 | |
|     my $agenot = param('age_invert') || '0';
 | |
|     my ( $start, $end ) = param('daterange') =~ m/(\d+-\d+-\d+)/gmx;
 | |
| 
 | |
|     my $mac = NetAddr::MAC->new(mac => ($node || ''));
 | |
|     undef $mac if
 | |
|       ($mac and $mac->as_ieee
 | |
|       and (($mac->as_ieee eq '00:00:00:00:00:00')
 | |
|         or ($mac->as_ieee !~ m/$RE{net}{MAC}/)));
 | |
| 
 | |
|     my @active = (param('archived') ? () : (-bool => 'active'));
 | |
|     my (@times, @wifitimes, @porttimes);
 | |
| 
 | |
|     if ( $start and $end ) {
 | |
|         $start = $start . ' 00:00:00';
 | |
|         $end   = $end   . ' 23:59:59';
 | |
| 
 | |
|         if ($agenot) {
 | |
|             @times = (-or => [
 | |
|               time_first => [ undef ],
 | |
|               time_last => [ { '<', $start }, { '>', $end } ]
 | |
|             ]);
 | |
|             @wifitimes = (-or => [
 | |
|               time_last => [ undef ],
 | |
|               time_last => [ { '<', $start }, { '>', $end } ],
 | |
|             ]);
 | |
|             @porttimes = (-or => [
 | |
|               creation => [ undef ],
 | |
|               creation => [ { '<', $start }, { '>', $end } ]
 | |
|             ]);
 | |
|         }
 | |
|         else {
 | |
|             @times = (-or => [
 | |
|               -and => [
 | |
|                   time_first => undef,
 | |
|                   time_last  => undef,
 | |
|               ],
 | |
|               -and => [
 | |
|                   time_last => { '>=', $start },
 | |
|                   time_last => { '<=', $end },
 | |
|               ],
 | |
|             ]);
 | |
|             @wifitimes = (-or => [
 | |
|               time_last  => undef,
 | |
|               -and => [
 | |
|                   time_last => { '>=', $start },
 | |
|                   time_last => { '<=', $end },
 | |
|               ],
 | |
|             ]);
 | |
|             @porttimes = (-or => [
 | |
|               creation => undef,
 | |
|               -and => [
 | |
|                   creation => { '>=', $start },
 | |
|                   creation => { '<=', $end },
 | |
|               ],
 | |
|             ]);
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     my ($likeval, $likeclause) = sql_match($node, not param('partial'));
 | |
|     my $using_wildcards = (($likeval ne $node) ? 1 : 0);
 | |
| 
 | |
|     my @where_mac =
 | |
|       ($using_wildcards ? \['me.mac::text ILIKE ?', $likeval]
 | |
|                         : ((!defined $mac or $mac->errstr) ? \'0=1' : ('me.mac' => $mac->as_ieee)) );
 | |
| 
 | |
|     my $sightings = schema(vars->{'tenant'})->resultset('Node')
 | |
|       ->search({-and => [@where_mac, @active, @times]}, {
 | |
|           order_by => {'-desc' => 'time_last'},
 | |
|           '+columns' => [
 | |
|             'device.dns',
 | |
|             'device.name',
 | |
|             { time_first_stamp => \"to_char(time_first, 'YYYY-MM-DD HH24:MI')" },
 | |
|             { time_last_stamp =>  \"to_char(time_last, 'YYYY-MM-DD HH24:MI')" },
 | |
|           ],
 | |
|           join => 'device',
 | |
|       });
 | |
| 
 | |
|     my $ips = schema(vars->{'tenant'})->resultset('NodeIp')
 | |
|       ->search({-and => [@where_mac, @active, @times]}, {
 | |
|           order_by => {'-desc' => 'time_last'},
 | |
|           '+columns' => [
 | |
|             'oui.company',
 | |
|             'oui.abbrev',
 | |
|             { time_first_stamp => \"to_char(time_first, 'YYYY-MM-DD HH24:MI')" },
 | |
|             { time_last_stamp =>  \"to_char(time_last, 'YYYY-MM-DD HH24:MI')" },
 | |
|           ],
 | |
|           join => 'oui'
 | |
|       });
 | |
| 
 | |
|     my $netbios = schema(vars->{'tenant'})->resultset('NodeNbt')
 | |
|       ->search({-and => [@where_mac, @active, @times]}, {
 | |
|           order_by => {'-desc' => 'time_last'},
 | |
|           '+columns' => [
 | |
|             'oui.company',
 | |
|             'oui.abbrev',
 | |
|             { time_first_stamp => \"to_char(time_first, 'YYYY-MM-DD HH24:MI')" },
 | |
|             { time_last_stamp =>  \"to_char(time_last, 'YYYY-MM-DD HH24:MI')" },
 | |
|           ],
 | |
|           join => 'oui'
 | |
|       });
 | |
| 
 | |
|     my $wireless = schema(vars->{'tenant'})->resultset('NodeWireless')->search(
 | |
|         { -and => [@where_mac, @wifitimes] },
 | |
|         { order_by   => { '-desc' => 'time_last' },
 | |
|           '+columns' => [
 | |
|             'oui.company',
 | |
|             'oui.abbrev',
 | |
|             {
 | |
|               time_last_stamp => \"to_char(time_last, 'YYYY-MM-DD HH24:MI')"
 | |
|             }],
 | |
|           join => 'oui'
 | |
|         }
 | |
|     );
 | |
| 
 | |
|     my $rs_dp = schema(vars->{'tenant'})->resultset('DevicePort');
 | |
|     if ($sightings->has_rows or $ips->has_rows or $netbios->has_rows) {
 | |
|         my $ports = param('deviceports')
 | |
|           ? $rs_dp->search({ -and => [@where_mac] }, { order_by => { '-desc' => 'creation' }}) : undef;
 | |
| 
 | |
|         return template 'ajax/search/node_by_mac.tt', {
 | |
|           ips       => $ips,
 | |
|           sightings => $sightings,
 | |
|           ports     => $ports,
 | |
|           wireless  => $wireless,
 | |
|           netbios   => $netbios,
 | |
|         }, { layout => 'noop' };
 | |
|     }
 | |
|     else {
 | |
|         my $ports = param('deviceports')
 | |
|           ? $rs_dp->search({ -and => [@where_mac, @porttimes] }, { order_by => { '-desc' => 'creation' }}) : undef;
 | |
| 
 | |
|         if (defined $ports and $ports->has_rows) {
 | |
|             return template 'ajax/search/node_by_mac.tt', {
 | |
|               ips       => $ips,
 | |
|               sightings => $sightings,
 | |
|               ports     => $ports,
 | |
|               wireless  => $wireless,
 | |
|               netbios   => $netbios,
 | |
|             }, { layout => 'noop' };
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     my $set = schema(vars->{'tenant'})->resultset('NodeNbt')
 | |
|         ->search_by_name({nbname => $likeval, @active, @times});
 | |
| 
 | |
|     unless ( $set->has_rows ) {
 | |
|         if ($node =~ m{^(?:$RE{net}{IPv4}|$RE{net}{IPv6})(?:/\d+)?$}i
 | |
|             and my $ip = NetAddr::IP::Lite->new($node)) {
 | |
| 
 | |
|             # search_by_ip() will extract cidr notation if necessary
 | |
|             $set = schema(vars->{'tenant'})->resultset('NodeIp')
 | |
|               ->search_by_ip({ip => $ip, @active, @times});
 | |
|         }
 | |
|         else {
 | |
|             $set = schema(vars->{'tenant'})->resultset('NodeIp')
 | |
|               ->search_by_dns({
 | |
|                   ($using_wildcards ? (dns => $likeval) :
 | |
|                   (dns => "${likeval}.\%",
 | |
|                    suffix => setting('domain_suffix'))),
 | |
|                   @active,
 | |
|                   @times,
 | |
|                 });
 | |
| 
 | |
|             # if the user selects Vendor search opt, then
 | |
|             # we'll try the OUI company name as a fallback
 | |
| 
 | |
|             if (param('show_vendor') and not $set->has_rows) {
 | |
|                 $set = schema(vars->{'tenant'})->resultset('NodeIp')
 | |
|                   ->with_times
 | |
|                   ->search(
 | |
|                     {'oui.company' => { -ilike => ''.sql_match($node)}, @times},
 | |
|                     {'prefetch' => 'oui'},
 | |
|                   );
 | |
|             }
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     return unless $set and $set->has_rows;
 | |
|     $set = $set->search_rs({}, { order_by => 'me.mac' });
 | |
| 
 | |
|     return template 'ajax/search/node_by_ip.tt', {
 | |
|       macs => $set,
 | |
|       archive_filter => {@active},
 | |
|     }, { layout => 'noop' };
 | |
| };
 | |
| 
 | |
| true;
 |