arabcoders / ytptube

A WebGUI for yt-dlp with concurrent downloads support.

Geek Repo:Geek Repo

Github PK Tool:Github PK Tool

YTPTube

Build Status

Web GUI for yt-dlp with playlist & channel support.

YTPTube started as a fork of meTube, Since then it went under heavy changes, and it supports many new features.

YTPTube Features.

  • A built in video player that can play any video file regardless of the format.
  • New /add_batch endpoint that allow multiple links to be sent.
  • Completely redesigned the frontend UI.
  • Switched out of binary file storage in favor of SQLite.
  • Handle live streams.
  • Support per link, yt-dlp config and cookies. and output format.
  • Tasks Runner. It allow you to queue channels for downloading using simple json file.
  • Webhook sender. It allow you to add webhook endpoints that receive events related to downloads using simple json file.
  • Multi-downloads support.

Tips

Your yt-dlp config should include the following options for optimal working conditions.

{
    "windowsfilenames": true,
    "live_from_start": true,
    "format_sort": [
        "codec:avc:m4a"
    ]
}
  • Note, the format_sort, forces YouTube to use x264 instead of vp9 codec, you can ignore it if you want. i prefer the media in x264.

Short screenshot

Run using Docker

docker run -d --name ytptube -p 8081:8081 -v ./config:/config:rw -v ./downloads:/downloads:rw ghcr.io/arabcoders/ytptube

Run using docker-compose

version: "3.9"
services:
  ytptube:
    user: "1000:1000"
    image: ghcr.io/arabcoders/ytptube
    container_name: ytptube
    restart: unless-stopped
    ports:
      - "8081:8081"
    volumes:
      - ./config:/config:rw
      - ./downloads:/downloads:rw
    tmpfs:
      - /tmp

Configuration via environment variables

Certain values can be set via environment variables, using the -e parameter on the docker command line, or the environment: section in docker-compose.

  • YTP_CONFIG_PATH: path to where the queue persistence files will be saved. Defaults to /config in the docker image, and ./var/config otherwise.
  • YTP_DOWNLOAD_PATH: path to where the downloads will be saved. Defaults to /downloads in the docker image, and ./var/downloads otherwise.
  • YTP_TEMP_PATH: path where intermediary download files will be saved. Defaults to /tmp in the docker image, and ./var/tmp otherwise.
  • YTP_TEMP_KEEP: Whether to keep the Individual video temp directory or remove it. Defaults to false.
  • YTP_URL_PREFIX: base path for the web server (for use when hosting behind a reverse proxy). Defaults to /.
  • YTP_OUTPUT_TEMPLATE: the template for the filenames of the downloaded videos, formatted according to this spec. Defaults to %(title)s.%(ext)s. This will be the default for all downloads unless the request include output template.
  • YTP_KEEP_ARCHIVE: Whether to keep history of downloaded videos to prevent downloading same file multiple times. Defaults to true.
  • YTP_YTDL_DEBUG: Whether to turn debug logging for the internal yt-dlp package. Defaults to false.
  • YTP_ALLOW_MANIFESTLESS: Allow yt-dlp to download live streams videos which are yet to be processed by YouTube. Defaults to false
  • YTP_HOST: Which IP address to bind to. Defaults to 0.0.0.0.
  • YTP_PORT: Which port to bind to. Defaults to 8081.
  • YTP_LOG_LEVEL: Log level. Defaults to info.
  • YTP_MAX_WORKERS: How many works to use for downloads. Defaults to 1.
  • YTP_MAX_WORKERS: How many works to use for downloads. Defaults to 1.
  • YTP_STREAMER_VCODEC: The video codec to use for in-browser streaming. Defaults to libx264.
  • YTP_STREAMER_ACODEC: The audio codec to use for in-browser streaming. Defaults to aac.

Running behind a reverse proxy

It's advisable to run YTPTube behind a reverse proxy, if authentication and/or HTTPS support are required.

When running behind a reverse proxy which remaps the URL (i.e. serves YTPTube under a subdirectory and not under root), don't forget to set the YTP_URL_PREFIX environment variable to the correct value.

NGINX

location /ytptube/ {
  proxy_pass http://ytptube:8081;
  proxy_http_version 1.1;
  proxy_set_header Upgrade $http_upgrade;
  proxy_set_header Connection "upgrade";
  proxy_set_header Host $host;
}

Note: the extra proxy_set_header directives are there to make web socket connection work.

Caddy

The following example Caddyfile gets a reverse proxy going behind caddy.

