The year is 2016. Giant clouds rule the internet. Microsoft supports Linux. Yet in this strange new world, not all is well. Your home network is still stuck behind an IPv4 NAT gateway. And your apps still haven't all figured out how to do secure HTTPS encrypted connections yet.
But wait! There in the distance... a savior emerges! The Road Warrior VPN!
This module sets up a StrongSwan-based IKEv2 VPN suitable for use with the native IKEv2 VPN client available on devices like iOS and Android, as well as traditional operating systems like Windows, MacOS and GNU/Linux.
It intentionally tries not to do everything for everyone, the module is smart in some areas (eg automatic generation of keys/certs) but dumb in other areas (eg limited configurability to keep things simple for users).
If you're wanting the simplest possible way to configure a VPN for your iOS or Android device this is the module for you. If you want a module that exposes every possible StrongSwan option, it's not.
... time passes ...
It's now the future, 2024 to be precise. Some stuff happened between 2016 and now, but I wouldn't look too deeply into it if I were you... just pretend everything is fine.
One big change, it's now really easy to connect to devices securely without a RoadWarrior VPN... haha, only kidding. But there are at least lots of other options to make it simple to create and manage configs... haha, got you again.
You still need this.
- Extremely simple configuration.
- IKEv2 using StrongSwan.
- Certificate-based authentication with automatic setup of CA & certs.
- Generates client certs for you
- Generates
.mobileconfig
files for easy import on iOS devices.
Tested and confirmed on:
* Debian 8/Jessie [Server]
* Debian 9/Stretch [Server]
* Ubuntu 16.04 [Server]
* iOS 9.3.1 [Client]
* iOS 10.0.1 [Client]
* iOS 11.4.1 [Client]
* MacOS X 10.11.4 [Client]
* Android 5 w/ StrongSwan [Client]
The VPN should work on any OS released in 2015-2016 onwards, but many
earlier OS releases didn't ship with IKEv2 VPN support. The following are
known minimum versions for working clients:
* iOS 9+
* MacOS X 10.11 (El Capitan)
* Android 4.4.3+ (with use of third party StrongSwan client)
Uncertain on above compatibility.
Tested and confirmed on:
- Ubuntu 24.04 [Server]
- strongswan/noble,now 5.9.13-2ubuntu4 all [installed]
- ruby 3.2.3 (2024-01-18 revision 52bb2ac0a6) [x86_64-linux-gnu]
- Puppet: 8.6.0
- NOTE: Puppet 8.x is installed when using
gem install puppet
& Puppet 8 has legacy facts disabled by default. Built-in legacy facts have been removed, but external dependencythias-sysctl
appears to have been abandoned. Alternative packagepuppet-augeasproviders_sysctl
does not currently work for24.04
might be usable in future, but for now, re-enable legacy facts with:echo "include_legacy_facts=true" >> /etc/puppetlabs/puppet/puppet.conf
- Ubuntu 22.04 [Server]
- strongswan/jammy-updates,jammy-security 5.9.5-2ubuntu2.2 all
- ruby 3.0.2p107 (2021-07-07 revision 0db68f0233) [x86_64-linux-gnu]
- Puppet: 7.30.0
Command to list versions
apt list strongswan 2>&1 | grep -v "Listing" | grep -v "CLI" && ruby --version && echo Puppet: $(puppet --version)
The following is an example of a basic configuration that sets up the firewall rules, defines the VPN name and both the IP range to use for the clients as well as the IP range to route back to the client devices.
class { 'roadwarrior':
manage_firewall_v4 => true,
manage_firewall_v6 => true,
vpn_name => 'vpn.example.com',
vpn_range_v4 => '10.10.10.0/24',
vpn_route_v4 => '192.168.0.0/16',
}
Any destination within vpn_route_v4
CIDR range with route through the VPN.
To route all traffic through it, set this to 0.0.0.0/0
. Make sure you also
set vpn_dns_servers
to either your internal DNS server (if you wish to access
anything in your network via DNS name) or to external public DNS server(s).
class { 'roadwarrior':
manage_firewall_v4 => true,
manage_firewall_v6 => true,
vpn_name => 'vpn.example.com',
vpn_range_v4 => '10.10.10.0/24',
vpn_route_v4 => '0.0.0.0/0',
vpn_dns_servers => '1.1.1.1,8.8.8.8',
}
It is recommended that you consider backing up the /etc/ipsec.d
directory. If
replacing/autoscaling the server running your roadwarror VPN, you will want to
populate the directory with the same data across the fleet, otherwises certs would
be re-generated.
You will want to make sure your server can be reached by the clients on UDP port 500 and UDP port 4500. If you use the managed firewall, this will be configured for you with iptables (using puppetlabs/firewall module).
Currently all traffic will egress your VPN server with an IP from the vpn_range_v4
defined above without any masqurading (NAT), so it's important that the
destination servers on your network know how to route back to the VPN range via
the server. (TODO: add a NAT option to this module?)
Refer to manifests/params.pp
for the full list of configurable options and what
they mean.
If you enable manage_firewall_v6
, you may also want to include the following
on your server if you rely on Route Advertisement rather than static IPv6
addresses, otherwise that autodiscovery will cease to function. The following is
an example of re-enabling RA for int eth0
.
# By default, IPv6 will ignore RA if forwarding is on which can cause what
# looks like breakage of autodiscovery of IPv6 on your interfaces. The following
# will ensure eth0 still accepts RAs, even if forwarding is enabled.
sysctl { 'net.ipv6.conf.eth0.accept_ra':
value => '2',
}
Simply define each client you wish to use, in addition to the main roadwarrior
class above.
roadwarrior::client { 'myiphone': }
roadwarrior::client { 'bobiphone': }
roadwarrior::client { 'androidftw': }
This module will export out the certs in a range of formats and sets up a mobile config file.
Most of the params you won't need to set, however the following three are useful. By default the iOS configuration will be "connect on request" only, however you can adjust to ensure the VPN always automatically establishes a connection
roadwarrior::client { 'examplephone':
ondemand_connect => false,
ondemand_ssid_excludes => undef,
ondemand_cellular_excludes => false,
}
For example, to generate configuration for iOS that will always reconnect unless you are on WiFi network "home" or "bach" which presumably don't require the VPN. Additionally, you can specify that you trust your cellular provider and connection on cellular network don't require the VPN (additionally, this gives a positive effect on battery)
roadwarrior::client { 'examplephone':
ondemand_connect => true,
ondemand_ssid_excludes => ['home', 'bach'],
ondemand_cellular_excludes => true,
}
The module will build and collect certs and configuration for your clients in
/etc/ipsec.d/dist/
, for example:
find /etc/ipsec.d/dist/
/etc/ipsec.d/dist/
/etc/ipsec.d/dist/examplephone
/etc/ipsec.d/dist/examplephone/CACert.der
/etc/ipsec.d/dist/examplephone/CACert.pem
/etc/ipsec.d/dist/examplephone/ios-examplephone.mobileconfig
/etc/ipsec.d/dist/examplephone/examplephone.p12
/etc/ipsec.d/dist/examplephone/examplephoneCert.pem
/etc/ipsec.d/dist/examplephone/examplephoneKey.pem
The purpose of these various files are:
-
ios-examplephone.mobileconfig
- A "ready to import" configuration for iOS devices that includes the CA and PKCS12 certs, along with all the config that you might want. -
examplephone.p12
- A PCKS/P12 file that includes the client's cert and key capable of being imported to most devices for configuring. If your device can't import, try renaming from.p12
to.pfx
and see if that helps. The default password on this file ispassword
. -
examplephone(Cert|Key).pem
- PEM format client cert and key. Try these if your device refuses to import the .p12 file above. -
CACert.(pem|der)
- The CA certificate. Many devices will want this to validate the authenticity of your VPN endpoint. If your device can't import the PEM, try the DER file.
To configure iOS clients (version 9+) email the .mobileconfig
file that has
been generated to the phone. Note that Apple Mail and the stock Mail app handle
this properly, but third party clients (on sender or reciever) possibly may fail
to set the weird content types needed by iOS. Sharing via iCloud drive doesn't
seem to work properly (unable to import from iCloud on iOS end).
From the mail client, tap and import the file and follow the prompts to import the certificate. Once complete, there will now be a VPN you can turn on/off in the settings screen.
If ondemand has been enabled, the VPN should automatically connect if all conditions are appropiate.
The MacOS process is a bit fiddly, make sure you carefully follow the instructions, especially in regard to trusting the certs. Also ensure you are on MacOS 10.11 (El Capitan) or later.
-
Import the
.p12
file generated for the client (eg "examplephone.p12"). Find the newly imported cert inside keychain access and make it as "Always Trusted". -
Import the
CACert.pem
file generate for the client. Go into Keychain Access and find the CA under "Certificates". Now mark it as "Always Trusted". -
These first two steps are CRITICAL, if either the client cert or CA cert are not trusted, the VPN will fail to connect without any visible message. Sometimes it can be hard to find where the certs are in Keychain access, recommend using the search box to search based on the vpn name (eg "vpn.example.com").
-
Access
System Preferences -> Network
and add a new VPN interface, making sure to select IKEv2 specifically. -
Populate
Server Address
andRemote ID
with the VPN Name you're using and set the local ID toclient@vpnname
where client is the name of the client resource defined (eg "examplephone" and vpnname is the mainvpn_name
defined for theroadwarrior
class. -
Under
Authentication Settings
select certificate authentication using the one we imported before. -
Establish your first connection and enjoy!
IKEv2 isn't supported natively on Android yet, so you'll have to install the StrongSwan Android app. If you're on Android versions below 4.4.3, the support is buggy and you'll probably need to consider another option like OpenVPN or a newer OS/phone instead.
-
Import the
.p12
certificate package (can be done via most email clients) into the OS cert/key store. -
Install the "StrongSwan" VPN client from the Play store.
-
Add a new VPN using type "IKEv2 Certificate".
The StrongSwan client does not appear to have ondemand logic like the iOS native client, which means the VPN will either have to be connected when required, or left running at all times - which could cause issues if you push routes from home/work down the VPN and then connect to the WiFi at those locations...
If you want to use a different configuration manager (eg Ansible or Chef), you'll have to port this module across, I can't help you with that.
However if you just want to run it like a dumb script since you don't use any configuration manager, you can do so by installing Puppet, installing the module and then running your config directly from a file.
The following is an example of how to do this:
## Note: apt-get install puppet installs an older version of Puppet (v5.x).
apt-get update && apt-get upgrade -y
apt-get install ruby
gem install puppet
## Ubuntu 24.04 only:
echo "include_legacy_facts=true" >> /etc/puppetlabs/puppet/puppet.conf
puppet module install jethrocarr/roadwarrior
cat > /root/configure_vpn.pp << EOF
class { 'roadwarrior':
manage_firewall_v4 => false,
manage_firewall_v6 => false,
vpn_name => 'vpn.example.com',
vpn_range_v4 => '10.10.10.0/24',
vpn_route_v4 => '192.168.0.0/16',
}
roadwarrior::client { 'myiphone':
ondemand_connect => true,
ondemand_ssid_excludes => ['wifihouse'],
}
roadwarrior::client { 'android': }
EOF
# Edit configuration to suit requirements.
# vim /root/configure_vpn.pp
# Apply configuration. You can re-run this if you make changes to the above file
# in future (such as adding new clients).
puppet apply /root/configure_vpn.pp
Contributions in the form of Pull Requests (or beer donations) are always welcome. However please keep in mind the goal of this module is to support road warrior style setups rather than being a general all-purpose StrongSwan/IKEv2 VPN configuration module.
This module is licensed under the Apache License, Version 2.0 (the "License"). See the LICENSE or http://www.apache.org/licenses/LICENSE-2.0
Copyright 2016-2024 jethrocarr
Copyright 2024 sec-ml
Licensed under the Apache License, Version 2.0 (the "License");
you may not use this file except in compliance with the License.
You may obtain a copy of the License at
http://www.apache.org/licenses/LICENSE-2.0
Unless required by applicable law or agreed to in writing, software
distributed under the License is distributed on an "AS IS" BASIS,
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
See the License for the specific language governing permissions and
limitations under the License.