NAME
    pkg - transparently use packages and inner packages

SYNOPSIS
      # standard operations
      # works on either inner or normal packages
      # -------------------------------       ----------------
      use pkg 'A';                       #=>  use A;
      use pkg 'A', 'a', 'b';             #=>  use A 'a', 'b';
      use pkg 'A', [];                   #=>  use A ();

      # extra operations

      # default alias for a class package
      use pkg -alias => 'A::B::C';
      C->new(...); #equivalent to A::B::C->new();

      # specific alias for a class package
      use pkg 'A::B::C' => -as => 'ABC';
      ABC->new( ); # equivalent to A:B::C->new;

      # multiple packages
      use pkg [ 'A::B::C' => -as => 'ABC'],
              [ 'A::B'    => -as => 'AB' ];

      # operate on A and its inner packages
      use pkg 'A', '-inner';

      # operate only on the inner packages of A
      use pkg 'A', '-only_inner';

      # operate on A and its inner packages, excluding anything below A::B
      use pkg 'A', -inner, -exclude => qr/^A::B::/;

DESCRIPTION
    pkg extends the standard "use" statement to more transparently handle
    inner packages, additionally incorporating some of the functionality of
    the aliased and use pragmata with a few extra twists.

    An inner package is one which does not have the same name as the (fully
    qualified) module in which it is defined. For example, if A.pm contains

      package A;

      sub a { ... }

      package A::B;

      sub ab { ... }

      package A::C;

      sub ac { ... }

      1;

    packages "A::B" and "A::C" are inner packages. The use statement (as
    well as most pragmata dealing with modules) does not handle inner
    packages. Some, such as parent, do, but require the user (via the
    "-norequire" option) to know if the package is inner or not.

    For example, after loading the above module:

      use A;

    You could simply call

      A::a();
      A::B::ab();
      A::C::ac();

    But, what if package A::B exported ab? Its import routine is not
    automatically called when A is loaded. If you try to do this

      use A::B 'ab';

    you'll get an error from Perl as it tries to search for a file named
    e.g., A/B.pm. It doesn't check to see if the "A::B" package has been
    loaded.

    Instead, you'd need to do this:

      A::B->import( 'ab' );
      ab();

    Or, using pkg:

      use pkg [ 'A' ], [ 'A::B' => qw[ ab ] ];

  Simple Usage
    In its simplest form, pkg accepts a *list* of a package name (*as a
    string*) and its imports.

      use pkg 'A::B', qw( funca funcb );

    This loads the package "A::B" (if necessary) and imports the functions
    funca and funcb. Note that if "A::B" is an inner package, the module
    (file) which contains it must be loaded prior to this e.g.

      # either of these is sufficient
      use A;
      use pkg 'A';

    This needs to be done only *once* (not every time an inner package is
    used). Of course it can be combined with loading "A::B":

      use pkg [ 'A' ], [ 'A::B' => qw( funca funcb ) ];

  Controlling imports
    There is a subtlety in how the standard use statement handles empty or
    non-existent import lists:

       use A;           # call A->import();
       use A 'a', 'b'   # call A->import( 'a', 'b' );
       use A ();        # do *not* call A->import;

    This mechanism isn't available to pkg as it cannot tell the difference
    between:

       use pkg 'A';
       use pkg 'A', ();

    Instead, use "[]" instead of "()":

       use pkg 'A', [];

    What if you need to pass a "[]" to "A->import()"? Use the "-import"
    package option:

      use pkg 'A', -import => [];        #=> use A [];
      use pkg 'A', -import => '-import'; #=> use A '-import';

    "-import" instructs pkg that all remaining arguments should be passed to
    the package's import routine.

    Note that the following are equivalent

      use A (), 'a';
      use pkg 'A', [], 'a';

    and result in

      A->import( 'a' );

  Multiple packages
    Multiple packages may be operated on by passing each package's
    specifications as separate array references:

      use pkg ['A'], ['A::B', qw( funca funcb ) ];

