是否可以从单个流字节范围块创建新的mp4文件?

时间:2014-04-18 20:32:32

标签: node.js video stream ffmpeg video-streaming

如果我在支持字节范围的服务器上有一个远程mp4文件,是否可以检索单个字节范围并从该范围数据创建一个新的/自包含的mp4?

如果我尝试使用fs.createWriteStream(remoteFilename)将返回的字节范围数据直接写入mp4文件,则无法获得需要播放的视频元数据(持续时间,尺寸等)。

当我得到一个以0开头并以XX结尾的字节范围时,输出mp4是可播放的,但是将具有整个视频长度的持续时间元数据,并且当剩余的字节范围完成时将冻结屏幕持续时间。

我还能如何获取字节范围并从该流对象创建独立的.mp4文件?

这一点的重点是避免在我使用ffmpeg创建5秒剪辑之前下载整个10分钟文件。如果我可以计算并下载字节范围,应该有办法将其写入独立的mp4文件。

提前感谢您提供的任何帮助。

2 个答案:

答案 0 :(得分:6)

MP4文件由框组成。其中两个主要是moov和mdat(非碎片MP4的一般情况):

  • moov box:包含其他框:) - 每个框都包含有关mdat框中存在的编码数据的信息(moov =关于MP4文件的元数据)。典型的metadatas是持续时间,帧率,编解码器信息,对视频/音频帧的引用......
  • mdat box:包含文件的实际编码数据。它可以来自各种编解码器,包括音频和视频数据(如果恰好是其中只有一个)。对于H264,NAL单元包含在mdat框中。

moov框是(应该)在MP4文件Web传送文件的开头,所以如果你写一个从0到XX的字节范围请求,你可能会得到整个moov框+一定数量的mdat数据。因此,文件可以播放到某一点。如果你的字节范围从YY到XX,那么你将无法得到一个像样的moov盒,但是很多mdat不能被使用,除非它们在M​​P4文件中重新包装,并带有适当的moov盒,引用有关“cut”mdat的信息

可以从字节范围块重新创建有效的MP4文件,但它需要MP4文件格式结构的高级知识(您还需要检索moov框以使其可以忍受)。 MP4文件格式基于ISO base media file format - 指定为ISO / IEC 14496-12(MPEG-4 Part 12)。

我知道2个lib可以帮助你做你想做的事:一个在PHP,一个在Java。我不知道node.js是否存在这样的lib(我猜它可以被移植)。即使您不使用它们,上面的2个库也包含有关该主题的有价值信息。

要提供问题的答案,您可以从不同角度解决问题。知道你想要的文件的哪一部分在几毫秒内你可以执行一个ffmpeg命令将全长MP4文件服务器端拼接成一个较小的文件,然后用这个新的较小的MP4文件做你需要的(因此你不需要)在客户端下载不必要的数据)。

ffmpeg命令是(在这种情况下从文件开头1分钟切割):

ffmpeg -i input.mp4 -ss 00:00:00.000 -t 00:01:00.000 -c:a copy -c:v copy output.mp4

有关above command line

的更多信息,请参阅此帖子

这很快就完成了,因为MP4文件结构只是重新组织而没有重新转码。

编辑: 或者我可以在远程文件上使用ffmpeg并在本地创建新剪辑吗?

ffmpeg -ss 00:01:00.000 -i "http://myfile.mp4" -t 00:02:00.000 -c:a copy -c:v copy output.mp4

假设您在客户端(app / web)上有ffmpeg,如果运行上述命令,ffmpeg会将mp4提​​取到输入URL,然后寻找1分钟并从那里切2分钟,然后将生成的内容写入输出。 mp4 本地(当然没有下载完整文件)。

需要构建ffmpeg,并支持http protocol input(您可以在大多数二进制文件中找到)。您可以阅读here以获取有关-ss参数放置位置的更多信息(优点/缺点)。

答案 1 :(得分:0)

取决于您的远程 MP4 文件是碎片 mp4 文件还是平面 mp4 文件。如果它是一个碎片化的 mp4 文件(我认为它是基于您的字节范围注释),您只需下载 init 字节范围和您感兴趣的片段并将它们连接在一起。

如果您有一个平面 mp4 文件,那么接受的答案是正确的方法。