python ffmpeg moov atom在处理输入时找不到无效数据

标签 python python-3.x audio ffmpeg video-processing

我有一个录制屏幕和麦克风音频的进度,然后将视频和音频录制(.mp4 和 .wav)合并到一个 mkv 文件中。

我正在使用 python 3.6 和 ffmpeg 来实现这个目标。对于短视频(<20 秒),它可以工作,但对于较长的录制,它会显示以下错误消息:

[mov,mp4,m4a,3gp,3g2,mj2 @ 0x55abb3a52540] moov atom not found
tmp/tmp_0.mp4: Invalid data found when processing input

完整输出:

ffmpeg version 3.3.7 Copyright (c) 2000-2018 the FFmpeg developers
built with gcc 7 (GCC)
configuration: --prefix=/usr --bindir=/usr/bin -- 
datadir=/usr/share/ffmpeg --docdir=/usr/share/doc/ffmpeg -- 
incdir=/usr/include/ffmpeg --libdir=/usr/lib64 --mandir=/usr/share/man -- 
arch=x86_64 --optflags='-O2 -g -pipe -Wall -Werror=format-security -Wp, 
-D_FORTIFY_SOURCE=2 -fexceptions -fstack-protector-strong --param=ssp- 
buffer-size=4 -grecord-gcc-switches -specs=/usr/lib/rpm/redhat/redhat- 
hardened-cc1 -m64 -mtune=generic -fasynchronous-unwind-tables' --extra- 
ldflags='-Wl,-z,relro -specs=/usr/lib/rpm/redhat/redhat-hardened-ld ' -- 
extra-cflags='-I/usr/include/nvenc ' --enable-libopencore-amrnb -- 
enable- 
libopencore-amrwb --enable-libvo-amrwbenc --enable-version3 --enable-bzlib 
--disable-crystalhd --enable-fontconfig --enable-frei0r --enable-gcrypt -- 
enable-gnutls --enable-ladspa --enable-libass --enable-libbluray --enable- 
libcdio --enable-indev=jack --enable-libfreetype --enable-libfribidi -- 
enable-libgsm --enable-libmp3lame --enable-nvenc --enable-openal --enable- 
opencl --enable-opengl --enable-libopenjpeg --enable-libopus --enable- 
libpulse --enable-libschroedinger --enable-libsoxr --enable-libspeex -- 
enable-libtheora --enable-libvorbis --enable-libv4l2 --enable- libvidstab - 
-enable-libvpx --enable-libx264 --enable-libx265 --enable-libxvid -- 
enable-avfilter --enable-avresample --enable-postproc --enable-pthreads -- 
disable-static --enable-shared --enable-gpl --disable-debug --disable- 
stripping --shlibdir=/usr/lib64 --enable-libmfx --enable-runtime-cpudetect
  libavutil      55. 58.100 / 55. 58.100
  libavcodec     57. 89.100 / 57. 89.100
  libavformat    57. 71.100 / 57. 71.100
  libavdevice    57.  6.100 / 57.  6.100
  libavfilter     6. 82.100 /  6. 82.100
  libavresample   3.  5.  0 /  3.  5.  0
  libswscale      4.  6.100 /  4.  6.100
  libswresample   2.  7.100 /  2.  7.100
  libpostproc    54.  5.100 / 54.  5.100
[wav @ 0x55abb3a0b880] Ignoring maximum wav data size, file may be invalid
[wav @ 0x55abb3a0b880] Estimating duration from bitrate, this may be 
inaccurate
Guessed Channel Layout for Input Stream #0.0 : stereo
Input #0, wav, from 'tmp/tmp_0.wav':
  Metadata:
    encoder         : Lavf57.71.100
  Duration: 00:00:21.97, bitrate: 768 kb/s
Stream #0:0: Audio: pcm_mulaw ([7][0][0][0] / 0x0007), 48000 Hz, 
stereo, s16, 768 kb/s
[mov,mp4,m4a,3gp,3g2,mj2 @ 0x55abb3a52540] moov atom not found
tmp/tmp_0.mp4: Invalid data found when processing input

python文件(ffmpeg.py)如下。类 AV_COMPILE 尚未完成,被上述错误阻止,因此仍使用初始测试文件作为默认文件。但除此之外它应该可以工作:

import os, time, glob

TMP_DIR = "tmp"
DISPLAY = os.environ['DISPLAY']
EXT = {
    'Video':'mp4',
    'Audio':'wav',
    'AV':'mkv',
}

