A Rust library for parsing and downloading media content from a DASH MPD file, as used by video services such as on-demand replay of TV content and video streaming services like YouTube.
DASH (dynamic adaptive streaming over HTTP), also called MPEG-DASH, is a technology used for media streaming over the web, commonly used for video on demand (VOD) services. The Media Presentation Description (MPD) is a description of the resources (manifest or “playlist”) forming a streaming service, that a DASH client uses to determine which assets to request in order to perform adaptive streaming of the content. DASH MPD manifests can be used both with content encoded as MPEG and as WebM. There is a good explanation of adaptive bitrate video streaming at howvideo.works.
This library provides a serde-based parser for the DASH MPD format, as formally defined in ISO/IEC standard 23009-1:2019. XML schema files are available for no cost from ISO. When MPD files in practical use diverge from the formal standard, this library prefers to interoperate with existing practice.
If the library feature fetch
is enabled (which it is by default), the library also provides
support for downloading content (audio or video) described by an MPD manifest. This involves
selecting the alternative with the most appropriate encoding (in terms of bitrate, codec, etc.),
fetching segments of the content using HTTP or HTTPS requests (this functionality depends on the
reqwest
crate) and muxing audio and video segments together.
If the library feature libav
is enabled, muxing support (combining audio and video streams, which
are often separated out in DASH streams) is provided by ffmpeg’s libav library, via the ac_ffmpeg
crate. Otherwise, muxing is implemented by calling an external muxer, mkvmerge (from the
MkvToolnix suite), ffmpeg or
vlc as a subprocess. Note that these commandline applications
implement a number of checks and workarounds to fix invalid input streams that tend to exist in the
wild. Some of these workarounds are implemented here when using libav as a library, but not all of
them, so download support tends to be more robust with the default configuration (using an external
application as a subprocess).
The choice of external muxer depends on the filename extension of the path supplied to download_to()
(will be ".mp4" if you call download()
):
```rust use std::time::Duration; use dash_mpd::{MPD, parse};
fn main() { let client = reqwest::blocking::Client::builder() .timeout(Duration::new(10, 0)) .gzip(true) .build() .expect("creating reqwest HTTP client"); let xml = client.get("http://rdmedia.bbc.co.uk/dash/ondemand/testcard/1/client_manifest-events.mpd") .header("Accept", "application/dash+xml,video/vnd.mpeg.dash.mpd") .send() .expect("requesting MPD content") .text() .expect("fetching MPD content"); let mpd: MPD = parse(&xml) .expect("parsing MPD"); if let Some(pi) = mpd.ProgramInformation { if let Some(title) = pi.Title { println!("Title: {:?}", title.content); } if let Some(source) = pi.Source { println!("Source: {:?}", source.content); } } for p in mpd.periods { if let Some(d) = p.duration { println!("Contains Period of duration {:?}", d); } } } ```
To download content from an MPD manifest:
```rust use dash_mpd::fetch::DashDownloader;
let url = "https://storage.googleapis.com/shaka-demo-assets/heliocentrism/heliocentrism.mpd"; match DashDownloader::new(url) .worst_quality() .download() { Ok(path) => println!("Downloaded to {:?}", path), Err(e) => eprintln!("Download failed: {:?}", e), } ```
An application that provides a commandline interface for this library is available separately in the dash-mpd-cli crate.
Add to your Cargo.toml
file:
toml
[dependencies]
dash-mpd = "0.4"
If you don’t need the download functionality and wish to reduce code size, use:
toml
[dependencies]
dash-mpd = { version = "0.4", default-features = false }
This crate is tested on the following platforms:
This project is licensed under the MIT license. For more information, see the LICENSE-MIT
file.