Home | Send Feedback

First configuration steps on a new VPS Server

Published: 10. June 2018  •  linux, selfhost

I like to self-host my web applications and other applications like blog software and Git servers. This gives you a bit of independence from third-party services and the freedom to install any software with any configuration. Managing your own server has the drawback that it requires more effort than using a third-party service. There is the initial setup that takes time, and after that, you have to manage and update the server periodically.

But even with all these drawbacks, managing your own server is interesting, and you can learn a lot and improve your skills.

You don't need a server farm in your apartment to run your own Internet-connected servers, although you could. Instead, you visit the homepage of a VPS provider and rent your VPS server there. There are a lot of VPS hosting providers on the Internet, to name a few: OVH, Linode, DigitalOcean, Scaleway, Vultr.

A VPS is a virtual machine that runs its own copy of an operating system, and you have root access to it. Most VPSs run with a Linux operating system, but you can also find servers running Windows.

Renting a VPS server is not that expensive. The cheapest VPS offers I've seen cost between 2 and 5 US Dollars per month. Often, you can rent them for just one month. A good way to figure out if self-hosting is something for you without wasting much money.

In this blog post, I'm going to describe the initial configuration steps after I rent a new VPS. The provider often gives you a plain operating system with just the SSH server installed. The initial steps I cover in this blog post include updating the operating system, installing a firewall, and hardening the SSH server.

For this blog post, I'm going to use a VPS from OVH. I'm not affiliated with OVH; it's just the provider that I have the most experience with and where I've rented a few VPSs. I'm satisfied with the service they provide, and they offer VPSs for a very low price. You can find the price list here: https://www.ovhcloud.com/en/vps/cheap-vps/

For the operating system, I'm choosing Ubuntu 18.04. I like to use a Debian-based Linux, either Debian directly or Ubuntu, because many tutorials you find on the World Wide Web are written for this operating system.

Rent VPS

Visit the homepage of the VPS hosting provider of your choice and select a VPS server. The offers differ in CPU, RAM, and disk space. Often, these VPSs are not scalable, and the RAM, disk space, and number of CPUs are fixed. If you run out of resources, the only option then is to buy the next bigger VPS and relocate your software. Therefore, before you rent a VPS, you should think about the software you want to install. If you only want to run a self-hosted Git server or a WordPress instance, a VPS with 512MB RAM should be sufficient. If you are planning to run multiple services, a VPS with more RAM or more disk space might be a better choice.

As mentioned before, I'm going to use a VPS from OVH, and I choose the cheapest server (VPS SSD 1). With this VPS, you get 2GB RAM, 20GB SSD, a bandwidth of 100 Mbps, unlimited traffic, one static IPv4, and one IPv6 address. OVH also supports an upgrade to the bigger VPS SSD2 and VPS SSD3 without reinstalling the operating system.

During the ordering process, I can choose the location of the data center where my VPS should run, the operating system, and additional offerings like more disk space and snapshots. I can select if I want to rent the VPS for just one month or a longer period. These options differ from provider to provider. You also have to create an account with the provider. At the end of the ordering process, you have to pay.

After the payment is processed, the provider sets up the VPS and sends you an email with information about your new VPS. This process can take a few minutes. In my case, it took about 10 minutes.

Connect with SSH

The email that OVH sends contains the IPv4 and IPv6 addresses of the VPS and the SSH username and password. Managing a Linux server is usually done over Secure Shell (SSH). SSH provides a secure (encrypted) channel from your computer to the server. When you work on Linux, macOS, or Windows 10 with the latest April 2018 update, you have an SSH client already installed (see this follow-up article about Putty if you work on an older Windows).

In a shell or command prompt, issue the following command. Replace 51.38.124.133 with your IPv4 address.

ssh root@51.38.124.133

The first time you connect with SSH to a server, the client presents a prompt with the server's fingerprint. The fingerprint is a hash of the public key of the server.

The authenticity of host '51.38.124.133 (51.38.124.133)' can't be established.
ECDSA key fingerprint is SHA256:BXdQQAsjdS9P91cCwR5yRRelfyynpsfY+OxTkLSlr/A.
Are you sure you want to continue connecting (yes/no)? yes

Unfortunately, we can't verify the fingerprint; the email from OVH does not contain this information, so we have to trust that this is our server and type yes. The fingerprint is a security measure to prevent man-in-the-middle attacks, and if you can find this information before you connect, you should compare it.

Next, enter the password, and Ubuntu greets you with a welcome screen.

