Ecosyste.ms: Awesome

An open API service indexing awesome lists of open source software.

Awesome Lists | Featured Topics | Projects

https://github.com/georgegkas/iaas-debian

Set up a clean Debian server.
https://github.com/georgegkas/iaas-debian

debian iaas-cloud ssh vm

Last synced: about 1 month ago
JSON representation

Set up a clean Debian server.

Awesome Lists containing this project

README

        

# IAAS Debian

Configure a clean Debian server.
---

This repository holds the configuration procedure I used to setup a web server in one of my [Virtual Machines](https://en.wikipedia.org/wiki/Virtual_machine) ([IAAS service model](https://en.wikipedia.org/wiki/Cloud_computing#Infrastructure_as_a_service_.28IaaS.29)). Some techniques described bellow can be used on other [VM](https://en.wikipedia.org/wiki/Virtual_machine) running the same distro too, although the initial state may differ. In my case I received the VM with a clean Debian Server installed on top, among with a plain text password and username to make the first sign in -written in a piece of paper and given to me by hand-.

I tried to use the best practices, but that does not guarantee my configuration settings are suitable for every use. I took the time to document, as much as possible, each step. A basic understanding of Linux systems is good to have, but not required. Moreover, I go beyond than simple system configuration and analyze installation procedures of software packages that you might no need.

The system is used to connect to virtual machine is Ubuntu 20.04 (LTS).

## Table Of Contents

- [Assumptions](#assumptions)
- [First Sign In](#first-sign-in)
- [Change root Password](#change-root-password)
- [Updating The System](#updating-the-system)
- [Adding New User](#adding-new-user)
- [Delete a User](#delete-a-user)
- [Generating ssh key-pair](#generating-ssh-key-pair)
- [Using ssh key-pairs](#using-ssh-key-pairs)
- [Configuring ufw Firewall](#configuring-ufw-firewall)
- [Setting Up Bash Shell](#setting-up-bash-shell)
- [Installing Fail2Ban](#installing-fail2ban)

## Assumptions

I don't want to expose my server credentials so lets make the following assumptions regarding to our VM (use the credentials provided to you):

- **USER**: user
- **PASSWORD**: secretPass0022
- **IP**: 88.23.54.32
- **DOMAIN**: example.com
- **vmNAME**: debian-jessie-server

The **USER** and **PASSWORD** info were given just for the first sign in. We'll create our own user later in the procedure and will delete the default credentials.

You may not have registered a domain name for your server. That's totaly fine, as we only need the IP address to do our work, but if you already have one associated with your IP then even better. I won't cover how to get a domain name. In my case I asked my VM provider to register one.

## First Sign In

Lets open a new terminal (Ctrl+T) and type the following command (do not press enter yet):

```
ssh [email protected]
```

We make use of ***ssh*** command to connect to our server. *[Secure Shell (SSH)](https://en.wikipedia.org/wiki/Secure_Shell)* is a UNIX-based command and cryptographic network protocol that allow us to remote login and execute commands on a computer securely.

**Note**: You can also replace the **IP** parameter in the command with the **DOMAIN** name.

We can press enter to execute the command. The first time it will promt a message.

```
The authenticity of host 'example.com (88.23.54.32)' can't be established.
ECDSA key fingerprint is 12:a7:28:01:25:aa:73:33:d6:20:67:91:52:55:3e:ff.
Are you sure you want to continue connecting (yes/no)?
```

This is a really important message. You've been asked to validate the authenticity of your server. To be said, that means that you have to ask your VM provider to send you the **key fingerprint** of your VM. Then you just check if all letters are match with the ones you see in your screen. If this is not the case, contact your VM provider and ask for additional help.

Type `yes`.

```
Password:
```

You will now be prompted to enter your **PASSWORD**. Please note that you will NOT see your cursor moving, or any characters, when typing your password. This is a standard Terminal security feature. If you typed the password correctly, you should established a new ssh session successfully.

```
[user@debian-jessie-server ~]#
```

Congratulations! You just made your first sign in. You can type `exit` any time to disconnect from the server.

## Change root Password

Before we continue our configurations we have to change the root password. There is only one acount named root. Is created by default in every Unix-like operating system (like ours). By default it has access to all commands available in the system. We also refer to root user as superuser.

We access the superuser account by executing the command bellow:

```
sudo su -
```

We are prompt to enter the root password. For me, the root password was the same as my user's one. You might be given a different root password.

```
Password:
```

If the password was entered successfully you gained access to root's account.

```
root@debian-jessie-server:/home/user#
```

We won't use the root account very much. The only reason we might do so, is to change security settings. The bellow command change the password of any user in the system. In this case it'll change the password of our superuser.

```
passwd
```

A new line will be appeared.

```
Enter new UNIX password:
```

Enter the new password you want to use, press enter and retype it. Choose a long and difficult password (eg 20-characters long).

```
Retype new UNIX password:
```

If the two passwords matches, you expect to see a confirmation message.

```
passwd: password updated successfully
```

Else you'll have to repeat the last procedure again.

## Updating our system

Next we are going to handle our system updates. The first thing you have to do is to sign in as root.

```
root@debian-jessie-server:/home/user#
```

The command that check for new updates is

```
apt update
```

and goes along with

```
apt upgrade
```

The second command do the actual installation. Every time you want to update your system you have to type those two commands. I advice you to do it often. If your system gets outdated, you can have potential security risks and that's something we don't want to.

If you haven't execute the above commands until now it's time to do so.

Note that when you execute `apt-get upgrade` command you are prompt with a question.

```
Do you want to continue? [Y/n]
```

Just type `y` or hit enter.

In this stage lets consider something. What will happen if you forget to connect to your server for a long time? For a month or two? Remember the security issues we might encounter if we don't update our system often.

To bypass this problem we are going to install a package that will automate the procedure of checking and installing **security** updates. That means, you still have to do a manual update when you connect to your VM, but now you know that the critical updates will be installed on your system.

The package is named `unattended-upgrades`. Install it using the bellow command.

```
apt-get install unattended-upgrades
```

**Note**: You can install any package you like using the `apt-get install` command.

When you promt for confirmation, type `y` and hit enter.

Now we have to configure the package. Fortunately, `unattended-upgrades` comes with default configs. To enable them type:

```
dpkg-reconfigure unattended-upgrades
```

Select `yes` when you prompt for confirmation.

## Adding New User

Next we're going to create a new user. Right now I use the one called `user`. We'd like to create a new one for ourselves to manage the tasks in our server.

First we have to be already signed into the root's account. For this example, we're going to create a new user named `georgegkas` (pick your own username).

We start by executing the bellow command *-Be sure to replace georgegkas with the user that you want to create.-*:

```
useradd georgegkas
```

Every user in Unix-based distros have it's home directory. Let's create one for our new user.

```
mkdir /home/georgegkas
```

Next we have to set home folder permissions.

```
chown georgegkas:georgegkas /home/georgegkas -R
```

The command `chown` is used to change the ownership of any file and folder in the system. We can also combine the above command with `chmod` to allow only our user to modify the contents of he's home folder.

```
chmod 700 -R /home/georgegkas
```

Next, we're going to set a password. Pick a strong one.

```
passwd georgegkas
```

Recall the command `passwd`. You can pass a parameter to indicate which user's password want to change.

Right now our new user has limited privileges on the system. Most of the time we have to run commands available only by root (such as `apt-get update`). To sign in as root every time is not a good practice and can expose our server's security.

To bypass this signin-as-root restriction we are going to add our new user to `sudo` group. Sudo stands for either "substitute user do" or "super user do". Sudo allow us to run commands as any other users (typically as root), we just have to add the `sudo` keyword at the beginning of any command.

By default, we have to install sudo on the system.

```
apt-get install sudo
```

Now add the new user to the sudo group:

```
usermod -a -G sudo georgegkas
```

Lets make a try. Sign out and sign in using the new user. In our case this is what we're going to execute:

```
ssh [email protected]
```

Type the password you set for your user and run the system update command. Typically you're going to get the following messages:

```
E: Could not open lock file /var/lib/apt/lists/lock - open (13: Permission denied)
E: Unable to lock directory /var/lib/apt/lists/
E: Could not open lock file /var/lib/dpkg/lock - open (13: Permission denied)
E: Unable to lock the administration directory (/var/lib/dpkg/), are you root?
```

As we already have said, `apt-get update` can be executed only by root or a sudoer (a sudo user). Try to run the update again using the `sudo` command like so:

```
sudo apt-get update
```

We can now update our system without the need to sign in as root. Moreover, we can run any other command we'd like as a root user, just by inserting `sudo` at the beginning.

From now on, every time we want to create a new user and allow him to execute commands as root, we have to run the `useradd` command we introduced above.

## Delete a User

Now that we have set up our primary user, we're going to delete the user that was provided to us in the beginning (named `user` in my case).

```
deluser -–remove-all-files user
```

The above command remove `user` from the system, deletes his home directory and all the files owned by this user.

## Generating ssh key-pair

SSH keys provide a more secure way of connecting to a VM. Right now, we have to type our user's password every time we want to sign in to the server. Passwords can be cracked with brute-force methods and eventually anyone who has access to a computer can connect to our server. SSH key pair provides us with two different text files containing the public and the private key respectively. We place the public key to our server and keep the private key in our machine. With this method, the attacker cannot connect to our server without the private key. We also do not have to enter the user's password every time we connect to the server. Finally, we can use the same ssh public-private key pair for as many VMs as we like.

Let's start by creating a ssh key-pair in our machine.

```
ssh-keygen -t ed25519 -a 100
```

I'm not going to analyze what the above command do. A nice post can be found on [Security stackexchange forum](https://security.stackexchange.com/questions/143442/what-are-ssh-keygen-best-practices).

```
Enter file in which to save the key (/home/georgegkas/.ssh/id_ed25519):
```

First prompt allow us to change the filename of the new generated key-pair. We'll use the default location, so hit enter to continue. By default, all ssh keys are saved under `/home/user/.ssh` folder, in this example `user` is `georgegkas`.

```
Enter passphrase (empty for no passphrase):
Enter same passphrase again:
```

Next, we've been asked to enter a passphrase for the private key. Even if we can skip this option, securing our private key with a password is highly recommended. We're going to enter that password every time we want to connect to the server that using this private key. Choose a strong one.

```
Your identification has been saved in /home/georgegkas/.ssh/id_ed25519.
Your public key has been saved in /home/georgegkas/.ssh/id_ed25519.pub.
The key fingerprint is:
SHA256:/qWwgLHsmcQBao8TuL8zlfE8j1omb0+Q2C6JzP8Yqfw me@mymachine
The key's randomart image is:
+--[ED25519 256]--+
| |
| |
| . |
|.. ..o . |
|oo +=+ S |
|.+.+oO+O |
|.o%o% =+. . |
| o+= @oo.+ o |
| ==E+ooo o |
+----[SHA256]-----+
```

We see now that we have successfully generated a new key pair. The private key have been saved as `id_ed25519` and the public key as `id_ed25519.pub`. The **key fingerprint** is used to identify the authenticity of the key pair and `me@mymachine` is the our localhost user who is able to use this key pair to connect into a server.

## Using ssh key-pairs

In this section we're going to add our ssh public key to our VM. First we're going to connect to our server as root.

```
root@debian-jessie-server:/home/georgegkas#
```

Let's create the `.ssh` folder for `georgegkas` and set the appropriate ownership. Note that, the ssh key pair can be used only with the user it has been attached to. That means, if we have another user on the server, called `alex` for example, and we store our public key only in `georgegkas`, we are not allow to connect as `alex` with that key. To do so, we have to follow the same procedure on `alex`'s account as well. Execute the following commands. *Remember to change `georgegkas` to your user account.*

```
mkdir /home/georgegkas/.ssh
chmod 700 /home/georgegkas/.ssh
touch /home/georgegkas/.ssh/authorized_keys
```

Next we're going to copy our public key to our `authorized_keys` file of our server. In your server type:

```
nano /home/georgegkas/.ssh/authorized_keys
```

to use nano text editor.

Open `id_ed25519.pub` file we created earlier with `ssh-keygen`, copy the text and add it to the nano console of the server. To paste the string type `Ctrl+Shift+V`. Save the file with `Ctrl+O` and pressing enter and close the nano window with `Ctrl+X`.

Next step is to edit the `sshd_config` file.

```
nano /etc/ssh/sshd_config
```

Using the arrow keys navigate to that line:

```
Port 22
```

Port 22 is the default port ssh uses. Change the number into something else greater than 1024, I chose 2456 for our example. This adds a layer to security by obscurity. Next nagivate until you see the line:

```
PermitRootLogin yes
```

If the value is set to `yes` change it to `no`. We won't allow anyone to connect as superuser.

Finally, navigate until you see this line:

```
#PasswordAuthentication yes
```

Change the above line to `PasswordAuthentication no`. By the time we use the key pair, we won't allow anyone to connect using the user's password.

Next change will be to only allow th user `georgegkas` to be able to connect. This is ok if we only have one user who needs to connect into our server. If we should allow more users, we can append them using the `AllowUsers` config property.

```
AllowUsers georgegkas
```

Last but not least, we should only allow secure algorithms when initializing a connection to the ssh server.

```
Ciphers aes128-ctr,aes192-ctr,aes256-ctr
HostKeyAlgorithms ecdsa-sha2-nistp256,ecdsa-sha2-nistp384,ecdsa-sha2-nistp521,ssh-rsa
KexAlgorithms ecdh-sha2-nistp256,ecdh-sha2-nistp384,ecdh-sha2-nistp521,diffie-hellman-group-exchange-sha256
```

Save the file `Ctrl+O` and enter, and close it `Ctrl+X`. We have to note in this point that the configurations saved in `sshd_config` will affect all the users in our system. That means, every user who wants to connect to the server needs to have a public ssh key installed on his home directory and use the port which defined above. We have to reload the ssh service.

```
systemctl restart ssh
```

Finally we need to reconfigure the permissions of our home directory.

```
chmod 400 /home/georgegkas/.ssh/authorized_keys
chown georgegkas:georgegkas /home/georgegkas -R
```

Exit the current ssh session. Bellow is the new method to login in your VM:

```
ssh [email protected] -p 2456
```

Point the parameter `-p` we added in our command. This indicates on what port the ssh server listens to (remember that we change the default port in `sshd_config`).

When you execute the above command, you will be prompt to enter the ssh private key password:

```
Enter passphrase for key '/home/georgegkas/.ssh/id_ed25519':
```

And now you are signed into the server.

## Configuring ufw Firewall

Now that we added ssh key pair in our server we are going to configure firewall. This is a very important step. `UFW` stands for ***Uncomplicated Firewall*** that allow us to secure our server without typing complicated commands. UFW is not installed by default on debian, so the first thing to be done is to add it on our system. Sign in as your primary user and type:

```
sudo apt-get install ufw
```

By default, after installation ufw will be disabled. You can check the state of the firewall using the following command:

```
sudo ufw status
```

You can activate ufw by typing:

```
sudo ufw enable
```
If you check the status again you'll see that ufw was enabled:

```
Status: active
```

Right now there is no active rule. We are going to block all incoming traffic by default and explicitly allow only specific traffic to known services. This strategy provides good control over the traffic and reduces the possibility of a breach because of service misconfiguration. To deny all incoming traffic type:

```
sudo ufw default deny incoming
```

We have denied all the incoming connections to the server. But wait a second! We still want to be able to connect to our VM through ssh. That means, we have to add ssh port to whitelist. To do that type:

```
sudo ufw allow 2456/tcp
```

Change `2456` with the port number you use to connect through ssh. The above command allows incoming tcp packets on port 2456, basically we are able to connect with ssh.

Now that we configured the incoming connections let's configure the outgoing.

```
sudo ufw default allow outgoing
```

We're going to allow all the connection coming from the server.

That's all about basic firewall configuration.

## Setting Up Bash Shell

This section includes advices on how to make our workplace easier to use. We're going to activate [Bash shell](https://en.wikipedia.org/wiki/Bash_(Unix_shell)) which is a very handy shell with great features. It's also the default shell for Ubuntu distros, but we have to configure it on debian. Start by editing `.profile` file in your user's home directory:

```
nano .profile
```

Add the bellow lines if their not already exist in the file:

```
# ~/.profile: executed by the command interpreter for login shells.
# This file is not read by bash(1), if ~/.bash_profile or ~/.bash_login
# exists.
# see /usr/share/doc/bash/examples/startup-files for examples.
# the files are located in the bash-doc package.

# the default umask is set in /etc/profile; for setting the umask
# for ssh logins, install and configure the libpam-umask package.
#umask 022

# if running bash
if [ -n "$BASH_VERSION" ]; then
# include .bashrc if it exists
if [ -f "$HOME/.bashrc" ]; then
. "$HOME/.bashrc"
fi
fi
```

Next create a new `.bashrc` file using `touch` command.

```
touch .bashrc
```

Add the bellow lines using nano editor.

```
# ~/.bashrc: executed by bash(1) for non-login shells.
# see /usr/share/doc/bash/examples/startup-files (in the package bash-doc)
# for examples

# If not running interactively, don't do anything
case $- in
*i*) ;;
*) return;;
esac

# don't put duplicate lines or lines starting with space in the history.
# See bash(1) for more options
HISTCONTROL=ignoreboth

# append to the history file, don't overwrite it
shopt -s histappend

# for setting history length see HISTSIZE and HISTFILESIZE in bash(1)
HISTSIZE=1000
HISTFILESIZE=2000

# check the window size after each command and, if necessary,
# update the values of LINES and COLUMNS.
shopt -s checkwinsize

# If set, the pattern "**" used in a pathname expansion context will
# match all files and zero or more directories and subdirectories.
#shopt -s globstar

# make less more friendly for non-text input files, see lesspipe(1)
[ -x /usr/bin/lesspipe ] && eval "$(SHELL=/bin/sh lesspipe)"

# set variable identifying the chroot you work in (used in the prompt below)
if [ -z "${debian_chroot:-}" ] && [ -r /etc/debian_chroot ]; then
debian_chroot=$(cat /etc/debian_chroot)
fi

# set a fancy prompt (non-color, unless we know we "want" color)
case "$TERM" in
xterm-color|*-256color) color_prompt=yes;;
esac

# uncomment for a colored prompt, if the terminal has the capability; turned
# off by default to not distract the user: the focus in a terminal window
# should be on the output of commands, not on the prompt
#force_color_prompt=yes

if [ -n "$force_color_prompt" ]; then
if [ -x /usr/bin/tput ] && tput setaf 1 >&/dev/null; then
# We have color support; assume it's compliant with Ecma-48
# (ISO/IEC-6429). (Lack of such support is extremely rare, and such
# a case would tend to support setf rather than setaf.)
color_prompt=yes
else
color_prompt=
fi
fi

if [ "$color_prompt" = yes ]; then
PS1='${debian_chroot:+($debian_chroot)}\[\033[01;32m\]\u@\h\[\033[00m\]:\[\033[01;34m\]\w\[\033[00m\]\$ '
else
PS1='${debian_chroot:+($debian_chroot)}\u@\h:\w\$ '
fi
unset color_prompt force_color_prompt

# If this is an xterm set the title to user@host:dir
case "$TERM" in
xterm*|rxvt*)
PS1="\[\e]0;${debian_chroot:+($debian_chroot)}\u@\h: \w\a\]$PS1"
;;
*)
;;
esac

# enable color support of ls and also add handy aliases
if [ -x /usr/bin/dircolors ]; then
test -r ~/.dircolors && eval "$(dircolors -b ~/.dircolors)" || eval "$(dircolors -b)"
alias ls='ls --color=auto'
#alias dir='dir --color=auto'
#alias vdir='vdir --color=auto'

alias grep='grep --color=auto'
alias fgrep='fgrep --color=auto'
alias egrep='egrep --color=auto'
fi

# colored GCC warnings and errors
#export GCC_COLORS='error=01;31:warning=01;35:note=01;36:caret=01;32:locus=01:quote=01'

# some more ls aliases
alias ll='ls -alF'
alias la='ls -A'
alias l='ls -CF'

# Add an "alert" alias for long running commands. Use like so:
# sleep 10; alert
alias alert='notify-send --urgency=low -i "$([ $? = 0 ] && echo terminal || echo error)" "$(history|tail -n1|sed -e '\''s/^\s*[0-9]\+\s*//;s/[;&|]\s*alert$//'\'')"'

# Alias definitions.
# You may want to put all your additions into a separate file like
# ~/.bash_aliases, instead of adding them here directly.
# See /usr/share/doc/bash-doc/examples in the bash-doc package.

if [ -f ~/.bash_aliases ]; then
. ~/.bash_aliases
fi

# enable programmable completion features (you don't need to enable
# this, if it's already enabled in /etc/bash.bashrc and /etc/profile
# sources /etc/bash.bashrc).
if ! shopt -oq posix; then
if [ -f /usr/share/bash-completion/bash_completion ]; then
. /usr/share/bash-completion/bash_completion
elif [ -f /etc/bash_completion ]; then
. /etc/bash_completion
fi
fi
```

Finally enable the bash shell for our user:

```
chsh -s /bin/bash
```

Exit the ssh session and sign in again. You should see a friendlier terminal, the same one we use in our Ubuntu machine.

## Installing Fail2Ban

Fail2Ban is an application that reads log files on the system and can detect any potential automated attack. Using a predefined set of rules, when Fail2Ban suspect that someone try to compromise the system, will add rules on your firewall, thus blocking IP address of the attacker, either for a set amount of time or permanently. Fail2Ban can monitor a wide variety of system services (such as http, ssh, sftp, etc), but we will configure only ssh at that point.

To install Fail2Ban type:

```
apt-get install fail2ban
```

*Hint:* Remember to run the above command with sudo.

First, we're going to create a copy of the default configuration settings located in `fail2ban.conf` file into a new file named `fail2ban.local`. If we want to make any changes in the configuration settings, we only should change the `.local` file.

```
sudo cp /etc/fail2ban/fail2ban.conf /etc/fail2ban/fail2ban.local
```

The same rule applies for the `jail.conf` file. In `jail.local` we are going to set the restriction rules.

```
sudo cp /etc/fail2ban/jail.conf /etc/fail2ban/jail.local
```

Let's open `jail.local` and configure it.

```
sudo nano /etc/fail2ban/jail.local
```

Under the `[DEFAULT]` section find the settings `bantime`, `findtime`, `maxretry`. The first option sets the number of seconds the IP address will be banned. The second option specifies the length of time between failed login attempts before a ban is set. The third option reveals the maximum number of failed attempts before the IP will be blocked.

By default, we allow to make 3 failed login attempts withing 600 seconds (= 10 minuets). If we exceed this number of tries, our IP will be banned for 10 minuets. The default options are well defined, but I'd like to change the `bantime` to some higher value (eg 3600 = 1 hour).

Next move down until you see the bellow lines:

```
[ssh]

enabled = true
port = ssh
filter = sshd
logpath = /var/log/auth.log
maxretry = 6
```

The above entry configures ssh. What we have to do here is to change the `port` option to the port we use to connect over ssh. We also can lower the max tries. In the end we change to something like this:

```
[ssh]

enabled = true
port = 2456 # change that number with your ssh port
filter = sshd
logpath = /var/log/auth.log
maxretry = 3 # we lowered the max tries
```

We can now save the file. To check if we configured the application correctly run:

```
sudo fail2ban-client status
```

The output should be:

```
Status
|- Number of jail: 1
`- Jail list: ssh
```