Asynchronous shell exec in PHP

PhpAsynchronousShell

Php Problem Overview


I've got a PHP script that needs to invoke a shell script but doesn't care at all about the output. The shell script makes a number of SOAP calls and is slow to complete, so I don't want to slow down the PHP request while it waits for a reply. In fact, the PHP request should be able to exit without terminating the shell process.

I've looked into the various exec(), shell_exec(), pcntl_fork(), etc. functions, but none of them seem to offer exactly what I want. (Or, if they do, it's not clear to me how.) Any suggestions?

Php Solutions


Solution 1 - Php

If it "doesn't care about the output", couldn't the exec to the script be called with the & to background the process?

EDIT - incorporating what @AdamTheHut commented to this post, you can add this to a call to exec:

" > /dev/null 2>/dev/null &"

That will redirect both stdio (first >) and stderr (2>) to /dev/null and run in the background.

There are other ways to do the same thing, but this is the simplest to read.


An alternative to the above double-redirect:

" &> /dev/null &"

Solution 2 - Php

I used at for this, as it is really starting an independent process.

<?php
    `echo "the command"|at now`;
?>

Solution 3 - Php

To all Windows users: I found a good way to run an asynchronous PHP script (actually it works with almost everything).

It's based on popen() and pclose() commands. And works well both on Windows and Unix.

function execInBackground($cmd) {
    if (substr(php_uname(), 0, 7) == "Windows"){
        pclose(popen("start /B ". $cmd, "r")); 
    }
    else {
        exec($cmd . " > /dev/null &");  
    }
} 

Original code from: http://php.net/manual/en/function.exec.php#86329

Solution 4 - Php

On linux you can do the following:

$cmd = 'nohup nice -n 10 php -f php/file.php > log/file.log & printf "%u" $!';
$pid = shell_exec($cmd);

This will execute the command at the command prompty and then just return the PID, which you can check for > 0 to ensure it worked.

This question is similar: https://stackoverflow.com/questions/209774/does-php-have-threading

Solution 5 - Php

php-execute-a-background-process has some good suggestions. I think mine is pretty good, but I'm biased :)

Solution 6 - Php

In Linux, you can start a process in a new independent thread by appending an ampersand at the end of the command

mycommand -someparam somevalue &

In Windows, you can use the "start" DOS command

start mycommand -someparam somevalue

Solution 7 - Php

the right way(!) to do it is to

  1. fork()
  2. setsid()
  3. execve()

fork forks, setsid tell the current process to become a master one (no parent), execve tell the calling process to be replaced by the called one. so that the parent can quit without affecting the child.

 $pid=pcntl_fork();
 if($pid==0)
 {
   posix_setsid();
   pcntl_exec($cmd,$args,$_ENV);
   // child becomes the standalone detached process
 }

 // parent's stuff
 exit();

Solution 8 - Php

I used this...

/** 
 * Asynchronously execute/include a PHP file. Does not record the output of the file anywhere.  
 * Relies on the PHP_PATH config constant.
 *
 * @param string $filename	file to execute
 * @param string $options	(optional) arguments to pass to file via the command line
 */ 
function asyncInclude($filename, $options = '') {
	exec(PHP_PATH . " -f {$filename} {$options} >> /dev/null &");
}

(where PHP_PATH is a const defined like define('PHP_PATH', '/opt/bin/php5') or similar)

It passes in arguments via the command line. To read them in PHP, see argv.

Solution 9 - Php

I also found Symfony Process Component useful for this.

use Symfony\Component\Process\Process;

$process = new Process('ls -lsa');
// ... run process in background
$process->start();

// ... do other things

// ... if you need to wait
$process->wait();

// ... do things after the process has finished

See how it works in its GitHub repo.

Solution 10 - Php

The only way that I found that truly worked for me was:

shell_exec('./myscript.php | at now & disown')

Solution 11 - Php

You can also run the PHP script as daemon or cronjob: #!/usr/bin/php -q

Solution 12 - Php

Use a named fifo.

#!/bin/sh
mkfifo trigger
while true; do
    read < trigger
    long_running_task
done

Then whenever you want to start the long running task, simply write a newline (nonblocking to the trigger file.

As long as your input is smaller than PIPE_BUF and it's a single write() operation, you can write arguments into the fifo and have them show up as $REPLY in the script.

Solution 13 - Php

without use queue, you can use the proc_open() like this:

    $descriptorspec = array(
        0 => array("pipe", "r"),
        1 => array("pipe", "w"),
        2 => array("pipe", "w")    //here curaengine log all the info into stderror
    );
    $command = 'ping stackoverflow.com';
    $process = proc_open($command, $descriptorspec, $pipes);

Attributions

All content for this solution is sourced from the original question on Stackoverflow.

The content on this page is licensed under the Attribution-ShareAlike 4.0 International (CC BY-SA 4.0) license.

Content TypeOriginal AuthorOriginal Content on Stackoverflow
QuestionAdamTheHuttView Question on Stackoverflow
Solution 1 - PhpwarrenView Answer on Stackoverflow
Solution 2 - PhpCzimiView Answer on Stackoverflow
Solution 3 - PhpLucaMView Answer on Stackoverflow
Solution 4 - PhpDarryl HeinView Answer on Stackoverflow
Solution 5 - PhpMark BiekView Answer on Stackoverflow
Solution 6 - PhpLeoView Answer on Stackoverflow
Solution 7 - Phppierre_moonView Answer on Stackoverflow
Solution 8 - PhpphilfreoView Answer on Stackoverflow
Solution 9 - PhpAnton PelykhView Answer on Stackoverflow
Solution 10 - PhpGordon ForsytheView Answer on Stackoverflow
Solution 11 - PhpRonald ConcoView Answer on Stackoverflow
Solution 12 - PhpgeocarView Answer on Stackoverflow
Solution 13 - PhpLF00View Answer on Stackoverflow