如果我尝试使用fs.createWriteStream(remoteFilename)直接将返回的字节范围数据写入mp4文件,则它不会获得需要可播放的视频元数据(持续时间,维度等).
当我得到一个以0开头并以XX结尾的字节范围时,输出mp4是可播放的,但是将具有整个视频长度的持续时间元数据,并且当字节范围在剩余的持续时间内完成时将冻结屏幕时间.
这一点的重点是避免在使用ffmpeg制作5秒剪辑之前下载整个10分钟的文件.如果我可以计算并下载字节范围,应该有办法将其写入独立的mp4文件.
提前感谢您提供的任何帮助.
解决方法
> moov Box:包含其他框:) – 每个框都包含有关mdat框中存在的编码数据的信息(moov =关于MP4文件的元数据).典型的Metadatas是持续时间,帧率,编解码器信息,对视频/音频帧的引用……
> mdat Box:包含文件的实际编码数据.它可以来自各种编解码器,包括音频和视频数据(如果恰好是其中只有一个).对于H264,NAL单元包含在mdat框中.
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.我不知道这个lib是否存在node.js(我猜它可以被移植).即使您不使用它们,上面的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分钟,然后将生成的内容写入output.mp4本地(当然没有下载完整的文件).
ffmpeg需要在http protocol input(大多数二进制文件中都可以找到)的支持下构建.您可以阅读here以获取有关-ss参数放置位置的更多信息(优点/缺点).