root@51.38.124.133's password:
Welcome to Ubuntu 18.04 LTS (GNU/Linux 4.15.0-20-generic x86_64)

 * Documentation:  https://help.ubuntu.com
 * Management:     https://ubuntu.com/landscape
 * Support:        https://ubuntu.com/pricing

  System information as of Thu Jun  7 15:58:40 CEST 2018

  System load:  0.0               Processes:           82
  Usage of /:   5.9% of 19.21GB   Users logged in:     0
  Memory usage: 6%                IP address for ens3: 51.38.124.133
  Swap usage:   0%


  Get cloud support with Ubuntu Advantage Cloud Guest:
    https://ubuntu.com/pricing

8 packages can be updated.
8 updates are security updates.

The first thing we should do is change the root password. OVH sent us the password in an unsecured email. Issue the command passwd, and it prompts for a new password. To verify that the new password works, close the SSH connection with exit and try to reconnect.

If something went wrong and you can no longer log in, open the web management console of your VPS provider. There, you should find an option where you can either reboot the VPS into a special rescue mode or another way to reset the root password.
In the OVH manager, you find the Reboot in rescue mode function. A click on this link reboots the VPS into a special mode where you can fix problems with your Linux installation. The reboot process takes about 3 minutes, and OVH sends you an email with the IP address, SSH username, and password when the rescue mode is ready. Visit this link to find more information about how to reset the root password in this mode.

As a last resort, if everything fails, most VPS providers support a Reinstall function, which deletes everything and reinstalls the operating system.

Update Operating System

In the previous welcome screen, you see that 8 installed packages can be updated. On Debian and Ubuntu, you install and update software packages with the apt command.

First, you have to update the local apt database and then upgrade all outdated installed packages.

apt update
apt full-upgrade

The local apt database contains information about all available apt packages. If you update or install a new package, apt looks in this database and installs the latest version that is listed in this database. When the database is outdated, apt installs old packages. Therefore, you must call apt update before you update or install packages. Ubuntu updates the apt database automatically once a day.

The full-upgrade command lists the outdated packages, and before it installs the updates, it asks if you want to continue. Type y, and apt installs the updates.

When the update process installs a new kernel (packages starting with linux-*), you should reboot the server. Ubuntu also tells you on the welcome screen if it needs a reboot.

To reboot the server, you issue the command reboot. This command closes the SSH connection.

Keeping the installed packages up-to-date is, in my opinion, very important, especially for a server that is connected to the Internet. Every server is going to be attacked, and if a crucial application like the SSH server has a weakness, attackers will find it and exploit it. Fortunately, Ubuntu provides a way to install critical updates automatically.

Install the unattended-upgrades package and then open the configuration file.

apt install unattended-upgrades
nano /etc/apt/apt.conf.d/50unattended-upgrades

Configure the Allowed-Origins section. If you only want Ubuntu to install important security updates automatically, comment out everything but the "${distro_id}:${distro_codename}-security"; line

Unattended-Upgrade::Allowed-Origins {
//      "${distro_id}:${distro_codename}";
        "${distro_id}:${distro_codename}-security";
        // Extended Security Maintenance; doesn't necessarily exist for
        // every release and this system may not have it installed, but if
        // available, the policy for updates is such that unattended-upgrades
        // should also install from here by default.
        "${distro_id}ESM:${distro_codename}";
//      "${distro_id}:${distro_codename}-updates";
//      "${distro_id}:${distro_codename}-proposed";
//      "${distro_id}:${distro_codename}-backports";
};

