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

时间:2018-05-02 12:34:06

标签: 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,这似乎有效。

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

1 个答案:

答案 0 :(得分:2)

关于moov atom not found我检查了你的文件并且原子在那里,但是在文件的后端(但是在处理剩余的MP4数据之前它需要先可用)。 / p>

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',