Pritunl is a VPN server with a bunch of advanced security and access control features.
In fact, it is just a wrapper over OpenVPN, adding such Access Control Lists to it in the form of Organizations, users, and routes.
The task is to deploy a Pritunl test instance in Kubernetesб so we can take a closer look at it.
For now, we will use the free version and later will look at the paid one. Differences and costs can be found here>>>.
Will run it in Minikube, and for installation, we will use the Helm chart from Dysnix.
Contents
Running Pritunl in Kubernetes
Create a namespace:
[simterm]
$ kubectl create ns pritunl-local namespace/pritunl-local created
[/simterm]
Add a repository:
[simterm]
$ helm repo add dysnix https://dysnix.github.io/charts
[/simterm]
And install the chart with Pritunl:
[simterm]
$ helm -n pritunl-local install pritunl dysnix/pritunl ... Pritunl default access credentials: export POD_ID=$(kubectl get pod --namespace pritunl-local -l app=pritunl,release=pritunl -o jsonpath='{.items[0].metadata.name}') kubectl exec -t -i --namespace pritunl-local $POD_ID pritunl default-password ... export VPN_IP=$(kubectl get svc --namespace pritunl-local pritunl --template "{{ range (index .status.loadBalancer.ingress 0) }}{{.}}{{ end }}") echo "VPN access IP address: ${VPN_IP}"
[/simterm]
Check the pods:
[simterm]
$ kubectl -n pritunl-local get pod NAME READY STATUS RESTARTS AGE pritunl-54dd47dc4d-672xw 1/1 Running 0 31s pritunl-mongodb-557b7cd849-d8zmj 1/1 Running 0 31s
[/simterm]
Get the login-password from the master pod:
[simterm]
$ kubectl exec -t -i --namespace pritunl-local pritunl-54dd47dc4d-672xw pritunl default-password ... Administrator default password: username: "pritunl" password: "zZymAt1tH2If"
[/simterm]
Find its Services:
[simterm]
$ kubectl -n pritunl-local get svc NAME TYPE CLUSTER-IP EXTERNAL-IP PORT(S) AGE pritunl LoadBalancer 10.104.33.93 <pending> 1194:32350/TCP 116s pritunl-mongodb ClusterIP 10.97.144.132 <none> 27017/TCP 116s pritunl-web ClusterIP 10.98.31.71 <none> 443/TCP 116s
[/simterm]
Here, the LoadBalancer pritunl is for client access to the VPN server, and the pritunl-web ClusterIP service is for accessing the web interface.
Forward a port to the web:
[simterm]
$ kubectl -n pritunl-local port-forward svc/pritunl-web 8443:443 Forwarding from 127.0.0.1:8443 -> 443 Forwarding from [::1]:8443 -> 443
[/simterm]
Open https://localhost:8443:
Log in and get into the initial settings:
Here, in the Public Address, the public address of the host on which Prytunl itself is running will be automatically set, and then it will be substituted into the client configs as the VPN host address.
Since Pritunl is working in Kubernetes, which is running in VirtualBox, which is running on Linux on a regular home PC, it does not suit us, but we will return to it later. For now, you can leave it as it is.
The rest of the settings are not interesting to us yet.
Setting up Pritunl VPN
Organization, Users
See Initial Setup.
Go group users, Pritunl has Groups, but it’s available in the full (paid) version, we will see it later.
Also, users can be grouped through Organizations.
Go to Users, add an Organization:
Add a user:
PIN, email – optional, not needed now.
Pritunl Server and routes
See Server configuration.
Go to Servers, add a new one:
Here:
- DNS Server: a DNS server for clients
- Port, Protocol: port and protocol for OpenVPN, which will run “inside” Prytunl and will accept connections from our users
- Virtual Network: a network from the address pool of which we will allocate private IPs for clients
I would single out Virtual Network 172.16.0.0 – then our home network, Kuber’s network, and client IPs will differ – it will be more simple to debug, see IPv4 Private Address Space and Filtering.
At the same time, it is important that the Server port here must match the port and protocol on the LoadBalancer – 1194 TCP.
I.e. requests from the working machine will go through the route:
- 192.168.3.0/24 – home network
- hits the VirtualBox network 192.168.59.1/24 (see Proxy )
- will go to LoadBalancer in the Kuber network 10.96.0.0/12
- and LoadBalancer will send a request to the Kubernetes Pod, where we have OpenVPN listening on the TCP port 1194
Check LoadBalancer itself:
[simterm]
$ kubectl -n pritunl-local get svc pritunl NAME TYPE CLUSTER-IP EXTERNAL-IP PORT(S) AGE pritunl LoadBalancer 10.104.33.93 <pending> 1194:32350/TCP 22m
[/simterm]
Port 1194 – TCP. We will deal with the Pending status a bit later.
Set the Virtual Network, port, and protocol for the Server:
Next, connect the Organization with all its users:
Start the server:
Check the process and port in the Kubernetes Pod – we see our OpenVPN Server on the port 1194:
[simterm]
$ kubectl -n pritunl-local exec -ti pritunl-54dd47dc4d-672xw -- netstat -anp | grep 1194 Defaulted container "pritunl" out of: pritunl, alpine (init) tcp6 0 0 :::1194 :::* LISTEN 1691/openvpn
[/simterm]
And let’s go to fix LoabBalancer.
minikube tunnel
See Kubernetes: Minikube, and a LoadBalancer in the Pending status for full details, for now just call minikube tunnel
:
[simterm]
$ minikube tunnel [sudo] password for setevoy: Status: machine: minikube pid: 1467286 route: 10.96.0.0/12 -> 192.168.59.108 minikube: Running services: [pritunl] errors: minikube: no errors router: no errors loadbalancer emulator: no errors ...
[/simterm]
Check Loadbalancer:
[simterm]
$ kubectl -n pritunl-local get svc pritunl NAME TYPE CLUSTER-IP EXTERNAL-IP PORT(S) AGE pritunl LoadBalancer 10.104.33.93 10.104.33.93 1194:32350/TCP 139m
[/simterm]
The EXTERNAL-IP
has the correct value now, so check the connection:
[simterm]
$ telnet 10.104.33.93 1194 Trying 10.104.33.93... Connected to 10.104.33.93. Escape character is '^]'.
[/simterm]
Return to the main Settings, specify Public Address == LoadBalancer IP :
OpenVPN – connect to the server
Go to Users, click Download profile:
Unpack the archive:
[simterm]
$ tar xfp local-user.tar
[/simterm]
And connect using a common OpenVPN client:
[simterm]
$ sudo openvpn --config local-org_local-user_local-server.ovpn [sudo] password for setevoy: ... 2022-10-04 15:58:32 Attempting to establish TCP connection with [AF_INET]10.104.33.93:1194 [nonblock] 2022-10-04 15:58:32 TCP connection established with [AF_INET]10.104.33.93:1194 ... 2022-10-04 15:58:33 net_addr_v4_add: 172.16.0.2/24 dev tun0 2022-10-04 15:58:33 WARNING: this configuration may cache passwords in memory -- use the auth-nocache option to prevent this 2022-10-04 15:58:33 Initialization Sequence Completed
[/simterm]
But now the network will not work:
[simterm]
$ traceroute 1.1.1.1 traceroute to 1.1.1.1 (1.1.1.1), 30 hops max, 60 byte packets 1 * * * 2 * * * 3 * * * ...
[/simterm]
Since in our VPN the route to 0.0.0.0/0 is directed through the same host on which the VPN actually works, so we got a “ring”.
Go to Servers, stop the server and delete the Default route:
Click on the Add Route – add a route to 1.1.1.1 through our VPN, and all other requests from the client will go through the usual routes:
Restart the connection:
[simterm]
$ sudo openvpn --config local-org_local-user_local-server.ovpn
[/simterm]
Check the routes on the host machine, locally:
[simterm]
$ route -n Kernel IP routing table Destination Gateway Genmask Flags Metric Ref Use Iface 0.0.0.0 192.168.3.1 0.0.0.0 UG 100 0 0 enp38s0 1.1.1.1 172.16.0.1 255.255.255.255 UGH 0 0 0 tun0 ...
[/simterm]
And check the network – the request went through the VPN:
[simterm]
$ traceroute 1.1.1.1 traceroute to 1.1.1.1 (1.1.1.1), 30 hops max, 60 byte packets 1 172.16.0.1 (172.16.0.1) 0.211 ms 41.141 ms 41.146 ms 2 * * * ...
[/simterm]
“It works!” (с)
Done.