The following use cases assume:
- A Linux network setup with loopback interface
lo
, an internal network interfacelan
and an external network interfacewan
on router. - An EIM + EIF(Full Cone) external network (like an EIM + EIF CGNAT) or a external network with public IPv4 address on
wan
. - The firewall is not blocking traffic of interfaces or ports that
einat
interacts. einat
NAPT44 is setup onwan
and have hairpin routing setup for traffic fromlo
andlan
, i.e.einat -i wan --hairpin-if lo lan
.
And we give the following example network configuration:
Host | interface | network | address | note |
---|---|---|---|---|
Router | wan |
233.252.0.0/24 |
233.252.0.200 |
Have default route setup |
Router | lan |
192.168.1.1/24 |
192.168.1.1 |
Forwarding to lo or wan |
Device 1 | eth0 |
192.168.1.1/24 |
192.168.1.100 |
Connected with lan on router |
192.168.1.200 |
Secondary address |
We call the address of wan
(233.252.0.200
here) on router as "external address".
Install stuntman which contains stunclient
and stunserver
.
Test NAT behavior on device 1:
$ stunclient stunserver.stunprotocol.org --mode full --verbosity 1 --protocol udp --localaddr 192.168.1.100 --localport 20000
# or test for TCP
$ stunclient stunserver.stunprotocol.org --mode full --verbosity 1 --protocol tcp --localport 192.168.1.200 --localport 20000
It should gives
Local address: 192.168.1.100:20000
Mapped address: 233.252.0.200:20000
Behavior test: success
Nat behavior: Endpoint Independent Mapping
Filtering test: success
Nat filtering: Endpoint Independent Filtering
Also if you perform NAT behavior test with the same local source port but from a different address in a short gap, the result should also be EIM + EIF and the resulting external port should not be the same with previous test. Otherwise, for a network setup with einat
behind an external NAT, the external NAT has a fake EIM behavior.
$ stunclient stunserver.stunprotocol.org --mode full --verbosity 1 --protocol udp --localaddr 192.168.1.200 --localport 20000
Local address: 192.168.1.200:20000
Mapped address: 233.252.0.200:[20001] <- this should changes
Behavior test: success
Nat behavior: Endpoint Independent Mapping
Filtering test: success
Nat filtering: Endpoint Independent Filtering
Natter is a STUN-based port mapping daemon, you can use this tool to hold an external TCP/UDP port and forwarding the traffic to a specified target(e.g. a local TCP listening service).
Run Natter on device 1 with test forwarder.
$ python natter.py -b 20000 -m test
[I] Natter v2.0.0
[I]
[I] tcp://192.168.1.100:20000 <--Natter--> tcp://233.252.0.200:20000
[I]
[I] Test mode in on.
[I] Please check [ http://233.252.0.200:20000 ]
...
# curl on test HTTP service served on the external port should gives:
$ curl http://233.252.0.200:20000
<html><body><h1>It works!</h1><hr/>Natter</body></html>
Similar work can also be done with natmap.
You can optionally start a STUN server for port mapping on private external address, which is not reachable(i.e. behind an external NAT) by public STUN server. With hairpin routing setup, we can reach the STUN server with external address. Note the STUN server port(default is 3478
) should be excluded from NAT port ranges that einat
uses.
Start STUN server on router:
stunserver --protocol tcp
Run Natter with router STUN server specified on device 1:
python natter.py -b 20001 -m test -s 233.252.0.200
Note
This section does not follow example network configuration above.
For WireGuard peer in server role, commonly we setup private IP on WireGuard interface(i.e. wg0
) and do iptables/nftables masquerading for traffic forwarding between WireGuard interface and default external interface from which can reach public Internet. Example nftables rules below.
table ip nat {
chain postrouting {
type nat hook postrouting priority srcnat; policy accept;
iifname wg0 oifname eth0 masquerade
}
}
However this only gives WireGuard "client" connecting this "server" an Address and Port-Dependent Mapping (with Endpoint-Independent Mapping characteristic1) and Address and Port-Dependent Filtering network environment that provided by Netfilter NAT system.
To have Endpoint-Independent Mapping and Endpoint-Independent Filtering NAT behavior for traffic forwarding between WireGuard "client" and external interface on WireGuard "server", you can use einat
instead of iptables/nftables masquerading. Also make sure iptables/nftables masquerade rules like above are removed if you are migrating from iptables/nftables masquerade to einat
.
$ sudo einat -i eth0 --nat44 --ports 20000-29999 --hairpin-if wg0 --internal <wg0 private network CIDR>
You might also want to enable NAT66 for IPv6 with --nat66
flag if you assign private IPv6 addresses instead of public IPv6 addresses to WireGuard peers.
Note if you are hosting TCP/UDP services on same server, you might want to tweak external ports that einat
uses so inbound initiated traffic towards unoccupied external ports are accepted. For example, the listening port of WireGuard service should not in range of external ports that einat
uses(e.g. the default 20000-29999), tweak the port setting of either einat
or WireGuard so inbound initiated traffic toward WireGuard service are accepted without filtering by einat
.