The following function takes an array of shell commands and executes them. It is able to execute up to $nb_max_process at the same time. As soon as one process is terminated, another one is executed. Quite useful if you want to batch process commands on a multi-processor or multi-core environment.
The example below tries to convert to PNG a list of SVG files submitted on the command line (using Inkscape).
(it's quick and dirty but works very well for me)
#!/usr/bin/php
<?php
function pool_execute($commandes,$nb_max_process) {
$pool=array();
for($i=0;$i<$nb_max_process;$i++) {
$pool[$i]=FALSE;
}
while(count($commandes)>0) {
$commande=array_shift($commandes);
$commande_lancee=FALSE;
while($commande_lancee==FALSE) {
usleep(50000);
for($i=0;$i<$nb_max_process and $commande_lancee==FALSE;$i++) {
if($pool[$i]===FALSE) {
$pool[$i]=proc_open($commande,array(),$foo);
$commande_lancee=TRUE;
} else {
$etat=proc_get_status($pool[$i]);
if($etat['running']==FALSE) {
proc_close($pool[$i]);
$pool[$i]=proc_open($commande,array(),$foo);
$commande_lancee=TRUE;
}
}
}
}
}
}
$fichiers=$argv;
array_shift($fichiers);
$commandes=array();
foreach($fichiers as $fichier) {
$entree=$fichier;
$sortie=basename($fichier,'.svg').".png";
$commandes[]='inkscape --file='.escapeshellarg($entree).' --export-area-canvas --export-png='.escapeshellarg($sortie);
}
pool_execute($commandes,4);
proc_get_status
(PHP 5)
proc_get_status — Lit les informations concernant un processus ouvert par proc_open()
Description
array proc_get_status
( resource $process
)
proc_get_status() lit les données concernant le processus process créé avec la fonction proc_open().
Valeurs de retour
Un tableau contenant les informations collectées en cas de succès, et FALSE en cas d'échec. Le tableau retourné contient les éléments suivants :
| Élément | Type | Description |
|---|---|---|
| command | chaîne de caractères | La commande passée à la fonction proc_open(). |
| pid | entier | identifiant du processus |
| running | booléen | TRUE si le processus fonctionne toujours et FALSE s'il est terminé. |
| signaled | booléen | TRUE si le processus fils a été terminé par un signal inconnu. Toujours défini à FALSE sous Windows. |
| stopped | booléen | TRUE si le processus fils a été stoppé par un signal. Toujours défini à FALSE sous Windows. |
| exitcode | entier | Le code retourné par le processus (uniquement si l'élément running vaut FALSE). Seul le premier appel à cette fonction retourne une valeur réelle, les prochains appels retournent -1. |
| termsig | entier | le numéro du signal qui a causé la fin de l'exécution du processus fils (uniquement significatif si signaled vaut TRUE). |
| stopsig | entier | le numéro du signal qui a causé l'arrêt de l'exécution du processus fils (uniquement significatif si signaled vaut TRUE). |
proc_get_status
webmaster at rouen dot fr
09-Jul-2008 10:49
09-Jul-2008 10:49
strrev xc.noxeh@ellij
05-Jun-2008 10:12
05-Jun-2008 10:12
You can NOT rely on pid+1.
You could prefix exec to the command string, this will replace the /bin/sh script with the real thing you want to exec (use only if you don't do 'scary things' like pipes, output redirection, multiple commands, however if you know how they work, go ahead).
If you prefix exec, the /bin/sh process will only start your process, and the PID will be the same.
damien at cyg dot net
13-Jun-2007 03:06
13-Jun-2007 03:06
Alternatively, if you're calling a subsequent php script using proc_open, you can have that process echo its own actual PID in the output.
Also, if you go through the /proc filesystem on linux, you can read through /proc/12345 where 12345 is the pid returned by proc_get_status (the pid of the /bin/sh instance) and it will list its child processes within.
andy dot shellam at mailnetwork dot co dot uk
05-Apr-2007 02:16
05-Apr-2007 02:16
Further to my previous note, I've found out the PID returned is the PID of the shell (/bin/sh) that then runs the actual command requested.
I've raised this as bug #41003.
andy dot shellam at mailnetwork dot co dot uk
05-Apr-2007 01:58
05-Apr-2007 01:58
To the poster above, same here on FreeBSD 6.1, PHP 5.2.1.
To get the correct PID to use for posix_kill I have to add 1 to the PID returned from proc_get_status.
