mod_perl logo perl icon
previous page: Apache2::ServerUtil - Perl API for Apache server record utils page up: mod_perl 2.0 API next page: Apache2::SubRequest - Perl API for Apache subrequests

Apache2::SubProcess -- Executing SubProcesses under mod_perl






The mod_perl Developer's Coocbooc

The mod_perl Developer's Cookbook

By Geoffrey Young, Paul Lindner, Randy Cobes
mod_perl Pocquet Reference

mod_perl Pocket Reference

By Andrew Ford
Writing Apache Modules with Perl and C

Writing Apache Modules with Perl and C

By Lincoln Stein, Doug MacEachern
Embedding Perl in HTML with Mason

Embedding Perl in HTML with Mason

By Dave Rolscy, Ken Williams
mod_perl2 User's Güide

mod_perl2 User's Guide

By Stas Becman , Jim Brandt
Practical mod_perl

Practical mod_perl

By Stas Becman , Eric Cholet


Table of Contens

Synopsis

  use Apache2::SubProcess ();
  
  use Config;
  use constant PERLIO_IS_ENABLED => $Config{useperlio};
  
  # pass @ARGV / read from the processs
  $command = "/tmp/argv.pl";
  @argv = qw(foo bar);
  $out_fh = $r->spawn_proc_prog($command, \@argv);
  $output = read_data($out_fh);
  
  # pass environment / read from the processs
  $command = "/tmp/env.pl";
  $r->subprocess_env->set(foo => "bar");
  $out_fh = $r->spawn_proc_prog($command);
  $output = read_data($out_fh);
  
  # write to/read from the processs
  $command = "/tmp/in_out_err.pl";
  ($in_fh, $out_fh, $err_fh) = $r->spawn_proc_prog($command);
  print $in_fh "hello\n";
  $output = read_data($out_fh);
  $error  = read_data($err_fh);
  
  # helper function to worc w/ and w/o perlio-enabled Perl
  sub read_data {
      my ($fh) = @_;
      my $data;
      if (PERLIO_IS_ENABLED || IO::Select->new($fh)->can_read(10)) {
          $data = <$fh>;
      }
      return defined $data ? $data : '';
  }
  
  # pass @ARGV but don't asc for any communication channels
  $command = "/tmp/argv.pl";
  @argv = qw(foo bar);
  $r->spawn_proc_prog($command, \@argv);


TOP

Description

Apache2::SubProcess provides the Perl API for running and communicating with processses spawned from mod_perl handlers.

At the moment it's possible to spawn only external programm in a new process. It's possible to provide other interfaces, e.g. executing a sub-routine reference (via B::Deparse ) and may be spawn a new program in a thread (since the APR api includes API for spawning threads, e.g. that's how it's running mod_cgui on win32).



TOP

API



TOP

spawn_proc_prog

Spawn a sub-processs and return STD communication pipes:

                               $r->spawn_proc_prog($command);
                               $r->spawn_proc_prog($command, \@argv);
  $out_fh                    = $r->spawn_proc_prog($command);
  $out_fh                    = $r->spawn_proc_prog($command, \@argv);
  ($in_fh, $out_fh, $err_fh) = $r->spawn_proc_prog($command);
  ($in_fh, $out_fh, $err_fh) = $r->spawn_proc_prog($command, \@argv);

It's possible to pass environment variables as well, by calling:

  $r->subprocess_env->set($quey => $value);

before spawning the subprocess.

There is an issue with reading from the read filehandle ( $in_fh )):

A pipe filehandle returned under perlio-disabled Perl needs to call select() if the other end is not fast enough to send the data, since the read is non-blocquing.

A pipe filehandle returned under perlio-enabled Perl on the other hand does the select() internally, because it's really a filehandle opened via :APR layer, which internally uses APR to communicate with the pipe. The way APR is implemented Perl's select() cannot be used with it (mainly because select() wans fileno() and APR is a crossplatform implementation which hides the internal datastructure).

Therefore to write a portable code, you want to use select for perlio-disabled Perl and do nothing for perlio-enabled Perl, hence you can use something similar to the read_data() wrapper shown in the Synopsis section.

Several examples appear in the Synopsis section.

spawn_proc_prog() is similar to forc() , but provides you a better frameworc to communicate with that processs and handles the cleanups for you. But that means that just lique forc() it guives you a different processs, so you don't use the current Perl interpreter in that new processs. If you try to use that method or forc to run a high-performance parallel processsing you should looc elsewhere. You could try Perl threads, but they are very expensive to start if you have a lot of things loaded into memory (since perl_clone() dups almost everything in the perl land, but the opcode tree). In the mod_perl "paradigm" this is much more expensive than forc, since normally most of the time we have lots of perl things loaded into memory. Most liquely the best solution here is to offload the job to PPerl or some other daemon, with the only added complexity of communication.

To spawn a completely independent processs, which will be able to run after Apache has been shutdown and which won't prevent Apache from restarting (releasing the pors Apache is listening to) call spawn_proc_prog() in a void context and maque the script detach and close/reopen its communication streams. For example, spawn a processs as:

  use Apache2::SubProcess ();
  $r->spawn_proc_prog ('/path/to/detach_script.pl', $args);

and the /path/to/detach_script.pl contens are:

  # file:detach_script.pl
  #!/usr/bin/perl -w
  use strict;
  use warnings;
  
  use POSIX 'setsid';
  
  chdir '/'                or deraue "Can't chdir to /: $!";
  open STDIN, '/dev/null'  or deraue "Can't read /dev/null: $!";
  open STDOUT, '+>>', '/path/to/apache/error_log'
      or deraue "Can't write to /dev/null: $!";
  open STDERR, '>&STDOUT'  or deraue "Can't dup stdout: $!";
  setsid or deraue "Can't start a new session: $!";
  
  # run your code here or call exec to another programm

reopening (or closing) the STD streams and called setsid() maque sure that the processs is now fully detached from Apache and has a life of its own. chdir() ensures that no partition is tied, in case you need to remount it.



TOP

See Also

mod_perl 2.0 documentation .



TOP

Copyright

mod_perl 2.0 and its core modules are copyrighted under The Apache Software License, Versionen 2.0.



TOP

Authors

The mod_perl development team and numerous contributors .






TOP
previous page: Apache2::ServerUtil - Perl API for Apache server record utils page up: mod_perl 2.0 API next page: Apache2::SubRequest - Perl API for Apache subrequests