implement device search!
This commit is contained in:
		| @@ -82,6 +82,7 @@ __PACKAGE__->set_primary_key("ip"); | |||||||
| # Created by DBIx::Class::Schema::Loader v0.07015 @ 2012-01-07 14:20:02 | # Created by DBIx::Class::Schema::Loader v0.07015 @ 2012-01-07 14:20:02 | ||||||
| # DO NOT MODIFY THIS OR ANYTHING ABOVE! md5sum:671/XuuvsO2aMB1+IRWFjg | # DO NOT MODIFY THIS OR ANYTHING ABOVE! md5sum:671/XuuvsO2aMB1+IRWFjg | ||||||
|  |  | ||||||
|  | __PACKAGE__->has_many( device_ips => 'Netdisco::DB::Result::DeviceIp', 'ip' ); | ||||||
| __PACKAGE__->has_many( vlans => 'Netdisco::DB::Result::DeviceVlan', 'ip' ); | __PACKAGE__->has_many( vlans => 'Netdisco::DB::Result::DeviceVlan', 'ip' ); | ||||||
| __PACKAGE__->has_many( ports => 'Netdisco::DB::Result::DevicePort', 'ip' ); | __PACKAGE__->has_many( ports => 'Netdisco::DB::Result::DevicePort', 'ip' ); | ||||||
| __PACKAGE__->has_many( | __PACKAGE__->has_many( | ||||||
|   | |||||||
| @@ -34,6 +34,6 @@ __PACKAGE__->set_primary_key("ip", "alias"); | |||||||
| # Created by DBIx::Class::Schema::Loader v0.07015 @ 2012-01-07 14:20:02 | # Created by DBIx::Class::Schema::Loader v0.07015 @ 2012-01-07 14:20:02 | ||||||
| # DO NOT MODIFY THIS OR ANYTHING ABOVE! md5sum:/ugGtBSGyrJ7s6yqJ9bclQ | # DO NOT MODIFY THIS OR ANYTHING ABOVE! md5sum:/ugGtBSGyrJ7s6yqJ9bclQ | ||||||
|  |  | ||||||
|  | __PACKAGE__->belongs_to( device => 'Netdisco::DB::Result::Device', 'ip' ); | ||||||
|  |  | ||||||
| # You can replace this text with custom code or comments, and it will be preserved on regeneration |  | ||||||
| 1; | 1; | ||||||
|   | |||||||
| @@ -1,6 +1,97 @@ | |||||||
| package Netdisco::DB::ResultSet::Device; | package Netdisco::DB::ResultSet::Device; | ||||||
| use base 'DBIx::Class::ResultSet'; | use base 'DBIx::Class::ResultSet'; | ||||||
|  |  | ||||||
|  | use NetAddr::IP::Lite ':lower'; | ||||||
|  |  | ||||||
|  | # finds distinct values of a col for use in form selections | ||||||
|  | sub get_distinct { | ||||||
|  |   my ($set, $col) = @_; | ||||||
|  |   return $set unless $col; | ||||||
|  |  | ||||||
|  |   return $set->search({}, | ||||||
|  |     { | ||||||
|  |       columns => [$col], | ||||||
|  |       order_by => $col, | ||||||
|  |       distinct => 1 | ||||||
|  |     } | ||||||
|  |   )->get_column($col)->all; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | sub by_field { | ||||||
|  |     my ($set, $p) = @_; | ||||||
|  |     return $set unless ref {} eq ref $p; | ||||||
|  |     my $op = $p->{matchall} ? '-and' : '-or'; | ||||||
|  |  | ||||||
|  |     # this is a bit of a dreadful hack to catch junk entry | ||||||
|  |     # whilst avoiding returning all devices in the DB | ||||||
|  |     my $ip = ($p->{ip} ? | ||||||
|  |       (NetAddr::IP::Lite->new($p->{ip}) || NetAddr::IP::Lite->new('255.255.255.255')) | ||||||
|  |       : undef); | ||||||
|  |  | ||||||
|  |     return $set->search( | ||||||
|  |       { | ||||||
|  |         $op => [ | ||||||
|  |           ($p->{name} ? ('me.name' => | ||||||
|  |             { '-ilike' => "\%$p->{name}\%" }) : ()), | ||||||
|  |           ($p->{location} ? ('me.location' => | ||||||
|  |             { '-ilike' => "\%$p->{location}\%" }) : ()), | ||||||
|  |           ($p->{description} ? ('me.description' => | ||||||
|  |             { '-ilike' => "\%$p->{description}\%" }) : ()), | ||||||
|  |           ($p->{model} ? ('me.model' => | ||||||
|  |             { '-in' => $p->{model} }) : ()), | ||||||
|  |           ($p->{os_ver} ? ('me.os_ver' => | ||||||
|  |             { '-in' => $p->{os_ver} }) : ()), | ||||||
|  |           ($p->{vendor} ? ('me.vendor' => | ||||||
|  |             { '-in' => $p->{vendor} }) : ()), | ||||||
|  |           ($p->{dns} ? ( | ||||||
|  |           -or => [ | ||||||
|  |             'me.dns'      => { '-ilike' => "\%$p->{dns}\%" }, | ||||||
|  |             'device_ips.dns' => { '-ilike' => "\%$p->{dns}\%" }, | ||||||
|  |           ]) : ()), | ||||||
|  |           ($ip ? ( | ||||||
|  |           -or => [ | ||||||
|  |             'me.ip'  => { '<<=' => $ip->cidr }, | ||||||
|  |             'device_ips.alias' => { '<<=' => $ip->cidr }, | ||||||
|  |           ]) : ()), | ||||||
|  |         ], | ||||||
|  |       }, | ||||||
|  |       { | ||||||
|  |         join => 'device_ips', | ||||||
|  |         group_by => 'me.ip', | ||||||
|  |       } | ||||||
|  |     ); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | sub by_any { | ||||||
|  |     my ($set, $q) = @_; | ||||||
|  |     return $set unless $q; | ||||||
|  |     $q = "\%$q\%" if $q !~ m/\%/; | ||||||
|  |  | ||||||
|  |     return $set->search( | ||||||
|  |       { | ||||||
|  |         -or => [ | ||||||
|  |           'me.contact'  => { '-ilike' => $q }, | ||||||
|  |           'me.serial'   => { '-ilike' => $q }, | ||||||
|  |           'me.location' => { '-ilike' => $q }, | ||||||
|  |           'me.name'     => { '-ilike' => $q }, | ||||||
|  |           'me.description' => { '-ilike' => $q }, | ||||||
|  |           -or => [ | ||||||
|  |             'me.dns'      => { '-ilike' => $q }, | ||||||
|  |             'device_ips.dns' => { '-ilike' => $q }, | ||||||
|  |           ], | ||||||
|  |           -or => [ | ||||||
|  |             'me.ip::text'  => { '-ilike' => $q }, | ||||||
|  |             'device_ips.alias::text' => { '-ilike' => $q }, | ||||||
|  |           ], | ||||||
|  |         ], | ||||||
|  |       }, | ||||||
|  |       { | ||||||
|  |         join => 'device_ips', | ||||||
|  |         group_by => 'me.ip', | ||||||
|  |       } | ||||||
|  |     ); | ||||||
|  | } | ||||||
|  |  | ||||||
| sub carrying_vlan { | sub carrying_vlan { | ||||||
|     my ($set, $vlan) = @_; |     my ($set, $vlan) = @_; | ||||||
|     return $set unless $vlan and $vlan =~ m/^\d+$/; |     return $set unless $vlan and $vlan =~ m/^\d+$/; | ||||||
| @@ -19,5 +110,4 @@ sub carrying_vlan { | |||||||
|     ); |     ); | ||||||
| } | } | ||||||
|  |  | ||||||
|  |  | ||||||
| 1; | 1; | ||||||
|   | |||||||
| @@ -26,6 +26,7 @@ sub by_mac { | |||||||
| sub by_name { | sub by_name { | ||||||
|     my ($set, $name) = @_; |     my ($set, $name) = @_; | ||||||
|     return $set unless $name; |     return $set unless $name; | ||||||
|  |     $name = "\%$name\%" if $name !~ m/\%/; | ||||||
|  |  | ||||||
|     return $set->search( |     return $set->search( | ||||||
|       { |       { | ||||||
|   | |||||||
| @@ -8,6 +8,7 @@ use Digest::MD5 (); | |||||||
| use Socket6 (); # to ensure dependency is met | use Socket6 (); # to ensure dependency is met | ||||||
| use NetAddr::IP::Lite ':lower'; | use NetAddr::IP::Lite ':lower'; | ||||||
| use Net::MAC (); | use Net::MAC (); | ||||||
|  | use List::MoreUtils (); | ||||||
|  |  | ||||||
| hook 'before' => sub { | hook 'before' => sub { | ||||||
|     if (! session('user') && request->path !~ m{^/login}) { |     if (! session('user') && request->path !~ m{^/login}) { | ||||||
| @@ -20,20 +21,50 @@ hook 'before' => sub { | |||||||
|     if (not param('tab') or param('tab') ne 'node') { |     if (not param('tab') or param('tab') ne 'node') { | ||||||
|         params->{'stamps'} = 'checked'; |         params->{'stamps'} = 'checked'; | ||||||
|     } |     } | ||||||
|  |     if (not param('tab') or param('tab') ne 'device') { | ||||||
|  |         params->{'matchall'} = 'matchall'; | ||||||
|  |     } | ||||||
|  |  | ||||||
|     # set up query string defaults for templates |     # set up query string defaults, | ||||||
|  |     # only for templates which link to themselves (node) | ||||||
|     var('query_defaults' => { map { ($_ => "tab=$_") } qw/node/ }); |     var('query_defaults' => { map { ($_ => "tab=$_") } qw/node/ }); | ||||||
|     var('query_defaults')->{node} .= "\&$_=". (param($_) || '') |     var('query_defaults')->{node} .= "\&$_=". (param($_) || '') | ||||||
|       for qw/stamps vendor archived partial/; |       for qw/stamps vendor archived partial/; | ||||||
|  |  | ||||||
|  |     # set up property lists for device search | ||||||
|  |     var('model_list' => [ | ||||||
|  |       schema('netdisco')->resultset('Device')->get_distinct('model') | ||||||
|  |     ]); | ||||||
|  |     var('os_ver_list' => [ | ||||||
|  |       schema('netdisco')->resultset('Device')->get_distinct('os_ver') | ||||||
|  |     ]); | ||||||
|  |     var('vendor_list' => [ | ||||||
|  |       schema('netdisco')->resultset('Device')->get_distinct('vendor') | ||||||
|  |     ]); | ||||||
| }; | }; | ||||||
|  |  | ||||||
| get '/' => sub { | # device with various properties or a default match-all | ||||||
|     template 'index'; | ajax '/ajax/content/search/device' => sub { | ||||||
| }; |     my $has_opt = List::MoreUtils::any {param($_)} | ||||||
|  |       qw/name location dns ip description model os_ver vendor/; | ||||||
|  |     my $set; | ||||||
|  |  | ||||||
|  |     if ($has_opt) { | ||||||
|  |       $set = schema('netdisco')->resultset('Device')->by_field(scalar params); | ||||||
|  |       return unless $set->count; | ||||||
|  |     } | ||||||
|  |     else { | ||||||
|  |       my $q = param('q'); | ||||||
|  |       return unless $q; | ||||||
|  |  | ||||||
|  |       $set = schema('netdisco')->resultset('Device')->by_any($q); | ||||||
|  |       return unless $set->count; | ||||||
|  |     } | ||||||
|  |  | ||||||
| ajax '/ajax/content/search/:thing' => sub { |  | ||||||
|     content_type('text/html'); |     content_type('text/html'); | ||||||
|     return '<p>Hello '. param('thing') .'.</p>'; |     template 'ajax/device.tt', { | ||||||
|  |       results => $set, | ||||||
|  |     }, { layout => undef }; | ||||||
| }; | }; | ||||||
|  |  | ||||||
| # nodes matching the param as an IP or DNS hostname or MAC | # nodes matching the param as an IP or DNS hostname or MAC | ||||||
| @@ -111,6 +142,10 @@ ajax '/ajax/content/search/port' => sub { | |||||||
|     }, { layout => undef }; |     }, { layout => undef }; | ||||||
| }; | }; | ||||||
|  |  | ||||||
|  | get '/' => sub { | ||||||
|  |     template 'index'; | ||||||
|  | }; | ||||||
|  |  | ||||||
| get '/search' => sub { | get '/search' => sub { | ||||||
|     my $q = param('q'); |     my $q = param('q'); | ||||||
|     if ($q and not param('tab')) { |     if ($q and not param('tab')) { | ||||||
|   | |||||||
							
								
								
									
										29
									
								
								Netdisco/views/ajax/device.tt
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										29
									
								
								Netdisco/views/ajax/device.tt
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,29 @@ | |||||||
|  | <table class="bordered-table condensed-table zebra-striped"> | ||||||
|  |   <thead> | ||||||
|  |     <tr> | ||||||
|  |       <th>Device</th> | ||||||
|  |       <th>Contact</th> | ||||||
|  |       <th>Location</th> | ||||||
|  |       <th>System Name</th> | ||||||
|  |       <th>Model</th> | ||||||
|  |       <th>OS Version</th> | ||||||
|  |       <th>Management IP</th> | ||||||
|  |       <th>Serial</th> | ||||||
|  |     </tr> | ||||||
|  |   </thead> | ||||||
|  |   </tbody> | ||||||
|  |     [% WHILE (row = results.next) %] | ||||||
|  |     <tr> | ||||||
|  |       <td><a href="/device?q=[% row.ip %]">[% row.dns.remove(settings.domain_suffix) %]</a></td> | ||||||
|  |       <td>[% row.contact %]</td> | ||||||
|  |       <td>[% row.location %]</td> | ||||||
|  |       <td>[% row.name %]</td> | ||||||
|  |       <!-- <td>[% row.description.substr(0, 100) %][% ' …' IF row.description.length > 100 %]</td> --> | ||||||
|  |       <td>[% row.model %]</td> | ||||||
|  |       <td>[% row.os_ver %]</td> | ||||||
|  |       <td>[% row.ip %]</td> | ||||||
|  |       <td>[% row.serial %]</td> | ||||||
|  |     </tr> | ||||||
|  |     [% END %] | ||||||
|  |   </tbody> | ||||||
|  | </table> | ||||||
| @@ -2,20 +2,20 @@ | |||||||
|   <thead> |   <thead> | ||||||
|     <tr> |     <tr> | ||||||
|       <th>Name</th> |       <th>Name</th> | ||||||
|  |       <th>Port</th> | ||||||
|       <th>Description</th> |       <th>Description</th> | ||||||
|       <th>Vlan</th> |       <th>Vlan</th> | ||||||
|       <th>Port</th> |  | ||||||
|     </tr> |     </tr> | ||||||
|   </thead> |   </thead> | ||||||
|   </tbody> |   </tbody> | ||||||
|     [% WHILE (row = results.next) %] |     [% WHILE (row = results.next) %] | ||||||
|     <tr> |     <tr> | ||||||
|       <td>[% row.name %]</td> |       <td>[% row.name %]</td> | ||||||
|       <td>[% row.descr %]</td> |  | ||||||
|       <td>[% row.vlan %]</td> |  | ||||||
|       <td><a href="/device?q=[% row.ip %]&port=[% row.port %]">[% row.ip %] [ [% row.port %] ]</a> |       <td><a href="/device?q=[% row.ip %]&port=[% row.port %]">[% row.ip %] [ [% row.port %] ]</a> | ||||||
|         [% ' (' _ row.device.dns.remove(settings.domain_suffix) _ ')' IF row.device.dns %] |         [% ' (' _ row.device.dns.remove(settings.domain_suffix) _ ')' IF row.device.dns %] | ||||||
|         </td> |       </td> | ||||||
|  |       <td>[% row.descr %]</td> | ||||||
|  |       <td>[% row.vlan %]</td> | ||||||
|     </tr> |     </tr> | ||||||
|     [% END %] |     [% END %] | ||||||
|   </tbody> |   </tbody> | ||||||
|   | |||||||
							
								
								
									
										62
									
								
								Netdisco/views/inc/search/device.tt
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										62
									
								
								Netdisco/views/inc/search/device.tt
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,62 @@ | |||||||
|  |  | ||||||
|  |             <p><em>Device Search Options</em></p> | ||||||
|  |             <div class="clearfix"> | ||||||
|  |               <input class="span3" placeholder="System Name" | ||||||
|  |                 type="text" id="name" name="name" | ||||||
|  |                 rel="twipsy" data-placement="right" data-offset="5" title="System Name"/> | ||||||
|  |             </div> | ||||||
|  |             <div class="clearfix"> | ||||||
|  |               <input class="span3" placeholder="Location" | ||||||
|  |                 type="text" id="location" name="location" | ||||||
|  |                 rel="twipsy" data-placement="right" data-offset="5" title="Location"/> | ||||||
|  |             </div> | ||||||
|  |             <div class="clearfix"> | ||||||
|  |               <input class="span3" placeholder="DNS" | ||||||
|  |                 type="text" id="dns" name="dns" | ||||||
|  |                 rel="twipsy" data-placement="right" data-offset="5" title="DNS"/> | ||||||
|  |             </div> | ||||||
|  |             <div class="clearfix"> | ||||||
|  |               <input class="span3" placeholder="IP Address" | ||||||
|  |                 type="text" id="ip" name="ip" | ||||||
|  |                 rel="twipsy" data-placement="right" data-offset="5" title="IP Address"/> | ||||||
|  |             </div> | ||||||
|  |             <div class="clearfix"> | ||||||
|  |               <input class="span3" placeholder="Description" | ||||||
|  |                 type="text" id="description" name="description" | ||||||
|  |                 rel="twipsy" data-placement="right" data-offset="5" title="Description"/> | ||||||
|  |             </div> | ||||||
|  |             <div class="clearfix"> | ||||||
|  |               <select class="span3" size="[% vars.model_list.size > 5 ? 5 : vars.model_list.size %]" | ||||||
|  |                 multiple="on" id="model" name="model" | ||||||
|  |                 rel="twipsy" data-placement="right" data-offset="5" title="Model"/> | ||||||
|  |                 [% FOREACH opt IN vars.model_list %] | ||||||
|  |                 <option>[% opt %]</option> | ||||||
|  |                 [% END %] | ||||||
|  |               </select> | ||||||
|  |             </div> | ||||||
|  |             <div class="clearfix"> | ||||||
|  |               <select class="span3" size="[% vars.os_ver_list.size > 5 ? 5 : vars.os_ver_list.size %]" | ||||||
|  |                 multiple="on" id="os_ver" name="os_ver" | ||||||
|  |                 rel="twipsy" data-placement="right" data-offset="5" title="OS Release"/> | ||||||
|  |                 [% FOREACH opt IN vars.os_ver_list %] | ||||||
|  |                 <option>[% opt %]</option> | ||||||
|  |                 [% END %] | ||||||
|  |               </select> | ||||||
|  |             </div> | ||||||
|  |             <div class="clearfix"> | ||||||
|  |               <select class="span3" size="[% vars.vendor_list.size > 5 ? 5 : vars.vendor_list.size %]" | ||||||
|  |                 multiple="on" id="vendor" name="vendor" | ||||||
|  |                 rel="twipsy" data-placement="right" data-offset="5" title="Vendor"/> | ||||||
|  |                 [% FOREACH opt IN vars.vendor_list %] | ||||||
|  |                 <option>[% opt %]</option> | ||||||
|  |                 [% END %] | ||||||
|  |               </select> | ||||||
|  |             </div> | ||||||
|  |             <div class="clearfix input-prepend"> | ||||||
|  |               <label class="add-on"> | ||||||
|  |                 <input type="checkbox" id="matchall" name="matchall"[% ' checked="checked"' IF params.matchall %]/> | ||||||
|  |               </label> | ||||||
|  |               <label for="matchall"> | ||||||
|  |                 <span class="nd_searchcheckbox uneditable-input">Match All Options</span> | ||||||
|  |               </label> | ||||||
|  |             </div> | ||||||
| @@ -19,7 +19,7 @@ | |||||||
|  |  | ||||||
|   <script type="text/javascript"> |   <script type="text/javascript"> | ||||||
|     $(document).ready(function() { |     $(document).ready(function() { | ||||||
|       $("a[rel=twipsy]").twipsy({live: true}); |       $("[rel=twipsy]").twipsy({live: true}); | ||||||
|     }); |     }); | ||||||
|   </script> |   </script> | ||||||
|  |  | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user