PHP:立即返回响应,但在后台运行多个冗长的shell_exec函数


PHP: Return response immediately, but run multiple lengthy shell_exec functions in the background

我正在创建一个简单的web服务,它接受视频上传,在视频上运行多个不同的编码(mp4、webm、ogv),然后将新创建的文件上传到我们的视频主机。

假设我有多个命令。。

shell_exec('ffmpeg -i input.mp4 -f mp4 -c:v libx264 -preset slow -crf 24 -s 1280x720 -c:a libfdk_aac -profile:a aac_he -ar 22050 -b:a 64k -movflags +faststart output-1280x720.mp4');
shell_exec('ffmpeg -i input.mp4 -f mp4 -c:v libx264 -preset slow -crf 24 -s 1920x1080 -c:a libfdk_aac -profile:a aac_he -ar 22050 -b:a 64k -movflags +faststart output-1920x1080.mp4');
shell_exec('ffmpeg -i input.mp4 -f ogg -c:v libtheora -q:v 5 -s 1280x720 -c:a libvorbis -ar 22050 -b:a 64k -movflags +faststart output-1280x720.ogv');
shell_exec('ffmpeg -i input.mp4 -f ogg -c:v libtheora -q:v 5 -s 1920x1080 -c:a libvorbis -ar 22050 -b:a 64k -movflags +faststart output-1920x1080.ogv');

总之,我想…

  1. 打印即时响应:{success:true}
  2. 同步启动多个ffmpeg作业
  3. 每个作业完成后,向另一台服务器发送一个POST(每个shell_exec一个POST)

如果作业成功,只发送POST也很好,但我可以通过检查服务器上是否存在输出文件来轻松解决这个问题。

我知道,我可以通过简单地将>/dev/null 2>/dev/null &附加到每个命令上来强制shell_exec在后台运行——这将允许我立即打印响应——但我认为这样做会导致所有作业并行运行,而且,由于这会转移输出,所以当作业完成时,我不会得到任何真正的回调。

有什么想法吗??

我决定采用异步请求文件的想法。为了保持干净,我能够将其全部包含在一个文件中,该文件使用不同的POST请求"重新调用"自己。

$root_path = $_SERVER['DOCUMENT_ROOT'] . '/';
$root_http = 'http' . (isset($_SERVER['HTTPS']) ? 's' : '') . '://' . $_SERVER['HTTP_HOST'] . '/';
function request_async($url, $params, $type='POST', $timeout=30) {
  foreach ($params as $key => &$val) {
    if (is_array($val)) $val = implode(',', $val);
    $post_params[] = $key.'='.urlencode($val);
  }
  $post_string = implode('&', $post_params);
  $parts = parse_url($url);
  $fp = fsockopen(
    $parts['host'],
    isset($parts['port']) ? $parts['port'] : 80,
    $errno,
    $errstr,
    $timeout
  );
  if ('GET' == $type) $parts['path'] .= '?'.$post_string;
  $out = "$type ".$parts['path']." HTTP/1.1'r'n";
  $out.= "Host: ".$parts['host']."'r'n";
  $out.= "Content-Type: application/x-www-form-urlencoded'r'n";
  $out.= "Content-Length: ".strlen($post_string)."'r'n";
  $out.= "Connection: Close'r'n'r'n";
  if ('POST' == $type && isset($post_string)) $out .= $post_string;
  fwrite($fp, $out);
  fclose($fp);
}
function get_http_path($path, $relative = FALSE){
  global $root_http, $root_path;
  return str_replace($root_path, $relative ? '/' : $root_http, $path);
}
$url = $_POST['url'];
$media = $_POST['media'];
if (isset($url)) {
  request_async(get_http_path(__FILE__), array(media => $url), 'POST', 1800);
  echo '{"success": true}';
} else
if (isset($media)) {
  $data = file_get_contents($media);
  // KICK OFF JOBS HERE
  echo '{"success": true}';
} else {
  echo '{"success": false}';
}