97

I have a process intensive task that I would like to run in the background.

The user clicks on a page, the PHP script runs, and finally, based on some conditions, if required, then it has to run a shell script, E.G.:

shell_exec('php measurePerformance.php 47 844 email@yahoo.com');

Currently I use shell_exec, but this requires the script to wait for an output. Is there any way to execute the command I want without waiting for it to complete?

Jason Plank
  • 2,322
  • 4
  • 29
  • 39
ToughPal
  • 2,141
  • 4
  • 24
  • 29
  • NOTE: The process has to start immediately so a cron is not an option. – ToughPal Jun 19 '09 at 20:27
  • 2
    Just a quick note: Make sure the page will not be accessible by everyone (search engines/ fraudulent users) and if you are on a shared environment, the execution time of all your scripts (in particular php!) probably is limited. In any case you may want to take a look at set_time_limit/php.ini. – merkuro Jun 19 '09 at 20:35
  • Possible duplicate of [php execute a background process](https://stackoverflow.com/questions/45953/php-execute-a-background-process) – Sean the Bean Jan 31 '18 at 21:33

8 Answers8

152

How about adding.

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

shell_exec('php measurePerformance.php 47 844 email@yahoo.com > /dev/null 2>/dev/null &');

Note this also gets rid of the stdio and stderr.

jitter
  • 51,939
  • 11
  • 106
  • 120
  • It works perfectly. Are there any problems / side effects of using this?? – ToughPal Jun 19 '09 at 20:39
  • 2
    No. No side effects. If you sometime decide you want the stdio or stderr output of your process consider http://stackoverflow.com/questions/45953/php-execute-a-background-process/45966#45966 – jitter Jun 19 '09 at 21:00
  • 1
    Question: That's just discarding the output, right? Does PHP still wait for the process to finish before continuing execution, or does it fire and forget? – Alan Storm Jun 19 '09 at 21:06
  • 5
    Yes discards out. Yes fire and forget – jitter Jun 19 '09 at 21:17
  • Works like a charm! Thanks! – buzznfrog Jan 15 '13 at 14:03
  • it is not working for the following is there any mistake I did? shell_exec('ffmpeg -f dshow -i video="Logitech HD Pro Webcam C920":audio="Rear Input (SoundMAX Integrated" -vcodec libx264 -b:v 600k -b:a 128k -f flv rtmp://127.0.0.1/live/mystream > /dev/null 2>/dev/null &'); – Thirumalai murugan Feb 26 '13 at 13:39
  • 5
    Note: `> /dev/null 2>&1 &` as seen in other answers is a terse form of `> /dev/null 2>/dev/null &` given here. This is basically saying "redirect STDERR (2) to the same place as STDOUT (&1)". – rymo Jul 10 '15 at 21:26
  • 3
    I usually get PID of running script by adding echo $! >> /tmp/pid.txt at the end of exec command but by redirecting output & error to /dev/null, no more pid file : does any of you can get PID of script launched by exec while not waiting for output ? – hugsbrugs Feb 22 '16 at 10:21
  • This answer provides a more comprehensive solution if you need to be able to check if the process is still running: https://stackoverflow.com/a/45966/814160 – Sean the Bean Jan 31 '18 at 21:49
  • Can't get any more perfect than this. Wish I saw this a few weeks ago. Thanks – user2993497 Oct 20 '19 at 01:21
28

This will execute a command and disconnect from the running process. Of course, it can be any command you want. But for a test, you can create a php file with a sleep(20) command it.

exec("nohup /usr/bin/php -f sleep.php > /dev/null 2>&1 &");
Brent Baisley
  • 700
  • 4
  • 4
11

You can also give your output back to the client instantly and continue processing your PHP code afterwards.

This is the method I am using for long-waiting Ajax calls which would not have any effect on client side:

ob_end_clean();
ignore_user_abort();
ob_start();
header("Connection: close");
echo json_encode($out);
header("Content-Length: " . ob_get_length());
ob_end_flush();
flush();
// execute your command here. client will not wait for response, it already has one above.

You can find the detailed explanation here: http://oytun.co/response-now-process-later

Oytun
  • 433
  • 5
  • 15
9

On Windows 2003, to call another script without waiting, I used this:

$commandString = "start /b c:\\php\\php.EXE C:\\Inetpub\\wwwroot\\mysite.com\\phpforktest.php --passmsg=$testmsg"; 
pclose(popen($commandString, 'r'));

This only works AFTER giving changing permissions on cmd.exe - add Read and Execute for IUSR_YOURMACHINE (I also set write to Deny).

Jason Plank
  • 2,322
  • 4
  • 29
  • 39
7

Sure, for windows you can use:

$WshShell = new COM("WScript.Shell");
$oExec = $WshShell->Run("C:/path/to/php-win.exe -f C:/path/to/script.php", 0, false);

Note:

If you get a COM error, add the extension to your php.ini and restart apache:

[COM_DOT_NET]
extension=php_com_dotnet.dll
Pedro Lobito
  • 75,541
  • 25
  • 200
  • 222
6

Use PHP's popen command, e.g.:

pclose(popen("start c:\wamp\bin\php.exe c:\wamp\www\script.php","r"));

This will create a child process and the script will excute in the background without waiting for output.

Jason Plank
  • 2,322
  • 4
  • 29
  • 39
  • 4
    It waits for child process ends. At least on Win – Max Chernopolsky Feb 15 '16 at 05:17
  • The backgrounding here is achieved with the help of the Windows "start" command that prefixes the exe you want to run, it wont work if you omit it, and I wouldn't expect it to work on another OS... but it DOES works for me, thanks! :-) – Antony Dec 02 '16 at 15:42
1

If it's off of a web page, I recommend generating a signal of some kind (dropping a file in a directory, perhaps) and having a cron job pick up the work that needs to be done. Otherwise, we're likely to get into the territory of using pcntl_fork() and exec() from inside an Apache process, and that's just bad mojo.

chaos
  • 115,791
  • 31
  • 292
  • 308
1

That will work but you will have to be careful not to overload your server because it will create a new process every time you call this function which will run in background. If only one concurrent call at the same time then this workaround will do the job.

If not then I would advice to run a message queue like for instance beanstalkd/gearman/amazon sqs.

Alfred
  • 56,245
  • 27
  • 137
  • 181