Added SSHCollector/Platform/ASAContext.pm
* based on the work of @haught mentioned in #754 * added some missing ASA.pm improvements from dc9feb747f..b58a62f300 * separate module for now since it's untested, if confirmed working by ASA owners it can replace the original ASA.pm
This commit is contained in:
		
							
								
								
									
										6
									
								
								Changes
									
									
									
									
									
								
							
							
						
						
									
										6
									
								
								Changes
									
									
									
									
									
								
							| @@ -1,3 +1,9 @@ | ||||
| 2.052000 - ? | ||||
|  | ||||
|   [NEW FEATURES] | ||||
|  | ||||
|   * Added SSHCollector::Platform::ASAContext (@haught) | ||||
|  | ||||
| 2.051005 - 2021-11-25 | ||||
|  | ||||
|   [ENHANCEMENTS] | ||||
|   | ||||
							
								
								
									
										169
									
								
								lib/App/Netdisco/SSHCollector/Platform/ASAContext.pm
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										169
									
								
								lib/App/Netdisco/SSHCollector/Platform/ASAContext.pm
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,169 @@ | ||||
| package App::Netdisco::SSHCollector::Platform::ASAContext; | ||||
|  | ||||
|  | ||||
| =head1 NAME | ||||
|  | ||||
| App::Netdisco::SSHCollector::Platform::ASAContext | ||||
|  | ||||
| =head1 DESCRIPTION | ||||
|  | ||||
| Collect IPv4 ARP and IPv6 neighbor entries from Cisco ASA devices. | ||||
|  | ||||
| You will need the following configuration for the user to automatically enter | ||||
| C<enable> status after login: | ||||
|  | ||||
| aaa authorization exec LOCAL auto-enable | ||||
|  | ||||
| To use an C<enable> password seaparate from the login password, add an | ||||
| C<enable_password> under C<device_auth> in your configuration file: | ||||
|  | ||||
| device_auth: | ||||
|     - tag: sshasa | ||||
|       driver: cli | ||||
|       platform: ASAContext | ||||
|       only: '192.0.2.1' | ||||
|       username: oliver | ||||
|       password: letmein | ||||
|       enable_password: myenablepass | ||||
|  | ||||
| =cut | ||||
|  | ||||
| use strict; | ||||
| use warnings; | ||||
|  | ||||
| use Dancer ':script'; | ||||
| use Expect; | ||||
| use Moo; | ||||
|  | ||||
| =head1 PUBLIC METHODS | ||||
|  | ||||
| =over 4 | ||||
|  | ||||
| =item B<arpnip($host, $ssh)> | ||||
|  | ||||
| Retrieve ARP and neighbor entries from device. C<$host> is the hostname or IP | ||||
| address of the device. C<$ssh> is a Net::OpenSSH connection to the device. | ||||
|  | ||||
| Returns a list of hashrefs in the format C<{ mac =E<gt> MACADDR, ip =E<gt> IPADDR }>. | ||||
|  | ||||
| This was kindly created by @haught and mentioned in https://github.com/netdisco/netdisco/issues/754  | ||||
| as being a context-aware version of ASA.pm. | ||||
|  | ||||
| The code is imported from https://github.com/haught/netdisco/blob/ASAContext/lib/App/Netdisco/SSHCollector/Platform/ASAContext.pm. | ||||
| However this version did not have some ASA.pm improvements added in dc9feb747f..b58a62f300,  | ||||
| so we tried to merge all of this in here. However we lack the ability to try it, so we also  | ||||
| left in place the original ASA.pm which is confirmed to work. | ||||
|  | ||||
| =back | ||||
|  | ||||
| =cut | ||||
|  | ||||
| sub arpnip { | ||||
|     my ($self, $hostlabel, $ssh, $args) = @_; | ||||
|  | ||||
|     debug "$hostlabel $$ arpnip()"; | ||||
|  | ||||
|     my ($pty, $pid) = $ssh->open2pty; | ||||
|     unless ($pty) { | ||||
|         debug "unable to run remote command [$hostlabel] " . $ssh->error; | ||||
|         return (); | ||||
|     } | ||||
|     my $expect = Expect->init($pty); | ||||
|  | ||||
|     my ($pos, $error, $match, $before, $after); | ||||
|     my $prompt; | ||||
|  | ||||
|     if ($args->{enable_password}) { | ||||
|         $prompt = qr/>/; | ||||
|         ($pos, $error, $match, $before, $after) = $expect->expect(10, -re, $prompt); | ||||
|  | ||||
|         $expect->send("enable\n"); | ||||
|  | ||||
|         $prompt = qr/Password:/; | ||||
|         ($pos, $error, $match, $before, $after) = $expect->expect(10, -re, $prompt); | ||||
|  | ||||
|         $expect->send( $args->{enable_password} ."\n" ); | ||||
|     } | ||||
|  | ||||
|     $prompt = qr/#\s*$/; | ||||
|     ($pos, $error, $match, $before, $after) = $expect->expect(10, -re, $prompt); | ||||
|  | ||||
|     $expect->send("terminal pager 2147483647\n"); | ||||
|     ($pos, $error, $match, $before, $after) = $expect->expect(5, -re, $prompt); | ||||
|  | ||||
|     $expect->send("changeto system\n"); | ||||
|     ($pos, $error, $match, $before, $after) = $expect->expect(10, -re, $prompt); | ||||
|  | ||||
|     $expect->send("show run | include ^context\n"); | ||||
|     ($pos, $error, $match, $before, $after) = $expect->expect(60, -re, $prompt); | ||||
|     my @contexts = split(m/\n/, $before); | ||||
|  | ||||
|  | ||||
|     my @arpentries = (); | ||||
|     for (my $i = 1; $i < scalar @contexts; $i++) { | ||||
|         $contexts[$i] =~ s/context //; | ||||
|         debug "$hostlabel/$contexts[$i]"; | ||||
|         $expect->send("changeto context $contexts[$i]\n"); | ||||
|         ($pos, $error, $match, $before, $after) = $expect->expect(10, -re, $prompt); | ||||
|  | ||||
|         $expect->send("show names\n"); | ||||
|         ($pos, $error, $match, $before, $after) = $expect->expect(60, -re, $prompt); | ||||
|  | ||||
|         my @names = split(m/\n/, $before); | ||||
|  | ||||
|         $expect->send("terminal pager 2147483647\n"); | ||||
|         ($pos, $error, $match, $before, $after) = $expect->expect(5, -re, $prompt); | ||||
|  | ||||
|         $expect->send("show arp\n"); | ||||
|         ($pos, $error, $match, $before, $after) = $expect->expect(60, -re, $prompt); | ||||
|  | ||||
|         my @lines = split(m/\n/, $before); | ||||
|  | ||||
|         # ifname 192.0.2.1 0011.2233.4455 123 | ||||
|         my $linereg = qr/[A-z0-9\-\.]+\s([A-z0-9\-\.]+)\s | ||||
|             ([0-9a-fA-F]{4}\.[0-9a-fA-F]{4}\.[0-9a-fA-F]{4})/x; | ||||
|  | ||||
|         foreach my $line(@lines) { | ||||
|             if ($line =~ $linereg) { | ||||
|                 my ($ip, $mac) = ($1, $2); | ||||
|                 if ($ip !~ m/^[0-9]{1,3}\.[0-9]{1,3}\.[0-9]{1,3}\.[0-9]{1,3}$/) { | ||||
|                     foreach my $name (@names) { | ||||
|                         if ($name =~ qr/name\s([0-9]{1,3}\.[0-9]{1,3}\.[0-9]{1,3}\.[0-9]{1,3})\s([\w-]*)/x) { | ||||
|                             if ($ip eq $2) { | ||||
|                                 $ip = $1; | ||||
|                             } | ||||
|                         } | ||||
|                     } | ||||
|                 } | ||||
|                 if ($ip =~ m/^[0-9]{1,3}\.[0-9]{1,3}\.[0-9]{1,3}\.[0-9]{1,3}$/) { | ||||
|                     push @arpentries, { mac => $mac, ip => $ip }; | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         # start ipv6 | ||||
|         $expect->send("show ipv6 neighbor\n"); | ||||
|         ($pos, $error, $match, $before, $after) = $expect->expect(60, -re, $prompt); | ||||
|  | ||||
|         @lines = split(m/\n/, $before); | ||||
|  | ||||
|         # IPv6 age MAC state ifname | ||||
|         $linereg = qr/([0-9a-fA-F\:]+)\s+[0-9]+\s | ||||
|             ([0-9a-fA-F]{4}\.[0-9a-fA-F]{4}\.[0-9a-fA-F]{4})/x; | ||||
|  | ||||
|         foreach my $line (@lines) { | ||||
|             if ($line =~ $linereg) { | ||||
|                 my ($ip, $mac) = ($1, $2); | ||||
|                 push @arpentries, { mac => $mac, ip => $ip }; | ||||
|             } | ||||
|         } | ||||
|         # end ipv6 | ||||
|     } | ||||
|     $expect->send("exit\n"); | ||||
|     $expect->soft_close(); | ||||
|     return @arpentries; | ||||
| } | ||||
|  | ||||
| 1; | ||||
|  | ||||
|  | ||||
		Reference in New Issue
	
	Block a user