Securely Self-Host Ghost on Proxmox Using Cloudflared - No Open Ports Needed
Selfhost without exposing ports to the internet

Self-hosting applications can be challenging because of the security risks associated with open ports. This article will show you how to self-host a Ghost blog on Proxmox without opening any ports. We'll achieve this by using Cloudflared, which securely tunnels traffic to your server, eliminating the need for open ports.
Prerequisites
Before you begin, ensure you have the following:
- A VM on Proxmox: A Linux-based VM (e.g., Ubuntu) to host Docker and Ghost.
- A domain name: Registered and managed through Cloudflare.
- Cloudflare account: Sign up at Cloudflare if you don't have one
Step 1: Setup Ghost
Install Docker and Docker Compose on your VM:
sudo apt install -y docker.io docker-compose
sudo systemctl start docker
sudo systemctl enable docker
Update your VM's system packages:
sudo apt update && sudo apt upgrade -y
Create a docker-compose.yml
file:
nano docker-compose.yml
Add the following to the file:
version: '3.1'
services:
ghost:
image: ghost
restart: always
ports:
- 8080:2242
environment:
database__client: mysql
database__connection__host: db
database__connection__user: root
database__connection__password: example
database__connection__database: ghost
# this url value is just an example, and is likely wrong for your environment!
url: https://your_domain.com # Set this value here before moving on
db:
image: mysql:8.0
restart: always
environment:
MYSQL_ROOT_PASSWORD: example_password
Start ghost container
docker-compose up -d
Step 3: Install Cloudflared
wget https://github.com/cloudflare/cloudflared/releases/latest/download/cloudflared-linux-amd64.deb
sudo dpkg -i cloudflared-linux-amd64.deb
Step 4: Configure Cloudflare
Log in to your Cloudflare account, add your domain, and follow the setup wizard. This will include changing your domain's nameservers to Cloudflare's nameservers as instructed
Authenticate Cloudflared with Cloudflare
This command will open a browser window and prompt you to log in to your Cloudflare account. After logging in, select the domain you want to use.
cloudflared login
Create your tunnel
cloudflared tunnel create my-tunnel
Get your tunnel id
cd ~/.cloudflared
There should be a .json file in this directory. Grab the file name (tunnel id)
Configure the Tunnel
Create a configuration file for your tunnel. Replace my-tunnel
with the name of your tunnel and your_domain
with your actual domain.
sudo nano /etc/cloudflared/config.yml
Add the following lines:
Replace my-tunnel.json
with the tunnel id you copied above. If you're using a non-root user, you'll also need to update the base path
tunnel: my-tunnel
credentials-file: /root/.cloudflared/my-tunnel.json
ingress:
- hostname: your_domain.com
service: http://localhost:2242
- service: http_status:404
Start your tunnel
cloudflared tunnel run my-tunnel &
Step 4: Setup DNS resolution to your tunnel
Go to the DNS section in your Cloudflare dashboard. Add a CNAME record for your domain pointing to <my-tunnel-id>.cfargotunnel.com