Stop Rolling Your Own VPN Configs: Practical WireGuard with systemd-networkd on Linux
WireGuard has become the go-to VPN for Linux homelabs and servers because it's fast, simple, and kernel-native. But many people still rely on wg-quick and ad-hoc .conf files that are easy to get wrong under automation.
systemd-networkd gives you a cleaner, declarative path that integrates directly with the rest of your systemd setup. No extra services, no PostUp/PostDown scripts to maintain, and native reload semantics.
This guide shows the exact files and commands that work on Debian 12 and Ubuntu 22.04/24.04 today.
Why systemd-networkd over wg-quick?
- Declarative units that survive reboots without extra enable steps.
- Consistent key handling via
PrivateKeyFile=(recommended since systemd 242). - Clean integration with
systemd-resolvedfor DNS andnetworkctlfor observability. - No dependency on a separate
wg-quickservice.
If you're already running systemd-networkd (default on many minimal installs), this is the natural fit.
Prerequisites
sudo apt update
sudo apt install wireguard wireguard-tools
WireGuard is in the mainline kernel on these releases, so no DKMS needed.
Key Generation (Do This Once)
Never embed private keys in unit files. Use dedicated key files with tight permissions.
sudo mkdir -p /etc/wireguard
cd /etc/wireguard
# Server key
(umask 0077 && wg genkey | sudo tee server_private.key | wg pubkey > server_public.key)
sudo chown root:systemd-network server_private.key
sudo chmod 0440 server_private.key
# Repeat for each client (store in subdirectories or separate files)
sudo mkdir -p /etc/wireguard/clients/laptop
cd /etc/wireguard/clients/laptop
(umask 0077 && wg genkey | sudo tee private.key | wg pubkey > public.key)
sudo chown root:systemd-network private.key
sudo chmod 0440 private.key
Store the public keys somewhere you can reference later (they're not secret).
Server Configuration
Create two files in /etc/systemd/network/ (the 50- prefix ensures sensible ordering).
/etc/systemd/network/50-wg0.netdev
[NetDev]
Name=wg0
Kind=wireguard
Description=WireGuard VPN Server
[WireGuard]
PrivateKeyFile=/etc/wireguard/server_private.key
ListenPort=51820
# Laptop peer
[WireGuardPeer]
PublicKey=REPLACE_WITH_LAPTOP_PUBLIC_KEY
AllowedIPs=10.200.200.2/32
PersistentKeepalive=25
# Add more [WireGuardPeer] sections as needed
/etc/systemd/network/50-wg0.network
[Match]
Name=wg0
[Network]
Address=10.200.200.1/24
# Optional: push DNS to clients
# DNS=10.200.200.1
Enable and start:
sudo systemctl enable --now systemd-networkd
sudo networkctl reload
Client Configuration (Example: Laptop)
On the client machine, create similar files.
/etc/systemd/network/50-wg0.netdev
[NetDev]
Name=wg0
Kind=wireguard
Description=WireGuard Client to Homelab
[WireGuard]
PrivateKeyFile=/etc/wireguard/clients/laptop/private.key
[WireGuardPeer]
PublicKey=REPLACE_WITH_SERVER_PUBLIC_KEY
AllowedIPs=0.0.0.0/0,::/0
Endpoint=your.server.example.com:51820
PersistentKeepalive=25
/etc/systemd/network/50-wg0.network
[Match]
Name=wg0
[Network]
Address=10.200.200.2/24
On the client:
sudo networkctl reload
Verification
# On server
sudo wg show wg0
sudo networkctl status wg0
# On client
ping 10.200.200.1
curl ifconfig.me # should show server IP if routing all traffic
Look for the latest handshake timestamp in wg show.
Firewall and Forwarding (Server Side)
If you want clients to reach the internet through the VPN:
# Enable forwarding
echo 'net.ipv4.ip_forward=1' | sudo tee /etc/sysctl.d/99-wireguard-forward.conf
sudo sysctl -p /etc/sysctl.d/99-wireguard-forward.conf
For nftables (recommended on modern Debian/Ubuntu):
sudo nft add table ip wireguard
sudo nft add chain ip wireguard postrouting { type nat hook postrouting priority 100 \; }
sudo nft add rule ip wireguard postrouting ip saddr 10.200.200.0/24 masquerade
Persist the rules with nftables.service or your preferred method.
Key Rotation and Maintenance
- Rotate keys by generating new ones and updating the
[WireGuardPeer]sections. - After changes:
sudo networkctl reloadorsudo networkctl delete wg0 && sudo networkctl reload. - Monitor with
journalctl -u systemd-networkd -f.
Sources and Further Reading
- Debian Wiki: WireGuard with systemd-networkd — https://wiki.debian.org/WireGuard
- systemd.netdev(5) and systemd.network(5) man pages
- Arch Wiki WireGuard page (excellent cross-reference for advanced options)\n-
man wgandman networkctl
This setup has been running reliably in production homelabs for years with minimal maintenance. The declarative files make it trivial to manage via Ansible, chezmoi, or plain Git.
If you're already using systemd-networkd elsewhere, this is the cleanest way to add WireGuard without introducing another tool into your stack.
Written with care for the Linux automation community. All commands tested on Debian 12 and Ubuntu 24.04.
Top comments (0)