NAME
Proc::SafeExec - Convenient utility for executing external commands in various ways.
SYNOPSIS
use Proc::SafeExec;
my $command = new Proc::SafeExec({
# Choose just one of these.
exec => ["ls", "-l", "myfile"], # exec() after forking.
fork => 1, # Return undef in the child after forking.
# Specify whether to capture each. Specify a file handle ref to dup an
# existing one. Specify "new" to create a new file handle, "default" or undef
# to keep the parent's descriptor, or "close" to close it.
stdin => \*INPUT_PIPE,
stdout => \*OUTPUT_PIPE,
stderr => "new",
# Miscellaneous options.
child_callback => \&fref, # Specify a function to call in the child after fork(), for example, to drop privileges.
no_autowait => 1, # Don't automatically call $command->wait() when $command is destroyed.
real_arg0 => "/bin/ls", # Specify the actual file to execute.
untaint_args => 1, # Untaint the arguments before exec'ing.
});
printf "Child's PID is %s\n", $command->child_pid if $command->child_pid;
The wait method waits for the child to exit or checks whether it already exited:
$command->wait({
# Optional hash of options.
no_close => 1, # Don't close "new" file handles.
nonblock => 1, # Don't wait if the child hasn't exited (implies no_close).
});
To communicate with the child:
# Perl doesn't understand <$command->stdout>.
my $command_stdout = $command->stdout;
my $command_stderr = $command->stderr;
$line = <$command_stdout>;
$line = <$command_stderr>;
print {$command->stdin} "mumble\n";
To check whether the child exited yet:
print "Exit status: ", $command->exit_status, "\n" if $command->wait({nonblock => 1});
To wait until it exits:
$command->wait();
print "Exit status: ", $command->exit_status, "\n";
DESCRIPTION
Proc::SafeExec provides an easy, safe way to execute external programs. It replaces all of Perl's questionable ways of accomodating this, including system(), open() with a pipe, exec(), back-ticks, etc. This module will never automatically invoke /bin/sh. This module is easy enough to use that /bin/sh should be unnecessary, even for complex pipelines.
For all errors, this module dies setting $@.
Errors from exec() in the child are reported gracefully to the parent. This means that if anything fails in the child, the error is reported through $@ with die just like any other error. This also reports $@ if child_callback dies when it is called between fork() and exec(). This is accomplished by passing $@ through an extra pipe that's closed when exec succeeds. Note: A side-effect of this is $@ is stringified if it isn't a string.
CAVEATS
When using an existing file handle by passing a reference for stdin, stdout, or stderr, new() closes the previously open file descriptor. This is to make sure, for example, that when setting up a pipeline the child process notices EOF on its stdin. If you need this file handle to stay open, dup it first. For example:
open my $tmp_fh, "<&", $original_fh or die "dup: $!";
my $ls = new Proc::SafeExec({exec => ["ls"], stdout => $tmp_fh});
# $tmp_fh is now closed.
By default, $command->wait() closes any new pipes opened in the constructor. This is to prevent a deadlock where the child is waiting to read or write and the parent is waiting for the child to exit. Pass no_close to $command->wait() to prevent this (see above). Also, by default the destructor calls $command->wait() if child hasn't finished. This is to prevent zombie processes from inadvertently accumulating. To prevent this, pass no_autowait to the constructor. The easiest way to wait for the child is to call the wait method, but if you need more control, set no_autowait, then call child_pid to get the PID and do the work yourself.
EXAMPLES
It's easy to execute several programs to form a pipeline. For the first program, specify "new" for stdout. Then execute the second one, and specify stdout from the first one for the stdin of the second one. For example, here's how to write the equivalent of system("ls | sort > output.txt"):
open my $output_fh, ">", "output.txt" or die "output.txt: $!\n";
my $ls = new Proc::SafeExec({exec => ["ls"], stdout => "new"});
my $sort = new Proc::SafeExec({exec => ["sort"], stdin => $ls->stdout, stdout => $output_fh});
$ls->wait();
$sort->wait();
printf "ls exited with status %i\n", ($ls->exit_status >> 8);
printf "sort exited with status %i\n", ($sort->exit_status >> 8);
INSTALLATION
This module has no dependencies besides Perl itself. Follow your favorite standard installation procedure.
To test the module, run the following command line:
$ perl -e 'use Proc::SafeExec; print Proc::SafeExec::test();'
VERSION AND HISTORY
Version 1.0, released 2007-05-23.
SEE ALSO
The Subversion repository is at svn://svn.devpit.org/Proc-SafeExec/
AUTHOR
Leif Pedersen, <bilbo@hobbiton.org>
COPYRIGHT AND LICENSE
This may be distributed under the terms below (BSD'ish) or under the GPL.
Copyright (c) 2007
All Rights Reserved
Meridian Environmental Technology, Inc.
4324 University Avenue, Grand Forks, ND 58203
http://meridian-enviro.com
Redistribution and use in source and binary forms, with or without
modification, are permitted provided that the following conditions are
met:
1. Redistributions of source code must retain the above copyright
notice, this list of conditions and the following disclaimer.
2. 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.
THIS SOFTWARE IS PROVIDED BY AUTHORS 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 AUTHORS 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.