Linux VPS servers have their advantages. In fact, Linux VPS are much more secure when compared to other operating systems like Windows because of Linux’s security model (LSM). But they’re not perfect, and definitely not invulnerable. In this post we’ll go over most important steps you can do to secure your VPS and protect it from hackers.
Use a non-root user
First thing I usually do to secure a new VPS instance is to create a new user. I use this user instead of root
to log into the VPS and run commands. It's good to avoid using the root
user because it has unrestricted access to the machine and can potentially cause serious damage.
Using a non-root user follows the principle of least privilege and adds a layer of protection against intruders looking to gain root
access.
Creating a new user
While logged into the VPS as root
, create a new user with the following command:
# Choose any username you want
adduser username
passwd username
# Provide some strong password
All commands in this article are for Ubuntu. Other Linux distributions usually have equivalent programs that can do the same thing. If you Google <command> equivalent <distribution>
(e.g.: adduser equivalent fedora
) you have a good chance of finding what you're looking for.
Test the create user
$ id username
uid=1000(username) gid=1000(username) groups=1000(username)
We'll give this user the ability to run commands as root
with sudo
since we will need that later in this article. You can accomplish that by adding the user to the sudo
group:
# Use your username here
usermod -aG sudo username
-aG
stands for append to group. If you run id <user>
again you can see sudo
has been added to this user's group list.
$ id username
uid=1000(username) gid=1000(username) groups=1000(username),27(sudo)
Setup SSH for the new user
Next step is to add your public key to the user's .ssh
folder so you can log in as the user with SSH.
Let's switch to the new user and create a .ssh
folder in the user's home directory:
# Switch to the new user
su - username
# Create the SSH folder
mkdir ~/.ssh
SSH is nit-picky about files and folders permissions and it will fail with an error if they are too permissive. Let's change the folder's permission so only this user is allowed to read and write to it:
# Restrict access to this user only
chmod 700 ~/.ssh
In this folder, we'll create a file named authorized_keys
which will contain your public key. SSH will look for the key in this file when you connect to the server.
# Create and open the file where SSH will look for your public key
nano ~/.ssh/authorized_keys
The above command will open the file in the nano
editor allowing you to paste your public key. Hit CTRL + X
to quit, then Y
and enter
to save.
A quick way to copy the contents of a file in the terminal is to use pbcopy
. To copy your public key, navigate to the .ssh
folder on your machine and run pbcopy <aside id_rsa.pub
(assuming id_rsa.pub
is the name of your public key file). The public key is then copied to the clipboard and you can paste it anywhere you want.
Confirm the key is saved by running cat ~/.ssh/authorized_keys
. This will print the contents of the file to the terminal and you should see your public key.
Let's change the permissions of this file as well:
# Restrict access to this user only
chmod 600 ~/.ssh/authorized_keys
Open a second terminal session and confirm you can SSH into the server with ssh <user>@<server_ip>
. If you've followed the above steps correctly, you should see the Ubuntu welcome message.
Close the first terminal session by running exit
twice and proceed to the next section.
Disable password and root login
A private/public key pair has many more bits of data making them impossible to guess in within the human lifespan.
It's good practice to disable password login and protect your VPS from these type of attacks. It also gives you peace of mind that, unless an attacker gets a hold of your private key, the only person who will have access to the server is you.
While we're at it, we will also disable logging in as root
since we will not be using it anymore.
Both settings are in a global SSH configuration file. Changing this file requires you to use sudo
which will ask you for a password. This is the same password you chose when you created the user.
# Open the SSH config file in the `nano` editor
sudo nano /etc/ssh/sshd_config
Scroll down until you find PermitRootLogin
and change its value to no
. Do the same for PasswordAuthentication
. Make sure they're not commented out by removing any leading #
(e.g.: #PasswordAuthentication
). Press CTRL + X
to quit, then Y
and enter
to save.
Finally, you need to restart the SSH service for the changes to take effect:
# Restart the SSH service
sudo systemctl reload sshd
Verify that root and password login are disabled by trying to log in as the root user ssh root@<server_ip>
. You should see Permission denied (publickey).
printed in the terminal.
Block incoming traffic on non-public ports
Using a firewall to close all ports, except those that need to be public, is an essential part of server security. Every application running on a machine is a potential vulnerability that can be exploited by an intruder.
Fewer exposed applications means a smaller attack surface resulting in a more secure server.
A firewall consists of a set of rules. Each rule serves as a filter for incoming/outgoing traffic deciding whether a package is allowed to continue to its destination or not.
We'll use UFW (Uncomplicated FireWall) to create firewall rules. UFW is an easy-to-use program built on top of iptables
— the traditional and more difficult interface for configuring firewall rules. UFW comes pre-installed on Ubuntu.
For most newly-created VPS instances, we want to begin by opening up three ports: 22 for SSH, 80 for HTTP traffic and 443 for HTTPS traffic. You can open a specific port or you can specify a service name that is mapped to a port. In the examples below we use service names. sudo ufw allow http
is the same as sudo ufw allow 80/tcp
.
# Allow SSH connections
sudo ufw allow ssh
# Allow HTTP traffic
sudo ufw allow http
# Allow HTTPS traffic
sudo ufw allow https
You will see two rules added after each command, one for IPv4 and another for IPv6. The rules don't go into effect until we enable the firewall:
# Enable firewall
sudo ufw enable
You will get a warning that enabling the firewall might close the active SSH connection. We've added a rule for SSH so we don't have to worry about that. Type y
followed by enter
to proceed.
Let's verify the firewall is up and running:
$ sudo ufw status
Status: active
To Action From
-- ------ ----
22/tcp ALLOW Anywhere
80/tcp ALLOW Anywhere
443/tcp ALLOW Anywhere
22/tcp (v6) ALLOW Anywhere (v6)
80/tcp (v6) ALLOW Anywhere (v6)
443/tcp (v6) ALLOW Anywhere (v6)
You might add additional services in the future that need to be open to the public. The syntax for exposing a port is: sudo ufw allow <port>/tcp
. To close a port, you run: sudo ufw deny <port>/tcp
. Changes will take effect immediately without needing to restart the firewall.
Enable automatic security updates
The final step to making the server more secure is to enable automatic security updates. The open-source community frequently releases security updates making sure vulnerabilities are patched soon after they are found. Conducting manual updates regularly is time-consuming and often forgotten.
By automating this process you're making sure the system is up-to-date with the latest security fixes.
Note: Automatic updates are only applied to software installed through the Ubuntu package manager and don't include application-specific dependencies (e.g.: npm dependencies for Node.js). It's good practice to update your app's dependencies from time to time.
Ubuntu comes pre-installed with a package named unattended-upgrades
that lets us configure automatic updates. This tool will regularly check if the packages installed on your machine have new releases and install them if needed. You can granularly specify which updates you want to be installed and which ones to skip. We'll turn automatic updates only for security releases to minimise the chance of introducing breaking changes.
The following command enables automatic security updates:
# Enable automatic security updates
sudo dpkg-reconfigure --priority=low unattended-upgrades
You will see an interactive dialog asking if you want to enable automatic updates, select Yes
.
This command edits two files:
/etc/apt/apt.conf.d/20auto-upgrades
(configures when and how the script should run)/etc/apt/apt.conf.d/50unattended-upgrades
(configures whether an update should be installed or skipped)
If you want to learn more, visit the automatic security updates documentation on the Ubuntu website.
To verify that automatic updates are enabled:
$ apt-config dump APT::Periodic::Unattended-Upgrade
APT::Periodic::Unattended-Upgrade "1";
The "1"
stands for every 1
day. You've now enabled daily automatic security updates for your server and won't have to worry about that anymore
Disable protocol 1
The SSH service works with 2 protocols namely protocol 1 and protocol 2. Protocol 1 has lesser security compared to the other, so it's better to use protocol 2 in your communications.
In order to disable protocol 1, open the sshd_config file with an editor.
nano /etc/ssh/sshd_config
Find the following statement and change to "protocol 2".
# Protocol 2,1
Protocol 2
Restart the SSH service after making changes and save the file.
Service ssh restart
Use non-standard ports for SSH
The default SSH service port is 22, so hackers will check this port before anything else. In some cases, administrators change the SSH port to 2222, but you should know that hackers will surely scan the port 22 and if they get no result, their second choice will be the port 2222. It's better to use the ports with a lot of digits that are not reserved for other services. The best choice is between 10,000 and 65,000, in which most of them are free.
Additional security measures include file auditing, isolated execution environments (with Docker), rate-limiting with fail2ban, and more.