OpenVPN is a great tool to ensure traffic is not eavesdropped. You can use this to ensure a secure connection from your laptop to your DigitalOcean VPS (droplet) as well as between cloud servers. You can also have both done simultaneously.
This is not a foolproof, definitive, perfectly-secure, life-depends-on-it set of instructions. We will be taking three shortcuts here, which in my opinion are reasonable tradeoffs between ease of use and security, but I, nor DigitalOcean can be held responsible for security of your VPS, even if you follow these instructions.
To quote a cryptography rock-star, "You have to know what you are doing every step of the way, from conception through installation." — Bruce Schneier
This article is to help get you started on your way to setting up a Virtual Private Network. You have been warned. I'll point out the shortcuts taken and the general sequence to avoid making these shortcuts at Appendix 1.
If you only want to have two cloud servers to connect to each other, you may want to find a simpler (yet less secure) tutorial — though this is a good compromise between ease of setup and security.
Note: This tutorial covers IPv4 security. In Linux, IPv6 security is maintained separately from IPv4. For example, "iptables" only maintains firewall rules for IPv4 addresses but it has an IPv6 counterpart called "ip6tables", which can be used to maintain firewall rules for IPv6 network addresses.
If your VPS is configured for IPv6, please remember to secure both your IPv4 and IPv6 network interfaces with the appropriate tools. For more information about IPv6 tools, refer to this guide: How To Configure Tools to Use IPv6 on a Linux VPS
You'll need at least two droplets or VPS for this OpenVPN setup, and will work up to around 60 VPS without major modifications. So to get started, create two droplets. For the rest of this tutorial, I'll refer to them as Droplet 1 and Droplet 2.
• Create the droplet with Ubuntu 13.04 x32.
This should work without modification on any version of Ubuntu that DigitalOcean offers, but was only tested on 13.04.
Connect to the VPS via secure shell. We're going to update packages and install a few things.
aptitude update && aptitude dist-upgrade -y && aptitude install openvpn firehol -y && reboot
Note, if your shell goes purple during this, just choose "Install Package Maintainer's Version" twice.
• Create the droplet with Ubuntu 13.04 x32.
Again, this should work on any version of Ubuntu.
Connect to the VPS via secure shell. We're going to update packages in install a few things.
aptitude update && aptitude dist-upgrade -y && aptitude install openvpn -y && reboot
Again, if your shell goes purple during this, just choose "Install Package Maintainer's Version" twice.
The key generation is going to be done exclusively on Droplet 1. Type the following commands into the shell:
cd /etc/openvpn/ mkdir easy-rsa cd easy-rsa cp -r /usr/share/doc/openvpn/examples/easy-rsa/2.0/* .
Next, we're going to type in some presets which will vastly speed up the key generation process. Type the following command:
nano /etc/openvpn/easy-rsa/vars
Go ahead and edit the following values (you only need do to these, although there are several more present):
You may adjust the KEY_SIZE to 2048 or higher for added protection.
Save and exit with Control-O, Enter, and Control-X.
Next, type the following commands:
source vars ./clean-all ./build-ca
You should be able to hit Enter though all of the questions.
Note: if you ever have to go back and create more keys, you'll need to retype source vars but don't type ./clean-all or you'll erase your Certificate Authority, undermining your whole VPN setup.
Generate the server certificate and key with the following command:
./build-key-server server
You should be able to hit Enter on defaults, but make sure the Common Name of the certificate is "server".
It will ask you to add a pass phrase, but just hit Enter without typing one.
When it asks you "Sign the certificate?", type y and hit Enter.
When it says "1 out of 1 certificate requests certified, commit?", type y and hit Enter.
Next is generating the certificate and keys for the clients. For security purposes, each client will get its own certificate and key.
I'm naming the first client "client1", so if you change this, you'll have to adjust it several times later. So type in the following:
./build-key client1
As with the server key, when it asks you "Sign the certificate?", type y and hit Enter.
When it says "1 out of 1 certificate requests certified, commit?", type y and hit Enter.
Go ahead and repeat this for as many clients as you need to make. You can also come back to this later (though remember to "source var" again if you do so).
This is used after authentication, to determine the encryption parameters. Simply type the following line:
./build-dh
Next, we copy the various keys and certificates into place on the cloud server:
cd /etc/openvpn/easy-rsa/keys cp ca.crt dh1024.pem server.crt server.key /etc/openvpn
It's very important that keys are kept secure. Double check that only root has permission to read. So type:
ls -lah /etc/openvpn
What you're looking for is that server.key has -rw------- for permissions (read/write for owner, none for group, and none everyone). If you need to change it, use this command:
chmod 600 /etc/openvpn/server.key
The following table shows which files go onto which client.
client1 | client2 |
---|---|
ca.crt | ca.crt |
client1.crt | client2.crt |
client1.key (SECRET) | client2.key (SECRET) |
We'll securely copy the files to the second VPS using secure copy. (You could also cat, then copy and paste across SSH windows. But this is a nice technique to securely copy files.)
Generate SSH keys with the following command:
ssh-keygen -t rsa
It will choose a default filename and then ask you for a secure passphrase, which you should set. Find the SSH public key you just generated and type:
cat ~/.ssh/id_rsa.pub
Copy the results onto the clipboard. It's a few lines of letters and numbers looking like:
ssh-rsa AAAAB3NzaC1yc2EAAAADAQABAAABAQCo249TgbI1gYP42RbLcDhsNN28r/fNT6ljdFOZxhk+05UAPhxq8bASaqSXZI3K8EEI3wSpigaceNUu65pxLEsZWS8xTtjY4AVxZU2w8GIlnFDSQYr3M2A77ZAq5DqyhGmnnB3cPsIJi5Q6JQNaQ/Meg1v7mYR9prfEENJeXrDiXjxUqi41NlVdb5ZQnPL1EdKM+KN/EPjiTD5XY1q4ICmLJUB8RkffHwH2knEcBoSZW2cNADpMu/IqtxTZpFL0I1eIEtoCWg4mGIdIo8Dj/nzjheFjavDhiqvUEImt1vWFPxHEXt79Iap/VQp/yc80fhr2UqXmxOa0XS7oSGGfFuXz root@openvpn1
But USE YOUR OWN, not mine. Your id_rsa.pub doesn't need to be kept secure, but if you use the key above, that would allow me access to your VPS.
cd ~/.ssh
(If you get an error, create the folder with mkdir ~/.ssh
).
nano authorized_keys
Paste the public key that is in your clipboard onto a new line, then save and exit with Control-O, Enter, Control-X.
Next, we copy the appropriate keys onto the second server:
scp /etc/openvpn/easy-rsa/keys/ca.crt \ /etc/openvpn/easy-rsa/keys/client1.crt \ /etc/openvpn/easy-rsa/keys/client1.key \ root@droplet2ip:~/
It will ask you "Are you sure you want to continue connecting (yes/no)?", so type yes and hit Enter.
Then input the passphrase you've just created.
Next, we move the certificates and keys into their final location:
cd ~ mv ca.crt client1.crt client1.key /etc/openvpn ls -l /etc/openvpn
As the key must be kept secure, let's make sure client1.key has the correct permissions (-rw-------).
Again, if need be, the permissions can be reset with the following command:
chmod 600 /etc/openvpn/client1.key
Next comes the excitement that is networking on a VPN. You can use OpenVPN using routing or bridging. If you know what the difference is, you don't need my help choosing. For this tutorial, we'll use routing. We'll also use OpenVPN's default network range, which is 10.8.0.0/24. Unless you already use this network range somewhere, this will be fine. If you do need a different range, pick a private range and make sure you adjust all the later configuration steps accordingly.
On the OpenVPN server, we need to configure routing and setup a firewall as well. I use a tool called firehol to configure iptables, which makes it very simple to set up a complex firewall. So, type the following commands:
nano /etc/firehol/firehol.conf
While we could allow incoming OpenVPN connections from any address, we're going to limit these connections to the IP addresses of the computers you want to connect. Make this list of your IP addresses now.
Note: The following configuration only allows incoming SSH and OpenVPN connections. If you have other services that need to receive incoming connections, you'll need to modify the firewall to support these.
version 5 interface eth0 inet client all accept // allow all outgoing connections server ssh accept // allow all incoming SSH connections server openvpn accept src "1.2.3.4 2.3.4.5" // allow incoming OpenVPN connections // from these designated addresses // NOTE: EDIT THESE ADDRESSES interface tun0 vpn server all accept // allow all incoming connections on the VPN client all accept // allow all outgoing connections on the router inet2vpn inface eth0 outface tun0 route all accept // route freely to the VPN router vpn2inet inface tun0 outface eth0 masquerade // use NAT masquerading from the VPN route all accept // route freely to the VPN
Then, start the firewall with the following command:
firehol start
If you have an issue with your firewall, you can restart your VPS and the firewall configuration will be cleared. To make the firewall permanent, input the following:
nano /etc/default/firehol
Find the following line:
START_FIREHOL=NO
Now, change NO to YES. Save and exit with Control-O, Enter, Control-X.
The next step is to copy the example server configuration into place and edit it to our needs.
cd /etc/openvpn cp /usr/share/doc/openvpn/examples/sample-config-files/server.conf.gz . gunzip server.conf.gz nano /etc/openvpn/server.conf
The OpenVPN server will start as root, but we can set it to drop to lower privileges after startup, which is a good security measure. To configure this, find the following lines and uncomment them by removing the semicolons:
;user nobody ;group nogroup
If you have multiple servers that should communicate to each other, find the following line and remove the semicolon:
;client-to-client
If you increased the key size of your DH key, find the line:
dh dh1024.pem
and change 1024 to 2048 (or whatever number you selected).
We're going to assign the different clients static IP addresses from the OpenVPN server, so to do that, uncomment the following line:
;client-config-dir ccd
Save with Control-O, Enter, Control-X. Next, make the client configuration directory:
mkdir /etc/openvpn/ccd
and we'll add configuration for the first client here:
nano /etc/openvpn/ccd/client1
Type the following command, which assigns client1 to IP address, 10.8.0.5:
ifconfig-push 10.8.0.5 10.8.0.6
Save and exit with Control-O, Enter, Control-X.
For reasons that require an in-depth knowledge of networking to understand, use the following addresses for additional clients:
/etc/openvpn/ccd/client2 ifconfig-push 10.8.0.9 10.8.0.10
/etc/openvpn/ccd/client3: ifconfig-push 10.8.0.13 10.8.0.14
Simply, add 4 to each IP for each new set. A more technical explanation is at Appendix 2.
Now we can start the OpenVPN server with the following command:
service openvpn start
Give it a second, then type the following commands to ensure OpenVPN is running:
ifconfig
And among the network interfaces, you should see that the interface tun0 look like this:
tun0 Link encap:UNSPEC HWaddr 00-00-00-00-00-00-00-00-00-00-00-00-00-00-00-00 inet addr:10.8.0.1 P-t-P:10.8.0.2 Mask:255.255.255.255 UP POINTOPOINT RUNNING NOARP MULTICAST MTU:1500 Metric:1 RX packets:140 errors:0 dropped:0 overruns:0 frame:0 TX packets:149 errors:0 dropped:0 overruns:0 carrier:0 collisions:0 txqueuelen:100 RX bytes:13552 (13.5 KB) TX bytes:14668 (14.6 KB)
You can also type:
service openvpn status
and if OpenVPN is running, you'll see the following:
* VPN 'server' is running
If both of these are in order, then the server is up and running and we'll configure the client connection next.
First, let's copy the sample client configuration file to the proper locations:
cd /etc/openvpn cp /usr/share/doc/openvpn/examples/sample-config-files/client.conf .
It’s mostly configured, but we have to “tell” the address of our OpenVPN server, Droplet 1:
nano /etc/openvpn/client.conf
Find the line that says:
remote my-server-1 1194
And change my-server-1 to the IP address of Droplet 1. Next, we have to ensure that the client key and the certificate matches the actual file names. Search for the following lines:
cert client.crt key client.key
and adjust them to the keys copied over (e.g. client1.crt and client1.key).
Save and exit with Control-O, Enter, Control-X.
And next, let's start up the VPN:
service openvpn start
Again, we can test it with the following commands:
service openvpn status
ifconfig
Now that both ends of the VPN are up, we should test the network. Use the following command:
ping 10.8.0.1
And if successful, you should see something like:
PING 10.8.0.1 (10.8.0.1) 56(84) bytes of data. 64 bytes from 10.8.0.1: icmp_req=1 ttl=64 time=0.102 ms 64 bytes from 10.8.0.1: icmp_req=2 ttl=64 time=0.056 ms
Any traffic you do not need encrypted, you can connect via the public-facing IP address. Any traffic between cloud servers you want encrypted, connect to the Internet network address, e.g. Droplet 1 connect to 10.8.0.1. Droplet 2 is 10.8.0.5, Droplet 3 is 10.8.0.9, and so on.
Encrypted traffic will be slower than unencrypted, especially if your cloud servers are in different datacenters, but either traffic methods are available simultaneously, so choose accordingly.
Also, now is a good time to learn more about OpenVPN and encryption in general. The OpenVPN website has some good resources for this.
There were three shortcuts used here which if security is of the utmost importance, you should not do.
Additionally, OpenVPN supports loads of other hardening features, beyond the scope of this tutorial. Reading up at openvpn.org should be done.
So the first client will use 10.8.0.6 as its IP address, and 10.8.0.5 is the VPN tunnel endpoint. The second address is only used to route traffic through the tunnel. This is because each client uses a CIDR /30 network, meaning 4 IP addresses are used per client computer.
So the VPN server will use the 10.8.0.0/30 network:
10.8.0.0 | Network |
10.8.0.1 | Server IP address |
10.8.0.2 | Tunnel Endpoint |
10.8.0.3 | Broadcast |
And the first client, client1, will use the 10.8.0.4/30 network:
10.8.0.4 | Network |
10.8.0.5 | Server IP address |
10.8.0.6 | Tunnel Endpoint |
10.8.0.7 | Broadcast |
And so on...
Thanks for learning with the DigitalOcean Community. Check out our offerings for compute, storage, networking, and managed databases.
This textbox defaults to using Markdown to format your answer.
You can type !ref in this text area to quickly search our full set of tutorials, documentation & marketplace offerings and insert the link!
Hi guy’s I have a guestion about this configuration.
I created a openvpn server and all connections work fine, but … when I connect my droplet to the openvpn the “external IP address” Doesn’t answer anymore, making all service unavailable! Any ideia how to make the IP usable again and keep the openvpn connected at the same time?
Thanks in advance.
You’ve got a typo in the Networking section. You write “While we could allow incoming OpenVPN connections from any address, we’re going to limit these connections to the IP addresses of the computers you want to correct.” Instead of “correct” you mean “connect”
@cloudyvdb
Are you able to access the public IP address when you are not using the OpenVPN assigned IP addresses? Or are you just never able to use the public IP addresses? Are you having trouble with only one computer when it is using the OpenVPN that it can’t also use the public IP addresses? IOW, are other computers still able to use the public IP address even when there is (at least) one computer connected via OpenVPN? Did you follow the setup for the firehol firewall? If so, as the author of the article states, those rules are designed to allow only the computers on the OpenVPN network and computers using ssh.
In the Copy Keys into Place (and Distribute Client Certificate and Key) section(s) you write "What you’re looking for is that server.key has -rw------- for permissions (read/write for owner, none for group, and none everyone). If you need to change it, use this command:
chmod 700 /etc/openvpn/server.key"
The command should actually be
chmod 600 /etc/openvpn/server.key (and similarly chmod 600 /etc/openvpn/client1.key)
@Marc: Thanks, I’ll update the article. :]
@kamal also see my note above about another typo in the article re: correct -> connect
@Kamal you forgot to change the chmod 700 -> chmod 600 for the client1.key
@Marc: Oops! Thanks again.
@Kamal: The openvpn install not longer has a /usr/share/doc/openvpn/examples/easy-rsa directory:
Which of these needs to be copied for the following instruction in the guide:
Thanks!
@Kamal: Update: to get the easy-rsa files on Ubuntu 14.04 you need to install it separately:
Cheers.