NAME Command::Template - A template to build command line arrays, and run them VERSION This document describes Command::Template version 0.001. SYNOPSIS use Command::Template qw< command_runner command_template cr ct >; #### COMMAND EXPANSION GENERATION (NO EXECUTION) #### # command_template is aliased to ct my $ct = command_template(qw{ ls [options=-l] <dir> }); # @c1 = qw< ls -l / > my @c1 = $ct->generate(dir => '/'); # @c2 = qw< ls -la / > my @c2 = $ct->generate(dir => '/etc', options => '-la'); # @c3 = @c4 = qw< ls /usr/bin > my @c3 = $ct->generate(dir => '/usr/bin', options => undef); my @c4 = $ct->generate(dir => '/usr/bin', options => []); # @c5 = qw< ls -l -a /usr/bin > my @c5 = $ct->generate(dir => '/usr/bin', options => [qw< -l -a >]); #### COMMAND RUNNING (VIA IPC::Run) #### # command_runner is aliased to cr my $runner = command_runner(qw{ ls [options=-l] <dir> }); # run command qw< ls -l / >, returns a Command::Template::RunRecord my $r = $runner->run(dir => '/'); my $run_successful = $r->success; my $exit_code = $r->exit_code; # 0 is OK as usual in UNIX my $received_signal = $r->signal; # e.g. if killed, ... my $stdout = $r->stdout; my $stderr = $r->stderr; my $merged = $r->merged; # stderr then stdout, no newlines # added # command_runner is aliased to cr my $other_runner = cr(qw< cat >); # gives stdin as stdout my $message = "blah\nblah"; $r = $other_runner->run(-stdin => $message); my $equal = $r->stdout eq $message; # true DESCRIPTION Command::Template eases the creation of objects that help building up command-lines and/or to run them. As an example, it might be needed to generate multiple commands according to the following template: ls [options=-l] <dir> i.e. the ls command, where there might be some options (defaulting to option -l) and there MUST be a directory dir (which in this case has no default). The following possible expansions adhere to the template above: ls -l / ls -la /usr ls -l -a /etc ls /run while the following does not (because there is no directory, which we want because we used the <...> form): ls -l The module provides two different interfaces: * "command_template" allows generating the commands as lists, which are then supposed to be used somehow (e.g. provided to system); * "command_runner" allows running the commands via IPC::Run. Both adhere to the same "Expansion Rules" explained below and implemented in Command::Template::Instance. Expansion Rules A template in Command::Template resembles the way commands are explained in documentation: * required parameters are enclosed by <...> delimiters; * optional parameters are enclosed by [...] delimiters. Parameters name can only start with a letter or an underscore, then can be followed by any sequence of underscores, letters, and digits. The following names are valid: foo bar dir baz1 baz_2 name_or_surname Any parameter can be optionally set with a default value, which can be set with the = sign, like in the following examples: <foo=this is FOO> <bar= BAR > Spaces are preserved starting immediately after the = sign. When Template::Command::Instance expands a template, it uses a hash that maps each name to the value it should take: * if undef, then the corresponding item is removed from the command. This allows getting rid of a default value set for an optional parameter; it is an error (leading to an exception) if the parameter is required; * if any other plain scalar, it is set as the value in the specific command line parameter location; * if an array reference, all items inside are put in the specific command line parameter location. The last possibility provides the flexibility to e.g. set options that require a parameter (e.g. it would be the case of option -name for command find). Running Commands While Command::Template::Instance only allows generating command lines as lists of strings, Command::Template::Runner allows running them by means of IPC::Run. All the heavy work for expanding is provided by Command::Template::Instance behind the scenes; the "SYNOPSIS" contains examples of actual running of commands. INTERFACE command_runner # parse a command line template and use it for running commands my $cr = command_runner(qw{ ls [options=-l] <dir> }); Returns a Command::Template::Runner object with a run method that allows actually running generated commands via IPC::Run. Each run returns a Command::Template::Runner::Record object that allows peeking into the actual result of the run. my $r = $cr->run(options => '-la', '/etc'); say $r->stdout if $r->success; cr Alias for "command_runner". command_template my $ct = command_template(qw{ galook <foo> <bar=N> [baz] [sil=Y] }) Returns a Command::Template::Instance object with a generate method that performs the actual expansion, returning a list of strings representing the command line. ct Alias for "command_template". EXTENDING If extending Command::Template is of interest, here are a few hints: * class Command::Template::Instance encapsulates parsing a command template and expanding it for generating command lists, based on a hash with values binding; * class Command::Template::Runner provides a simple running facility around Command::Template::Instance that uses IPC::Run to run expanded commands, returning the result as a Command::Template::Runner::Record object. * class Command::Template::Runner::Record provides accessors into the outcome of a single run from Command::Template::Runner. Possible reasons for expanding might be: * different or additional rules for expansions are needed. If this is the case, the alternative to Command::Template::Instance must provide a generate method; Comman::Template::Runner can still be used, as long as a reference to the alternative object for the instance is passed to the constructor. * a different running facility is needed. In this case, it suffices to impelement a new runner class, leveraging Command::Template::Instance for the command expansion. BUGS AND LIMITATIONS Minimum perl version 5.24. Report bugs through GitHub (patches welcome) at https://github.com/polettix/Command-Template. AUTHOR Flavio Poletti <flavio@polettix.it> COPYRIGHT AND LICENSE Copyright 2021 by Flavio Poletti <flavio@polettix.it> Licensed under the Apache License, Version 2.0 (the "License"); you may not use this file except in compliance with the License. You may obtain a copy of the License at http://www.apache.org/licenses/LICENSE-2.0 Unless required by applicable law or agreed to in writing, software distributed under the License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the License for the specific language governing permissions and limitations under the License.