MeTube is a self-hosted web UI for yt-dlp, for downloading media from YouTube and dozens of other sites.
Key capabilities: * Download videos, audio, captions, and thumbnails from a browser UI. * Download playlists and channels, with configurable output and download options. * Subscribe to channels and playlists, periodically check for new items, and queue new uploads automatically.

docker run -d -p 8081:8081 -v /path/to/downloads:/downloads ghcr.io/alexta69/metube
services:
metube:
image: ghcr.io/alexta69/metube
container_name: metube
restart: unless-stopped
ports:
- "8081:8081"
volumes:
- /path/to/downloads:/downloads
Certain values can be set via environment variables, using the -e parameter on the docker command line, or the environment: section in docker-compose.
5, then at most five downloads will run concurrently, and any additional downloads will wait until one of the active downloads completes. Defaults to 3. true, downloaded files are deleted on the server, when they are trashed from the "Completed" section of the UI. Defaults to false.0 (no limit).60.50.50000.0 (disabled)./downloads in the Docker image, and . otherwise.DOWNLOAD_DIR.true.true.(^|/)[.@].*$, which means directories starting with . or @.true, the download directories (DOWNLOAD_DIR and AUDIO_DOWNLOAD_DIR) are indexable on the web server. Defaults to false.queue.json, pending.json, completed.json, subscriptions.json). Defaults to /downloads/.metube in the Docker image, and . otherwise./downloads in the Docker image, and . otherwise.tmpfs) for better performance.false, ownership of DOWNLOAD_DIR, STATE_DIR, and TEMP_DIR (and their contents) will not be set on container start. Ensure user under which MeTube runs has necessary access to these directories already. Defaults to true.%(title)s.%(ext)s.%(title)s - %(section_number)s %(section_title)s.%(ext)s.%(playlist_title)s/%(title)s.%(ext)s. Set to empty to use OUTPUT_TEMPLATE instead.%(channel)s/%(title)s.%(ext)s. Set to empty to use OUTPUT_TEMPLATE instead.false. See Configuring yt-dlp options for details and security considerations.HH:MM, 24-hour) when you want the daily upgrades and MeTube restart to happen. Defaults to empty (disabled).0.0.0.0 (all interfaces).8081./.https instead of http (CERTFILE and KEYFILE required). Defaults to false.* to allow all origins. This must be configured for browser extensions, bookmarklets, and any other browser-based tools that contact MeTube from a different origin. For browser extensions use * (see below); for bookmarklets you can list specific sites, e.g. https://www.youtube.com,https://www.vimeo.com.robots.txt file mounted in the container.1000 (legacy UID also supported).1000 (legacy GID also supported).022.light, dark, or auto. Defaults to auto.DEBUG, INFO, WARNING, ERROR, CRITICAL, or NONE. Defaults to INFO. false.MeTube lets you customize how yt-dlp behaves at three levels, from broadest to most specific:
When a download starts, these layers are combined in order. If the same option appears in more than one layer, the more specific one wins: per-download overrides beat presets, and presets beat global options.
In JSON presets and overrides, setting an option to null clears that option for that download (for example, "download_archive": null overrides a global archive path so the archive is not used). This follows yt-dlp’s usual meaning of None for that option.
yt-dlp options in MeTube are expressed as JSON objects. The keys are yt-dlp API option names, which roughly correspond to command-line flags with dashes replaced by underscores. For example, the command-line flag --write-subs becomes "writesubtitles": true in JSON.
Tip: Some command-line flags don't have a direct single-key equivalent — for instance,
--embed-thumbnailand--recode-videomust be expressed via"postprocessors". A full list of available API options can be found in the yt-dlp source, and this conversion script can help translate command-line flags to their API equivalents.
Global options form the baseline for every download. There are two ways to define them, and you can use either or both:
Inline via environment variable (YTDL_OPTIONS) — pass a JSON object directly:
environment:
- 'YTDL_OPTIONS={"writesubtitles": true, "subtitleslangs": ["en", "de"], "updatetime": false, "writethumbnail": true}'
Via a JSON file (YTDL_OPTIONS_FILE) — mount a file into the container and point to it:
volumes:
- /path/to/ytdl-options.json:/config/ytdl-options.json
environment:
- YTDL_OPTIONS_FILE=/config/ytdl-options.json
where ytdl-options.json contains:
{
"writesubtitles": true,
"subtitleslangs": ["en", "de"],
"updatetime": false,
"writethumbnail": true
}
The file is monitored for changes and reloaded automatically — no container restart needed. If you use both methods and they define the same key, the file takes precedence.
Presets let you define named bundles of options that appear in the web UI under Advanced Options as "Option Presets". Users can select one or more presets per download, making it easy to apply common option combinations without editing global settings.
Like global options, presets can be set inline or via a file:
YTDL_OPTIONS_PRESETS — a JSON object where each key is a preset name and its value is a set of yt-dlp options.YTDL_OPTIONS_PRESETS_FILE — path to a JSON file containing presets, monitored and reloaded on changes.If both are used and they define a preset with the same name, the file's version takes precedence.
Example — a presets file defining three presets:
{
"sponsorblock": {
"postprocessors": [
{ "key": "SponsorBlock", "categories": ["sponsor", "selfpromo", "interaction"] },
{ "key": "ModifyChapters", "remove_sponsor_segments": ["sponsor", "selfpromo", "interaction"] }
]
},
"embed-subs": {
"writesubtitles": true,
"writeautomaticsub": true,
"subtitleslangs": ["en", "de"],
"postprocessors": [{ "key": "FFmpegEmbedSubtitle" }]
},
"limit-rate": {
"ratelimit": 5000000
}
}
This makes three presets available in the UI: * sponsorblock — strips sponsor, self-promo, and interaction segments from videos. * embed-subs — downloads English and German subtitles and embeds them into the video file. * limit-rate — caps download speed to ~5 MB/s.
When multiple presets are selected for a download, they are applied in order. If two presets set the same option, the later one wins.
For one-off tweaks, MeTube can expose a free-text JSON field in the UI ("Custom yt-dlp Options") where users type yt-dlp options that apply only to that single download. This is disabled by default:
environment:
- ALLOW_YTDL_OPTIONS_OVERRIDES=true
Once enabled, the field appears under Advanced Options. Any options entered there take the highest priority, overriding both global options and selected presets.
⚠️ Security note: Enabling this allows arbitrary yt-dlp API options to be supplied by anyone with access to the UI. Depending on the options used, this may enable arbitrary command execution inside the container. Enable only in trusted environments.
When a download starts, the final set of yt-dlp options is built in this order:
YTDL_OPTIONS / YTDL_OPTIONS_FILE).$ claude mcp add metube \
-- python -m otcore.mcp_server <graph>