Docker can help you build a Home Media Server in just minutes without complex setups. In this post, I will show you how to build a perfect Docker media server using Docker and Ubuntu.
When I say, Media Server, I mean an all-in-one media server built with Docker wthat ill automate media download, organization, streaming, and sharing with family/friends.
Note that this is a "basic" level post on how to set up a perfect home media server using Docker only, which is better suited for accessing stuff from inside your network. Reverse Proxy and secure external access will be covered later in the series.
Ultimate Docker Server Series:
This post is part of the Docker Server Tutorial Series, which includes the following individual chapters/parts:- Ultimate Docker Server: Getting Started with OS Preparation [VIDEO] [2024]
- Docker Media Server Ubuntu/Debian with 60+ Awesome Apps [VIDEO] [2024]
- ZeroTier VPN Ubuntu, Docker, Synology, Windows: Secure on-the-go access [VIDEO] [2024]
- Nginx Proxy Manager Docker Compose Guide: Simplest Reverse Proxy [coming soon]
- Ultimate Traefik v3 Docker Compose Guide: Best Reverse Proxy [VIDEO] [2024]
- Authelia Docker Compose Guide: Secure 2-Factor Authentication [VIDEO] [2024]
- Ultimate Authentik Docker Compose Guide with Traefik [2025]
- Google OAuth Docker Compose Guide: Multi-Factor Authentication [VIDEO] [2024]
- Docker Security Practices for Homelab: Secrets, Firewall, and more
- Cloudflare Settings for Docker Traefik Stacks
- Implementing a Backup System for Docker Traefik Stack [coming soon]
- Automate Homelab Setup Deployarr: 110+ Apps (Traefik, Authentik, and more) in Minutes
This post is written with a lot of details to help newbies get started on this journey. It may look long but the process itself should take less than an hour.
If you have already reviewed part 1 of this series, let's get started with building the perfect Docker media server stack with Ubuntu 22.04 LTS Jammy Jellyfish.
Table of Contents
- Docker Media Server Guide
- What is Docker?
- Requirements for Media Server Docker Stack
- Setting Up the Docker Environment
- Building Docker Media Server
- Membership Required
- Membership Required
- Adding Additional Apps to the Docker Stack
- 6. Jellyfin - Free and Awesome Media Server
- 7. SABnzbd - Binary newsgrabber (NZB downloader)
- 8. qBittorrent - Torrent downloader
- 9. Radarr - Movie management
- 10. Sonarr - TV Shows Management
- 11. Bazarr - Subtitle Management
- 12. Tautulli - Plex Statistics and Monitoring
- 13. Uptime Kuma - Status Page & Monitoring Server
- 14. MariaDB - MySQL Database
- 15. File Browser - File Explorer
- 16. Docker Garbage Collector
- The Final Docker Compose File
- 17-60 and More: Rest of the Apps
- Adding Additional Apps to the Docker Stack
- External Access to Apps
- Troubleshooting
- Membership Required
- Closing Thoughts
Docker Media Server Guide
Here are the previous versions of this guide:
- Ultimate Smart Home Media Server with Docker and Ubuntu 18.04 – Basic
- Docker Media Server Ubuntu 22.04 with 23 Awesome Apps
If you prefer watching a video, check out the video version of this guide:
This guide is an update based on the evolution of my own setup and to align with the best practices implemented in my Auto-Traefik script.
In addition, I am also taking the "secure-by-design" approach and including Socket Proxy and Docker Secrets in this basic guide.
Objectives of this Home and Media Server with Docker
- I constantly mess with my home server and I do not want my tinkering to bring down the media server.
- [Begin Rant] Avoid having to deal with all my media consumers who take the whole setup for granted and do not appreciate what I do. [End Rant]
But for a typical user, both services may reside on just one Docker host.
One of the big tasks of a completely automated media server is media aggregation. For example, when a TV show episode becomes available, automatically download it, collect its poster, fanart, subtitle, etc., put them all in a folder of your choice (eg. inside your TV Shows folder), update your media library (eg. on Jellyfin, Emby, or Plex) and then send a notification to you (eg. Email, Mobile notification, etc.) saying your episode is ready to watch.
Sounds awesome right?
How Everything Fit Together - The Big Picture
Here is a list of functions I want my comprehensive autonomous media server to do:
- Automated TV Show download and organization
- Automated Movie download and organization
- On-demand or automated torrent download
- On-demand or automated NZB (Usenet) download
- Serve and Stream Media to Devices in the house and outside through internet
- On demand torrent and NZB search interface
- Act as a personal cloud server with secure file access anywhere
- Provide a unified interface to access all the apps
- Update all the apps automatically
My Proxmox Home Server
My Proxmox Server runs my Home Server, Media/Database Server, and AdBlock/DNS Server as Ubuntu 22.04 LXC Containers. [Read: 5 Best Mini PC for Proxmox Home Server [2024]]
I recently upgraded to the TopTon V700 Mini PC as Proxmox Host (for just $481) and it's been killing it.
- TopTon V700 Mini PC with Intel 13th Gen "Raptor Lake" i7-13800H - $481
- 2x32GB Crucial DDR5 4800MHz SO-DIMM RAM - $163
- 2x2TB Crucial T500 PCIe Gen4 M.2 NVME Drives in ZFS RAID - $230
- 4TB Crucial MX500 SATA III SSD for non-critical data, cache, etc. - $200
Apps for Docker Media Server
There are several apps that can do such tasks and we have compiled them in our list of 60+ best Docker containers for home server beginners.
In this guide, I will cover a few examples starting with Socket Proxy, Portainer, Dozzle, and Plex to explain the Docker Concepts. Then, present a few more apps as concise examples. Finally, I will show you how to add literally 10s, 100s, of apps to your stack.
It may seem like a complex setup, but trust me, docker (along with Docker Compose) can make installation, migration, and maintenance of these home server apps easier.
There are separate posts focused on many of these apps that explain specific configurations in detail. These will be linked o along the way.
What is Docker?
Before we get started with building a docker media server, it only makes sense to touch on Docker. We have already covered What is Docker and how it compares to a Virtual Machine such as VirtualBox. Therefore, we won't go into much detail here.
Briefly, Docker allows for operating-system-level virtualization. What this means is that applications can be installed inside virtual "containers", completely isolated from the host operating system.
Unlike a virtual machine, which needs guest OS for each of the virtual machines, a Docker container does not need a separate Operating system. So docker containers can be created and destroyed in seconds. The containers also boot in seconds and so your app is ready to roll very quickly.
Docker works natively on Linux, but is also available for Mac and Windows.
Recommended Guides on Docker:
OK Great, but why build a Media Server on Docker?
Again, this has been explained in detail in my original docker guide.
The traditional way of building a Home Media Server involves setting up the operating system, adding repositories, downloading the apps, installing the pre-requisites/dependencies, installing the app, and configuring the app.
This is cumbersome on Linux and requires extensive commandline work.
In Docker, home server apps such as Radarr, Sonarr, Plex, etc. can be installed with ease without worrying about pre-requisites or incompatibilities. All requirements are already pre-packaged with each container.
Most well-known apps are already containerized by the Docker community and available through the Docker Store. Many of them even have example Docker compose files. [Read: Podman vs Docker]
What is Docker Compose?
Docker already makes installation of applications easier. But it gets even better. With Docker Compose, you can edit the compose file to set some configuration parameters (e.g.. download directory, seed ratio, etc.) and run the file and all your containerized apps can be configured and started with just one command.
But wait, there is more. Once you create your docker compose files, it becomes so much easier to migrate your apps, rebuild your servers, etc. I have moved my setup to many servers. All I have to do is install Ubuntu Server, copy over my Docker data folder (or Docker Root Folder as we will call it in this guide), edit my environmental variables, and start the stack from compose file.
All the 50 or so apps I have defined in my compose files are up in minutes and continue from where I left off in my previous server.
A Docker Compose file is like a template of all the apps in your docker stack with their basic configuration. You can even share your template with others, just like I am doing in this guide.
The scope of this post is to build a Docker-Compose media server. However, there are other methods to simplify installation of Docker containers (e.g. Portainer, Ansible, etc.). Explaining those methods is outside the scope of this post.
Requirements for Media Server Docker Stack
Having a Ubuntu or Debian system ready is a basic requirement of this guide. Windows and Mac users check out Docker Desktop.
At this point, the assumption is that you already have Ubuntu running and have prepared the operating system as detailed in the preparatory guide (Part 1 of this series).
You could install any flavor of Ubuntu or Debian. Previously, I have used Pop OS and Linux Mint with this setup, with no issues.
1. Install Docker and Docker Compose
First, we need to install Docker and Docker Compose. This has been covered in detail in my Ubuntu Docker installation guide.
But, the easiest way to install Docker and Docker Compose, irrespective of the OS, is using the official convenience script. It is as easy as running the following two commands:
curl -fsSL https://get.docker.com -o get-docker.sh sudo sh get-docker.sh
After Docker is installed, you may verify their installation and versions using the following commands:
2. Adding User to Docker Group
In part 1, we added a new user called "anand". Notice in the picture above that the user had to use sudo to see Docker and Docker Compose versions.
One way to get around using sudo with every Docker command is to add the user "anand" to "docker" group:
sudo adduser anand docker
In my first Docker home server guide, I suggested adding yourself or the user that will run the docker stack to the docker group to avoid having to add sudo for all docker commands.
It is still OK to do this in a Docker homelab environment if you implement other Docker best security practices.
However, I do not do this (I have gotten used to sudoing on demand) and so in this guide, I will include sudo in front of docker commands.
Plus, I recommend setting up Bash Aliases, which simplifies issuing long/complex commands. Here is a video guide on how I use bash aliases.
Setting Up the Docker Environment
The groundwork is done. We have the server, operating system, and Docker in place. Let us start laying the foundation for us to start building our perfect media server Docker stack.
1. Folders and Files
I have a specific folder structure I use for my setup; with everything I need to manage the server in one place. This is the base of all of my Docker guides and the GitHub repo.
So here it goes:
As you can see above, I have a docker folder in my home directory. This is the root Docker data folder. Let us call this DOCKER ROOT FOLDER. This will house all our docker-related folders and files:
- appdata - this folder will store the data for all our apps and services. Create this (and the folders listed below) using the following command:
mkdir /home/anand/docker/appdata
- compose - this folder will have a subfolder for each host, inside which all the individual Docker Compose files will be stored.
- logs - to centralize all relevant logs. I use this to store my script logs, traefik logs, etc. Although you can customize your apps (e.g. Nginx Proxy Manager) to store logs in this folder, we won't cover that in this guide. So, you can safely ignore it for this guide.
- scripts - to store all scripts. I use this folder to store my scripts for rClone, systemd, backup, etc. You can safely ignore it folder for this guide.
- secrets - to store credentials used by apps securely. See Docker secrets. Notice that the folder is owned by root and permissions are set to 600, using the following commands:
sudo chown root:root /home/anand/docker/secrets sudo chmod 600 /home/anand/docker/secrets
- shared - to store shared information. I save a lot of things in this folder that I share between 5 docker hosts (e.g. SSH config, .bash_aliases, etc.). For this guide, you can ignore it.
- .env - to store credentials used by apps securely as variable names. This way I won't have to use real values in docker-compose-udms.yml (for security). And, I can use the variable names in many places. Notice that the file is owned by root and permissions are set to 600. Create the empty file and set its permissions as shown below:
touch /home/anand/docker/.env sudo chown root:root /home/anand/docker/.env sudo chmod 600 /home/anand/docker/.env
- docker-compose-udms.yml - this is our template or configuration file for all our services. We will create this later in the guide. Note that my actual Docker Compose files (e.g. in the Github repo) have hostname as the suffix to separate them by host. We will call this file the Master Docker Compose File - you will see why later. Create this empty file using the following command:
touch /home/anand/docker/docker-compose-udms.yml
As in the description of each folder (and in bold font), you will need only a few from the above list for this basic docker media server guide.
But, if you start here and continue to follow my other guides, you will have to create the rest later on.
2. Docker Root Folder Permissions
Assuming that you have created the files and folders listed above, let us set the right permissions for them. We will need acl for this. If it is not installed, install it using:
sudo apt install acl
Next, set the permission for /home/anand/docker folder (anand being the username of the user) as follows:
sudo chmod 775 /home/anand/docker sudo setfacl -Rdm u:anand:rwx /home/anand/docker sudo setfacl -Rm u:anand:rwx /home/anand/docker sudo setfacl -Rdm g:docker:rwx /home/anand/docker sudo setfacl -Rm g:docker:rwx /home/anand/docker
The above commands provide access to the contents of the docker root folder (both existing and new stuff) to the docker group. Some may disagree with the liberal permissions above but again this is for home use and it is restrictive enough.
In my experience, this has addressed many permissions issues I have faced in the past due to containers not being able to access the contents of docker root folder.
3. Environmental Variables (.env) + Permissions
We are going to put some frequently used information in a common location and call it up as needed using variable names. This is what setting up environmental variables means in simple terms.
So, if you haven't already created, create and set restrictive permissions for .env file. The dot in front is not a typo, it hides the file in directory listings. From inside docker root folder:
touch .env sudo chown root:root .env sudo chmod 600 .env
From now on, to edit the .env file you will have to be either logged in as root or elevate your privileges by using sudo. Let us now open the file for editing:
sudo nano /home/anand/docker/.env
Add the following environmental variables to it:
PUID=1000 PGID=1000 TZ="Europe/Zurich" USERDIR="/home/anand" DOCKERDIR="/home/anand/docker" DATADIR="/media/storage" HOSTNAME="udms"
Replace/Configure:
- PUID and PGID - the user ID and group ID of the Linux user (anand), who we want to run the home server apps as. Both of these can be obtained using the id command as shown below.
As in the above picture, we are going to use 1000 for both PUID and PGID.
- TZ - the time zone that you want to set for your containers. Get your TZ from this timezone database.
- USERDIR - the path to the home folder of the current user (typically /home/USER).
- DOCKERDIR - the docker root folder that will house all persistentt data folders for docker apps. We created this in the steps above.
- DATADIR - the data folder that stores your media, downloads, and other stuff. This could be an external drive or a network folder. [Read: Install and configure NFS server on Ubuntu for serving files]
- HOSTNAME - is the name of your docker host. While it can be any name you choose, I recommend using the same as your Docker Host's hostname (can be found using the hostname command).
These would be hs, mds, ws, ds918, and dns in my Github Repo files. For this tutorial, let's set it to "UDMS" for ultimate Docker media server.
Save and exit nano (Ctrl X followed by Y and Enter).
These environmental variables will be referred to using $VARIABLE_NAME throughout the docker-compose file. Their values will be automatically pulled from the environment file that we created/edited above.
As we go through this guide, we will continue to add more environmental variables to the .env file. You will find an example .env in my GitHub repo.
That's it, the basic prep work to build our docker home server is done.
4. Create Docker Compose Folder
A big change from my past guide is: previously I had one Docker Compose file with all services defined inside. With the launch of Auto-Traefik and the step to align both of them, I changed my setup to break up all the services into individual YML files.
Therefore, let's create the following folder for our host "udms":
mkdir /home/anand/docker/compose/udms
This udms folder will host all our docker compose YML files.
5. Docker and Docker Compose Usage
In the past, we have listed a few good docker and docker compose commands to know. But I will expand on them here anyway.
Starting Containers using Docker Compose
This section is an intro to some of the commands you will use later in this guide. Running them at this point in the guide will throw errors. After adding compose options for each container (note that we have not added these yet), I recommend saving, exiting, and running the compose file using the following command to check if the container app starts correctly.
At the time of writing this guide, only Synology is on the older version of Docker and requires manual installation of Docker Compose and docker-compose command.
sudo docker compose -f /home/anand/docker/docker-compose-udms.yml up -d
Replace docker-compose-udms.yml with the actual name of the compose file (mine varies depending on the host).
The -d
option daemonizes it in the background. Without it, you will see real-time logs, which is another way of making sure no errors are thrown. Press Ctrl + C
to exit out of the real-time logs.
Also notice we are using sudo in front because we chose not to add the user (anand) to docker group.
See Docker Containers
At any time, you can check all the docker containers you have on your system (both running and stopped) using the following command:
sudo docker ps -a
As an example, here is a list of my containers for now. "STATUS" column shows whether a container is running (for how long) or exited. The last column shows the friendly name of the container.
Check Docker Container Logs
If you want to check the logs while the container starts you can use the following command:
sudo docker compose -f /home/anand/docker/docker-compose-udms.yml logs
Or,
sudo docker logs
In addition, you can also specify the name of the specific container at the end of the previous command if you want to see logs of a specific container. Here is a screenshot of the docker logs for my transmission-vpn
container that was generated using the following command:
sudo docker compose -f /home/anand/docker/docker-compose-udms.yml logs transmission-vpn
Finally, if you want to follow the logs in real-time (tailing), then use:
sudo docker compose -f /home/anand/docker/docker-compose-udms.yml logs -tf --tail="50" crowdsec
This will show you the last 50 lines of the log, while following it in real-time.
At any time, you can exit from the real-time logs screen by pressing Ctrl + C
.
Stopping / Restarting Containers using Docker Compose
To stop any running docker container, use the following command:
sudo docker compose -f /home/anand/docker/docker-compose-udms.yml stop CONTAINER-NAME
Replace CONTAINER-NAME
with the friendly name of the container. You can also replace stop
with restart
. To completely stop and remove containers, images, volumes, and networks (go back to how it was before running docker compose file), use the following command:
sudo docker compose -f /home/anand/docker/docker-compose-udms.yml down
Docker Cleanup
Remember, one of the biggest benefits of Docker is that it is extremely hard to mess up your host operating system. So, you can create and destroy containers at will. But over time leftover Docker images, containers, and volumes can take several GBs of space. So, at any time, you can run the following clean up scripts and re-run your docker-compose as described above.
sudo docker system prune sudo docker image prune sudo docker volume prune
These commands will remove any stray containers, volumes, and images that are not running or are not associated with any containers. Remember, even if you remove something that was needed you can always recreate it by just running the docker compose file.
Be the 1 in 200,000. Help us sustain what we do.You will gain benefits such as Deployarr access, discord roles, exclusive content, ad-free browsing, and more.🔥 Holiday Sale! 25% Off Platinum Membership$399.99$299.99 (ends December 31).Join the Geek Army (starting from just $1.67/month)
Building Docker Media Server
Finally, we are now ready to start building our media server with Docker. Let us look at docker-compose examples for a comprehensive autonomous media server setup.
Start the Docker Compose File
1. Define Docker Compose File Basics
Open the docker-compose-udms.yml
file:
nano /home/anand/docker/docker-compose-udms.yml
Add the following line at the top:
version: "3.9"
It basically says we are going to use Docker Compose file format 3.9.
2. Define Default Network
Next, we will add the network block right below the version line, as shown below:
########################### NETWORKS networks: default: driver: bridge
Note that ########################### NETWORKS is ignored by Docker compose. With # in the front, this line is basically a comment and a visual demarcation within our long docker-compose-udms.yml file.
We are only defining one network called "default".
Start Adding Docker Media Server Containers
As mentioned above, we will create individual YML files for each service. We will accomplish this using the include block in Docker Compose.
Right below the network block, let us start adding our services/containers. First, begin by adding the following lines, starting from include::
########################### NETWORKS networks: default: driver: bridge include: ########################### SERVICES # PREFIX udms = Ultimate Docker Media Server # HOSTNAME=udms - defined in .env
Notice all the comments. As you will see below, we are going to use udms as prefix for the individual compose files.
Core Services
I call these core services because they are kind of the most basic/core apps in my stack. Let us begin by adding the comment line (# CORE) below (don't ignore the 2 blank spaces in the front) under the include block.
########################### NETWORKS networks: default: driver: bridge include: ########################### SERVICES # PREFIX udms = Ultimate Docker Media Server # HOSTNAME=udms - defined in .env # CORE
There are many services that I call "Core", in my GitHub repo. In this Docker server tutorial, I am only going to show Socket Proxy and Portainer.
1. Socket Proxy - Secure Proxy for the Docker Socket
I did not include Socket Proxy in the previous version of this guide. Instead, it was explained as an improvement. In this version of the guide, I am going to make it "secure-by-design" as much as possible. This is also the approach I have taken with Auto-Traefik.
So, what is a Docker Socket Proxy? Any time you expose the Docker socket to a service, you are making it easier for the container to gain root access on the host system.
But some apps require access to Docker socket and API (eg. Traefik, Glances, Dozzle, Watchtower, etc.).
If Traefik gets compromised, then your host system could be compromised. Traefik's own documentation lists using a Socket Proxy as a solution.
A socket proxy is like a firewall for the docker socket/API. You can allow or deny access to certain API.
Define socket_proxy Network
First, let us define a separate network for Socket Proxy called socket_proxy. Only those services connected to this network will have access to the Docker Socket.
Add socket_proxy network right below the default network (while respecting the indentations), as shown below:
########################### NETWORKS networks: default: driver: bridge socket_proxy: name: socket_proxy driver: bridge ipam: config: - subnet: 192.168.91.0/24 include: ########################### SERVICES # PREFIX udms = Ultimate Docker Media Server # HOSTNAME=udms - defined in .env # CORE
All services behind socket_proxy network will use IP addresses between 192.168.91.1 and 192.168.91.254. You may customize the subnet if you prefer.
Create Socket Proxy Docker Compose
First, let's create the Socket Proxy Docker compose file. Head over to the compose folder in my Github Repository, and then into any of the host folders. Find the compose file for Socket Proxy and copy the contents.
Create a file called socket-proxy.yml inside /home/anand/docker/compose/udms. Copy-paste the contents into socket-proxy.yml compose file (pay attention to blank spaces at the beginning of each line).
services: # Docker Socket Proxy - Security Enchanced Proxy for Docker Socket socket-proxy: container_name: socket-proxy image: tecnativa/docker-socket-proxy security_opt: - no-new-privileges:true restart: unless-stopped # profiles: ["core", "all"] networks: socket_proxy: ipv4_address: 192.168.91.254 # You can specify a static IP privileged: true # true for VM. false for unprivileged LXC container on Proxmox. ports: - "127.0.0.1:2375:2375" # Do not expose this to the internet with port forwarding volumes: - "/var/run/docker.sock:/var/run/docker.sock" environment: - LOG_LEVEL=info # debug,info,notice,warning,err,crit,alert,emerg ## Variables match the URL prefix (i.e. AUTH blocks access to /auth/* parts of the API, etc.). # 0 to revoke access. # 1 to grant access. ## Granted by Default - EVENTS=1 - PING=1 - VERSION=1 ## Revoked by Default # Security critical - AUTH=0 - SECRETS=0 - POST=1 # Watchtower # Not always needed - BUILD=0 - COMMIT=0 - CONFIGS=0 - CONTAINERS=1 # Traefik, Portainer, etc. - DISTRIBUTION=0 - EXEC=0 - IMAGES=1 # Portainer - INFO=1 # Portainer - NETWORKS=1 # Portainer - NODES=0 - PLUGINS=0 - SERVICES=1 # Portainer - SESSION=0 - SWARM=0 - SYSTEM=0 - TASKS=1 # Portainer - VOLUMES=1 # Portainer
Here are some notes about the Socket Proxy Docker Compose:
- I have commented out profiles. If you copy-paste from my GitHub Repository, remember to do the same for all other apps. I use the Docker profiles for some automations. You do not need it when you start out.
- Unless you are running in an unprivileged Proxmox LXC container, privileged should be set to true.
- We are defining a static IP of 192.168.91.254 for the socket-proxy container, in the subnet we specified for socket_proxy network previously. Note the difference in container name (hyphen) and network name (underscore).
- Expose port 2375 only to the internal network (127.0.0.1:2375). The ports line instructs that port 2375 from inside the container (right side) is mapped to the port 2375 on the host machine (left side) only on the internal network interface (meaning other machines won't be able to connect to the port).Do not ever expose port 2375 to the internet. You will get hacked (here is an example). This is even more important for virtual private servers that typically expose all ports.
- In the environment: block we specify the Docker API section that we want to open up or close. I have added comments to describe which services require what API sections. For example, if you do not use WatchTower, you can enter 0 for several of the API sections.
- If you need to use a port other than 2375 on the host machine (may be its already occupied), you may customize the left side (to the colon) of the ports section (e.g. 127.0.0.1:2376:2375).
Port Availability
You may check if a port is occupied on the host machine using the following command:
This will show all ports on which some service is listening, as shown below:
Add Socket Proxy to the Docker Stack
We created the socket-proxy.yml file. Now we need to add it to our docker-compose-udms.yml (Master Docker Compose) file. To do so, add the path to the socket-proxy.yml file (compose/udms/socket-proxy.yml) under the include block, as shown below:
########################### NETWORKS networks: default: driver: bridge socket_proxy: name: socket_proxy driver: bridge ipam: config: - subnet: 192.168.91.0/24 include: ########################### SERVICES # PREFIX udms = Ultimate Docker Media Server # HOSTNAME=udms - defined in .env # CORE - compose/$HOSTNAME/socket-proxy.yml
Save the Master Docker Compose file. Reminder that $HOSTNAME here will be replaced with udms automatically (as defined in the .env file).
Starting and Testing Containers
After saving the docker-compose-udms.yml
file, run the following command to start the container and check if the app is accessible:
sudo docker compose -f /home/anand/docker/docker-compose-udms.yml up -d
If you prefer to use Docker Profiles, then "--profile profile_name" should be included with all Docker Compose commands (e.g. sudo docker compose --profile all -f /home/anand/docker/docker-compose-udms.yml up -d).
If Socket Proxy starts successfully, you should see the following:
Usually, I also like to check the logs to ensure there are no errors using the following command (or dclogs or dclogs socket-proxy aliases in my case):
sudo docker compose -f /home/anand/docker/docker-compose-udms.yml logs -tf --tail="50" socket-proxy
If everything looks OK in the logs, press Ctrl+C to exit.
docker-compose-udms.yml
file.Not Using Socket Proxy
In case you decide not to use Socket Proxy, the way services such as Portainer, Dozzle, Traefik, etc. access the Docker Socket will change.
Instead of adding the DOCKER_HOST environmental variable (to services that support it) as shown below:
DOCKER_HOST: tcp://socket-proxy:2375
you have to provide direct access to Docker socket under volumes section:
- /var/run/docker.sock:/var/run/docker.sock:ro
2. Portainer - WebUI for Containers
We have covered Portainer installation docker run command and docker-compose in detail. Portainer provides a WebUI to manage all your docker containers. I strongly recommend this for newbies.
It even allows several advanced admin tasks, including setting up stacks, managing containers, volumes, networks, etc.
Create Portainer Docker Compose
Let's create the Portainer Docker compose file. Head over to the compose folder in my Github Repository, and then into any of the host folders. Find the compose file for Portainer and copy the contents.
Create a file called portainer.yml inside /home/anand/docker/compose/udms. Copy-paste the contents into portainer.yml compose file (pay attention to blank spaces at the beginning of each line).
services: # Portainer - WebUI for Containers portainer: container_name: portainer image: portainer/portainer-ce:latest # Use portainer-ee if you have a Business Edition license key security_opt: - no-new-privileges:true restart: unless-stopped # profiles: ["core", "all"] networks: - socket_proxy # command: -H unix:///var/run/docker.sock # # Use Docker Socket Proxy instead for improved security command: -H tcp://socket-proxy:2375 ports: - "9000:9000" volumes: # - /var/run/docker.sock:/var/run/docker.sock:ro # # Use Docker Socket Proxy instead for improved security - $DOCKERDIR/appdata/portainer/data:/data # Change to local directory if you want to save/transfer config locally environment: - TZ=$TZ
Here are some notes about the Portainer Docker Compose:
- Once again, Docker profiles is commented out as explained previously.
- We are going to let Portainer access the Docker Socket via Socket Proxy. For this:
- We add Portainer to the socket_proxy network.
- We add command: -H tcp://socket-proxy:2375 so portainer knows to connect to socket-proxy host on port 2375, instead of directly connecting to the Docker Socket.
- With the above two done, we can comment out access to docker.sock under volumes section.
- The environmental variable $DOCKERDIR is already defined in our .env file.
- All Portainer data is being stored in a portainer-specific folder within appdata.
- You may check the availability of port 9000 on the Docker host as explained previously and if needed change port on the host machine (left side of the colon) to something else (e.g. - "9001:9000").
Add Portainer to the Docker Stack
We created the portainer.yml file. Now we need to add it to our docker-compose-udms.yml file. To do so, add the path to the portainer.yml (compose/udms/portainer.yml) file under the include block, as shown below:
########################### NETWORKS networks: default: driver: bridge socket_proxy: name: socket_proxy driver: bridge ipam: config: - subnet: 192.168.91.0/24 include: ########################### SERVICES # PREFIX udms = Ultimate Docker Media Server # HOSTNAME=udms - defined in .env # CORE - compose/$HOSTNAME/socket-proxy.yml - compose/$HOSTNAME/portainer.yml
Save the Master Docker Compose file. $HOSTNAME here will be replaced with udms automatically (as defined in the .env file).
Starting and Accessing Portainer
As explained previously, start the container and check the logs to make sure Portainer docker container is working fine before proceeding. If Portainer started properly, it can be accessed in the following ways:
- Using Docker Host IP: In this tutorial, it is http://192.168.1.100:9000, since we are connecting container's port 9000 to the host's port 9000.
Forwarding Ports
You may be tempted to port forward 9000 from your router to the Docker host. This would allow you to access Portainer from the internet, using your WAN/Public IP. DO NOT DO THIS. It is a security risk. A secure way to access services outside your internet network is to use a reverse proxy like Nginx Proxy Manager or Traefik, which will be covered later in this Docker server tutorial series.
3. Dozzle - Real-time Docker Log Viewer
Now that you understand the basic workflow of adding containers, let's simplify it a bit more for convenience using Dozzle.
Dozzle is a simple and responsive application that provides you with a web-based interface to monitor your Docker container logs live. It doesn’t store log information; it is for live monitoring of your container logs only.
I quickly became a big fan of it. Dozzle allows you to monitor logs of all your docker containers in real-time. This helps to troubleshoot and fix issues.
I have covered Dozzle logs viewer for real-time logs viewing.">Dozzle Docker Compose installation in a separate guide for beginners, if you are interested.
Create Dozzle Docker Compose
Let's create the Dozzle Docker compose file. Head over to the compose folder in my Github Repository, and then into any of the host folders. Find the compose file for Dozzle and copy the contents.
Create a file called dozzle.yml inside /home/anand/docker/compose/udms. Copy-paste the contents into dozzle.yml compose file (pay attention to blank spaces at the beginning of each line).
services: # Dozzle - Real-time Docker Log Viewer dozzle: image: amir20/dozzle:latest container_name: dozzle security_opt: - no-new-privileges:true restart: unless-stopped # profiles: ["apps", "all"] networks: - socket_proxy - default ports: - "8082:8080" environment: DOZZLE_LEVEL: info DOZZLE_TAILSIZE: 300 DOZZLE_FILTER: "status=running" # DOZZLE_FILTER: "label=log_me" # limits logs displayed to containers with this label DOCKER_HOST: tcp://socket-proxy:2375 # volumes: # - /var/run/docker.sock:/var/run/docker.sock # Use Docker Socket Proxy instead for improved security
Here are some notes about the Dozzle Docker Compose:
- Third Reminder: Docker profiles is commented out as explained previously.
- We are going to let Dozzle access the Docker Socket via Socket Proxy. For this, again:
- We add Dozzle to the socket_proxy network.
- We specify a custom Docker Host using the environment variable DOCKER_HOST: tcp://socket-proxy:2375.
- With the above two done, we can comment out access to docker.sock under volumes section.
- We are also adding Dozzle to default network so it can see containers on that network.
- You may check the availability of port 8080 on the Docker host as explained previously. Mine was occupied so I changed it to 8082 on the Docker Host side.
There are a few more things you can customize in the environment variables, to your liking.
Add Dozzle to the Docker Stack
We created the dozzle.yml file. Now we need to add it to our docker-compose-udms.yml file. To do so, add the path to the dozzle.yml (compose/udms/dozzle.yml) file under the include block, as shown below:
########################### NETWORKS networks: default: driver: bridge socket_proxy: name: socket_proxy driver: bridge ipam: config: - subnet: 192.168.91.0/24 include: ########################### SERVICES # PREFIX udms = Ultimate Docker Media Server # HOSTNAME=udms - defined in .env # CORE - compose/$HOSTNAME/socket-proxy.yml - compose/$HOSTNAME/portainer.yml - compose/$HOSTNAME/dozzle.yml
Save the Master Docker Compose file.
Starting and Accessing Dozzle
As explained previously, start the container and check the logs to make sure Dozzle docker container is working fine before proceeding. If Dozzle started properly, it can be accessed in the following ways:
- Using Docker Host IP: In this tutorial, it is http://192.168.1.100:8082, since we are connecting container's port 8080 to the host's port 8082.
Easier Log Viewing
Moving forward, you can leave the Dozzle window open while you add more services. As and when you add more containers, you can refresh the Dozzle window and follow the logs on Dozzle instead of command line interface.
4. Homepage - Application Dashboard
We are going to be adding a bunch of apps to our Docker server, which are only available on the internal network using http://DOCKER-HOST-IP:PORT. But what if you do not remember or want to remember all the port numbers?
This is where an application dashboard such as Homepage can come in handy. I started with Organizer, then moved to Heimdall. I tried a few others such as Dashy, Homarr, and Flame. To me, Homepage seems simple and easy to use.
Create Homepage Docker Compose
Let's create the Homepage Docker compose file. Head over to the compose folder in my Github Repository, and then into any of the host folders. Find the compose file for Homepage and copy the contents.
Create a file called homepage.yml inside /home/anand/docker/compose/udms. Copy-paste the contents into homepage.yml compose file (pay attention to blank spaces at the beginning of each line).
services: # Homepage - Application Dashboard homepage: image: ghcr.io/gethomepage/homepage:latest container_name: homepage security_opt: - no-new-privileges:true restart: unless-stopped # profiles: ["apps", "all"] networks: - socket_proxy - default ports: - "3000:3000" volumes: - $DOCKERDIR/appdata/homepage:/app/config environment: TZ: $TZ PUID: $PUID PGID: $PGID
Here are some notes about the Homepage Docker Compose:
- Fourth Reminder: Docker profiles is commented out as explained previously.
- We are adding Homepage to socket_proxy network so it can interact with the Docker socket to read information on Docker containers. We are also making it part of the default network. Essentially, Homepage should be part of the networks of all the apps that you want it to interact with and get metrics. But if you are just adding a shortcut link to the app, then adding those networks is not needed.
- You may check the availability of port 3000 on the Docker host as explained previously. Mine was free on the Docker Host side.
Add Homepage to the Docker Stack
We created the homepage.yml file. Now we need to add it to our docker-compose-udms.yml file. To do so, add the path to the homepage.yml (compose/udms/homepage.yml) file under the include block, as shown below:
########################### NETWORKS networks: default: driver: bridge socket_proxy: name: socket_proxy driver: bridge ipam: config: - subnet: 192.168.91.0/24 include: ########################### SERVICES # PREFIX udms = Ultimate Docker Media Server # HOSTNAME=udms - defined in .env # CORE - compose/$HOSTNAME/socket-proxy.yml - compose/$HOSTNAME/portainer.yml - compose/$HOSTNAME/dozzle.yml - compose/$HOSTNAME/homepage.yml
Save the Master Docker Compose file.
Starting and Accessing Homepage
As explained previously, start the container and check the logs to make sure Homepage docker container is working fine before proceeding. If Homepage started properly, it can be accessed in the following ways:
- Using Docker Host IP: In this tutorial, it is http://192.168.1.100:3000, since we are mapping container's port 3000 to the host's port 3000.
Media Server Apps
This section is why you are probably here. There are several media server, live TV server, and music server options.
In my opinion, the best media server for videos are Plex and Jellyfin, depending on your needs. For music, I use Airsonic-Advanced due to its one main unique feature: being able to provide folder level access to users.
Let add the comment line (# MEDIA) to the master Docker Compose file (don't ignore the 2 blank spaces in the front) under the include block.
########################### NETWORKS networks: default: driver: bridge socket_proxy: name: socket_proxy driver: bridge ipam: config: - subnet: 192.168.91.0/24 include: ########################### SERVICES # PREFIX udms = Ultimate Docker Media Server # HOSTNAME=udms - defined in .env # CORE - compose/$HOSTNAME/socket-proxy.yml - compose/$HOSTNAME/portainer.yml - compose/$HOSTNAME/dozzle.yml - compose/$HOSTNAME/homepage.yml # MEDIA
There are many services that I call "Media", in my GitHub repo. In this Docker server tutorial, I am only going to show Plex and Jellyfin.
5. Plex - Media Server
Let's see how to add Plex Media Server to our Docker Stack. Here, I am going to introduce Docker Secrets.
Plex media server is a free media server that can stream local and internet content to you several of your devices. It has a server component that catalogs your media (movies, tv shows, photos, videos, music, etc.). [Read: 10 Best Media Server for Plex + one SURPRISING bonus [2022]]
To stream, you need the client app installed on compatible Plex client devices. This can cost some money.
With free movies, TV, curated content, and Plexamp music, lifetime Plex Pass is a great value.
Best Plex Client Devices:
- NVIDIA SHIELD TV Pro Home Media Server - $199.99
- Amazon Fire TV Streaming Media Player - $89.99
- Roku Premiere+ 4K UHD - $83.99
- CanaKit Raspberry Pi 3 Complete Starter Kit - $69.99
- Xbox One 500 GB Console - $264.99
I have covered Plex docker setup with Docker Compose, separately if you are interested.
Create Additional Environment Variables for Plex
First, we need to add some environment variables. Edit your .env file using:
sudo nano /home/anand/docker/.env
Add the SERVER_IP and LOCAL_IPS variables at the end, as shown below:
PUID=1000 PGID=1000 TZ="Europe/Zurich" USERDIR="/home/anand" DOCKERDIR="/home/anand/docker" DATADIR="/media/storage" HOSTNAME="udms" SERVER_IP=192.168.1.100 LOCAL_IPS=127.0.0.1/32,10.0.0.0/8,192.168.0.0/16,172.16.0.0/12
You may leave the LOCAL_IPS typically. However, be sure to change the SERVER_IP to your Docker host's LAN IP address.
Create Plex Claim Docker Secret
In addition to the environmental variables defined above, you will also have to define PLEX_CLAIM, which is your Plex claim token.
Let's take this opportunity to learn about Docker Secrets. After you have obtained your claim token, add the following secrets section to docker-compose-udms.yml, right between the networks and include blocks:
########################### SECRETS secrets: plex_claim: file: $DOCKERDIR/secrets/plex_claim
So far, the master Docker Compose file should look like this;
########################### NETWORKS networks: default: driver: bridge socket_proxy: name: socket_proxy driver: bridge ipam: config: - subnet: 192.168.91.0/24 ########################### SECRETS secrets: plex_claim: file: $DOCKERDIR/secrets/plex_claim include: ########################### SERVICES # PREFIX udms = Ultimate Docker Media Server # HOSTNAME=udms - defined in .env # CORE - compose/$HOSTNAME/socket-proxy.yml - compose/$HOSTNAME/portainer.yml - compose/$HOSTNAME/dozzle.yml - compose/$HOSTNAME/homepage.yml # MEDIA
You have defined a secret with the name plex_claim, which refers to the file /home/anand/docker/secrets/plex_claim. But this file doesn't exist. So, let's create it.
sudo nano /home/anand/docker/secrets/plex_claim
Note that you have to use sudo as the file is inside a secrets folder that is owned by the user root. What goes inside the file is only your Plex Claim token, like so:
Press Ctrl X, Y, and Enter to save and exit.
Create Plex Docker Compose
Let's create the Plex Docker compose file. But first, add the following to your .env file:
LOCAL_IPS=127.0.0.1/32,10.0.0.0/8,192.168.0.0/16,172.16.0.0/12
We will use the LOCAL_IPS variable to let Plex know to treat the above IP ranges as local network.
Next, head over to the compose folder in my GitHub Repository, and then into any of the host folders. Find the compose file for Plex and copy the contents.
Create a file called plex.yml inside /home/anand/docker/compose/udms. Copy-paste the contents into plex.yml compose file (pay attention to blank spaces at the beginning of each line).
services: # Plex - Media Server plex: image: plexinc/pms-docker:plexpass container_name: plex networks: - default security_opt: - no-new-privileges:true restart: "no" # profiles: ["media", "all"] devices: - /dev/dri:/dev/dri # for hardware transcoding ports: - "32400:32400/tcp" - "3005:3005/tcp" - "8324:8324/tcp" - "32469:32469/tcp" - "1900:1900/udp" - "32410:32410/udp" - "32412:32412/udp" - "32413:32413/udp" - "32414:32414/udp" volumes: - $DOCKERDIR/appdata/plex:/config - $DATADIR/data/media:/data/media1 # Media Folder 1 - $DATADIR/data2/media:/data/media2 # Media Folder 2 - /dev/shm:/data/transcode # Offload transcoding to RAM if you have enough RAM environment: TZ: $TZ HOSTNAME: "myPlex" PLEX_CLAIM_FILE: /run/secrets/plex_claim # Not required initially PLEX_UID: $PUID PLEX_GID: $PGID ADVERTISE_IP: "http://$SERVER_IP:32400/" ALLOWED_NETWORKS: $LOCAL_IPS secrets: - plex_claim
Here are some notes about the Plex Docker Compose:
- Fourth Reminder: Docker profiles is commented out as explained previously.
- We are going to put Plex on the default network.
- Notice that I set the restart policy to "no" for Plex. You may change it to "unless-stopped" as with other containers in this guide.Why did I set Plex to not auto start? Because I want to ensure that my media folders are mounted before Plex (or other media servers) starts. I wrote a custom script for this, which you may find in the scripts folder of my GitHub repo. This script runs at boot time, waits for media folders to be mounted, and then starts all the media folder dependent containers.
This is where I use Docker Profiles.
- If your docker host has a graphics card (you will see /dev/dri) that you can use for hardware accelerated transcoding then you can pass this on. We do this using the devices. This is especially useful for NASes that support Plex (e.g. Synology).
- I am specifying two media folders, but you may have only one to pass on to Plex container.
- strong>/dev/shm is the RAM memory, which we are passing for faster transcoding purposes. If you are short on memory, you could comment this out.
- Previously we defined the Docker Secret for Plex Claim Token. Skip this step if you are setting up a brand new Plex media server. To use it inside the container, we have to do two things:
- Define a secrets section in the Docker Compose for Plex and list the name of the secret (plex_claim).
- Set the environment variable PLEX_CLAIM_FILE to /run/secrets/plex_claim.
- We are also specifying the advertise IP using the environment variable $SERVER_IP. If you have multiple IPs for the server (e.g. multiple network interfaces), you can list them here separated by comma.
- We are also passing all the typical LAN subnets as $LOCAL_IPS. This helps Plex to treat all these networks as local. This is also helpful when you want to disable authentication on local network.
- Plex uses lots of ports. You may check the availability of ports on the Docker host as explained previously and make changes if necessary.
Creating Additional Docker Secrets
As shown above, the typical workflow to add more secrets involves:
- Create the secret file inside /home/user/docker/secrets folder, owned by root:root and with 600 permissions.
- Define it globally in the master docker compose file
- Call it in the secrets section of the service
- Set relevant environment variable to /run/secrets/secret_name
Keep in mind that the Docker Image must support secrets feature for Docker Secrets to work. Not all of them do.
Add Plex to the Docker Stack
We created the plex.yml file. Now we need to add it to our docker-compose-udms.yml file. To do so, add the path to the plex.yml (compose/udms/plex.yml) file under the include block, as shown below:
########################### NETWORKS networks: default: driver: bridge socket_proxy: name: socket_proxy driver: bridge ipam: config: - subnet: 192.168.91.0/24 ########################### SECRETS secrets: plex_claim: file: $DOCKERDIR/secrets/plex_claim include: ########################### SERVICES # PREFIX udms = Ultimate Docker Media Server # HOSTNAME=udms - defined in .env # CORE - compose/$HOSTNAME/socket-proxy.yml - compose/$HOSTNAME/portainer.yml - compose/$HOSTNAME/dozzle.yml - compose/$HOSTNAME/homepage.yml # MEDIA - compose/$HOSTNAME/plex.yml
Save the Master Docker Compose file.
Starting and Accessing Plex
As explained previously, start the container and check the logs to make sure Plex docker container is working fine before proceeding. If Plex started properly, it can be accessed in the following ways:
- Using Docker Host IP: In this tutorial, it is http://192.168.1.100:32400/web/, since we are connecting container's port 32400 to the host's port 32400.
The first time you try to access a new Plex server you will either have to be on the local Plex server machine and use http://localhost:32400/web or use the IP address of the Plex server (e.g. http://192.168.1.100:32400/web). You will not be able to access the Plex server externally.
Adding Additional Apps to the Docker Stack
With the above 4 apps (Socket Proxy, Portainer, Dozzle, and Plex), I have covered all the basics of creating Docker Server Stacks. So, I hope you did not skip those sections.
Moving forward, the concepts and workflow are the same. So, I will keep the explanations to a minimum and just list the steps for a few more apps as examples.
6. Jellyfin - Free and Awesome Media Server
Jellyfin is the Free Software Media System that puts you in control of your media, allowing you to access, manage and stream from home or away. Jellyfin is the first choice for many because it is an alternative to the proprietary Plex and Emby home media servers.
Jellyfin also supports several client devices and can be supercharged with several awesome plugins.
Jellyfin was originally forked from Emby 3.5, but eventually became its own application quite different from Emby. Jellyfin does not require a premium license or features for the same remote access for Plex and Emby. Though you may have to be a bit tech-savvy to use Jellyfin, those who do often never return to Plex or Emby again.
Jellyfin Docker installation be found in my Jellyfin Docker Compose guide or GitHub repo.
Jellyfin Docker Compose
Create Jellyfin Docker Compose File (/home/anand/docker/compose/udms/jellfin.yml), with the following contents:
services: # Jellyfin - Media Server jellyfin: image: jellyfin/jellyfin:latest container_name: jellyfin networks: - default security_opt: - no-new-privileges:true restart: "no" # profiles: ["media", "all"] user: $PUID:$PGID devices: - /dev/dri:/dev/dri # for harware transcoding ports: - "8096:8096" # - "8920:8920" # Emby also uses same port if running both environment: UMASK_SET: 022 TZ: $TZ volumes: - $DOCKERDIR/appdata/jellyfin:/config - $DATADIR/data/media:/data/media1 - $DATADIR/data2/media:/data/media2 - $EXTDIR/ssd/home-server/downloads:/data/downloads - /dev/shm:/data/transcode # Offload transcoding to RAM if you have enough RAM
Add Jellfin to the Docker Stack
Add Jellyfin to the master docker-compose-udms.yml:
... # MEDIA - compose/$HOSTNAME/plex.yml - compose/$HOSTNAME/jellyfin.yml
Start and test the Jellyfin container. Jellyfin should be available at http://192.168.1.100:8096. Do not forward port 8096 on the router to the Jellyfin server. Instead use more secure alternatives.
7. SABnzbd - Binary newsgrabber (NZB downloader)
SABnzbd is an Open-Source Binary Newsreader that makes Usenet easy through automation. When you add an .nzb
file, SABnzbd automatically verifies, repairs, extracts, and archives your download. Sonarr, Radarr, and Lidarr integrate with SABnzbd easily. [Read: Complete Usenet Guide: What is Usenet, Usenet vs torrents, Downloading Files]
SABnzbd can also handle RSS feeds. An easy setup wizard and self-analysis tools can be used to verify your setup.
You can manage SABnzbd through a web browser from just about any PC, smartphone, or Android device through the Glitter interface. Alternatively, you can use apps like LunaSea for Android and iOS, nzb360 for Android, and nzbUnity on the iPhone and iPad. SABnzbd runs on Windows, macOS, Unix, and NAS devices.
SABnzbd Docker-Compose can be found in my GitHub repo.
SABnzbd Docker Compose
Create SABnzbd Docker Compose File (/home/anand/docker/compose/udms/sabnzbd.yml), with the following contents:
services: # SABnzbd - Binary newsgrabber (NZB downloader) sabnzbd: image: lscr.io/linuxserver/sabnzbd:latest container_name: sabnzbd security_opt: - no-new-privileges:true restart: unless-stopped # profiles: ["apps", "downloads", "all"] networks: - default ports: - "8084:8080" volumes: - $DOCKERDIR/appdata/sabnzbd:/config - $DATADIR/downloads:/data/downloads environment: PUID: $PUID PGID: $PGID TZ: $TZ UMASK_SET: 002
Add SABnzbd to the Docker Stack
Add SABnzbd to the master docker-compose-udms.yml:
... - compose/$HOSTNAME/jellyfin.yml # DOWNLOADERS - compose/$HOSTNAME/sabnzbd.yml
Start and test the SABnzbd container. SABnzbd should be available at http://192.168.1.100:8084. Do not forward port 8084 on the router to the SABnzbd host server. Instead use more secure alternatives.
Show your support for FREE. Be the 1 in 200,000.Get your VPN and USENET through our affiliate links (exclusive discounts):
- SurfShark VPN: 82% Off + 4 Months FREE, Wireguard support.
- NewsHosting Usenet: 58% Off, 5000+ Retention Days, Unlimited, VPN included.
- Eweka Usenet: Lifetime Discount: EU and US Servers, Unlimited.
8. qBittorrent - Torrent downloader
qBittorrent is one of the best torrent clients you can use. It is fast, easy, and free to use. You can install it as a native macOS GUI app, or on Windows, Linux, and BSD. We have covered qBittorrent installation on Linux previously. But Docker is so much easier.
qBittorrent uses very little memory with light overhead, which makes it well-suited for home media servers. You can configure watch directories, bad peer blocklists, UPnP and NAT-PMP port forwarding, webseed support, tracker editing, global and per-torrent speed limits, and more. With an easy-to-use web interface, encryption, and peer exchange, qBittorrent opens magnetic links and handles DHT and µTP.
Along with Prowlarr, torrent websites, or private torrent trackers, you will have yourself a content aggregation machine.
qBittorrent Docker-Compose can be found in my GitHub repo.
qBittorrent Docker Compose
Create qBittorrent Docker Compose File (/home/anand/docker/compose/udms/qbittorrent.yml), with the following contents:
services: # qBittorrent - Torrent downloader qbittorrent: image: lscr.io/linuxserver/qbittorrent:latest container_name: qbittorrent security_opt: - no-new-privileges:true restart: unless-stopped # profiles: ["apps", "downloads", "all"] networks: - default ports: - "8081:8080" volumes: - $DOCKERDIR/appdata/qbittorrent:/config - $DATADIR/downloads:/data/downloads # Ensure that downloads folder is set to /data/downloads in qBittorrent environment: TZ: $TZ PUID: $PUID PGID: $PGID UMASK_SET: 002
Add qBittorrent to the Docker Stack
Add qBittorrent to the master docker-compose-udms.yml:
... # DOWNLOADERS - compose/$HOSTNAME/sabnzbd.yml - compose/$HOSTNAME/qbittorrent.yml
Start and test the qBittorrent container. qBittorrent should be available at http://192.168.1.100:8081. Do not forward port 8081 on the router to the qBittorrent host server. Instead use more secure alternatives.
9. Radarr - Movie management
Radarr is a Movie PVR. You add the movies you want to see to Radarr and it will search various BitTorrent and Usenet providers for the movie. If it is available, Radarr will grab the index file and send it to your BitTorrent client or NZB client for downloading.
Once the download is complete it can rename your movie to a specified format and move it to a folder of your choice (movie library). It can even update your Plex library or notify you when a new movie is ready for you to watch. [Read: CouchPotato vs SickBeard, SickRage, or Sonarr for beginners]
Radarr Docker-Compose can be found in my GitHub repo.
Radarr Docker Compose
Create Radarr Docker Compose File (/home/anand/docker/compose/udms/radarr.yml), with the following contents:
services: # Radarr - Movie management radarr: image: lscr.io/linuxserver/radarr:latest container_name: radarr security_opt: - no-new-privileges:true restart: "no" # profiles: ["media", "arrs", "all"] networks: - default ports: - "7878:7878" volumes: - $DOCKERDIR/appdata/radarr:/config - $DATADIR/data/media/movies:/data/movies1 - $DATADIR/data2/media/movies:/data/movies2 - $DATADIR/downloads:/data/downloads - "/etc/localtime:/etc/localtime:ro" environment: TZ: $TZ PUID: $PUID PGID: $PGID
Add Radarr to the Docker Stack
Add Radarr to the master docker-compose-udms.yml:
... - compose/$HOSTNAME/qbittorrent.yml # PVRS - compose/$HOSTNAME/radarr.yml
Start and test the Radarr container. Radarr should be available at http://192.168.1.100:7878. Do not forward port 7878 on the router to the Radarr host server. Instead use more secure alternatives.
Referring to an App Inside Another App
Say you want to add SABnzbd as a download client inside Radarr, you can provide the host URL in two ways:
- Since both SABnzbd and Radarr belong to the same network (default), SABnzbd host URL can be http://sabnzbd:8080. Notice two things here:
- sabzbnd is the name of the SABnzbd service we provided in the sabnzbd.yml docker compose file (sabnzbd:).
- Since we are referring to SABnzbd using the hostname, we have to use the port number inside the container, which is 8080.
- You can also call SABnzbd inside Radarr, using the URL http://192.168.1.100:8084 (the port we mapped 8080 to on the host side). If both SABnzbd and Radarr were on different docker networks, then this is the method that will work.
10. Sonarr - TV Shows Management
Sonarr (formerly NzbDrone) is a PVR for TV Shows. You add the shows you want to see to Sonarr and it will search various BitTorrent and Usenet providers for the show episodes. If it is available, Sonarr will grab the index file and send it to your BitTorrent client or NZB client for downloading.
Once the download is complete it can rename your episode to a specified format and move it to a folder of your choice (TV Show library). It can even update your Plex library or notify you when a new episode is ready for you to watch.
Sonarr Docker-Compose can be found in my GitHub repo.
Sonarr Docker Compose
Create Sonarr Docker Compose File (/home/anand/docker/compose/udms/sonarr.yml), with the following contents:
services: # Sonarr - TV Shows Management # Set url_base in sonarr settings if using PathPrefix sonarr: image: lscr.io/linuxserver/sonarr:develop container_name: sonarr security_opt: - no-new-privileges:true restart: "no" # profiles: ["media", "arrs", "all"] networks: - default ports: - "8989:8989" volumes: - $DOCKERDIR/appdata/sonar:/config - $DATADIR/data/media/shows:/data/shows1 - $DATADIR/data2/media/shows:/data/shows2 - $DATADIR/downloads:/data/downloads - "/etc/localtime:/etc/localtime:ro" environment: TZ: $TZ PUID: $PUID PGID: $PGID
Add Sonarr to the Docker Stack
Add Sonarr to the master docker-compose-udms.yml:
... # PVRS - compose/$HOSTNAME/radarr.yml - compose/$HOSTNAME/sonarr.yml
Start and test the Sonarr container. Sonarr should be available at http://192.168.1.100:8989. Do not forward port 8989 on the router to the Sonarr host server. Instead use more secure alternatives.
11. Bazarr - Subtitle Management
Bazarr is a companion application to Sonarr and Radarr that manages and downloads subtitles. You can search automatically for missing subtitles and download them as they become available. You can find all matching subtitles, choose one, and download it directly to your media library.
Bazarr Docker-Compose can be found in my GitHub repo.
Bazarr Docker Compose
Create Bazarr Docker Compose File (/home/anand/docker/compose/udms/bazarr.yml), with the following contents:
services: # Bazarr - Subtitle Management bazarr: image: lscr.io/linuxserver/bazarr container_name: bazarr security_opt: - no-new-privileges:true restart: "no" # profiles: ["media", "arrs", "all"] networks: - default ports: - "6767:6767" volumes: - $DOCKERDIR/appdata/bazarr:/config - $DATADIR/media:/data/media environment: TZ: $TZ PUID: $PUID PGID: $PGID
Add Bazarr to the Docker Stack
Add Bazarr to the master docker-compose-udms.yml:
... - compose/$HOSTNAME/sonarr.yml # COMPLEMENTARY APPS - compose/$HOSTNAME/bazarr.yml
Start and test the Bazarr container. Bazarr should be available at http://192.168.1.100:6767. Do not forward port 6767 on the router to the Bazarr host server. Instead use more secure alternatives.
12. Tautulli - Plex Statistics and Monitoring
Tautulli is a third-party app for monitoring and tracking Plex Media Server. You can see what has been watched, who watched it, when they watched it, and what device it was watched on. All statistics are displayed in a well-designed interface made of tables and graphs for easy viewing.
With Tautulli Remote you can view activity and stats away from home. Intuitive graphs allow you to view streaming trends. You can create newsletters for recently added media. You can see the watch history for all users. Tuatulli also allows you to view statistics and detailed media info about your Plex library and delete synced content.
Tautulli Docker-Compose can be found in my GitHub repo.
Tautulli Docker Compose
Create Tautulli Docker Compose File (/home/anand/docker/compose/udms/tautulli.yml), with the following contents:
services: # Tautulli - Plex statistics and monitoring tautulli: image: lscr.io/linuxserver/tautulli:latest container_name: tautulli security_opt: - no-new-privileges:true restart: "no" # profiles: ["media", "all"] networks: - default ports: - "8181:8181" volumes: - $DOCKERDIR/appdata/tautulli/config:/config - $DOCKERDIR/appdata/plex/Library/Application Support/Plex Media Server/Logs:/logs:ro # For Tautulli Plex log viewer environment: TZ: $TZ PUID: $PUID PGID: $PGID
Add Tautulli to the Docker Stack
Add Tautulli to the master docker-compose-udms.yml:
... # COMPLEMENTARY APPS - compose/$HOSTNAME/bazarr.yml - compose/$HOSTNAME/tautulli.yml
Start and test the Tautulli container. Tautulli should be available at http://192.168.1.100:8181. Do not forward port 8181 on the router to the Tautulli host server. Instead use more secure alternatives.
13. Uptime Kuma - Status Page & Monitoring Server
Uptime Kuma is an open-source monitoring tool for services over HTTP, TCP, DNS, and other protocols. Uptime Kuma will send you notification alerts of downtime. You can also create custom status pages for users.
Combined with Prometheus, Uptime Kuma can display metrics about each monitoring target. There are pre-made Grafana dashboards available to pull the metrics from Prometheus and display them as nice monitoring dashboards.
Uptime Kuma Docker-Compose can be found in my GitHub repo.
Uptime-Kuma Docker Compose
Create Uptime-Kuma Docker Compose File (/home/anand/docker/compose/udms/uptime-kuma.yml), with the following contents:
services: # Uptime Kuma - Status Page & Monitoring Server uptime-kuma: image: louislam/uptime-kuma container_name: uptime-kuma security_opt: - no-new-privileges:true restart: unless-stopped # profiles: ["core", "all"] networks: - default volumes: - $DOCKERDIR/appdata/uptime-kuma:/app/data
Add Uptime-Kuma to the Docker Stack
Add Uptime-Kuma to the master docker-compose-udms.yml:
... - compose/$HOSTNAME/tautulli.yml # MONITORING - compose/$HOSTNAME/uptime-kuma.yml
Start and test the Uptime-Kuma container. Uptime-Kuma should be available at http://192.168.1.100:8182. Do not forward port 8182 on the router to the Uptime-Kuma host server. Instead use more secure alternatives.
14. MariaDB - MySQL Database
You may have never used SQL commands, but you probably have heard of MySQL because of its many uses. MySQL Community Edition is free but MySQL itself is a commercial application. The developers of MySQL developed MariaDB as an option that would always be free and open-source.
MariaDB Server is also developed to maintain compatibility with MySQL, allowing developers to use the same open-source code libraries for MariaDB as MySQL. Some names you might recognize, such as Wikipedia, DBS Bank, and ServiceNow use MariaDB, you can now recognize them as using MySQL.
MariaDB is a popular choice for self-hosting enthusiasts as a database for Home Assistant, Authelia, WordPress, Airsonic Advanced, and more.
MariaDB Docker-Compose can be found in my GitHub repo.
MariaDB Docker Compose
Create MariaDB Docker Compose File (/home/anand/docker/compose/udms/mariadb.yml), with the following contents:
services: # MariaDB - MySQL Database # After starting container for first time dexec and mysqladmin -u root password <password> mariadb: container_name: mariadb image: lscr.io/linuxserver/mariadb networks: - default security_opt: - no-new-privileges:true restart: unless-stopped # profiles: ["dbs", "all"] ports: - "3306:3306" volumes: - $DOCKERDIR/appdata/mariadb/data:/config environment: TZ: $TZ PUID: $PUID PGID: $PGID FILE__MYSQL_ROOT_PASSWORD: /run/secrets/mysql_root_password # Note FILE__ (double underscore) secrets: - mysql_root_password
Note that MariaDB root password is specified as a Docker Secret. In addition, FILE__MYSQL_ROOT_PASSWORD has a double underscore.
Add MariaDB to the Docker Stack
Add MariaDB to the master docker-compose-udms.yml:
... - compose/$HOSTNAME/uptime-kuma.yml # DATABASES - compose/$HOSTNAME/mariadb.yml
Start and test the MariaDB container.
15. File Browser - File Explorer
File Browser provides a web UI for managing files for multiple users in multiple directories. Each user can upload, delete, preview, rename and edit files in a specified directory. It is one the cleanest and easiest file managers for your home server.
All you must do is simply install File Browser on a server and point it to a directory path to access files through the web interface.
File Browser Docker-Compose can be found in my GitHub repo.
File Browser Docker Compose
Create File Browser Docker Compose File (/home/anand/docker/compose/udms/filebrowser.yml), with the following contents:
services: # File Browser - Explorer filebrowser: image: filebrowser/filebrowser:s6 container_name: filebrowser security_opt: - no-new-privileges:true restart: unless-stopped # profiles: ["apps", "all"] networks: - default ports: - "81:80" volumes: - $DOCKERDIR/appdata/filebrowser:/config - $USERDIR:/data/home environment: TZ: $TZ PUID: $PUID PGID: $PGID
Add File Browser to the Docker Stack
Add File Browser to the master docker-compose-udms.yml:
... - compose/$HOSTNAME/tautulli.yml # UTILITIES - compose/$HOSTNAME/filebrowser.yml
Start and test the File Browser container. File Browser should be available at http://192.168.1.100:81. Do not forward port 81 on the router to the File Browser host server. Instead use more secure alternatives.
16. Docker Garbage Collector
After a while of tinkering with docker containers, stored orphan images and volumes can take up several gigabytes of space. While you can use docker system prune
, Docker garbage collector automates the job for you. A good Docker garbage collection example can be found in the recently posted article Docker Media Server guide.
The best Docker images for garbage collection are docker-gc
and docker-gc-cron
.
If you are considering docker-gc
, one alternative you will immediately notice is spotify/docker-gc
. However, spotify/docker-gc
is no longer being developed. Using docker-gc-cron
is a great alternative because it is actively developed and, in fact, uses spotify/docker-gc
under the hood.
Docker-GC Docker Compose
Create Docker-GC Docker Compose File (/home/anand/docker/compose/udms/docker-gc.yml), with the following contents:
services: # Docker-GC - Automatic Docker Garbage Collection # Create docker-gc-exclude file docker-gc: image: clockworksoul/docker-gc-cron:latest container_name: docker-gc security_opt: - no-new-privileges:true restart: unless-stopped # profiles: ["apps", "all"] networks: - socket_proxy volumes: # - /var/run/docker.sock:/var/run/docker.sock # Use Docker Socket Proxy instead for improved security - $DOCKERDIR/appdata/docker-gc/docker-gc-exclude:/etc/docker-gc-exclude environment: CRON: 0 0 0 * * ? # Every day at midnight FORCE_IMAGE_REMOVAL: 1 FORCE_CONTAINER_REMOVAL: 0 GRACE_PERIOD_SECONDS: 604800 DRY_RUN: 0 CLEAN_UP_VOLUMES: 1 TZ: $TZ DOCKER_HOST: tcp://socket-proxy:2375
The cron is set to clean the unused docker data every day at midnight.
Add Docker-GC to the Docker Stack
Add Docker-GC to the master docker-compose-udms.yml:
... - compose/$HOSTNAME/filebrowser.yml # MAINTENANCE - compose/$HOSTNAME/docker-gc.yml
Start and test the Docker-GC container. There is no web interface for Docker Garbage Collector.
The Final Docker Compose File
We have added about 15 apps to our Docker Media Server. At this point, your docker-compose-udms.yml should look like what is shown below:
########################### NETWORKS networks: default: driver: bridge socket_proxy: name: socket_proxy driver: bridge ipam: config: - subnet: 192.168.91.0/24 ########################### SECRETS secrets: plex_claim: file: $DOCKERDIR/secrets/plex_claim mysql_root_password: file: $DOCKERDIR/secrets/mysql_root_password include: ########################### SERVICES # PREFIX udms = Ultimate Docker Media Server # HOSTNAME=udms - defined in .env # CORE - compose/$HOSTNAME/socket-proxy.yml - compose/$HOSTNAME/portainer.yml - compose/$HOSTNAME/dozzle.yml - compose/$HOSTNAME/homepage.yml # MEDIA - compose/$HOSTNAME/plex.yml - compose/$HOSTNAME/jellyfin.yml # DOWNLOADERS - compose/$HOSTNAME/sabnzbd.yml - compose/$HOSTNAME/qbittorrent.yml # PVRS - compose/$HOSTNAME/radarr.yml - compose/$HOSTNAME/sonarr.yml # COMPLEMENTARY APPS - compose/$HOSTNAME/bazarr.yml - compose/$HOSTNAME/tautulli.yml # MONITORING - compose/$HOSTNAME/uptime-kuma.yml # DATABASES - compose/$HOSTNAME/mariadb.yml # UTILITIES - compose/$HOSTNAME/filebrowser.yml # MAINTENANCE - compose/$HOSTNAME/docker-gc.yml
17-60 and More: Rest of the Apps
The media server apps listed above are what I consider as key to a kickass media server based on Docker. However, there are a few more apps that are nice to have.
This guide is already several thousand words long. So, I have included the docker compose examples for the following apps in my Github Repo.
Here are a few that are not listed in this article but available in the repository:
- Media/Music: Lidarr, Readarr, Prowlarr, Ombi, Handbrake, MakeMKV, MKVToolNix, Picard, Plex Meta Manager, Notifiarr, Ampache, Emby, Gonic, Navidrome, Funkwhale, Plex-Trakt-Sync, Tiny Media Manager, Cloudplow, Autoscan
- Monitoring: Grafana, Glances, Cadvisor, ha-dockermon, Youtube-DL Material, Loki, Promtail, Smokeping, Varken, Lidarr Exporter, Node Exporter, Prowlarr Exporter, Radarr Exporter, Sabnzbd Exporter, Sonarr Exporter
- DNS/AdBlockers/VPN/Networking: Unbound, Zerotier, Tailscale, DDNS Updater, PiHole
- Databases: PostgreSQL, InfluxDB, Prometheus, Redis, phpMyAdmin
- Web: php7, Nginx
- Admin: Guacamole
- Security: Crowdsec, Traefik Bouncer, Cloudflare Bouncer
- File Managers: Visual Studio Code Server, AutoIndex, Cloud9, Cloud-Commander, Dupeguru, PyRenamer
- Photo Management: Digikam, Photoprism, Photoshow
- Others: Homepage, Heimdall, jDownloader, APCUPSD, Cloud-Commander, Dupeguru, Flaresolver
By now, you should be quite familiar with copy-pasting it from my GitHub repo and customizing any needed environmental variables. So, it should be a breeze to add the following apps to your setup if you choose to.
Be the 1 in 200,000. Help us sustain what we do.You will gain benefits such as Deployarr access, discord roles, exclusive content, ad-free browsing, and more.🔥 Holiday Sale! 25% Off Platinum Membership$399.99$299.99 (ends December 31).Join the Geek Army (starting from just $1.67/month)
External Access to Apps
So, a 11,000-word post on how to setup a media server from scratch using Docker and Ubuntu. If this helped, I would greatly appreciate you showing your support of my work in one or more ways listed above.
As mentioned previously, the objective of this guide was to create a basic Docker media server stack that is accessible inside the home/local network. But what if you want to expose the apps to the internet so you have access to your apps on the go?
This topic will be covered in detail in the later parts of this tutorial series. But here is a summary of options. I will add more details and relevant links as those individual guides are published.
- Wireguard: With your server running Wireguard server and Wireguard client on the mobile, you VPN into your home/LAN network and have access to your Dockerized apps.
Other Posts in the Wireguard Series:
- Wireguard VPN Intro in 15 min: Amazing new VPN Protocol
- Complete Wireguard Setup in 20 min – Better Linux VPN Server
- Wireguard Windows Setup: Powerful VPN for Windows
- Wireguard Mac OS Client Setup – The sleek new VPN
- Wireguard Android Client Setup – Simple and Secure VPN
- Ultimate WireGuard Docker Compose: with CF and Traefik Support
- Overlay Mesh Network: I use ZeroTier to tie all my Docker Hosts and mobile device together. Using this I can access all my apps while I am away. Another example is Tailscale. Both ZeroTier and Tailscale have free plans. There is no need to open ports on the router, keeping malicious traffic away. I will cover this in more detail, later in this series.
- Cloudflare Tunnels and Access: With Cloudflare Tunnels and Access, you can secure have on the go access to your apps. There is no need to open ports on the router, which is awesome. Another benefit is you will also have SSL/HTTPS connection to your app for added security. This will also be covered in detail, later in this series.
- Reverse Proxy:, the last option is to add a Reverse Proxy. Nginx Proxy Manager and Traefik are two common ones. Updated guides for both of these are in the plan for this series.
So be sure to watch out for updates to this Docker Server tutorial series in the coming days.
Troubleshooting
If you follow the guide word-to-word, you should be fine. However, something may go wrong. Feel free to join our discord community to ask around or just chat with like-minded people. But before you do, here are some common mistakes and fixes for those.
Closing Thoughts
Congratulations! But if you thought that you are done. You are wrong. Your journey is just getting started. Here are on out the possibilities are endless.
You could improve security by implementing docker security practices and implement Cloudflare tweaks.
You could even replace Nginx Proxy Manager with Traefik Reverse Proxy for additional features. This would also enable you to add Google OAuth, or even your own Multi-factor authentication system with Authelia.
You could add a Docker WordPress stack, like I do, and host your own blog.
There are literally hundreds of self-hosted apps for your homelab: like ad-blocking with PiHole, or AdGuard Home. Add NextCloud for your own Cloud storage, Guacamole for VNC, SSH, SFTP, and RDP, or run UniFi Controller if you are into UniFi ecosystem. Checkout the many YML files in my GitHub Repo.
Whatever you decide to do, my hope was to share my knowledge and experience through this Docker media server guide and help you get started with your setup. If you have any comments or thoughts, feel free to comment below.