| 20 | rodolico | 1 | #!/usr/bin/env perl
 | 
        
           |  |  | 2 | use warnings;
 | 
        
           | 26 | rodolico | 3 | use strict;  
 | 
        
           | 2 | rodolico | 4 |   | 
        
           | 256 | rodolico | 5 | use version ; our $VERSION = '1.3';
 | 
        
           | 20 | rodolico | 6 |   | 
        
           | 2 | rodolico | 7 | # Linux network module for sysinfo client
 | 
        
           |  |  | 8 | # Author: R. W. Rodolico
 | 
        
           | 20 | rodolico | 9 | # Date:   2016-04-08
 | 
        
           | 2 | rodolico | 10 |   | 
        
           |  |  | 11 | # module to get network interface information for Linux systems
 | 
        
           |  |  | 12 | # assumes ifconfig is on the system and executable by the user
 | 
        
           |  |  | 13 | # NOTE: this takes the ifconfig output and parses it, so changes to 
 | 
        
           |  |  | 14 | #       this output invalidates this module
 | 
        
           | 65 | rodolico | 15 | #
 | 
        
           | 256 | rodolico | 16 | # Revision History
 | 
        
           | 65 | rodolico | 17 | # 2018-05-20 RWR
 | 
        
           |  |  | 18 | # Modified to use ip vs ifconfig, and changed the regex's.
 | 
        
           |  |  | 19 | # kept the original ifconfig code but commented it out
 | 
        
           | 2 | rodolico | 20 |   | 
        
           | 256 | rodolico | 21 | # find our location and use it for searching for libraries. library.pm must be in the same directory as the calling script
 | 
        
           |  |  | 22 | # or, if run interactively, in the parent of the modules
 | 
        
           | 2 | rodolico | 23 | BEGIN {
 | 
        
           | 251 | rodolico | 24 |    use FindBin;
 | 
        
           |  |  | 25 |    use File::Spec;
 | 
        
           | 256 | rodolico | 26 |    # prepend the bin directory and its parent
 | 
        
           |  |  | 27 |    use lib File::Spec->catdir($FindBin::Bin), File::Spec->catdir("$FindBin::Bin/..");
 | 
        
           |  |  | 28 |    eval( 'use library;' );
 | 
        
           |  |  | 29 |    die sprintf( "Could not find library.pm in %s, INC is %s\n", __FILE__, join( "\n", @INC ) ) if $@;
 | 
        
           | 2 | rodolico | 30 | }
 | 
        
           |  |  | 31 |   | 
        
           | 256 | rodolico | 32 | #####
 | 
        
           |  |  | 33 | ##### Change these to match your needs
 | 
        
           |  |  | 34 | #####
 | 
        
           | 2 | rodolico | 35 |   | 
        
           | 256 | rodolico | 36 | # Make this a list of all the modules we are going to use. You can replace undef with the version you need, if you like
 | 
        
           |  |  | 37 | my $modulesList = {
 | 
        
           |  |  | 38 |         'Data::Dumper'     => undef,
 | 
        
           |  |  | 39 |    };
 | 
        
           | 48 | rodolico | 40 |   | 
        
           | 256 | rodolico | 41 | # hash of commands that are needed for the system. key is the name of the command and, in some cases, the value will become
 | 
        
           |  |  | 42 | # the full path (from which or where)
 | 
        
           |  |  | 43 | my $commandsList = {
 | 
        
           |  |  | 44 |         'ip' => undef,
 | 
        
           |  |  | 45 |    };
 | 
        
           |  |  | 46 |   | 
        
           |  |  | 47 | # list of operating systems this module can be used on.
 | 
        
           |  |  | 48 | my $osList = {
 | 
        
           |  |  | 49 | #         'mswin32' => undef,
 | 
        
           |  |  | 50 | #         'freebsd' => undef,
 | 
        
           |  |  | 51 |          'linux'   => undef,
 | 
        
           |  |  | 52 |    };
 | 
        
           |  |  | 53 |   | 
        
           |  |  | 54 | # the category the return data should go into. See sysinfo for a list
 | 
        
           | 2 | rodolico | 55 | my $CATEGORY = 'network';
 | 
        
           | 65 | rodolico | 56 |   | 
        
           | 256 | rodolico | 57 | #####
 | 
        
           |  |  | 58 | ##### End of required
 | 
        
           |  |  | 59 | #####
 | 
        
           | 65 | rodolico | 60 |   | 
        
           | 256 | rodolico | 61 | # some variables needed for our system
 | 
        
           |  |  | 62 | my $errorPrepend = 'error: in ' . __FILE__; # this is prepended to any error messages
 | 
        
           |  |  | 63 | my @out; # temporary location for each line of output
 | 
        
           | 65 | rodolico | 64 |   | 
        
           | 256 | rodolico | 65 | # Try to load the modules we need. If we can not, then make a list of missing modules for error message.
 | 
        
           |  |  | 66 | for my $module ( keys %$modulesList ) {
 | 
        
           |  |  | 67 |    eval ( "use $module;" );
 | 
        
           |  |  | 68 |    push @out, "$errorPrepend Could not load $module" if $@;
 | 
        
           |  |  | 69 | }
 | 
        
           |  |  | 70 |   | 
        
           |  |  | 71 | if ( ! @out && ! checkOS ( $osList ) ) { # check if we are on an acceptible operating system
 | 
        
           |  |  | 72 |     push @out, "$errorPrepend Invalid Operating System";
 | 
        
           |  |  | 73 | }
 | 
        
           |  |  | 74 | if ( !@out && ! validCommandOnSystem ( $commandsList ) ) {
 | 
        
           |  |  | 75 |    push @out, "$errorPrepend Can not find some commands needed";
 | 
        
           |  |  | 76 | }
 | 
        
           |  |  | 77 | if ( !@out ) { # we made it, we have everything, so do the processing
 | 
        
           |  |  | 78 |    #####
 | 
        
           |  |  | 79 |    ##### Your code starts here. Remember to push all output onto @out
 | 
        
           |  |  | 80 |    #####
 | 
        
           |  |  | 81 |    # get list of all interfaces except lo
 | 
        
           |  |  | 82 |    my @interfaces = `ip -o a s | cut -d' ' -f2 | sort | uniq | grep -v 'lo'`;
 | 
        
           |  |  | 83 |    chomp @interfaces;
 | 
        
           |  |  | 84 |   | 
        
           |  |  | 85 |    my $ipv4Regex = '\b(?:[0-9]{1,3}\.){3}[0-9]{1,3}\b';
 | 
        
           |  |  | 86 |    my $ipv6Regex = '(([a-f0-9]+):{1,2})+[a-z0-9]+';
 | 
        
           |  |  | 87 |   | 
        
           |  |  | 88 |    foreach my $if ( @interfaces ) {
 | 
        
           |  |  | 89 |       foreach my $line ( `ip address show $if` ) {
 | 
        
           |  |  | 90 |          # mtu
 | 
        
           |  |  | 91 |          if ( $line =~ m/mtu\s+(\d+)/i ) {
 | 
        
           |  |  | 92 |             push @out, "$CATEGORY\t$if\tmtu\t$1";
 | 
        
           |  |  | 93 |          # MAC
 | 
        
           |  |  | 94 |          } elsif ( $line =~ m/((([a-f0-9]{2}):){5}[a-f0-9]+)/i ) {
 | 
        
           |  |  | 95 |             push @out, "$CATEGORY\t$if\tmac\t$1";
 | 
        
           |  |  | 96 |          # inet 6
 | 
        
           |  |  | 97 |          } elsif ( $line =~ m"inet6\s*((([a-f0-9]+):{1,2})+[a-z0-9]+)/(\d{1,2})"i ) {
 | 
        
           |  |  | 98 |             push @out, "$CATEGORY\t$if\tip6address\t$1";
 | 
        
           |  |  | 99 |             push @out, "$CATEGORY\t$if\tip6networkbits\t$4";
 | 
        
           |  |  | 100 |          # inet 4
 | 
        
           |  |  | 101 |          } elsif ( $line =~ m"inet.*($ipv4Regex)/(\d{2})" ) {
 | 
        
           |  |  | 102 |             push @out, "$CATEGORY\t$if\taddress\t$1";
 | 
        
           |  |  | 103 |             push @out, "$CATEGORY\t$if\tnetmask\t$2";
 | 
        
           |  |  | 104 |          }
 | 
        
           | 2 | rodolico | 105 |       }
 | 
        
           |  |  | 106 |    }
 | 
        
           | 256 | rodolico | 107 |    #####
 | 
        
           |  |  | 108 |    ##### Your code ends here.
 | 
        
           |  |  | 109 |    #####
 | 
        
           | 2 | rodolico | 110 | }
 | 
        
           | 65 | rodolico | 111 |   | 
        
           | 256 | rodolico | 112 | # If we are testing from the command line (caller is undef), print the results for debugging
 | 
        
           |  |  | 113 | print join( "\n", @out ) . "\n" unless caller;
 | 
        
           |  |  | 114 | # called by do, which has a value of the last assignment made, so make the assignment. The equivilent of a return
 | 
        
           |  |  | 115 | my $return = join( "\n", @out );
 | 
        
           | 65 | rodolico | 116 |   |