WGX is an authenticated and end-to-end encrypted Wireguard relay for hub-and-spoke networks.
- It allows two Wireugard peers residing in different local-area networks to communicate.
- It is compatible with all standard Wireguard implementations (Linux, Android, iOS, MacOS, Windows etc.).
- Relay authenticates every peer to prevent even a single foreign packet reach your local-area networks.
- Relay passes all the packets through without decrypting them (this is in contrast to a regular Wireguard peer that re-encrypts all the packets passed through it).
The goal of the project is to implement efficient NAT traversal for standard Wireguard implementations without patching them. Currently it works for hub-and-spoke networks, and all packets go over a relay. Future work is to send packets directly wherever possible.
The relay itself is a Wireguard peer that you add to your Wireguard configuration as any other peer. Once added the peers perform regular handshakes with the relay thus «punching a hole» in NAT. Then in your spoke peer you replace hub's endpoint with the relay's endpoint, and the packets will go back and forth through the punched hole.
Hub peer needs to tell the relay which public keys it wants to communicate with. This is done by sending a configuration command over the established Wireguard tunnel directly to the relay. These commands are sent as UDP packets.
After that the handshake between hub and spoke will succeed and you can use your Wireguard network as usual.
To summarize:
- add the relay as a peer to your hub and spoke Wireguard configuration (
wgx export
), - replace hub's endpoint with relay's endpoint in the spoke Wireguard configuration,
- tell the relay which peers the hub wants to communicate with (
wgx hub sync
).
Run these commands on the server with a public IP.
# generate the configuration file
umask 077
mkdir -p /etc/wgx
cat >/etc/wgx/relay.conf <<EOF
[Relay]
PrivateKey = $(wg genkey)
AllowedPublicKeys = # comma-separated list of hub's and spokes' public keys or "all" to allow all public keys
EOF
# run the relay in the background
docker run \
--name wgx \
--volume /etc/wgx:/etc/wgx \
--network host \
--restart unless-stopped \
--detach \
docker.io/igankevich/wgx:latest \
/bin/wgxd
# alias wgx command-line client
alias wgx='docker exec wgx /bin/wgx'
# print status
wgx relay status
# print active sessions
wgx relay sessions
Run these commands on the hub (a Wireguard peer to which every other peer is connected).
alias wgx='docker run -it --rm --network host docker.io/igankevich/wgx:latest /bin/wgx'
# configure relay
# IP:PORT - public ip address and optional port of the relay
wgx hub init IP:PORT
# add "spoke" peer
wgx hub add-spoke
# configure wireguard interface
wgx hub start
These commands will configure the relay as a peer, and then send the public keys of all the other peers to the relay for routing.
Run these commands on the spoke (a Wireguard peer that connects to the hub over the relay).
alias wgx='docker run -it --rm --network host docker.io/igankevich/wgx:latest /bin/wgx'
# configure relay
# IP:PORT - public ip address and optional port of the relay
wgx spoke init IP:PORT
# add "spoke" peer
# PUBLIC-KEY - hub's public key in BASE64 format
# PRESHARED-KEY-FILE - a file containing hub's preshared key in BASE64 format
wgx spoke add-hub PUBLIC-KEY PRESHARED-KEY-FILE
# configure wireguard interface
wgx spoke start