如果我在支持字节范围的服务器上有一个远程mp4文件,是否可以检索单个字节范围并从该范围数据创建一个新的/自包含的mp4?
如果我尝试使用fs.createWriteStream(remoteFilename)
将返回的字节范围数据直接写入mp4文件,则无法获得需要播放的视频元数据(持续时间,尺寸等)。
当我得到一个以0开头并以XX结尾的字节范围时,输出mp4是可播放的,但是将具有整个视频长度的持续时间元数据,并且当剩余的字节范围完成时将冻结屏幕持续时间。
我还能如何获取字节范围并从该流对象创建独立的.mp4文件?
这一点的重点是避免在我使用ffmpeg创建5秒剪辑之前下载整个10分钟文件。如果我可以计算并下载字节范围,应该有办法将其写入独立的mp4文件。
提前感谢您提供的任何帮助。
答案 0 :(得分:6)
MP4文件由框组成。其中两个主要是moov和mdat(非碎片MP4的一般情况):
moov框是(应该)在MP4文件Web传送文件的开头,所以如果你写一个从0到XX的字节范围请求,你可能会得到整个moov框+一定数量的mdat数据。因此,文件可以播放到某一点。如果你的字节范围从YY到XX,那么你将无法得到一个像样的moov盒,但是很多mdat不能被使用,除非它们在MP4文件中重新包装,并带有适当的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
的更多信息,请参阅此帖子
这很快就完成了,因为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 文件,那么接受的答案是正确的方法。