dash-mpd-cli

A commandline application for downloading media content from a DASH MPD file, as used for on-demand replay of TV content and video streaming services like YouTube.

Crates.io CI Dependency status LICENSE

Terminal capture

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 with content encoded in different formats and containers, including H264/MP4, HEVC/MP4 and VP9/WebM. There is a good explanation of adaptive bitrate video streaming at howvideo.works.

This commandline application allows you to download 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 and muxing audio and video segments together. There is also support for downloading subtitles (mostly WebVTT, TTML and SMIL formats, with some support for wvtt format).

This application builds on the dash-mpd crate.

Features

The following features are supported:

The following are not supported:

Installation

Binary releases are available on GitHub for GNU/Linux on AMD64 (statically linked against musl libc to avoid glibc versioning problems), Microsoft Windows on AMD64 and MacOS on aarch64 (“Apple Silicon”) and AMD64. These are built automatically on the GitHub continuous integration infrastructure.

You can also build from source using an installed Rust development environment:

shell cargo install dash-mpd-cli

This installs the binary to your installation root's bin directory, which is typically $HOME/.cargo/bin.

You should also install the following dependencies:

This crate is tested on the following platforms:

Usage

``` Download content from an MPEG-DASH streaming media manifest.

Usage: dash-mpd-cli [OPTIONS]

Arguments: URL of the DASH manifest to retrieve.

Options: -U, --user-agent

  --proxy <URL>
      URL of Socks or HTTP proxy (e.g. https://example.net/ or socks5://example.net/).

  --no-proxy
      Disable use of Socks or HTTP proxy even if related environment variables are set.

  --auth-username <USER>
      Username to use for authentication with the server(s) hosting the DASH manifest and the media segments (HTTP Basic authentication only).

  --auth-password <PASSWORD>
      Password to use for authentication with the server(s) hosting the DASH manifest and the media segments (HTTP Basic authentication only).

  --timeout <SECONDS>
      Timeout for network requests (from the start to the end of the request), in seconds.

  --sleep-requests <SECONDS>
      Number of seconds to sleep between network requests (default 0).

-r, --limit-rate Maximum network bandwidth in octets per second (default no limit), e.g. 200K, 1M.

  --max-error-count <COUNT>
      Maximum number of non-transient network errors that should be ignored before a download is aborted (default is 10).

  --source-address <source-address>
      Source IP address to use for network requests, either IPv4 or IPv6. Network requests will be made using the version of this IP address (e.g. using an IPv6 source-address will select IPv6 network traffic).

  --add-root-certificate <CERT>
      Add a root certificate (in PEM format) to be used when verifying TLS network connections.

  --client-identity-certificate <CERT>
      Client private key and certificate (in PEM format) to be used when authenticating TLS network connections.

  --prefer-video-width <WIDTH>
      When multiple video streams are available, choose that with horizontal resolution closest to WIDTH.

  --prefer-video-height <HEIGHT>
      When multiple video streams are available, choose that with vertical resolution closest to HEIGHT.

  --quality <quality>
      Prefer best quality (and highest bandwidth) representation, or lowest quality.

      [possible values: best, intermediate, worst]

  --prefer-language <LANG>
      Preferred language when multiple audio streams with different languages are available. Must be in RFC 5646 format (e.g. fr or en-AU). If a preference is not specified and multiple audio streams are present, the first one listed in the DASH manifest will be downloaded.

  --video-only
      If media stream has separate audio and video streams, only download the video stream.

  --audio-only
      If media stream has separate audio and video streams, only download the audio stream.

  --simulate
      Download the manifest and print diagnostic information, but do not download audio, video or subtitle content, and write nothing to disk.

  --write-subs
      Write subtitle file, if subtitles are available.

  --keep-video <VIDEO-PATH>
      Keep video stream in file specified by VIDEO-PATH.

  --keep-audio <AUDIO-PATH>
      Keep audio stream (if audio is available as a separate media stream) in file specified by AUDIO-PATH.

  --no-period-concatenation
      Never attempt to concatenate media from different Periods (keep one output file per Period).

  --key <KID:KEY>
      Use KID:KEY to decrypt encrypted media streams. KID should be either a track id in decimal (e.g. 1), or a 128-bit keyid (32 hexadecimal characters). KEY should be 32 hexadecimal characters. Example: --key eb676abbcb345e96bbcf616630f1a3da:100b6c20940f779a4589152b57d2dacb. You can use this option multiple times.

  --save-fragments <FRAGMENTS-DIR>
      Save media fragments to this directory (will be created if it does not exist).

  --ignore-content-type
      Don't check the content-type of media fragments (may be required for some poorly configured servers).

  --add-header <NAME:VALUE>
      Add a custom HTTP header and its value, separated by a colon ':'. You can use this option multiple times.

-H, --header

Add a custom HTTP header, in cURL-compatible format. You can use this option multiple times.

  --referer <URL>
      Specify content of Referer HTTP header.

-q, --quiet

-v, --verbose... Level of verbosity (can be used several times).

  --no-progress
      Disable the progress bar

  --no-xattr
      Don't record metainformation as extended attributes in the output file.

  --no-version-check
      Disable the check for availability of a more recent version on startup.

  --ffmpeg-location <PATH>
      Path to the ffmpeg binary (necessary if not located in your PATH).

  --vlc-location <PATH>
      Path to the VLC binary (necessary if not located in your PATH).

  --mkvmerge-location <PATH>
      Path to the mkvmerge binary (necessary if not located in your PATH).

  --mp4box-location <PATH>
      Path to the MP4Box binary (necessary if not located in your PATH).

  --mp4decrypt-location <PATH>
      Path to the mp4decrypt binary (necessary if not located in your PATH).

-o, --output Save media content to this file.

  --cookies-from-browser <BROWSER>
      Load cookies from BROWSER (Firefox, Chrome, ChromeBeta, Chromium).

  --list-cookie-sources
      Show valid values for BROWSER argument to --cookies-from-browser on this computer, then exit.

-h, --help Print help (see a summary with '-h')

-V, --version Print version ```

