Aaditya M Nair

Split a VPN

Published
4 min read

Earlier this month, for various reasons (gaming, restrictive local connection, etc), I bought a VPN service. I use openvpn to connect to the service. Now, openvpn by default routes all traffic through the tunnel. This was not good for me because I still wanted to access local resources.

Openvpn does have a property where locally destined packet won't go through a VPN while the rest of them do. This allows me to, say, SSH into a local machine while having a P2P application running in the background. But it doesn't allow for fine grained control over what applications use the service.

That is what I am aiming for. Much of this post is a modified/simpler version of this blog post.

Pre-requisites

Almost all of this post deals with iptables and routing, especially policy routing. So I am assuming that you are aware of how those work in linux and are atleast familiar with the commands that modify them (iptables * and ip route ). You are also aware of a bit on TCP/UDP protocols and the concept of ports.

Let's move on to the actual content.

Split Tunnelling a network connection

So, what we are going to do is a lot of iptables black magic. It has a module that can detect what linux user is associated with the application that is sending the packets. Hence we will create a user that is allowed to use VPN and send all packets from that user through the VPN.

One time stuff: Create users and the rule table.

Create a separate user which will access the VPN.

$ adduser -c "Split Tunnel User" -M -s /bin/bash stunnel

Also edit the /etc/iproute2/rt_tables file to add a new rule table. Mine now looks like this:

255 local
254 main
253 default
200 vpn

Creating the tunnel

Now you will need to actually start the openvpn client and connect to the server.

$ sudo openvpn --route-noexec --config config.ovpn

Note that the --route-noexec parameter is important. This makes sure that openvpn does not add any routes by itself. It will just create a new interface and leave it at that. Alternatively, you could add route-noexec option in the config file.

Also, note the interface name(probably of the form tunX) and the IP address of the said interface. We are going to assume the ip address is 10.8.8.192.

Selecting the packets

Now we need to select all the packets from the user stunnel to go through the tunnel. iptables conveniently has a MARK action that, you know, marks them with a number. Hence the rule,

$ iptables -t mangle -A OUTPUT -m owner --uid-owner stunnel -j MARK --set-mark 0x1

Sending them the right way.

Now that we have the packets, we need to make sure that they go through the tunnel. The following commands do it.

$ ip rule add from all fwmark 0x1 lookup vpn
$ ip route add default via 10.8.8.192 table vpn

The first one ensures that all packets marked with 0x1 (traffic from the stunnel user from the last step) go through the given table 'vpn'. The next adds the rule in the vpn table that makes sure that the packets go through the given IP. Since the IP is of the VPN interface, tunX, all such packets will go through the vpn.

Making sure they come back.

Now that packets have gone through the VPN interface, we still haven't changed the source IP to that of the VPN. So packets won't be able to return as of now. Hence, we mask the source IP with that of the interface.

$ iptables -t nat -I POSTROUTING -o tunX -j MASQUERADE

We also need to do this,

$ sysctl net/ipv4/conf/tunX/rp_filter=2

Here we have set reverse path filtering to loose mode. This I believe makes sure that kernel doesn't reject the packet after the source IP has been unmasked.

Conclusion and other stuff.

Now, if you, like my university, only allows for a local DNS server, then you still won't be able to resolve hostnames your resolv.conf still points to the local nameservers which you cannot access while using VPN. To get around this, redirect everything to any public DNS server of your choice. I myself use Google's DNS.

$ iptables -t nat -I OUTPUT -p udp --dport 53 -m owner --uid-owner stunnel -j DNAT --to-destination 8.8.8.8:53

Finally, if you want to run GUI applications or anything that gives out sound via the stunnel user, you have allow for it. That is a topic for a separate post

By now, you have a fully functional split tunnel. To run command with vpn, just go,

$ sudo -u stunnel -i -- <command>

Copyright 2022 by Aaditya M Nair. Source here