WireGuard Easy

VPN rootful 1 container
wg-easy ghcr.io/wg-easy/wg-easy:latest · 51820:51820, 51821:51821

Kubernetes YAML

apiVersion: v1
kind: Pod
metadata:
  name: wg-easy
  labels:
    app: wg-easy
spec:
  restartPolicy: Always
  containers:
  - name: wg-easy
    image: ghcr.io/wg-easy/wg-easy:latest
    ports:
    - hostPort: 51820
      containerPort: 51820
    - hostPort: 51821
      containerPort: 51821
    env:
    - name: WG_HOST
      value: changeme
    - name: PASSWORD_HASH
      value: changeme
    - name: WG_PORT
      value: '51820'
    - name: WG_DEFAULT_DNS
      value: 1.1.1.1
    - name: WG_ALLOWED_IPS
      value: 0.0.0.0/0
    volumeMounts:
    - name: vol-0
      mountPath: /etc/wireguard
    securityContext:
      capabilities:
        add:
        - NET_ADMIN
        - SYS_MODULE
  volumes:
  - name: vol-0
    persistentVolumeClaim:
      claimName: wg-easy-data

Deployment Guide rootful

Operating System:
!

Prerequisite as root

Install Podman

apt update && apt install -y podman
1

Save the YAML file

sudo mkdir -p /etc/containers/
# Copy the YAML above to:
sudo nano /etc/containers/wg-easy.yaml
2

Test the pod (without systemd)

sudo podman play kube /etc/containers/wg-easy.yaml

# Status:
sudo podman pod ps && sudo podman ps

# Stop:
sudo podman play kube --down /etc/containers/wg-easy.yaml
3

Create Quadlet .kube file (systemd)

Place it at /etc/containers/systemd/wg-easy.kube

sudo mkdir -p /etc/containers/systemd/
cat << 'EOF' | sudo tee /etc/containers/systemd/wg-easy.kube
[Unit]
Description=wg-easy Pod

[Kube]
Yaml=/etc/containers/wg-easy.yaml

[Install]
WantedBy=multi-user.target
EOF
4

Enable systemd service (rootful)

sudo systemctl daemon-reload
sudo systemctl enable --now wg-easy-pod.service
5

Status & Logs

sudo systemctl status wg-easy-pod.service
sudo journalctl -u wg-easy-pod.service -f
sudo podman pod ps
sudo podman ps
Rootful mode:
  • Containers run as root — only use trusted images
  • Ports < 1024 can be bound directly
  • No loginctl enable-linger needed — systemd manages the service
  • Quadlet path: /etc/containers/systemd/ (not ~/.config/)

Ports < 1024 (e.g. 80, 443)

Rootless cannot open privileged ports. Solution:

sysctl net.ipv4.ip_unprivileged_port_start=80

Make persistent in /etc/sysctl.d/99-podman.conf.

Containers communicate via localhost

All containers in the pod share the same network namespace. Always use localhost, not container names.

# Correct (e.g. app → db):
localhost:5432

# Wrong (doesn't work in a pod):
db-container:5432

List open ports

Which ports is the running pod listening on?

podman port wg-easy-pod

Custom DNS for the pod

Set a custom DNS server (e.g. local Pi-hole):

# In YAML under spec.dnsConfig:
spec:
  dnsConfig:
    nameservers:
      - 192.168.1.x

Set volume ownership

Fix permission errors by adjusting UID/GID in the user namespace:

podman unshare chown 1000:1000 /path/to/volume

SELinux volume labels

On SELinux systems (RHEL, Fedora) set the volume suffix:

/host/path:/container/path:Z   # private
/host/path:/container/path:z   # shared

List all volumes

podman volume ls
podman volume inspect <volume-name>

Volume backup

Back up data from a named volume:

podman run --rm \
  -v <volume-name>:/data:ro \
  -v $(pwd):/backup \
  busybox tar czf /backup/backup.tar.gz /data

Cleanup

Remove unused images, containers and volumes:

podman system prune -f        # containers + images
podman image prune -f         # untagged images only
podman volume prune -f        # unused volumes

Automatic image updates (podman-auto-update)

Podman can automatically update images and restart the pod. Enable once:

systemctl --user start podman.socket
systemctl --user daemon-reload
systemctl --user enable --now podman-auto-update.timer
systemctl --user status podman-auto-update.timer

Test without actually updating:

podman auto-update --dry-run

Manual update

Pull a new image version and restart the pod:

podman pull <image>:<tag>
podman play kube --replace \
  ~/.config/containers/wg-easy.yaml

Find outdated images

Check local images against the registry:

podman images --filter dangling=false
podman pull --all-tags <image>

Shell into a running container

podman exec -it wg-easy-<container> /bin/sh
# or bash:
podman exec -it wg-easy-<container> /bin/bash

Follow live logs

# All containers in the pod:
podman pod logs -f wg-easy-pod

# Single container:
podman logs -f wg-easy-<container>

Pod info & resource usage

podman pod inspect wg-easy-pod
podman stats wg-easy-pod

Restart pod without data loss

podman pod restart wg-easy-pod

# or via systemd:
systemctl --user restart wg-easy-pod.service

Customize this stack
Change ports, volumes, environment variables and more in the generator.
Open in Generator