class ffmpegVideo:

    FFMPEG_BIN = "ffmpeg"
    AUDIO = False

    def __init__(self, fps = 30, audio = True):
    global TMP_DIR, DISPLAY, EXT

    self.fps = fps

    if audio:
        self.AUDIO = True

    self.video_filename = self.unique_filename()

    self.command = [ self.FFMPEG_BIN,
        '-video_size', '1920x1080',
        '-framerate', str(fps),
        '-f', 'x11grab',
        '-i', DISPLAY,
        '-vcodec', 'libx264',
        '-qp', '0',
        '-preset', 'ultrafast',
        '-y', TMP_DIR + '/' + self.video_filename
    ]

def start(self):
    import threading as th

    thread = th.Thread(target=self.record)
    thread.start()

def record(self):
    import subprocess as sp

    self.pipe = sp.Popen(self.command, stderr=sp.PIPE)

    if self.AUDIO:
        ffmpegAudio().start()

def stop(self):
    self.pipe.terminate()

def unique_filename(self):
    global TMP_DIR, EXT

    i = 0

    while os.path.exists((TMP_DIR + '/' + 'tmp_%s.%s') % (i, EXT['Video'])):
        i += 1

    return ('tmp_%s.%s') % (i, EXT['Video'])

class ffmpegAudio:

    FFMPEG_BIN = "ffmpeg"

    def __init__(self):

        self.audio_filename = self.unique_filename()

        self.command = [ self.FFMPEG_BIN,
            '-f', 'pulse',
            '-ac', '2',
            '-ar', '48000',
            '-i', 'default',
           '-acodec', 'pcm_mulaw',
           '-y', TMP_DIR + '/' + self.audio_filename
        ]

    def start(self):
        import threading as th

        au_thread = th.Thread(target=self.record)
        au_thread.start()

    def record(self):
         import subprocess as sp

        self.pipe = sp.Popen(self.command, stderr=sp.PIPE)

    def stop(self):
        self.pipe.terminate()

    def unique_filename(self):
        global TMP_DIR, EXT

        i = 0

        while os.path.exists((TMP_DIR + '/' + 'tmp_%s.%s') % (i, EXT['Audio'])):
        i += 1

        return ('tmp_%s.%s') % (i, EXT['Audio'])

class AV_COMPILE:

    def __init__(self, au_in = TMP_DIR + '/' + 'out1.wav', vd_in = 
TMP_DIR + '/' + 'test4.mp4', out = TMP_DIR + '/' + 'av.mkv'):
        import subprocess as sp

        au_in = min(glob.iglob(TMP_DIR + '/*.wav'), key=os.path.getctime)
        vd_in = min(glob.iglob(TMP_DIR + '/*.mp4'), key=os.path.getctime)

        self.command = ('ffmpeg -i %s  -r 30 -i %s -shortest -c:a aac -c:v copy %s') % (au_in, vd_in, out)
        sp.call(self.command, shell=True)

如果您能提供任何帮助以帮助我理解为什么会发生这种情况以及如何解决错误,我将不胜感激。此外,我很高兴收到有关如何改进此代码的任何其他提示,或任何人可能注意到的任何其他问题。

编辑: 我现在认为,较长视频(有时较短)出现此错误的原因是程序正在继续尝试编译 av 输出,无论它是否已完成原始视频文件的编译。我测试了一个 time.sleep(10) 函数来延迟 AV_COMPILE,这似乎有效。

但是,随着视频文件变大,显然需要调整延迟。所以我想知道如何单独检查视频文件的完整性并确定继续下一步是安全的。

最佳答案

关于 moov atom not found 好吧,我检查了你的文件,原子在那里,但在你文件的后端(但它需要在其余 MP4 数据之前首先可用)可以处理)。

MP4 容器的结构使得 a/v 数据在没有帧头的情况下存储,Moov 原子是一个表,列出每个 a/v 帧的字节位置和字节长度。 Moov 原子只有在记录完成后才能存在,此时可以为存储的数据创建相关的元数据。

这就是引入分片 MP4 的原因。

解决方案:

不要将您的 a/v 数据放入 MP4,然后将相同的 a/v 从 MP4 传输到 MKV 容器中。我怀疑您使用 MP4 容器设置是为了获得使用 H.264 视频编解码器编码的输出(图片)?

我不使用 Python,但尝试设置:

'Video':'libx264',
'Audio':'wav',
'AV':'mkv',

关于python ffmpeg moov atom在处理输入时找不到无效数据,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/50134790/

相关文章:

python - 从 Tensorflow 模型获取预测

audio - 在关键帧之间剪切视频而不使用ffmpeg重新编码完整视频?

python-3.x - Pandas 合并意外产生后缀

python - 如何在 Python 中模拟实例方法

python - Django 无法识别 MEDIA_URL 路径?

python - 在 html 表中访问带有空格的 json 对象

python-3.x - 模块未找到错误 : No module named 'numpy'

c - Solaris 音频 API

java - 是否有 java API 来检索单词发音的声音?

python - 为什么这种方式获取http客户端失败呢?