Skip to content

An IKEv2/IPSEC VPN gateway that presents an application as if it was on a private network

License

Notifications You must be signed in to change notification settings

i-e-b/VirtualVpn

Repository files navigation

VirtualVpn

An IKEv2/IPSEC VPN gateway that presents an application as if it was on a private network

It will respond to StrongSwan opening a session from outside, and will correctly start a secured session. It allows communication to and from a configured web app.

It is slow compared to a hardware VPN device. It is slow compared to a pair of tin-cans with a tight string in between. It might be just fast enough for a basic API that is called a few times a second at most.

This VPN does not support all IKEv2 settings, and doesn't support IKEv1 at all.

Hopefully it's simple enough that any new parts can be added.

Starting

In the VirtualVPN project folder (where the VirtualVPN.csproj file is), call dotnet run .

Once the project is compiled and running, you should get a message like:

Starting up VirtualVPN. Current platform=Windows
Log level set to 3 (Info)
2022-09-06T09:07 (utc) Listening on 4500...
2022-09-06T09:07 (utc) Listening on 500...

If you get an error message, you might not have permissions to attach to the network (in which case, make sure you are running as root/administrator), or there is already VPN software running on these ports.

Press enter in the console to get a list of available commands.

You will want to load a configuration file before connecting to a remote gateway. The default settings can be saved or viewed by typing save mySettings.jsonenter into the console.

Load settings with load mySettings.jsonenter

Once settings are loaded, you can connect from the remote gateway, or request a connection by typing start 192.168.x.xenter at the console (with the IP address of the remote gateway).

Running as a service

If you run VirtualVPN with command-line arguments, it will enter 'non-interactive' mode, designed for running as a service.

  • See VirtualVpn.service file for information on setting up a systemd service.
  • See VirtualVpn/RunVpn.sh for an example of calling command-line arguments.

The command line arguments are the same as the interactive arguments, and are run in the order they are supplied.

If the first argument is int, VirtualVPN will run in interactive mode even when command line arguments are supplied.

Health check failure codes

The health check API end point gives text messages on failure, but in case you only have access to the HTTP status codes (looking at you, uptime-kuma), these are

  • 200 - All OK: VpnServer is up with an active session
  • 503 - VpnServer instance is null
  • 410 - VpnServer instance is not running
  • 502 - VpnServer is running, but there are no active sessions
  • 409 - VpnServer is up, a session is established, but tunnelled TCP connections are being terminated (indicates some filtering software is misconfigured at the remote side)
  • 408 - VpnServer is up, a session is established, but no traffic (including keep-alive pings) has been received for over 2 minutes

Layout

The projects ProtocolTests, and TestProxy are tools to help with development, and are not required when running Virtual VPN

Virtual VPN parts

  • Crypto - Diffie-Hellman key exchange, cryptographic routines, secure hashes etc.
  • Enums - Standard numeric values for IKE, ESP, and TCP/IP
  • EspProtocol - Handlers for Child SAs, ESP packets, IKE packets. This is where the gateway-to-gateway tunnelled traffic is handled
  • Helpers - Serialisation tools, low level C# bits
  • InternetProtocol - Various IP bits, including packet, addresses, and ICMP bits
  • Logging - Log level control, log formatting, and output to Loki servers
  • TcpProtocol - Embedded TCP stack to handle termination and re-routing to host web-app
  • TlsWrappers - Adaptors to decrypt and re-encrypt TLS/SSL traffic so VirtualVPN can pretend to be a different HTTPS server
  • Web - An api and a website for remotely lifting traffic captures off of the VirtualVPN
  • Program - The entry point
  • Settings - Configuration root and documentation. Read the comments to understand the setting values.
  • VpnServer - root listener for IKE and ESP traffic. This directs to one of a set of VpnSessions
  • VpnSession - handler for a single VPN session. This deals with IKEv2 protocol, but not tunnelled traffic.

Proxy API

VirtualVPN lets other systems call out through the remote gateway as if they were a machine on the virtual network. See TestProxy/TestProxyApiProgram.cs for an example of calling this. You will need to supply complete and correct HTTP headers and body.

Helpful Bash Commands

Posting a binary file with curl

https://curl.se/docs/manpage.html#-d

curl -X 'POST' 'http://55.55.55.55:5223/WeatherForecast/checksum' -H 'accept: */*' -H 'Content-Type: application/octet-stream' -d @CHANGES -v

Setup for test and development

Example ipsec.conf

This config goes in /etc/ipsec.conf for StrongSwan to use. "alice" is a server running both VirtualVPN and the web app. "bob" is a server running StrongSwan as a test 'remote' VPN gateway

