如何在浏览器上播放avi文件



我的团队需要开发一个在网络上播放avi文件的系统。这些文件是由我们无法访问其固件的硬件记录的。我们正试图与制造商协商,将文件格式更改为mp4,但到目前为止,我们什么都没有。

正因为如此,我们正在尝试另一种方式来让它发挥作用。我们的第一次尝试是使用FFMPEG将文件转换为mp4(或webm或ogg),但这个过程太长了,因为我们每天都要处理大量的视频。

我们还尝试使用FFMPEG的复制命令(速度快得多),但视频总是在某个点崩溃(主要是当我们需要在其时间线中导航时),我们不知道为什么。

所以现在我们正在考虑自定义videojs flash播放器来再现AVI文件,但因为我们没有太多的视频编程和flash经验,我们不知道这是否可能。有可能在动作脚本中编写解码器来读取avi文件吗?

我看到Youtube和Facebook可以播放AVI文件。。。他们是怎么做到的?我已经考虑了很多,但都没有成功。

编辑1

复制命令前的avi视频文件:

$ ffmpeg -i video.avi
ffmpeg version N-82324-g872b358 Copyright (c) 2000-2016 the FFmpeg developers
built with gcc 5.4.0 (GCC)
configuration: --enable-gpl --enable-version3 --disable-w32threads --enable-dx                                                                                                                                                                                               va2 --enable-libmfx --enable-nvenc --enable-avisynth --enable-bzlib --enable-lib                                                                                                                                                                                               ebur128 --enable-fontconfig --enable-frei0r --enable-gnutls --enable-iconv --ena                                                                                                                                                                                               ble-libass --enable-libbluray --enable-libbs2b --enable-libcaca --enable-libfree                                                                                                                                                                                               type --enable-libgme --enable-libgsm --enable-libilbc --enable-libmodplug --enab                                                                                                                                                                                               le-libmp3lame --enable-libopencore-amrnb --enable-libopencore-amrwb --enable-lib                                                                                                                                                                                               openh264 --enable-libopenjpeg --enable-libopus --enable-librtmp --enable-libschr                                                                                                                                                                                               oedinger --enable-libsnappy --enable-libsoxr --enable-libspeex --enable-libtheor                                                                                                                                                                                               a --enable-libtwolame --enable-libvidstab --enable-libvo-amrwbenc --enable-libvo                                                                                                                                                                                               rbis --enable-libvpx --enable-libwavpack --enable-libwebp --enable-libx264 --ena                                                                                                                                                                                               ble-libx265 --enable-libxavs --enable-libxvid --enable-libzimg --enable-lzma --e                                                                                                                                                                                               nable-decklink --enable-zlib
libavutil      55. 36.100 / 55. 36.100
libavcodec     57. 66.101 / 57. 66.101
libavformat    57. 57.100 / 57. 57.100
libavdevice    57.  2.100 / 57.  2.100
libavfilter     6. 66.100 /  6. 66.100
libswscale      4.  3.100 /  4.  3.100
libswresample   2.  4.100 /  2.  4.100
libpostproc    54.  2.100 / 54.  2.100
Input #0, avi, from 'video.avi':
Metadata:
encoder         :
Duration: 00:50:07.67, start: 0.000000, bitrate: 6 kb/s
Stream #0:0: Video: h264 (Constrained Baseline) (H264 / 0x34363248), yuv420p                                                                                                                                                                                               (progressive), 352x240, 3 fps, 3 tbr, 3 tbn, 6 tbc
At least one output file must be specified

复制命令(没有音频流,因为视频没有):