OPTIONS
    pkg accepts options to modify its behavior. "Global" options (which
    affect more than one package) can appear in multiple places if more than
    one package is manipulated. Package specific options always appear
    directly after the package name and apply only to that package.

    If there's only one package, the syntax is simple. Global options occur
    before the package name.

      use pkg -norequire => 'My::Package' -as => 'MyP';

    "-norequire" is a global option, and "-as" is a package option.

    If more than one package is specified, global options may occur both
    outside of the package specifications as well as inside of them. For
    example,

      use pkg
        -alias =>
        [ 'My::FirstClass' ],
        [ -noalias => 'My::SecondClass' ]
        [ 'My::ThirdClass' => -as => 'ThirdClassIsBetterThanFirst' ]
        -noalias =>
        [ 'My::Library1' ],
        [ 'My::Library2' ],
        [ 'My::Library3' ],
        ;

    The options appearing outside of the package specifications affect all
    packages which follow. The options inside a specification affect that
    package only. As shown, some options may be negated, and package options
    may override global ones.

  Global Options
    "-alias"
    "-noalias"
        Provide (or don't provide) shortened names for class names. These
        are simply the last component of the original name.

        The idea is borrowed from the "aliased" pragma; pkg constructs and
        exports a subroutine with the shortened name which returns the fully
        qualified name.

        For example,

          use pkg -alias => 'A::Long:Class';

          # these are equivalent
          A::Long::Class->new();
          Class->new();

        If multiple classes are loaded, no checks are performed to ensure
        that the shortened names are unique. Use the "-as" package option to
        specify specific names.

    "-strip"
        Created aliases by removing a prefix from the succeeding class
        names. The prefix may be specified in one of two ways:

        "-strip" *string*
            Remove a leading *string* from the class names. All component
            separators ("::") are also removed. For example,

              -strip => 'A::C', 'A::C::E::F::G'

            results in an alias of "EFG".

        "-strip { pfx => *string*, sep => *string* }"
            Remove *prefix* from class names, and replace the class
            component separators ("::") with the specified string. After
            prefix removal, a leading "::" sequence is removed.

    "-require"
    "-norequire"
        Try to load (or don't try to load) the packages with
        Class::Load::load_class. If you know that the package is an inner
        package and the file containing it has already been loaded,
        specifying "-norequire" can speed things up by not loading
        Class::Load.

        By default packages are loaded (i.e. "-require").

  Package Options
    "-as" => *string*
        Create an alias named *string* for the package. The aliased name
        must be a legal subroutine name.

        For example,

          use pkg 'A::Long:Class' => -as => 'ALC';

          # these are equivalent
          A::Long::Class->new();
          ALC->new();

    "-import"
        There's always a chance that a package's import list may be confused
        with pkg package options (perhaps it also has a "-as" option). To
        avoid this, a package's import list may be preceded with the
        "-import" option, which indicates to pkg that all of the following
        arguments are to be passed as is to the package's import routine.

          # these are equivalent
          use A ( '-as', 'func1', 'func2' );
          use pkg 'A' => -import => ( '-as', 'func1', 'func2' );

    "-require"
    "-norequire"
        This has the same functionality as the similarly named global
        options, but as a package option may be placed after the package
        name for aesthetics.

    "-inner"
        In addition to the package, process any of its currently loaded
        inner packages. Inner packages are discovered via
        Devel::InnerPackage, and must fall within the "hierarchy" of the
        package. For example, given a module with the following contents:

          package A;
          sub a {}

          package A::B;
          sub ab {}

          package B;
          sub b {}

        "A::B" is an inner package of "A", but "B" is not. Inner packages
        must have defined symbols, otherwise they will not be identified.

    "-only_inner"
        Similar to "-inner", but *only* the inner packages are processed,
        not the package itself.

        This *does not* affect whether the package is loaded; this is
        controlled by the "-require" option.

    "-include" *specification*
        Check the package name against the *specification* using the smart
        match operator ("~~") and ignore it if it does not match. If
        "-inner" or "-only_inner" are specified, inner packages are also
        checked.

        This *does not* affect whether the package is loaded; this is
        controlled by the "-require" option.

        This is most useful when either "-inner" or "-only_inner" is
        specified.

    "-exclude" *specification*
        Check the package name against the *specification* using the smart
        match operator ("~~") and ignore it if it matches. The "-exclude"
        match is processed after "-include" if both are specified. If
        "-inner" or "-only_inner" are specified, inner packages are also
        checked.

        This *does not* affect whether the package is loaded; this is
        controlled by the "-require" option.

        This is most useful when either "-inner" or "-only_inner" is
        specified.

    "-version" => *version*
        Specify the minimum acceptable version of the package.

DIAGNOSTICS
    "global option '%s': unknown option"
        The specified option wasn't recognized as a global option.

    "package option '%s': unknown option"
        The specified option wasn't recognized as a package option.

    "option '%s': cannot be negated"
        An illegal negation of the specified option was specified.

    "option '%s': not enough values"
        The specified option required more values than was specified.

    "can't use option "%s" when looping over inner packages"
        The specifed option cannot be used in conjunction with "-inner" or
        "-only_inner".

    "-strip: no prefix specified"
        The "-strip" option requires an argument specifying the prefix to
        remove.

    "internal error"
        Something really bad happened.

IMPLEMENTATION
    pkg does very little on its own. It uses the following modules:

    Class::Load
        Class::Load::load_class is used to load the package. It also takes
        care of checking package versions.

    Import::Into
        This is used to call a package's import routine

    aliased
        This provided the inspiration for the aliasing implementation.

    Devel::InnerPackages
        Discover a package's inner self.

DEPENDENCIES
    Class::Load, Import::Into, Devel::InnerPackages, Perl 5.10.1.

INCOMPATIBILITIES
    None reported.

BUGS AND LIMITATIONS
    pkg is focussed specifically on dealing with packages and is not
    intended as a general purpose replacement for the standard use
    statement. In particular it does not know how to deal with other
    pragmata, e.g.,

      use pkg strict;

    will probably not do anything useful and will most probably advance the
    heat death of the universe.

    Please report any bugs or feature requests to "bug-pkg@rt.cpan.org", or
    through the web interface at
    <http://rt.cpan.org/Public/Dist/Display.html?Name=pkg>.

SEE ALSO
    aliased, namespace, as, use.

VERSION
    Version 0.01

LICENSE AND COPYRIGHT
    Copyright (c) 2013 Diab Jerius

    pkg is free software: you can redistribute it and/or modify it under the
    terms of the GNU General Public License as published by the Free
    Software Foundation, either version 3 of the License, or (at your
    option) any later version.

    This program is distributed in the hope that it will be useful, but
    WITHOUT ANY WARRANTY; without even the implied warranty of
    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General
    Public License for more details.

    You should have received a copy of the GNU General Public License along
    with this program. If not, see <http://www.gnu.org/licenses/>.

AUTHOR
    Diab Jerius <djerius@cpan.org>