PHP仅调度监控,ffmpeg负责实际转码;必须异步执行、记录任务状态、用-progress输出JSON进度日志,并结构化记录错误归因。

PHP怎样处理视频转码请求_PHP处理视频转码请求流程【处理】  第1张

PHP 本身不转码,只调度和监控

PHP 没有内置视频编解码能力,ffmpeg 才是实际干活的。PHP 的角色是:接收请求 → 校验参数 → 启动后台转码进程 → 记录状态 → 提供进度查询接口。强行在 PHP 进程里用 exec() 同步跑 ffmpeg 会导致超时、阻塞、内存溢出,必须异步化。

proc_open() 或队列启动 ffmpeg 命令

推荐用 proc_open() 精确控制子进程(尤其需捕获错误输出),避免 shell_exec() 隐藏失败。关键点:

  • ffmpeg 命令必须带完整路径(如 /usr/bin/ffmpeg),不能依赖 $PATH
  • 输出重定向到文件(如 -loglevel quiet -y /tmp/output.mp4),避免 stdout/stderr 堆积阻塞
  • 设置超时(stream_set_timeout())和资源限制(setrlimit())防失控
  • 转码命令末尾加 & 不够可靠,应配合 nohup 或交由队列系统(如 Redis + php-resque
proc_open(
    '/usr/bin/ffmpeg -i /tmp/input.mp4 -c:v libx264 -preset fast -crf 23 -c:a aac /tmp/output.mp4 2>/tmp/ffmpeg.log',
    [
        ['pipe', 'r'],
        ['pipe', 'w'],
        ['pipe', 'w']
    ],
    $pipes,
    '/tmp',
    $_ENV
);

必须记录任务 ID 和状态,别靠轮询文件存在

用户请求后返回一个唯一 $task_id(如 uniqid('transcode_')),所有状态存数据库或 Redis,字段至少包含:status(pending/running/success/failed)、input_pathoutput_patherror_logcreated_atupdated_at。不要用 file_exists() 判断是否完成——文件可能写一半就中断,也难定位失败原因。

  • 启动转码前先插入 pending 记录
  • 子进程退出后,用 proc_get_status() 或信号回调更新状态
  • 失败时把 /tmp/ffmpeg.log 内容读入 error_log 字段,方便排查(常见如 No such file or directoryInvalid data found when processing input

前端轮询进度?不如让 ffmpeg 输出 JSON 日志

ffmpeg 从 4.3 开始支持 -progress 输出机器可读进度(JSON 或 KV 格式)。PHP 可开一个独立日志文件,让 ffmpeg 实时写入,再由另一个轻量接口读取最新行解析:frame=1234 fps=24.5 q=23.0 size=12345kB time=00:00:56.78 bitrate=1234.5kbits/s。比估算耗时或查文件大小靠谱得多。

立即学习“PHP免费学习笔记(深入)”;

  • 命令中加 -progress /tmp/transcode_abc123.progress
  • PHP 接口用 tail -n 1 /tmp/transcode_abc123.progressfseek($fp, -1024, SEEK_END) 读末尾几行
  • 注意并发请求时 progress 文件名必须唯一,且要处理文件未生成或权限问题(is_readable() 必须校验)
转码最麻烦的不是启动命令,而是错误归因:是源文件损坏?编码格式不支持?磁盘满?内存不足?还是 ffmpeg 版本太低不支持某参数?每种情况对应的日志特征和应对方式完全不同,得靠结构化记录和分类提取错误关键词来快速定位。