411 lines
		
	
	
		
			10 KiB
		
	
	
	
		
			Perl
		
	
	
	
	
	
			
		
		
	
	
			411 lines
		
	
	
		
			10 KiB
		
	
	
	
		
			Perl
		
	
	
	
	
	
package SNMP::Info::MRO;
 | 
						|
 | 
						|
use warnings;
 | 
						|
use strict;
 | 
						|
 | 
						|
our ($VERSION);
 | 
						|
$VERSION = '3.71';
 | 
						|
 | 
						|
use PPI;
 | 
						|
use Class::ISA;  ## no critic
 | 
						|
use Module::Info;
 | 
						|
use Module::Load ();
 | 
						|
# use Data::Printer;
 | 
						|
 | 
						|
sub _walk_global_data {
 | 
						|
    my $self = shift;
 | 
						|
    my $class = (ref $self ? ref $self : $self);
 | 
						|
 | 
						|
    my $ppi = PPI::Document->new( file($class) );
 | 
						|
    my $name    = shift or die "name (e.g. GLOBALS) required";
 | 
						|
    my $results = shift || {};
 | 
						|
    my $subref  = \&_walk_global_data;
 | 
						|
 | 
						|
    # get the hash declaration
 | 
						|
    my $declaration = $ppi->find_first(sub {
 | 
						|
      my ($doc, $tok) = @_;
 | 
						|
      return ($tok->isa('PPI::Token::Symbol')
 | 
						|
              and $tok->symbol eq "\%$name");
 | 
						|
    });
 | 
						|
 | 
						|
    # get the hash content
 | 
						|
    my $content = $declaration->snext_sibling->snext_sibling;
 | 
						|
 | 
						|
    # get relevant tokens in the hash content
 | 
						|
    my @tokens = $content->find(sub {
 | 
						|
      my ($doc, $tok) = @_;
 | 
						|
      return ($tok->isa('PPI::Token::Symbol')
 | 
						|
              or $tok->isa('PPI::Token::Quote'));
 | 
						|
    });
 | 
						|
 | 
						|
    return $results unless scalar @tokens and ref $tokens[0] eq ref [];
 | 
						|
    @tokens = @{ $tokens[0] };
 | 
						|
 | 
						|
    # walk tokens and build final result
 | 
						|
    while (my $tok = splice(@tokens, 0, 1)) {
 | 
						|
        if ($tok->isa('PPI::Token::Quote')) {
 | 
						|
            my $token = $tok->string;
 | 
						|
            my $leaf = splice(@tokens, 0, 1);
 | 
						|
            my $extract = (($leaf =~ m/^&/) ? 'symbol' : 'string');
 | 
						|
 | 
						|
            unshift @{ $results->{$token} },
 | 
						|
                     [ $class => $leaf->$extract ]
 | 
						|
              # we can sometimes see the same package twice
 | 
						|
              unless scalar grep { $_ eq $class }
 | 
						|
                            map  { $_->[0] }
 | 
						|
                                 @{ $results->{$token} };
 | 
						|
        }
 | 
						|
        elsif ($tok->isa('PPI::Token::Symbol')) {
 | 
						|
            # recurse to get the results of the mentioned package
 | 
						|
            (my $otherpkg = $tok->symbol) =~ s/^\%(.+)::$name$/$1/;
 | 
						|
            $results = $subref->($otherpkg, $name, $results);
 | 
						|
        }
 | 
						|
    }
 | 
						|
 | 
						|
    return $results;
 | 
						|
}
 | 
						|
 | 
						|
