Wrap a standalone FFmpeg binary in an intuitive Iterator interface.
The core goal of this project is to provide a method of interacting with any video as if it were an array of raw RGB frames.
Of course, that's what video is, fundamentally, but there is a whole pandora's box of complexity in terms of receiving and decoding video before you get there.
Using FFmpeg as the core engine provides interoperability between a massive range of formats, containers, extensions, protocols, encoders, decoders, hardware accelerations, and more.
One method of using FFmpeg is low-level bindings to the code used inside the CLI itself -- there are good crates in the Rust ecosystem that do this.
Low level bindings have drawbacks, though:
By wrapping the CLI, this crate avoids those downsides, and also solves some of the pain points that you would encounter if you were to use the CLI directly on its own:
The only remaining downside is the size of the FFmpeg binary itself, but it's less than 100MB when zipped.
First you need an FFmpeg binary. If you don't already have one, head to
https://ffmpeg.org. Either install it globally (e.g. add to PATH
on windows),
or simply place the executable adjacent to your Rust binary target. When you
package and distribute
On the Rust side, it has zero Cargo dependencies! 🎉
console
cargo add ffmpeg-sidecar
Read raw video frames.
```rust use ffmpeg_sidecar::{ child::FfmpegChild, command::FfmpegCommand, event::FfmpegEvent, iter::FfmpegIterator, };
/// Iterates over the frames of a testsrc.
fn main() {
// similar to std::process::Command
let mut command = FfmpegCommand::new();
command
.testsrc() // generate a test pattern video
.rawvideo(); // pipe raw video output
// similar to std::process::Child
let mut child: FfmpegChild = command.spawn().unwrap();
// Iterator over all messages and output let iter: FfmpegIterator = child.iter().unwrap(); iter.for_each(|event: FfmpegEvent| { match event { FfmpegEvent::OutputFrame(frame) => { let _pixels = frame.data; // <- raw RGB pixels! 🎨 } FfmpegEvent::Error(e) => eprintln!("Error: {}", e), _ => {} } }); } ```
Source: /examples/hello_world.rs
console
cargo run --example hello-world
Decode H265, modify the decoded frames, and then write back to H265.
Source: /examples/h265_transcode.rs
console
cargo run --example h265_transcode
Pipe an FFmpeg instance to FFplay for debugging purposes.
Source: /examples/ffplay_preview.rs
console
cargo run --example ffplay_preview
For a myriad of other examples, check any of the unit tests in /src/test.rs in this repo.
/examples
ffplay
for debuggingResult<_, String>
iter()
since
they require consuming stdout
directly📣 Pull requests are welcome!
Inspired loosely by Node.js
fluent-ffmpeg
, which does
something similar in Javascript.