Notes on Installing Lemmy With Docker and Caddy
Are you looking to run Lemmy behind an existing Caddy web server in a docker image? This one is for you!
Background
With recent changes Reddit is making to destroy third-party apps, I’ve been thinking more and more about why I’m using Reddit in the first place. And it looks like I’m definitely not alone. I’ve mostly unsubbed from popular front page subreddits that new accounts start out with and what kept me coming back were specialized subreddits with high-quality useful information like /r/LocalLLaMA.
Well for a little while now there has been a push to “federate” or decentralized social media software. I held off initially because, well, frankly the software situation just looked rough. But after diving into the LLM AI space, where things still are truly bleeding-edge, I feel like I’m much more tolerant of rough edges that have at least been sanded down by a lot of users before me.
This website is currently running on a pretty small VPS with only 1 CPU and 1 GB of RAM. I already had Caddy up and running via a docker-compose script mostly pulled from the offical Docker Hub image docs. That was easy enough and the Caddyfile I created was only a few lines of basic settings, yet still provided a super fast static site server with automatic Let’s Encrypt integration to make serving HTTPS ridiculously easy.
Getting Lemmy running wasn’t too hard, but I wanted to add my own installation notes to the blog in case it helps someone else in a similar situation.
Step 1: First pass on configuration files
The official documentation for installing from Docker is pretty succinct and helpful. I made the ~/Lemmy
directory as shown and downloaded the files as suggested. Technically, the last one won’t be necessary because that’s an Nginx configuration. You may find it useful to look at, I suppose.
mkdir ~/lemmy
cd ~/lemmy
wget https://raw.githubusercontent.com/LemmyNet/lemmy-ansible/main/templates/docker-compose.yml
wget https://raw.githubusercontent.com/LemmyNet/lemmy-ansible/main/examples/config.hjson -O lemmy.hjson
wget https://raw.githubusercontent.com/LemmyNet/lemmy-ansible/main/templates/nginx_internal.conf
At this point, you need to open up the docker-compose.yml
file and change everything inside {{ }}
chunks. I made a few more changes:
- I ripped out the ‘proxy’ service that was an nginx image
- for the lemmy image, I used ‘dessalines/lemmy:latest’
- for the lemmy-ui image, I used ‘dessalines/lemmy-ui:latest’
- I generated a long alphanumeric password and replaced
{{ postgres_password }}
with that (in pictrs and postgres) - I had setup an A record in DNS for
lemmy.animal-machine.com
so I replaced{{ domain }}
with that - I ripped out the
postfix
service since I will not be enabling email
There was a helpful YouTube video by Awesome Open Source that I watched while prepping for this install. In that video, he edits more user names than I think you have to. Following the directions replacing the marked segments and ripping out nginx and postfix segments were sufficient for me.
The last part of the initial configuration is to open up the lemmy.hjson
file created and put that same postgres password from the previous file in the {{ postgres_password }}
spots and replace the {{ domain }}
chunks with the same hostname from step 5 above.
The folder for pictrs
needs to be created and have permissions set:
mkdir -p volumes/pictrs
sudo chown -R 991:991 volumes/pictrs
The documentation describes how to optimize your database settings. I plugged in my meager setup and adjusted the values as suggested.
I think you would technically be able to start up the server with docker-compose
at this point, but there are a few other changes that need to be made.
Step 2: Caddy reverse proxy setup
The official documentation has a page for Using Caddy as a reverse proxy, but I found that it didn’t work for me. The configuration as given resulted in syntax errors when I brought up the Caddy image. Keep in mind that I already have a basic server running with TLS support and I didn’t want to add Caddy as a service to the new Lemmy one.
Tackling the first problem, take the template provided in the docs and just put most of it within the site handler like this:
(caddy-common) {
encode gzip
header {
-Server
Strict-Transport-Security "max-age=31536000; include-subdomains;"
X-XSS-Protection "1; mode=block"
X-Frame-Options "DENY"
X-Content-Type-Options nosniff
Referrer-Policy no-referrer-when-downgrade
X-Robots-Tag "none"
}
}
lemmy-site.com {
import caddy-common
reverse_proxy http://lemmy_lemmy-ui_1:1234
@lemmy {
path /api/*
path /pictrs/*
path /feeds/*
path /nodeinfo/*
path /.well-known/*
}
@lemmy-hdr {
header Accept application/*
}
handle @lemmy {
reverse_proxy http://lemmy_lemmy_1:8536
}
handle @lemmy-hdr {
reverse_proxy http://lemmy_lemmy_1:8536
}
@lemmy-post {
method POST
}
handle @lemmy-post {
reverse_proxy http://lemmy_lemmy_1:8536
}
}
For me, ’lemmy-site.com’ above is ’lemmy.animal-machine.com’ because that is the ‘{{ domain }}’ text replacement from above and what I had setup in my DNS record. All of this went right under my block I already had defined for my main static page website.
Step 3: Docker-compose networks
The last piece of the puzzle is to create a docker network for the two setups to use for communication. Open up the Caddy docker-compose.yml
you’re using and define services/caddy/networks
to have a network called internalwebnet
. I’m choosing to call this ‘internalwebnet’ but that’s arbitrary. Then make sure to define a network of the same name under networks
services:
caddy:
# [... snipped ...]
networks:
- internalwebnet
# This was also added
networks:
internalwebnet:
name: caddy_network
With that setup, we can open up the Lemmy docker-compose.yml
and add the internalwebnet
network to all the services and then the networks
section so that they can communicate.
services:
lemmy:
# [... snipped ...]
networks:
- internalwebnet
lemmy-ui:
# [... snipped ...]
networks:
- internalwebnet
pictrs:
# [... snipped ...]
networks:
- internalwebnet
postgres:
# [... snipped ...]
networks:
- internalwebnet
# This was also added
networks:
internalwebnet:
external:
name: caddy_network
Finished!
At this point you should be able to start the docker images.
cd ~/caddy
sudo docker-compose up -d
cd ~/lemmy
sudo docker-compose up -d && sudo docker-compose logs -f
The last command also automatically tails the logs so you can watch and make sure that everything is found appropriately. A ctrl-c
will kill that when you’re finished without stopping the Lemmy server.
[Editors note (Jan 2025): This all proved a little too much for my poor VPS with 1 GB of RAM, so my Lemmy has been taken down.]