本文将尝试指出在PHP Web脚本中使用FFmpeg时需要了解的所有重要事项。它还将显示一些使用示例,以使事情更清楚。这个想法也可以应用到其他web脚本语言。
从PHP脚本调用命令行工具选择一个模型
网页设计为快速执行,以便浏览您的网站的人不必等待太多的响应。因为如果他们无聊的等待,他们只会导航到另一个,更灵敏(通常是您的竞争对手的)网站。话虽如此,你可以做的最糟糕的事情是从你的web脚本运行一个命令行工具(如ffmpeg),并等待它完成处理,以便能够将结果返回到等待在线用户(异常这是小/快的工具,其执行时间太小而不能注意到)。
您想要做的是将命令行工具的长处理与Web脚本执行分开,以使脚本更具响应性。所以,你有至少两个选项:
- 运行命令行工具(使用了shell_exec()在后台为例),并继续与web脚本执行,定期刷新状态页面,显示当前进度
- 创建一个新的“作业”(通常是数据库表中的一个新行),并有一些背景“工作”进程(cron作业,shell脚本,批处理脚本),它将监视“作业列表”需要处理的新工作
这两种方法看起来是一样的,但是他们不是。最重要的区别是,第二种方法可以更好地扩展更高的网站流量,因为它允许“工作”进程在完全独立的机器上运行。此外,可以有几个“工作”机器,一起工作,分裂工作负载,涉及小的同步。
第一种方法通常是大多数想要快速完成工作的人的首选,但是在他们的网站变得流行的时间,他们的web服务器变得不那么敏感,由于不断的cpu饥饿,由多个命令行工具(ffmpeg实例)在后台运行。在那一刻,他们通常开始考虑第二种方法。
在后台运行的ffmpeg
以下示例适用于基于Linux的操作系统。对于基于Windows的操作系统,请继续阅读本文,而是你读了之后,也看到了如何使用START和CMD后台运行的过程。这将给你一个想法如何应用相同的逻辑到基于Windows的操作系统。
让我们考虑从PHP脚本运行ffmpeg的自然方式:
<?php echo "Starting ffmpeg...\n\n"; echo shell_exec("ffmpeg -i input.avi output.avi &"); echo "Done.\n"; ?>
这里有几个问题需要指出。第一个是,虽然我们指定我们希望在后台执行ffmpeg(使用和号运算符“&”),PHP脚本将不会继续执行,直到ffmpeg完成其执行。这是由于这样的事实,在该票据中提及了PHP的exec()函数,上面写着:
如果程序使用此函数启动,为了使其在后台继续运行,程序的输出必须重定向到文件或另一个输出流。如果不这样做,将导致PHP挂起,直到程序的执行结束。
不要混淆的例子显示了shell_exec()调用,而不是EXEC() 。所有的PHP的程序执行功能共享相同的代码库和限制。
所以,要解决这个问题,我们需要做这样的事情:
<?php echo "Starting ffmpeg...\n\n"; echo shell_exec("ffmpeg -i input.avi output.avi >/dev/null 2>/dev/null &"); echo "Done.\n"; ?>
该说的部分“ >的/ dev / null的 ”将FFmpeg的实例的标准输出(stdout)重定向到/ dev / null的(有效地忽略输出)和“ 2>的/ dev / null的 ”将重定向标准错误(错误)到/ dev / null(有效地忽略任何错误日志消息)。这两个可以组合成一个较短的表示:“ >的/ dev / null的2>&1 ”。如果你喜欢,你可以阅读更多关于I / O重定向。
这里应该提到一个重要的注意事项。ffmpeg命令行工具使用stderr输出错误日志消息,并且保留stdout用于可能使用的管道(将从ffmpeg生成的输出媒体流重定向到其他命令行工具)。也就是说,如果你在后台运行你的ffmpeg,你很可能想把stderr重定向到一个日志文件,以便能够检查它。
需要注意的另一件事是标准INPUT(stdin)。命令行ffmpeg工具被设计为一个交互式实用程序,它接受用户的输入(通常来自键盘),并在用户的当前屏幕/终端上报告错误日志。当我们在后台运行ffmpeg时,我们想要告诉ffmpeg没有从stdin接受任何输入(也不等待)。我们可以告诉这ffmpeg的,再次使用I / O重定向“ <的/ dev / null的 ”,这样最后一个例子,安全运行ffmpeg的命令行工具在后台将与此类似:
<?php echo "Starting ffmpeg...\n\n"; echo shell_exec("ffmpeg -y -i input.avi output.avi </dev/null >/dev/null 2>/var/log/ffmpeg.log &"); echo "Done.\n"; ?>
在“-y”选项用于自动覆盖输出文件(output.avi)不要求是/否确认。如果需要相反的情况,如果输出文件已经存在,要自动取消整个过程,请改用“-n”选项。
包装库一些PHP库允许将ffmpeg调用包装到PHP对象中,并且如果您不喜欢使用命令行,可以为您提供一个不错的语法。其中之一是在积极维护PHP-FFmpeg的。它只需要你下载一个最新的ffmpeg和ffprobe从安装PHP组件构建分开。然后你可以这样运行PHP代码:
$ffmpeg = FFMpeg\FFMpeg::create(); $video = $ffmpeg->open('video.mpg'); $video->filters() ->resize(new FFMpeg\Coordinate\Dimension(320, 240)) ->synchronize(); $video->save(new FFMpeg\Format\Video\X264(), 'export-x264.mp4')
当然,你需要在后台运行这样的任务。图书馆如GearmanClient促成。
注: ffmpeg的- PHP的是,2007年以来未开发的扩展名(和要求“的ffmpeg-0.4.9_pre1或更高版本”),这意味着你仅限于使用一种非常古老的版本的ffmpeg,没有可能把它更新到最新版本。由于进行了大量的更改/改进,在ffmpeg的代码中,每天,它使得ffmpeg-php与最新的ffmpeg不兼容。
案例成功测试结果:
<?php echo "正在启动ffmpeg...\n\n"; echo shell_exec("ffmpeg -y -i rtmp://tinywan.123.com/live/4001483413136 -c copy -f flv rtmp://131.180/live/4001483492781?vhost=13.168 </dev/null >/dev/null 2>/var/log/ffmpeg.log &"); echo "完成.\n";
正在启动ffmpeg...
完成.
日志文件:
ffmpeg version 2.3.git Copyright (c) 2000-2014 the FFmpeg developers built on Sep 1 2014 19:01:47 with gcc 4.8 (Ubuntu 4.8.2-19ubuntu1) configuration: --prefix=/usr/local/ffmpeg/build --extra-cflags=-I/usr/local/ffmpeg/build/include --extra-ldflags=-L/usr/local/ffmpeg/build/lib --bindir=/usr/local/ffmpeg/bin --enable-gpl --enable-libass --enable-libfaac --enable-libfdk-aac --enable-libfreetype --enable-libmp3lame --enable-libopus --enable-libtheora --enable-libvorbis --enable-libvpx --enable-libx264 --enable-nonfree --enable-openssl --enable-libspeex --enable-libxvid libavutil 54. 7.100 / 54. 7.100 libavcodec 56. 1.100 / 56. 1.100 libavformat 56. 3.100 / 56. 3.100 libavdevice 56. 0.100 / 56. 0.100 libavfilter 5. 0.103 / 5. 0.103 libswscale 3. 0.100 / 3. 0.100 libswresample 1. 1.100 / 1. 1.100 libpostproc 53. 0.100 / 53. 0.100 [flv @ 0x2862000] Stream discovered after head already parsed Last message repeated 1 times Input #0, flv, from 'rtmp://tinywan.3213.com/live/4001483413136': Metadata: Server : Tengine displayWidth : 1920 displayHeight : 1080 fps : 25 profile : level : Duration: 00:00:00.00, start: 765.097000, bitrate: N/A Stream #0:0: Video: h264 (High), yuv420p(tv, bt709), 1920x1080, 1048 kb/s, 25 fps, 25 tbr, 1k tbn, 2k tbc Stream #0:1: Data: none Stream #0:2: Audio: aac, 44100 Hz, stereo, fltp Output #0, flv, to 'rtmp://1232130/live/4001483492781?vhost=321312': Metadata: Server : Tengine displayWidth : 1920 displayHeight : 1080 fps : 25 profile : level : encoder : Lavf56.3.100 Stream #0:0: Video: h264 ([7][0][0][0] / 0x0007), yuv420p, 1920x1080, q=2-31, 1048 kb/s, 25 fps, 1k tbn, 1k tbc Stream #0:1: Audio: aac ([10][0][0][0] / 0x000A), 44100 Hz, stereo Stream mapping: Stream #0:0 -> #0:0 (copy) Stream #0:2 -> #0:1 (copy) Press [q] to stop, [?] for help
测试成功!