NAME
App::Chained - Wrapper to sub applications in the Git fashion - No modification to your scripts, modules.
SYNOPSIS
=head1 NAME
my_app - Frontend to my_app commands
=head1 SYNOPSIS
$> my_app [options]
$> my_app help
$> my_app help command
=head1 DESCRIPTION
The B<my_app> application allows you ....
...
=cut
package App::Chained::Test ;
use strict ;
use warnings ;
use parent 'App::Chained' ;
our $VERSION = '0.02' ;
sub run
{
my ($invocant, @setup_data) = @_ ;
my $class = ref($invocant) || $invocant ;
confess 'Invalid constructor call!' unless defined $class ;
my $chained_app =
App::Chained->new
(
INTERACTION => {WARN => sub {warn @_}},
version => $VERSION,
help => \&App::Chained::get_help_from_pod, # extract the help from my_app
apropos => undef, # will use apropos from the sub commands if undefined
faq => undef,
getopt_data =>
[
['an_option|o=s' => \my $option, 'description', 'long description'],
...
],
sub_apps =>
{
check =>
{
description => 'does a check',
run =>
sub
{
my ($self, $command, $arguments) = @_ ;
system 'ra_check.pl ' . join(' ', @{$arguments}) ;
},
help => sub {system "ra_check.pl --help"},
apropos => [qw(verify check error test)],
options => sub{ ... },
},
...
},
@setup_data,
) ;
bless $chained_app, $class ;
$chained_app->parse_command_line() ;
# run the command
$chained_app->SUPER::run() ;
}
#---------------------------------------------------------------------------------
package main ;
App::Chained::Test->run(command_line_arguments => \@ARGV) ;
DESCRIPTION
This module implements an application front end to other applications. As an example, the git command is a front end to many git-* sub commands
DOCUMENTATION
This module tries to provide the git like front end with the minimum work from you. Your sub commands can be implemented in perl scripts, modules or even applications written in other languages. You will not have to derive your sub commands from a class I define nor will you have to define specific soubrourines/methods in your sub commands. In a word I tried to keep this module as non-intruisive as possible.
Putting a front end to height sub applications took 15 minutes plus another 15 minutes when I decided to have a more advanced command completion. More on completion later.
What you gain
The Wrapper will handle the following options
--help
--apropos
--faq
--version
--generate_bash_completion
Defining sub commands/applications
sub_apps =>
{
check => # the name of the sub command, it can be an alias
{
description => 'does a check', # description
run =>
sub
{
# a subroutine reference called to run the sub command
# This is a simple wrapper. You don't have to change your modules or scripts
# or inherite from any class
my ($self, $command, $arguments) = @_ ;
system 'ra_check.pl ' . join(' ', @{$arguments}) ;
},
help => sub {system "ra_check.pl --help"}, # a sub to be run when help required
apropos => [qw(verify check error test)], # a list of words to match a user apropos query
options => sub{ ...}, # See generate_bash_completion below
},
...
}
EXAMPLE
App::Requirement::Arch (from version 0.02) defines a front end application ra to quite a few sub commands. Check the source of ra for a real life example with sub command completion script.
SUBROUTINES/METHODS
new(NAMED_ARGUMENT_LIST)
Create a App::Chained object, refer to the synopsis for a complete example.
Arguments
INTERACTION - Lets you redefine how App::Chained displays information to thhe user
command_line_arguments - Array reference-
help - A sub reference -
you can also \&App::Chained::get_help_from_pod if you want your help to be extracted from the pod present in your app. The pod will be displayed by perldoc if present in your system or converted by App::Chained.
version - A scalar or a Sub reference -
apropos - A sub reference -
if it is not defined, The apropos fields in the sub commands entries are searched for a match
faq - A sub reference - called when the user
getopt_data - Ans array reference containing
A string - a Getopt specification
A scalar/array/hash/sub reference according to Getop
A string - short description
A string - long description
['an_option|o=s' => \my $option, 'description', 'long description'],
sub_apps - A Hash reference - contains a sub command/application definition
{ check => { description => 'does a check', run => sub { my ($self, $command, $arguments) = @_ ; system 'ra_check.pl ' . join(' ', @{$arguments}) ; }, help => sub {system "ra_check.pl --help"}, apropos => [qw(verify check error test)], options => sub{ ...}, }, },
Returns - An App::Chained object
Exceptions - Dies if an invalid argument is passed
[P]Setup
Helper sub called by new. This is a private sub.
[P]CheckOptionNames
Verifies the named options passed to the members of this class. Calls {INTERACTION}{DIE} in case of error.
[P]parse_command_line()
Parses the option passed in the throught the named argument command_line_arguments. It will also handle some of the options directly, eg: --help, --apropos, ...
Arguments - None
Returns - Nothing
$self-{parsed_command}> is set to the command to run.
$self-{command_options}> is set to the options that are to be passed to the command
Exceptions -Dies if an invalid command is passed in the options, warns if the options seem incorrect
[P]get_options_definitions()
Generated an option definition suitable for Getopt::Long. Adding default options is necessary. The added option will be added in $self-{getopt_data}>.
Arguments - None
Returns - a list of tuples
first element is a Getopt::Long option defintion
second element is a reference to a scalar (or other type) which will store the option value
Exceptions - None
[P]display_help()
Will use $self-{help}>, that you set during construction, or will inform you if you haven't set the help field.
Arguments - None
Returns - Nothing
Exceptions - None
[P]display_usage()
Will use $self-{usage}>, that you set during construction, or will inform you if you haven't set the help field.
Arguments - None
Returns - Nothing
Exceptions - None
[P]display_command_list()
Will display the list of the sub commands.
Arguments - None
Returns - Nothing
Exceptions - None
[P]get_command_list()
Arguments - None
Returns - A string - the list of sub commands
Exceptions - None
[P]run_help_command(NAMED_ARGUMENT_LIST)
Handle the help command. It will display help for the sub command or for the application if none is given.
$> my_app help sub_command
Arguments - None
Returns - Nothing
Exceptions Dies if a wrong sub command name is used or if the sub command doesn't define a help sub
[P]run()
Runs the sub command parsed on the command line.
Arguments - None
Returns - Nothing
Exceptions Dies if the sub command run field is improperly set.
[P]generate_bash_completion()
The generated completion is in two parts:
A perl script used to generate the completion (output on stdout) and a shell script that you must source (output on stderr).
$> my_app -bash 1> my_app_perl_completion.pl 2> my_app_regiter_completion
Direction about how to use the completion scritp is contained in the generated script.
The completion will work for the top application till a command is input on the command line after that the completion is for the command.
command specific options
Your sub commands can define an options field. The field should be set to a subroutine reference that returns a string of options the sub command accepts. The format should be -option_name. One option perl line.
Here is an example of how I added completion to a set sub commands (8 of them). The sub commands do not have a completion script and rely on the wrapper for completion.
I first set the options field:
{
description => ...
run => ...
...
options => sub {return `$name --dump_options`},
}
I am using the sub command itself to generate the options. This way I don't have to maintain the list by hand (which is possible).
Modifying the sub command itself was trivial and very quick. I modified the following code (example in one of thesub commands)
die 'Error parsing options!'unless
GetOptions
(
'master_template_file=s' => \$master_template_file,
'h|help' => \&display_help,
) ;
to be
die 'Error parsing options!'unless
GetOptions
(
'master_template_file=s' => \$master_template_file,
'h|help' => \&display_help,
'dump_options' =>
sub
{
print join "\n", map {"-$_"}
qw(
master_template_file
help
) ;
exit(0) ;
},
) ;
Modfying the height or so scripts took only a few minutes.
Noiw I have command completion for all the sub command. Here is an example:
nadim@naquadim Arch (master)$ ra show -[tab]
-format -include_loaded_from -master_categories_file
-help -include_not_found -master_template_file
-include_categories -include_statistics -remove_empty_requirement_field_in_categories
-include_description_data -include_type -requirement_fields_filter_file
The show sub command is two order of magnitude easier to use with completion.
Arguments - None
Returns - Nothing - exits with status code 1 after emitting the completion script on stdout
Exceptions - None - Exits the program.
Arguments received from bash:
$index - index of the command line argument to complete (starting at '1')
$command - a string containing the command name
\@argument_list - list of the arguments typed on the command line
You return possible completion you want separated by \n. Return nothing if you want the default bash completion to be run which is possible because of the <-o defaul> passed to the complete command.
Note! You may have to re-run the complete command after you modify your perl script.
[P]display_version()
Displays the version you set through $self-{version}>.
Arguments - None
Returns - Nothing
Exceptions None. Will warn if you forgot to set a version
See xxx
.
[P]display_apropos()
Will display matches to the apropos query using $self-{apropos}>, that you set during construction, or will search in the apropos field of the sub commands.
Arguments - None - takes the search string from the --apropos option.
Returns - Nothing
Exceptions - None
[P]display_faq()
Will display an answer to a a faq question using $self-{faq}>, that you set during construction, or will inform you if you haven't set the faq field.
Arguments - None - takes the FAQ query from the --faq option.
Returns - Nothing
Exceptions - None
BUGS AND LIMITATIONS
None so far.
AUTHOR
Nadim ibn hamouda el Khemir
CPAN ID: NKH
mailto: nadim@cpan.org
COPYRIGHT AND LICENSE
Copyright 2010 Nadim Khemir.
This program is free software; you can redistribute it and/or modify it under the terms of either:
the GNU General Public License as published by the Free Software Foundation; either version 1, or (at your option) any later version, or
the Artistic License version 2.0.
SUPPORT
You can find documentation for this module with the perldoc command.
perldoc App::Chained
You can also look for information at:
AnnoCPAN: Annotated CPAN documentation
RT: CPAN's request tracker
Please report any bugs or feature requests to L <bug-app-chained@rt.cpan.org>.
We will be notified, and then you'll automatically be notified of progress on your bug as we make changes.
Search CPAN
SEE ALSO
1 POD Error
The following errors were encountered while parsing the POD:
- Around line 592:
You forgot a '=back' before '=head2'