一个可以将爱奇艺qsv格式视频转换为flv格式视频的简单命令行工具
学习Rust后的第一个项目,感谢原C#项目作者提供的针对QSV的解决方案,也欢迎大家fork或者提交PR,帮助完善功能或者修复bug
本程序仅支持QSV v2.0(可能无法处理某些较早的视频,比如说部分16年前的视频)
前段时间用爱奇艺看电视时,想要收藏一些影视资源,于是去网上找相关的转码工具。遗憾的是,有些工具要么各种骗钱,要么不好用卡顿错帧。后面发现了一些不错的能够转换QSV的代码,但总觉得他们的实现有些仓促,于是用Rust重写,并做出了改进
Rust的开发的体验确实不错,解决了很多C++的痛点,同时又可以较大程度保证运行性能
命令行运行 (最近实现了交叉编译,提供了多种系统平台下已编译好的程序)
```out qsv2flv 0.1.1 ZhangZhilin corex_public@outlook.com A tool for converting QSV to FLV
USAGE: qsv2flv [FLAGS]
FLAGS: -h, --help Prints help information -V, --version Prints version information -v, --verbose Print test information verbosely
ARGS: Sets the input file to use
仅供学习用途(切勿将该项目用于非法盈利)
提供了简单的命令行前端交互
转换速度快
不会生成临时文件
执行顺序:
```rust // 将QSV转换为FLV fn convertqsvtoflv(qsv: &mut File, flv: &mut File) -> Result<()> /* [步骤] 检查QSV文件是否正确 */ // 验证QSV格式 fn validateqsv_format(qsv: &mut File) -> Result<()>
/* [步骤] 解析QSV文件中 */
// 从QSV中解析得到每个TAG块的信息
fn tag_blocks_from_qsv(qsv: &mut File) -> io::Result<FlvTagBlocks>
// 将QSV文件seek至QSV的TAGS的起始处
fn seek_qsv_to_start(qsv: &mut File) -> io::Result<()>
// 使QSV文件的seek指针跳过元数据开头
fn skip_qsv_metadata(qsv: &mut File) -> io::Result<()>
// 从所有TAG块获取FLV的元数据
fn meta_data_from_tag_blocks(qsv: &mut File, tags: &[FlvTagBlock]) -> Result<MetaData>
// 解析视频TAG块信息 (是否为关键帧, 视频编码ID)
fn parse_video_tag(qsv: &mut File, tag: FlvTagBlock) -> Result<(bool, u8)>
// 解析音频TAG块信息 (音频编码ID, 音频采样率, 音频采样大小, 音频是否为立体声)
fn parse_audio_tag(qsv: &mut File, tag: FlvTagBlock) -> Result<(u8, u8, u8, bool)>
// 从TAG块中读取时间戳
fn get_time_stamp_from_tag(qsv: &mut File, tag: &FlvTagBlock) -> Result<i32>
/* [步骤] 正在写入到FLV文件中 */
// 根据已提取出的TAG块信息和FLV元数据,将QSV转换为FLV
fn write_from_qsv_to_flv(qsv: &mut File, tags: &[FlvTagBlock], flv: &mut File, meta: &MetaData)
-> Result<()>
```
错误类型:
rust
pub enum ErrorKind {
Io(std::io::Error), // 文件系统IO错误
IncorrectQsvVersion, // 错误的QSV版本:本程序无法处理
IncorrectQsvFormat, // 错误的QSV格式:不符合预期格式/该文件不是QSV
QsvTagsIsEmpty, // 该QSV文件TAG块数量小于1个
MediaDurationIsTooShort, // 媒体时长过短
}
检查输出路径是否已存在文件,询问覆盖,防止因File::create(...)导致意外覆盖
性能探查:valgrind, qcachegrind
rust trait FilePlus { fn read_byte(&mut self) -> std::io::Result<Option<u8>>; fn write_byte(&mut self, u8) -> io::Result<bool>; fn tell(&mut self) -> std::io::Result<u64>; } impl FilePlus for File { //... }
性能优化:unsafe优化
详细进度显示
确保宿主机跨平台可移植性:跨系统、大小端、32位/64位
完善注释、文档
- 支持管道输出/重定向输出
- 改进FLV生成,提供更多可设定的FLV元数据参数(建议引入:flavors && nom)
- 支持批量文件转码
简单测试样例:
| 文件名称 | 文件大小 | 存储介质 | 本项目[Rust]用时 | 其他版[C++]用时 | | -------------------------- | ----- | ---------------- | ----------- | ---------- | | iPartment 5第2集-蓝光1080P.qsv | 842MB | 2.5英寸希捷机械 | 34.095s | 65.914s | | iPartment 5第1集-蓝光1080P.qsv | 895MB | 2.5英寸Intel固态 | 14.048s | 13.537s | | Smooth Criminal.qsv | 18MB | 2.5英寸Intel固态 | 0.442s | 0.143s |
v0.1.0
更新日志
第一个可用版本
避免了临时文件的生成
备注
这是命令行程序,没有提供图形界面,但是使用方法并不难
v0.1.1
更新日志
增加了磁盘缓存同步步骤,防止在某些意外情况下可移动磁盘数据丢失
v0.1.2
更新日志
重构了少量代码,改善了代码风格,也因此改动了部分函数签名
备注
该版本代码行为上与v0.1.1一致,主要集中在代码风格、文档的改进
macros.rs