如何获取 ffprobe 元数据作为要在 python 中解析的变量

How to get ffprobe metadata as variable to parse in python

当我运行ffprobe <video>时,我得到如下标准元数据:

ffprobe version 4.3.1 Copyright (c) 2007-2020 the FFmpeg developers
  built with Apple clang version 11.0.3 (clang-1103.0.32.62)
  configuration: --prefix=/opt/local --enable-swscale --enable-avfilter --enable-avresample --enable-libmp3lame --enable-libvorbis --enable-libopus --enable-librsvg --enable-libtheora --enable-libopenjpeg --enable-libmodplug --enable-libvpx --enable-libsoxr --enable-libspeex --enable-libass --enable-libbluray --enable-lzma --enable-gnutls --enable-fontconfig --enable-libfreetype --enable-libfribidi --disable-libjack --disable-libopencore-amrnb --disable-libopencore-amrwb --disable-libxcb --disable-libxcb-shm --disable-libxcb-xfixes --disable-indev=jack --enable-opencl --disable-outdev=xv --enable-audiotoolbox --enable-videotoolbox --enable-sdl2 --disable-securetransport --mandir=/opt/local/share/man --enable-shared --enable-pthreads --cc=/usr/bin/clang --enable-libdav1d --arch=x86_64 --enable-x86asm --enable-libx265 --enable-gpl --enable-postproc --enable-libx264 --enable-libxvid
  libavutil      56. 51.100 / 56. 51.100
  libavcodec     58. 91.100 / 58. 91.100
  libavformat    58. 45.100 / 58. 45.100
  libavdevice    58. 10.100 / 58. 10.100
  libavfilter     7. 85.100 /  7. 85.100
  libavresample   4.  0.  0 /  4.  0.  0
  libswscale      5.  7.100 /  5.  7.100
  libswresample   3.  7.100 /  3.  7.100
  libpostproc    55.  7.100 / 55.  7.100
Input #0, mov,mp4,m4a,3gp,3g2,mj2, from '../directConversion/200mbs105_55.mp4':
  Metadata:
    major_brand     : isom
    minor_version   : 512
    compatible_brands: isomiso2avc1mp41
    encoder         : Lavf58.45.100
  Duration: 00:01:12.43, start: 0.000000, bitrate: 213963 kb/s
    Stream #0:0(und): Video: h264 (High) (avc1 / 0x31637661), yuv420p, 1080x1920 [SAR 1:1 DAR 9:16], 213828 kb/s, 30 fps, 30 tbr, 15360 tbn, 60 tbc (default)
    Metadata:
      handler_name    : Core Media Video
    Stream #0:1(und): Audio: aac (LC) (mp4a / 0x6134706D), 48000 Hz, stereo, fltp, 128 kb/s (default)
    Metadata:
      handler_name    : Core Media Audio

我想从 python 获取此信息,以便我可以解析此输出并在我的代码中使用其中的一些。但是,我无法让它工作。

我试过以下方法:

data = subprocess.call(f'ffprobe {video}', shell=True)
data = subprocess.check_output(f'ffprobe {vid}', shell=True)
data = subprocess.Popen(f'ffprobe {vid}', shell=True)
data = subprocess. Popen(f'ffprobe {vid}', stdout=subprocess.PIPE ).communicate()[0]
data = run(f'ffprobe {vid}', capture_output=True).stdout

如果我包含 shell=True,则终端会打印正确的信息,但我得到 data 的空字符串。对于没有 shell=True 的命令,我得到 file not found error 并且它中断了。我应该怎么办?我从 SO 那里得到了所有这些解决方案,它们似乎对其他人也有效。

您可以使用 shlex.split 或将 FFprobe 的参数放在列表中。

在WindowsOS中,可以使用sp.run(f'ffprobe {video}'...
在 Linux 和 Mac 中,Python 尝试以 spaces.
的文件名执行命令 例如:'ffprobe vid.mp4'被认为是单个可执行命令(文件名带有space)。

您可以使用参数列表:
sp.run(['ffprobe', f'{video}']...

或使用 shlex.split 将 shell 命令拆分为列表:
sp.run(shlex.split(f'ffprobe {video}'))...


对于Python中FFprobe输出的简单解析:

  • 使用 -of json 参数执行 ffprobe,并获得 JSON 格式的输出。
  • 使用 json.loads.
  • 将输出字符串转换为字典

下面是将 FFprobe 的输出读入字典的代码示例:

import subprocess as sp
import shlex
import json

video = 'vid.mkv'

# Execute ffprobe (to show streams), and get the output in JSON format
data = sp.run(shlex.split(f'ffprobe -loglevel error -show_streams -of json {video}'), capture_output=True).stdout

# Same - using a list of arguments:
# data = sp.run(['ffprobe', '-loglevel', 'error', '-show_streams', '-of', 'json', f'{video}'], capture_output=True).stdout

# Convert data from JSON string to dictionary
d = json.loads(data)

print(d)  # Print the dictionary for testing

备注:

  • 我以 -show_streams 参数为例。
    如果缺少(或太多)信息,请查找相关参数集。
  • 该示例假定 ffprobe 在执行路径中。