sub _print_global_data {
 | 
						|
    my $results = _walk_global_data(@_);
 | 
						|
 | 
						|
    foreach my $key (sort keys %$results) {
 | 
						|
        print $key, "\n";
 | 
						|
        my @defs = @{ $results->{$key} };
 | 
						|
 | 
						|
        my $first = 0;
 | 
						|
        while (my $classdef = splice(@defs, 0, 1)) {
 | 
						|
            my $class = $classdef->[0];
 | 
						|
            my $meth  = $classdef->[1];
 | 
						|
 | 
						|
            if ($first) {
 | 
						|
                printf "     %s ( %s )\n", $meth, $class;
 | 
						|
            }
 | 
						|
            else {
 | 
						|
                printf " `-- %s ( %s )\n", $meth, $class;
 | 
						|
                $first = 1;
 | 
						|
            }
 | 
						|
        }
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
=head1 NAME
 | 
						|
 | 
						|
SNMP::Info::MRO - Method resolution introspection for SNMP::Info
 | 
						|
 | 
						|
=head1 SYNOPSIS
 | 
						|
 | 
						|
 use SNMP::Info::MRO;
 | 
						|
 use Data::Printer;
 | 
						|
 | 
						|
 # SNMP::Info::MRO::print_* functions
 | 
						|
 SNMP::Info::MRO::print_superclasses ('SNMP::Info::Layer3::Juniper');
 | 
						|
 | 
						|
 # print output using Data::Printer for other functions
 | 
						|
 my $buff = SNMP::Info::MRO::all_methods('SNMP::Info::Layer3::Juniper');
 | 
						|
 p $buff;
 | 
						|
 | 
						|
=head1 DESCRIPTION
 | 
						|
 | 
						|
This is a set of helpers to show where a given method in SNMP::Info has been
 | 
						|
implemented, and which implementation is being used at runtime.
 | 
						|
 | 
						|
The following distributions are I<required> to run this code:
 | 
						|
 | 
						|
=over 4
 | 
						|
 | 
						|
=item *
 | 
						|
 | 
						|
PPI
 | 
						|
 | 
						|
=item *
 | 
						|
 | 
						|
Class::ISA
 | 
						|
 | 
						|
=item *
 | 
						|
 | 
						|
Module::Info
 | 
						|
 | 
						|
=item *
 | 
						|
 | 
						|
Module::Load
 | 
						|
 | 
						|
=back
 | 
						|
 | 
						|
=head1 FUNCTIONS
 | 
						|
 | 
						|
None of the functions are exported. For all helper functions, you can pass
 | 
						|
either the name of a Perl module, or an object instance of SNMP::Info.
 | 
						|
 | 
						|
=over 4
 | 
						|
 | 
						|
=item all_methods( $module )
 | 
						|
 | 
						|
Returns the location of methods defined in C<$module> and all its ancestor
 | 
						|
classes (superclasses), either as Perl subroutines or via C<%GLOBALS>
 | 
						|
or C<%FUNCS> configuration. The data structure looks like:
 | 
						|
 | 
						|
 {
 | 
						|
   method_name => {
 | 
						|
     globals => [
 | 
						|
       [ Package::Name        => 'mib_leaf.0' ],
 | 
						|
       [ Other::Package::Name => '1.3.6.1.4.1.9.2.1.58.0' ],
 | 
						|
     ],
 | 
						|
   },
 | 
						|
   other_method_name => [
 | 
						|
     subs => [
 | 
						|
       'Package::Name',
 | 
						|
     ],
 | 
						|
     funcs => [
 | 
						|
       [ Package::Name => 'mib_leaf_name' ],
 | 
						|
     ],
 | 
						|
   ],
 | 
						|
 }
 | 
						|
 | 
						|
It should be noted that the order of method resolution in SNMP::Info is to
 | 
						|
first look for a defined subroutine (this is done by Perl), then the
 | 
						|
AUTOLOAD sequence will search for a definition in C<%GLOBALS> followed by
 | 
						|
C<%FUNCS>.
 | 
						|
 | 
						|
The defining class or module at runtime is always the first entry in the
 | 
						|
list, if it exists:
 | 
						|
 | 
						|
 $data->{method_name}->{subs}->[0]
 | 
						|
   if exists $data->{method_name}->{subs};
 | 
						|
 | 
						|
=cut
 | 
						|
 | 
						|
sub all_methods {
 | 
						|
    my $self = shift;
 | 
						|
    my $class = (ref $self ? ref $self : $self);
 | 
						|
 | 
						|
    my $results = subroutines( $class );
 | 
						|
    $results = { map { $_ => { subs => $results->{$_} } }
 | 
						|
                     keys %$results };
 | 
						|
 | 
						|
    my $globals = globals( $class );
 | 
						|
    foreach my $key (keys %$globals) {
 | 
						|
        $results->{$key}->{globals} = $globals->{$key};
 | 
						|
    }
 | 
						|
 | 
						|
    my $funcs = funcs( $class );
 | 
						|
    foreach my $key (keys %$funcs) {
 | 
						|
        $results->{$key}->{funcs} = $funcs->{$key};
 | 
						|
    }
 | 
						|
 | 
						|
    #foreach my $key (keys %$results) {
 | 
						|
    #    $results->{$key}->{subs}    ||= [];
 | 
						|
    #    $results->{$key}->{globals} ||= [];
 | 
						|
    #    $results->{$key}->{funcs}   ||= [];
 | 
						|
    #}
 | 
						|
 | 
						|
    return $results;
 | 
						|
}
 | 
						|
 | 
						|
=item subroutines( $module )
 | 
						|
 | 
						|
Returns the set of subroutines defined in C<$module> and all its ancestor
 | 
						|
classes (superclasses). The data structure looks like:
 | 
						|
 | 
						|
 {
 | 
						|
   method_name => [
 | 
						|
     'Package::Name',
 | 
						|
     'Other::Package::Name',
 | 
						|
   ],
 | 
						|
   other_method_name => [
 | 
						|
     'Package::Name',
 | 
						|
   ],
 | 
						|
 }
 | 
						|
 | 
						|
Should a subroutine have been defined more than once,
 | 
						|
the defining classes are listed in reverse order, such that the definition
 | 
						|
used at runtime is always:
 | 
						|
 | 
						|
 $data->{method_name}->[0];
 | 
						|
 | 
						|
=cut
 | 
						|
 | 
						|
sub subroutines {
 | 
						|
    my $self = shift;
 | 
						|
    my $class = (ref $self ? ref $self : $self);
 | 
						|
    my $results = {};
 | 
						|
 | 
						|
    my @super = superclasses($class);
 | 
						|
    foreach my $parent (reverse @super) {
 | 
						|
        my %sh = Module::Info->new_from_module( $parent )->subroutines;
 | 
						|
        my @subs = grep { $_ !~ m/^_/ }
 | 
						|
                   map { $_ =~ s/^.+:://; $_ }  ## no critic
 | 
						|
                   keys %sh;
 | 
						|
 | 
						|
        foreach my $sub (@subs) {
 | 
						|
            unshift @{ $results->{$sub} }, $parent;
 | 
						|
        }
 | 
						|
    }
 | 
						|
 | 
						|
    return $results;
 | 
						|
}
 | 
						|
 | 
						|
=item globals( $module || $object )
 | 
						|
 | 
						|
Returns a data structure showing how L<SNMP::Info> will resolve MIB Leaf
 | 
						|
Nodes configured through the C<%GLOBALS> hashes in C<$module>.
 | 
						|
 | 
						|
The data structure looks like:
 | 
						|
 | 
						|
 {
 | 
						|
   method_name => [
 | 
						|
     [ Package::Name        => 'mib_leaf_name' ],
 | 
						|
     [ Other::Package::Name => '1.3.6.1.4.1.9.2.1.58.0' ],
 | 
						|
   ],
 | 
						|
   other_method_name => [
 | 
						|
     [ Package::Name => 'mib_leaf.0' ],
 | 
						|
   ],
 | 
						|
 }
 | 
						|
 | 
						|
Where a method has been defined in different packages, then they are listed in
 | 
						|
reverse order, such that the mapping used by SNMP::Info is always:
 | 
						|
 | 
						|
 $data->{method_name}->[0];
 | 
						|
 | 
						|
=cut
 | 
						|
 | 
						|
sub globals { _walk_global_data(shift, 'GLOBALS') }
 | 
						|
 | 
						|
=item funcs( $module || $object )
 | 
						|
 | 
						|
Returns a data structure showing how L<SNMP::Info> will resolve MIB Tables
 | 
						|
configured through the C<%FUNCS> hashes in C<$module>.
 | 
						|
 | 
						|
See L<SNMP::Info::Layer3/"GLOBALS"> for further detail.
 | 
						|
 | 
						|
=cut
 | 
						|
 | 
						|
sub funcs { _walk_global_data(shift, 'FUNCS') }
 | 
						|
 | 
						|
=item munge( $module || $object )
 | 
						|
 | 
						|
Returns a data structure showing the subroutines used for munging returned
 | 
						|
values for any method defined in C<%FUNCS> or C<%GLOBALS>.
 | 
						|
 | 
						|
The data structure looks like:
 | 
						|
 | 
						|
 {
 | 
						|
   method_name => [
 | 
						|
     [ Package::Name        => '&subroutine' ],
 | 
						|
     [ Other::Package::Name => '&Other::Package::subroutine' ],
 | 
						|
   ],
 | 
						|
   other_method_name => [
 | 
						|
     [ Package::Name => '&subroutine' ],
 | 
						|
   ],
 | 
						|
 }
 | 
						|
 | 
						|
Where a mapping has been defined in different packages, then they are listed
 | 
						|
in reverse order, such that the munge subroutine used by SNMP::Info is always:
 | 
						|
 | 
						|
 $data->{method_name}->[0];
 | 
						|
 | 
						|
=cut
 | 
						|
 | 
						|
sub munge { _walk_global_data(shift, 'MUNGE') }
 | 
						|
 | 
						|
=item file( $module )
 | 
						|
 | 
						|
Returns the filename from which Perl will load the given module.
 | 
						|
 | 
						|
=cut
 | 
						|
 | 
						|
sub file {
 | 
						|
    my $self = shift;
 | 
						|
    my $class = (ref $self ? ref $self : $self);
 | 
						|
 | 
						|
    return Module::Info->new_from_module( $class )->file;
 | 
						|
}
 | 
						|
 | 
						|
=item superclasses( $class || $object )
 | 
						|
 | 
						|
Returns the list (in order) of the names of classes Perl will search to find
 | 
						|
methods for this SNMP::Info class or object instance.
 | 
						|
 | 
						|
Note this B<requires> the L<Class::ISA> distribution to be installed.
 | 
						|
 | 
						|
=cut
 | 
						|
 | 
						|
sub superclasses {
 | 
						|
    my $self = shift;
 | 
						|
    my $class = (ref $self ? ref $self : $self);
 | 
						|
 | 
						|
    Module::Load::load( $class );
 | 
						|
    return Class::ISA::self_and_super_path( $class );
 | 
						|
}
 | 
						|
 | 
						|
=item print_globals( $module || $object )
 | 
						|
 | 
						|
Pretty print the output of C<globals()>.
 | 
						|
 | 
						|
=cut
 | 
						|
 | 
						|
sub print_globals { _print_global_data(shift, 'GLOBALS') }
 | 
						|
 | 
						|
=item print_funcs( $module || $object )
 | 
						|
 | 
						|
Pretty print the output of C<funcs()>.
 | 
						|
 | 
						|
=cut
 | 
						|
 | 
						|
sub print_funcs { _print_global_data(shift, 'FUNCS') }
 | 
						|
 | 
						|
=item print_munge( $module || $object )
 | 
						|
 | 
						|
Pretty print the output of C<munge()>.
 | 
						|
 | 
						|
=cut
 | 
						|
 | 
						|
sub print_munge { _print_global_data(shift, 'MUNGE') }
 | 
						|
 | 
						|
=item print_superclasses( $class || $object )
 | 
						|
 | 
						|
Pretty print the output of C<superclasses()>.
 | 
						|
 | 
						|
=cut
 | 
						|
 | 
						|
sub print_superclasses {
 | 
						|
    print join ("\n", superclasses(@_)), "\n";
 | 
						|
}
 | 
						|
 | 
						|
=back
 | 
						|
 | 
						|
=head1 AUTHOR
 | 
						|
 | 
						|
Oliver Gorwits <oliver@cpan.org>
 | 
						|
 | 
						|
=head1 COPYRIGHT AND LICENSE
 | 
						|
 | 
						|
This software is copyright (c) 2014 by The SNMP::Info Project.
 | 
						|
 | 
						|
 # Redistribution and use in source and binary forms, with or without
 | 
						|
 # modification, are permitted provided that the following conditions are met:
 | 
						|
 #
 | 
						|
 #    * Redistributions of source code must retain the above copyright notice,
 | 
						|
 #      this list of conditions and the following disclaimer.
 | 
						|
 #    * Redistributions in binary form must reproduce the above copyright
 | 
						|
 #      notice, this list of conditions and the following disclaimer in the
 | 
						|
 #      documentation and/or other materials provided with the distribution.
 | 
						|
 #    * Neither the name of the University of California, Santa Cruz nor the
 | 
						|
 #      names of its contributors may be used to endorse or promote products
 | 
						|
 #      derived from this software without specific prior written permission.
 | 
						|
 #
 | 
						|
 # THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
 | 
						|
 # AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
 | 
						|
 # IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
 | 
						|
 # ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE
 | 
						|
 # LIABLE FOR # ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
 | 
						|
 # CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
 | 
						|
 # SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
 | 
						|
 # INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
 | 
						|
 # CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
 | 
						|
 # ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
 | 
						|
 # POSSIBILITY OF SUCH DAMAGE.
 | 
						|
 | 
						|
=cut
 | 
						|
 | 
						|
1;
 |