conn alice
    # life cycle #
    auto=add
    dpdaction=clear
    dpddelay=300s
    rekey=no
    ## phase 1 ##
    keyexchange=ikev2
    ## phase 2 ##
    authby=secret
    mark=24
    type=tunnel
    # us (bob) #
    leftauth=psk
    left=192.168.0.3
    leftid=192.168.0.3
    leftsubnet=192.168.0.40/32
    lefthostaccess=yes
    leftallowany=yes
    leftupdown=/etc/ipsec-notify-bob.sh
    # them (alice) #
    rightauth=psk
    right=192.168.0.2
    rightid=192.168.0.2
    rightsubnet=55.55.0.0/16

Notify script

This goes with the ipsec.conf, in /etc/ipsec-notify-bob.sh. It needs to have execute permissions. This script adds a vti which routes traffic for 55.55.?.? to a VirtualVPN device on Bob. Because the sub-net is a /16 range, many IP addresses will route to the one VirtualVPN - but it does not care. All requests will get routed to the web app VirtualVPN is set up to use. Ports are also ignored, and the web app will get any requests -- except ICMP pings, which VirtualVPN will respond to itself.

#!/bin/bash
echo "###### BOB UP/DOWN SCRIPT #######"
echo "wake..." > /var/log/vti_state
set -o errexit
! echo "VERB = ${PLUTO_VERB-}" || true

case "${PLUTO_VERB-}" in
    "up-client")
        echo "cleaning old vti devices" > /var/log/vti_state
        ! ip tunnel del vti_h || true
        echo "creating vti device ${PLUTO_ME} -> ${PLUTO_PEER} mark=${PLUTO_MARK_OUT%%/*}->${PLUTO_MARK_IN%%/*}" > /var/log/vti_state
        #ip tunnel add vti_h mode vti local "${PLUTO_ME}" remote "${PLUTO_PEER}" key "${PLUTO_MARK_IN%%/*}"
        ip tunnel add vti_h mode vti local "${PLUTO_ME}" remote 0.0.0.0 key "${PLUTO_MARK_IN%%/*}"
        echo "linking" > /var/log/vti_state
        ip link set vti_h up mtu 1419
        echo "adding routes" > /var/log/vti_state

        ip addr add 192.168.0.40/32 remote 55.55.0.0/16 dev vti_h

        echo "setting sysctl" > /var/log/vti_state
        sysctl -w "net.ipv4.conf.vti_h.disable_policy=1"
        echo "up" > /var/log/vti_state
        ;;
    "up-host")
        echo "cleaning old vti devices" > /var/log/vti_state
        ! ip tunnel del vti_h || true
        echo "creating vti device ${PLUTO_ME} -> ${PLUTO_PEER} mark=${PLUTO_MARK_OUT%%/*}->${PLUTO_MARK_IN%%/*}" > /var/log/vti_state
        ip tunnel add vti_h mode vti local "${PLUTO_ME}" remote "${PLUTO_PEER}" okey "${PLUTO_MARK_OUT%%/*}" ikey "${PLUTO_MARK_IN%%/*}"
        echo "linking" > /var/log/vti_state
        ip link set vti_h up mtu 1419
        echo "adding routes" > /var/log/vti_state

        ip addr add 192.168.0.40/32 remote 55.55.0.0/16 dev vti_h

        echo "setting sysctl" > /var/log/vti_state
        sysctl -w "net.ipv4.conf.vti_h.disable_policy=1"
        echo "up" > /var/log/vti_state
        ;;
    "down-client")
        echo "removing vti_h" > /var/log/vti_state
        ! ip tunnel del vti_h || true
        echo "down" > /var/log/vti_state
        ;;
esac

Notes

If ports are blocked on Windows, call netstat -ano to get the process ids. It's probably the IKE service, which you'll need to turn off.

To get this fully working, you will either need a good modem that allow you to pass all traffic to your device, or not be behind any kind of NAT, and have a firewall that permits all traffic on ports 500 and 4500 regardless of protocol type.

Parts based on, or derived from:

References

https://datatracker.ietf.org/doc/html/rfc7296 https://www.rfc-editor.org/rfc/rfc9293 https://security.stackexchange.com/questions/56434/understanding-the-details-of-spi-in-ike-and-ipsec http://unixwiz.net/techtips/iguide-ipsec.html https://www.secfu.net/2017/12/23/the-ikev2-header-and-the-security-association-payload/

About

An IKEv2/IPSEC VPN gateway that presents an application as if it was on a private network

Topics

Resources

License

Stars

Watchers

Forks

Languages