If your filesystem supports extended attributes, the application will save the following metainformation in the output file:

You can examine these attributes using xattr -l (you may need to install your distribution's xattr package). Disable this feature using the --no-xattr commandline argument.

Muxing

The underlying library dash-mpd-rs has two methods for muxing audio and video streams together. If the library feature libav is enabled (which is not the default configuration), muxing support 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, vlc or MP4Box 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 libav feature currently only works on Linux.

The choice of external muxer depends on the filename extension of the path supplied to --output or -o (which will be ".mp4" if you don't specify the output path explicitly):

License

This project is licensed under the MIT license. For more information, see the LICENSE-MIT file.

Similar tools

Similar commandline tools that are able to download content from a DASH manifest:

However, dash-mpd-cli (this application) is able to download content from certain streams that do not work with other applications:

Building

$ git clone https://github.com/emarsden/dash-mpd-cli $ cd dash-mpd-cli $ cargo build --release $ target/release/dash-mpd-cli --help

The application can also be built statically with the musl-libc target on Linux. First install the MUSL C standard library on your system. Add linux-musl as a target to your Rust toolchain, then rebuild for the relevant target:

$ sudo apt install musl-dev $ rustup target add x86_64-unknown-linux-musl $ cargo build --release --target x86_64-unknown-linux-musl

Static musl-libc builds don’t work with OpenSSL, which is why we disable default features on the dash-mpd crate and build it with rustls support (a Rust TLS stack). You may encounter some situations where rustls fails to connect (handshake errors, for example) but other applications on your system can connect. These differences in behaviour are typically due to different configurations for the set of root certificates. If you prefer to use your machine’s native TLS stack, replace both instances of rustls-tls by native-tls in Cargo.toml and rebuild.