If you want Ubuntu to automatically upgrade everything, uncomment (remove //) all ${distro_id}:* lines. Save and close the file with ctrl+o and ctrl+x.

Next, we need to enable unattended-upgrades. Run the command nano /etc/apt/apt.conf.d/20auto-upgrades to open the configuration file and add these lines.

APT::Periodic::Update-Package-Lists "1";
APT::Periodic::Download-Upgradeable-Packages "1";
APT::Periodic::AutocleanInterval "7";
APT::Periodic::Unattended-Upgrade "1";

Visit the official documentation about unattended-upgrades for more information: https://ubuntu.com/server/docs

IPv6

OVH gives you one free static IPv6 address, but it's not configured yet. If you are not sure if IPv6 is enabled or not, issue this command

ping6 www.google.com

If the command prints out connect: Network is unreachable, IPv6 is not enabled.

In Ubuntu 18.04, netplan is responsible for managing the network interfaces. To add IPv6, we have to open the configuration file.

nano /etc/netplan/50-cloud-init.yaml

On my VPS, the file contains this code

network:
    version: 2
    ethernets:
        ens3:
            dhcp4: true
            match:
                macaddress: fa:16:3e:dd:b2:03
            set-name: ens3

ens3 is the name of the network interface. You see that only DHCP for IPv4 is enabled. OVH assigns static IPv4 addresses to the VPS over DHCP by maintaining a mapping in the DHCP server from MAC address to IP address, so even though it is DHCP, your server always gets the same IPv4 address.

To enable IPv6 support, we have to add an address and gateway6 element. You can find both IPv6 addresses in the OVH web console.

network:
    version: 2
    ethernets:
        ens3:
            addresses:
              - 2001:41d0:701:1100:0:0:0:e54/64
            gateway6: 2001:41d0:0701:1100:0000:0000:0000:0001
            dhcp4: true
            match:
                macaddress: fa:16:3e:dd:b2:03
            set-name: ens3

Save (ctrl+o) and close (ctrl+x) the text editor. Now, try to load the configuration.

netplan try

With the try argument, netplan activates the network configuration, and you have to press enter during the next 120 seconds if you want to keep the settings. If not, netplan reverts to the old configuration. This prevents any misconfiguration that accidentally disables IPv4 and kicks you out of the SSH session. Just wait 120 seconds, and you should be able to log in again.

To check if IPv6 is enabled, issue the command ping6 www.google.com. You can also see the configured addresses with ip -6 addr.

Harden SSH server

The SSH server is the main entry point to the server, and we should make this entry as secure as possible. The current login with username and password is not the most secure authentication method. In this section, we change the SSH configuration and make it more difficult for an attacker to gain access to our server.

Change default port

The first thing I do on a new server is to change the default listening port (22) of SSH.

Open the SSH configuration file, uncomment the Port line, and set it to a random port. For this example, I set the port to 44933.

nano /etc/ssh/sshd_config

Port 44933

Make sure that you do not accidentally choose a port that is used by another service. The following command lists all currently running services that listen on a port.

ss -tulpn | grep LISTEN

Ports from 0 to 1023 are reserved and should not be used. Choose a port higher than 1023 and less than 65535.

This change does not make your server more secure. A targeted attack is going to find all open ports easily. But it helps against drive-by scans. If you run your server for a few hours, you will notice in the /var/log/auth.log file warnings about users that tried to access the server. These are automated scripts that scan the Internet for open ports and look for ways to break in.

By changing the default SSH port from 22 to something else, we can prevent or at least significantly reduce these scans and attacks.

To activate the change, restart the SSH server sudo systemctl restart ssh and check the status sudo systemctl status ssh. You should see a green Active: active (running) message. If there is a syntax error in the configuration file, you will see a warning.

Close the SSH connection with exit and reconnect. Because we have changed the port, we need to specify the port with the -p option.

ssh -p 44933 root@51.38.124.133

Disable Protocol 1

SSH supports two protocols, 1 and 2. Version 1 is considered to be less secure and should no longer be used. Open the configuration file nano /etc/ssh/sshd_config and insert a new line Protocol 2. Ctrl+o and Ctrl+x save the change and close the text editor. Restart the SSH server with systemctl restart ssh.


Disable root login

So far, we logged in with the root user, which has superuser permissions on the server. From a security standpoint, it is better to create an unprivileged user for SSH access and then disable the root login.

Issue the command adduser manager to create a new Linux user. Instead of manager, you can name the user anything you want.

Enter a new password for the user. The command asks for information like name and phone. It is okay to accept the defaults and leave all fields blank.

This user is going to be the only user that is allowed to log in with SSH. But for managing a server, we need a user that has root privileges. The easiest way is to add this user to the sudo group. Users in this group can execute commands with the sudo command and then have the same privileges a root user has.

usermod -aG sudo manager

Close the SSH connection and try to log in with the new user ssh -p 44933 manager@51.38.124.133

Verify that the user can run commands with sudo. Run the command sudo ls -al /root; when successful, it lists the contents of the /root directory. The first time you use sudo in a session, Linux asks for the user's password.

When everything is okay, we can disable the root login in the SSH server and limit the login to the manager user. Because we are now logged in with the manager user and need to change a system file, we have to use sudo.

Issue the command sudo nano /etc/ssh/sshd_config, change the line PermitRootLogin yes to PermitRootLogin no, and insert a new line AllowUsers manager

Restart the server sudo systemctl restart ssh and check the status sudo systemctl status ssh.

Exit the SSH connection and try to log in with the root user. The client should not be able to establish a connection. Only the login with manager should work.

//FAIL
ssh -p 44933 root@51.38.124.133

//OK
ssh -p 44933 manager@51.38.124.133

Disable password login

Using public/private keys as authentication is considered to be more secure than using passwords. In this section, we're going to disable password authentication and switch to Ed25519 public/private key authentication (elliptic curve algorithm).

Check if the directory /etc/ssh contains a file with ed25519 in its name. If not, create the file with.

sudo ssh-keygen -f /etc/ssh/ssh_host_ed25519_key -N '' -t ed25519

This creates a public and a private key and stores them in the /etc/ssh/ directory. Open the SSH configuration sudo nano /etc/ssh/sshd_config and uncomment (remove #) the following line

HostKey /etc/ssh/ssh_host_ed25519_key

Save the change and restart the SSH server sudo systemctl restart ssh

Next, we need to create a public/private key pair on the client computer. Make sure that you open a new shell or close the SSH connection and issue this command

ssh-keygen -N "mysupersecretpassphrase" -t ed25519 -C "mydesktopcomputer"

This command generates a public and private key and stores them in two files (id_ed25519 and id_ed25519.pub) in the home directory of the logged-in user.

The -N option specifies a passphrase that protects the private key. You could create a private key that is not password protected, but I would advise against this because everybody that has access to the private key file can log in to the server without entering any password. The -C is a description of the key.

Next, we need to transfer the public key from the client to the server.

ssh -p 44933 manager@51.38.124.133
mkdir /home/manager/.ssh
nano /home/manager/.ssh/authorized_keys

Copy and paste the content from your public key <user_home>/.ssh/id_ed25519.pub into the editor. Close nano with ctrl+o and ctrl+x. We need to change the permission of the authorized_keys file; SSH ignores the file if it does not have the correct permissions.

chmod 0600 /home/manager/.ssh/authorized_keys

If you work on Linux or macOS, you can issue the ssh-copy-id command that does all the steps above in one command

ssh-copy-id -i ~/.ssh/id_ed25519.pub manager@51.38.124.133

If ssh-copy-id is not installed, you can run this one-liner

cat ~/.ssh/id_ed25519.pub | ssh manager@51.38.124.133 "mkdir -p ~/.ssh && touch ~/.ssh/authorized_keys && chmod -R go= ~/.ssh && cat >> ~/.ssh/authorized_keys"

Close the SSH connection and try to log in with the key.

ssh -i <user_home>/.ssh/id_ed25519 -p 44933 manager@51.38.124.133

The client asks for the private key's passphrase, and if everything is okay, it opens the connection without asking for another password.

With public/private authentication working, we can disable password authentication in the SSH server. Open the configuration file sudo nano /etc/ssh/sshd_config, search for the line PasswordAuthentication yes, and change it to PasswordAuthentication no. Save and close the editor and restart the SSH server.

When you want to access the SSH server from multiple clients, it is recommended to create a key pair on each device and add it to the /home/manager/.ssh/authorized_keys file. Insert a comment after the key so you know which client uses which key. If one of the devices gets lost or stolen, you can remove the entry in the authorized_keys file. This makes the private key useless, and nobody can access the server with this key, even when they can brute force the passphrase.


Simplify client access

To simplify the command on the client, you can enter all the command-line options in the <user_home>/.ssh/config file. Create the file if it does not already exist.

Host 51.38.124.133
  User manager
  IdentityFile ~/.ssh/id_ed25519
  IdentitiesOnly yes
  Port 44933

Now you can connect to the server with just this command: ssh 51.38.124.133

The only problem left is that we have to enter the passphrase for the private key each time we want to connect. It would be nice if we only had to do that once per session. For that, SSH provides the SSH agent. First, start the ssh-agent and then add the key with this command: ssh-add <user_home>\.ssh\id_ed25519

On Windows 10 (April 2018 Update), ssh-agent is installed as a service. It is not started automatically. To start it automatically, change the Startup Type in the Service panel to Automatic.

Install Firewall

As the last step in this tutorial, we install a firewall. I usually install the Uncomplicated Firewall UFW. UFW is a front end for iptables and eases the firewall configuration.

sudo apt install ufw

When we enable UFW, it blocks by default every incoming connection. Therefore, we must allow incoming connections to the SSH server before enabling the firewall.

sudo ufw allow 44933/tcp

Check that the rule is added

sudo ufw show added

Make sure that the port matches the listening port of the SSH server. If there is a mismatch, you can no longer connect to the server. Enable the firewall

sudo ufw enable

Check the current active firewall rules.

sudo ufw status

Close the SSH connection with exit and then try to reconnect. The firewall rule is configured correctly when the SSH client can open a connection.

For more information about UFW, visit the official homepage: https://help.ubuntu.com/community/UFW


Our plain server is ready for future endeavors. If there are errors in my description or other useful configuration steps that should be done on a new VPS server, send me a message.

If you can't wait to install software, here is an awesome list of software packages you can self-host: https://github.com/awesome-selfhosted/awesome-selfhosted

See also my other blog posts:

Self-hosted Git server with Gitea:
https://golb.hplar.ch/2018/06/self-hosted-git-server.html

Self-hosted Google Drive alternative with SparkleShare:
https://golb.hplar.ch/2018/06/self-hosted-gdrive-onedrive-dropbox-alternative.html

Sending emails:
https://golb.hplar.ch/2018/06/send-only-email.html

Self-hosted tile server:
https://golb.hplar.ch/2018/07/self-hosted-tile-server.html

Self-host Polyfill.io:
https://golb.hplar.ch/2018/01/Self-host-Polyfill-io.html