(PHP 4 >= 4.3.0, PHP 5, PHP 7)
proc_open — Execute a command and open file pointers for input/output
$cmd
      , array $descriptorspec
      , array &$pipes
      [, string $cwd = NULL
      [, array $env = NULL
      [, array $other_options = NULL
     ]]] ) : resourceproc_open() is similar to popen() but provides a much greater degree of control over the program execution.
cmdThe commandline to execute as string. Special characters have to be properly escaped, and proper quoting has to be applied.
Note: On Windows, unless
bypass_shellis set toTRUEinother_options, thecmdis passed to cmd.exe (actually,%ComSpec%) with the/cflag as unquoted string (i.e. exactly as has been given to proc_open()). This can cause cmd.exe to remove enclosing quotes fromcmd(for details see the cmd.exe documentation), resulting in unexpected, and potentially even dangerous behavior, because cmd.exe error messages may contain (parts of) the passedcmd(see example below).
       As of PHP 7.4.0, cmd may be passed as array of command parameters.
       In this case the process will be opened directly (without going through a shell)
       and PHP will take care of any necessary argument escaping.
      
Note:
On Windows, the argument escaping of the array elements assumes that the command line parsing of the executed command is compatible with the parsing of command line arguments done by the VC runtime.
descriptorspecAn indexed array where the key represents the descriptor number and the value represents how PHP will pass that descriptor to the child process. 0 is stdin, 1 is stdout, while 2 is stderr.
Each element can be:
pipe (the second
         element is either r to pass the read end of the pipe
         to the process, or w to pass the write end) and
         file (the second element is a filename).
        STDIN).
        The file descriptor numbers are not limited to 0, 1 and 2 - you may specify any valid file descriptor number and it will be passed to the child process. This allows your script to interoperate with other scripts that run as "co-processes". In particular, this is useful for passing passphrases to programs like PGP, GPG and openssl in a more secure manner. It is also useful for reading status information provided by those programs on auxiliary file descriptors.
pipesWill be set to an indexed array of file pointers that correspond to PHP's end of any pipes that are created.
cwd
       The initial working dir for the command. This must be an
       absolute directory path, or NULL
       if you want to use the default value (the working dir of the current
       PHP process)
      
env
       An array with the environment variables for the command that will be
       run, or NULL to use the same environment as the current PHP process
      
other_optionsAllows you to specify additional options. Currently supported options include:
suppress_errors (windows only): suppresses errors
         generated by this function when it's set to TRUE
        bypass_shell (windows only): bypass
         cmd.exe shell when set to TRUE
        blocking_pipes (windows only): force
         blocking pipes when set to TRUE
        create_process_group (windows only): allow the
         child process to handle CTRL events when set to TRUE
        create_new_console (windows only): the new process
         has a new console, instead of inheriting its parent's console
        
   Returns a resource representing the process, which should be freed using
   proc_close() when you are finished with it. On failure
   returns FALSE.
  
| Version | Description | 
|---|---|
| 7.4.4 | Added the create_new_consoleoption to theother_optionsparameter. | 
| 7.4.0 | proc_open() now also accepts an array
        for the cmd. | 
| 7.4.0 | Added the create_process_groupoption to theother_optionsparameter. | 
| 7.0.0 | Added the blocking_pipesoption to theother_optionsparameter. | 
| 5.2.1 | Added the bypass_shelloption to theother_optionsparameter. | 
Example #1 A proc_open() example
<?php
$descriptorspec = array(
   0 => array("pipe", "r"),  // stdin is a pipe that the child will read from
   1 => array("pipe", "w"),  // stdout is a pipe that the child will write to
   2 => array("file", "/tmp/error-output.txt", "a") // stderr is a file to write to
);
$cwd = '/tmp';
$env = array('some_option' => 'aeiou');
$process = proc_open('php', $descriptorspec, $pipes, $cwd, $env);
if (is_resource($process)) {
    // $pipes now looks like this:
    // 0 => writeable handle connected to child stdin
    // 1 => readable handle connected to child stdout
    // Any error output will be appended to /tmp/error-output.txt
    fwrite($pipes[0], '<?php print_r($_ENV); ?>');
    fclose($pipes[0]);
    echo stream_get_contents($pipes[1]);
    fclose($pipes[1]);
    // It is important that you close any pipes before calling
    // proc_close in order to avoid a deadlock
    $return_value = proc_close($process);
    echo "command returned $return_value\n";
}
?>
The above example will output something similar to:
Array
(
    [some_option] => aeiou
    [PWD] => /tmp
    [SHLVL] => 1
    [_] => /usr/local/bin/php
)
command returned 0
Example #2 proc_open() quirk on Windows
     While one may expect the following program to search the file
     filename.txt for the text search and
     to print the results, it behaves rather differently.
    
<?php
$descriptorspec = [STDIN, STDOUT, STDOUT];
$cmd = '"findstr" "search" "filename.txt"';
$proc = proc_open($cmd, $descriptorspec, $pipes);
proc_close($proc);
?>
The above example will output:
'findstr" "search" "filename.txt' is not recognized as an internal or external command, operable program or batch file.
     To work around that behavior, it is usually sufficient to enclose the
     cmd in additional quotes:
    
$cmd = '""findstr" "search" "filename.txt""';
Note:
Windows compatibility: Descriptors beyond 2 (stderr) are made available to the child process as inheritable handles, but since the Windows architecture does not associate file descriptor numbers with low-level handles, the child process does not (yet) have a means of accessing those handles. Stdin, stdout and stderr work as expected.
Note:
If you only need a uni-directional (one-way) process pipe, use popen() instead, as it is much easier to use.