$ ffmpeg -i video.avi -vcodec copy video.mp4
ffmpeg version N-82324-g872b358 Copyright (c) 2000-2016 the FFmpeg developers
built with gcc 5.4.0 (GCC)
configuration: --enable-gpl --enable-version3 --disable-w32threads --enable-dxva2 --enable-libmfx --enable-nvenc --enable-avisynth --enable-bzlib --enable-libebur128 --enable-fontconfig --enable-frei0r --enable-gnutls --enable-iconv --enable-libass --enable-libbluray --enable-libbs2b --enable-libcaca --enable-libfreetype --enable-libgme --enable-libgsm --enable-libilbc --enable-libmodplug --enable-libmp3lame --enable-libopencore-amrnb --enable-libopencore-amrwb --enable-libopenh264 --enable-libopenjpeg --enable-libopus --enable-librtmp --enable-libschroedinger --enable-libsnappy --enable-libsoxr --enable-libspeex --enable-libtheora --enable-libtwolame --enable-libvidstab --enable-libvo-amrwbenc --enable-libvorbis --enable-libvpx --enable-libwavpack --enable-libwebp --enable-libx264 --enable-libx265 --enable-libxavs --enable-libxvid --enable-libzimg --enable-lzma --enable-decklink --enable-zlib
libavutil      55. 36.100 / 55. 36.100
libavcodec     57. 66.101 / 57. 66.101
libavformat    57. 57.100 / 57. 57.100
libavdevice    57.  2.100 / 57.  2.100
libavfilter     6. 66.100 /  6. 66.100
libswscale      4.  3.100 /  4.  3.100
libswresample   2.  4.100 /  2.  4.100
libpostproc    54.  2.100 / 54.  2.100
Input #0, avi, from 'video.avi':
Metadata:
encoder         :
Duration: 00:50:07.67, start: 0.000000, bitrate: 6 kb/s
Stream #0:0: Video: h264 (Constrained Baseline) (H264 / 0x34363248), yuv420p(progressive), 352x240, 3 fps, 3 tbr, 3 tbn, 6 tbc
Output #0, mp4, to 'video.mp4':
Metadata:
encoder         : Lavf57.57.100
Stream #0:0: Video: h264 (Constrained Baseline) ([33][0][0][0] / 0x0021), yuv420p(progressive), 352x240, q=2-31, 3 fps, 3 tbr, 12288 tbn, 3 tbc
Stream mapping:
Stream #0:0 -> #0:0 (copy)
Press [q] to stop, [?] for help
[mp4 @ 0000000002513fc0] Timestamps are unset in a packet for stream 0. This is deprecated and will stop working in the future. Fix your code to set the timestamps properly
[NULL @ 0000000002524020] missing picture in access unit with size 16
Last message repeated 300 times
frame= 9324 fps=0.0 q=-1.0 Lsize=    1388kB time=01:38:27.66 bitrate=   1.9kbits/s speed=3.32e+004x
video:1354kB audio:0kB subtitle:0kB other streams:0kB global headers:0kB muxing overhead: 2.493988%

复制命令后的mp4视频文件:

$ ffmpeg -i video.mp4
ffmpeg version N-82324-g872b358 Copyright (c) 2000-2016 the FFmpeg developers
built with gcc 5.4.0 (GCC)
configuration: --enable-gpl --enable-version3 --disable-w32threads --enable-dx                                                                                                                                                                                               va2 --enable-libmfx --enable-nvenc --enable-avisynth --enable-bzlib --enable-lib                                                                                                                                                                                               ebur128 --enable-fontconfig --enable-frei0r --enable-gnutls --enable-iconv --ena                                                                                                                                                                                               ble-libass --enable-libbluray --enable-libbs2b --enable-libcaca --enable-libfree                                                                                                                                                                                               type --enable-libgme --enable-libgsm --enable-libilbc --enable-libmodplug --enab                                                                                                                                                                                               le-libmp3lame --enable-libopencore-amrnb --enable-libopencore-amrwb --enable-lib                                                                                                                                                                                               openh264 --enable-libopenjpeg --enable-libopus --enable-librtmp --enable-libschr                                                                                                                                                                                               oedinger --enable-libsnappy --enable-libsoxr --enable-libspeex --enable-libtheor                                                                                                                                                                                               a --enable-libtwolame --enable-libvidstab --enable-libvo-amrwbenc --enable-libvo                                                                                                                                                                                               rbis --enable-libvpx --enable-libwavpack --enable-libwebp --enable-libx264 --ena                                                                                                                                                                                               ble-libx265 --enable-libxavs --enable-libxvid --enable-libzimg --enable-lzma --e                                                                                                                                                                                               nable-decklink --enable-zlib
libavutil      55. 36.100 / 55. 36.100
libavcodec     57. 66.101 / 57. 66.101
libavformat    57. 57.100 / 57. 57.100
libavdevice    57.  2.100 / 57.  2.100
libavfilter     6. 66.100 /  6. 66.100
libswscale      4.  3.100 /  4.  3.100
libswresample   2.  4.100 /  2.  4.100
libpostproc    54.  2.100 / 54.  2.100
Input #0, mov,mp4,m4a,3gp,3g2,mj2, from 'video.mp4':
Metadata:
major_brand     : isom
minor_version   : 512
compatible_brands: isomiso2avc1mp41
encoder         : Lavf57.57.100
Duration: 01:38:28.00, start: 0.000000, bitrate: 1 kb/s
Stream #0:0(und): Video: h264 (Constrained Baseline) (avc1 / 0x31637661), yu                                                                                                                                                                                               v420p, 352x240, 1 kb/s, 1.58 fps, 3 tbr, 12288 tbn, 24576 tbc (default)
Metadata:
handler_name    : VideoHandler
At least one output file must be specified

编辑2

如果有人想测试,我上传了两个小样本:

  • AVI视频;这个不在浏览器上运行
  • MP4视频。这辆车在第5秒时坠毁

我们也尝试使用FFMPEG的copy命令(速度更快),但是视频总是在某个时刻崩溃(主要是在我们需要的时候在时间轴中导航),我们不知道为什么。