example.com {
  route /ytptube/* {
    uri strip_prefix ytptube
    reverse_proxy ytptube:8081
  }
}

Updating yt-dlp

The engine which powers the actual video downloads in YTPTube is yt-dlp. Since video sites regularly change their layouts, frequent updates of yt-dlp are required to keep up.

There's an automatic nightly build of YTPTube which looks for a new version of yt-dlp, and if one exists, the build pulls it and publishes an updated docker image. Therefore, in order to keep up with the changes, it's recommended that you update your YTPTube container regularly with the latest image.

Troubleshooting and submitting issues

Before asking a question or submitting an issue for YTPTube, please remember that YTPTube is only a UI for yt-dlp. Any issues you might be experiencing with authentication to video websites, postprocessing, permissions, other yt-dlp options configurations which seem not to work, or anything else that concerns the workings of the underlying yt-dlp library, need not be opened on the YTPTube project. In order to debug and troubleshoot them, it's advised to try using the yt-dlp binary directly first, bypassing the UI, and once that is working, importing the options that worked for you into yt-dlp options file.

In order to test with the yt-dlp command directly, you can either download it and run it locally, or for a better simulation of its actual conditions, you can run it within the YTPTube container itself.

Via HTTP

Simply go to Console button in your navbar and directly use the yt-dlp command.

Via CLI

Assuming your YTPTube container is called ytptube, run the following on your Docker host to get a shell inside the container:

docker exec -ti ytptube bash
cd /downloads
yt-dlp ....

Once there, you can use the yt-dlp command freely.

Building and running locally

Make sure you have nodejs and Python 3.11+ installed.

cd ytptube/frontend
# install Vue and build the UI
npm install
npm run build
# install python dependencies
cd ..
python -m venv .venv
source .venv/bin/activate
pip3 install pipenv
pipenv install
# run
python app/main.py

A Docker image can be built locally (it will build the UI too):

docker build . -t ytptube

ytdlp.json File

The config/ytdlp.json, is a json file which can be used to alter the default yt-dlp config settings. For example these are the options i personally use,

{
  // Make the final filename windows compatible.
  "windowsfilenames": true,
  // Write subtitles if the stream has them.
  "writesubtitles": true,
  // Write info.json file for each download. It can be used by many tools to generate info etc.
  "writeinfojson": true,
  // Write thumbnail if available.
  "writethumbnail": true,
  // Do not download automatically generated subtitles. 
  "writeautomaticsub": false,
  // MP4 is limited with the codecs we use, so "mkv" make sense.
  "merge_output_format": "mkv",
  // Record live stream from the start.
  "live_from_start": true,
  // For YouTube try to force H264 video codec & AAC audio.
  "format_sort": [
      "codec:avc:m4a"
  ],
  // Your choice of subtitle languages to download.
  "subtitleslangs": [ "en", "ar" ],
  // postprocessors to run on the file
  "postprocessors": [
      // this processor convert the downloaded thumbnail to jpg.
      {
          "key": "FFmpegThumbnailsConvertor",
          "format": "jpg"
      },
      // This processor convert subtitles to srt format.
      {
          "key": "FFmpegSubtitlesConvertor",
          "format": "srt"
      },
      // This processor embed metadata & info.json file into the final mkv file.
      {
          "key": "FFmpegMetadata",
          "add_infojson": true,
          "add_metadata": true
      },
      // This process embed subtitles into the final file if it doesn't have subtitles embedded.
      {
          "key": "FFmpegEmbedSubtitle",
          "already_have_subtitle": false
      }
  ]
}

tasks.json File

The config/tasks.json, is a json file, which can be used to queue URLs for downloading, it's mainly useful if you follow specific channels and you want it downloaded automatically, The schema for the file is as the following, Only the URL key is required.

[
  {
    // (URL: string) **REQUIRED**, URL to the content.
    "url": "", 
    // (Name: string) Optional field. Mainly used for logging. If omitted, random GUID will be shown.
    "name": "My super secret channel", 
    // (Timer: string) Optional field. Using regular cronjob timer, if the field is omitted, it will run every hour in random minute.
    "timer": "1 */1 * * *", 
    // (yt-dlp cookies: object) Optional field. A JSON cookies exported by flagCookies.
    "ytdlp_cookies": {}, 
    // (yt-dlp config: object) Optional field. A JSON yt-dlp config.
    "ytdlp_config": {},
    // (Output Template: string) Optional field. A File output format,
    "output_template": "",
    // (Folder: string) Optional field. Where to store the downloads relative to the main download path.
    "folder":"",
    // (Format: string) Optional field. Format as specified in Web GUI. Defaults to "any".
    "format": "",
    // (Quality: string), Optional field. Quality as specified in Web GUI. Defaults to "best".
    "quality": "", 
  },
  {
   // (URL: string) **REQUIRED**, URL to the content.
   "url": "https://..." // This is valid config, it will queue the channel for downloading every hour at random minute.
  },
  ...
]

The task runner is doing what you are doing when you click the add button on the WebGUI, this just fancy way to automate that.

WARNING: We strongly advice turning on YTP_KEEP_ARCHIVE option. Otherwise, you will keep re-downloading the items, and you will eventually get banned from the source or or you will waste space, bandwidth re-downloading content over and over.

webhooks.json File

The config/webhooks.json, is a json file, which can be used to add webhook endpoints that would receive events related to the downloads.

[
  {
    // (name: string) - REQUIRED - The webhook name.
    "name": "my very smart webhook receiver",
    // (on: array) - OPTIONAL - List of accepted events, if left empty it will send all events.
    // Allowed events ["added", "completed", "error", "not_live" ] you can choose one or all of them.
    "on": [ "added", "completed", "error", "not_live" ],
    "request": {
      // (url: string) - REQUIRED-  The webhook url
      "url": "https://mysecert.webhook.com/endpoint", 
      // (type: string) - OPTIONAL - The request type, it can be json or form.
      "type": "json",
      // (method: string) - OPTIONAL - The request method, it can be POST or PUT
      "method": "POST",
      // (headers: dictionary) - OPTIONAL - Extra headers to include.
      "headers": {
        "Authorization": "Bearer my_secret_token"
      }
  }
  ...
]

Social contact

If you have short or quick questions, you are free to join my discord server and ask the question. keep in mind it's solo project, as such it might take me a bit of time to reply.

Donation

If you feel like donating and appreciate my work, you can do so by donating to children charity. For example Make-A-Wish. I Personally don't need the money, but I do appreciate the gesture. Making a child happy is more worthwhile.

About

A WebGUI for yt-dlp with concurrent downloads support.

License:MIT License


Languages

Language:Python 62.5%Language:Vue 30.3%Language:JavaScript 2.5%Language:Dockerfile 2.0%Language:CSS 1.7%Language:Shell 0.5%Language:HTML 0.4%