这里有一个链接,指向您的重新修复的MP4。寻找现在正在发挥作用
我手动检查了字节,正如我所怀疑的,时间戳是错误的。

FFprobe(使用ffprobe video.avi)表示AVI持续时间为5.88秒MPC-HC也显示持续时间为6秒(四舍五入)。现在我们稍后期望MP4在使用copy命令时也具有相同的5.88持续时间。但是在copy之后,输出MP4具有新的10秒持续时间???

MP4将持续时间放在3个位置。每个软件都有自己的规则来检查哪个条目,所以为了每个人的利益,最好编辑所有3个地方。元数据被划分为"原子",因此我们编辑3个原子。

这些被称为mvhdtkhdelst的原子在持续时间内都具有字节值00 00 28 87,其等于10375毫秒(或10.375秒)。这是错误的,因为我们需要5880毫秒(或5.88秒)。我手动将它们全部从00 00 28 87更改为00 00 16 F8,这提供了5880毫秒。现在MPC-HC认为mp4是5秒(phew!!),但Chrome一直说它还是10秒(该死!!)。

作为最后的手段,我检查了一个叫stts的原子。这有3个项目,总共136帧的定时(注意FFprobe说AVI实际上有141帧)。无论如何,第一项所说的前128帧具有512毫秒的显示时间(这需要128 * 512 = 65536 ms的持续时间)。这是错误的。

长话短说。我编辑了stts说。。。

  • 只有01显示时间条目而不是03(但其他条目字节未删除)
  • 不是00 00 00 80=128,实际上是00 00 00 8D=141帧
  • 我仍然将这些(现在是141帧)的原始显示时间保持为00 00 02 00=512毫秒
  • 我将剩下的两个条目字节全部更改为零(因为我们不需要使用它们)

这就是你的MP4如何在Chrome浏览器中测试成为一个可工作的可查找文件的原因。

结论:

为了修复您自己的文件,我怀疑只编辑MP4元数据中stts部分的第一个条目就足够了。你告诉它"我有X帧数量,每帧显示X间隔,没有额外的!!">。然后只需编辑mvhdtkhdelst中的持续时间。在这里,您必须知道如何使用字节。

写一个桌面程序(如果你的团队中有人能做到)。。。

  • 从AVI中获取"持续时间"one_answers"总帧"(我建议ExifTool优于FFprobe,因为它可以正确显示这两个细节)。您的程序通过std-in/out读取此信息。

  • 一旦知道,就以UTF文本的形式搜索原子名称,以查找它们在字节内的起始位置。

  • 当你有了每个起始位置。。。您现在可以将数字编辑为4字节整数(用十六进制表示法编写)

    • 查找"mvhd"的起始位置,然后跳过20个字节查找持续时间
    • 查找"tkhd"的起始位置,然后跳过24个字节查找持续时间
    • 查找"elst"的起始位置,然后跳过24个字节查找持续时间
  • 最后找到"stts"的起始位置,然后跳过8个字节来找到总条目(4个字节)。在那之后,接下来的4个字节是"总帧数"(你用实数固定),然后接下来的4字节是帧的计时,但保持不变。。。

其他查询:。。。

有可能在动作脚本中编写解码器来读取avi文件吗?我看到Youtube和Facebook可以播放AVI文件。。。他们是怎么做的这

是的!!由于AVI有Mpeg(H.264)视频,Flash有一个接受Mpeg a/v字节的解码器,所以如果你从AVI中提取帧字节并将其提供给解码器,它就会显示出来。

这个旧的演示应用程序就是我将Mpeg的AAC音频字节馈送到解码器的一个例子。AVI的原理相同。我和Flash合作,但我要告诉你。。。与其世界上只有一个自定义应用程序可以播放这些文件,不如修复这些文件。固定文件可以在Flash、HTML5浏览器和其他任何地方使用。

附言:Facebook和Youtube正在重新编码你的文件。它不是直接的文件副本。

我们终于解决了这个问题。

尽管@VC。一个人的解决方案似乎奏效了,我们没有成功,因为我们认为有一种更好的方法可以解决它,而不必进行字节运算(一旦我们没有太多经验)。。。

经过多次尝试,我们意识到我们的AVI文件格式错误,因此我们使用了一个名为Mencoder的程序,该程序具有forceidx:功能

强制重建索引。对于索引已损坏的文件(A/V去同步等)非常有用。这将允许在无法进行查找的文件中进行查找。您可以使用MEncoder永久修复索引(请参阅文档)。注意:此选项仅在底层媒体支持查找(即不支持stdin、pipe等)时有效。

运行此命令后,我们使用FFMPEG的复制命令将(现在格式良好的)AVI文件"转换"为mp4。Voilá!魔法发生了!

谢谢大家!

谢谢,@VC。一个被你的时间驱散在这里!

最新更新