Skip to content

System Administration: Configuring a secure virtual machine with user authentication, firewall rules, and system monitoring.

License

Notifications You must be signed in to change notification settings

alx-sch/42_born2beroot

Folders and files

NameName
Last commit message
Last commit date

Latest commit

 

History

47 Commits
 
 
 
 
 
 
 
 

Repository files navigation

42_born2beroot

born2beroot_badge.png

This project serves as a practical exercise in system administration: A secure virtual machine is configured, involving user authentication, firewall rules, and system monitoring, with a focus on Linux system administration and network security.

Rather than providing another step-by-step installation guide, this documentation highlights certain challenges I noticed during evaluations, which are also overlooked by many online guides for this project.

Features

  • Operating System Configuration: Set up a Debian server with minimal services and create partitions using LVM.
  • User and Group Management: Manage users and groups through sudoers configuration for controlled access and privileges.
  • Security Measures: Implement robust security protocols, covering SSH configuration, root access restriction, UFW setup, and enforcing a stringent password policy.
  • System Monitoring Script: Develop a monitoring.sh bash script to provide regular system information updates on all terminals at predefined intervals.
  • WordPress Website Setup: Host a WordPress website using lighttpd, MariaDB, and PHP.

Guides Used

I used the following guides to set up my virtual machine, acknowledging certain limitations as highlighted in the sections below:

Installing Debian OS

  • Download the latest version of Debian here.
  • To install Debian with the specified partitions as mentioned in the bonus section, I followed this video guide by Youssef Oussama.
    • ⚠️ Ensure that you install the server on 42's shared storage space "sgoinfre." Be mindful not to exceed the disk space limits specified by your school's policy (the partition sizes presented in the project's subject are examples; a total size of ~8 GB should work just fine).
    • ⚠️ Note how the volume "var--log" (as listed with lstblk) is named "var-log" during setup (see 7:50 timestamp in the video guide). Naming it "var--log" would result in a volume called "var----log". Screenshot from 2023-12-15 15-11-14 (1)

Password Policy

During OS Installation:

  • I recommend adhering to the password policy when setting up passwords for both the root and main user. Also, please keep following the same policy for accessing the encrypted partition; even if it's not explicitly specified in the subject, it is still good practice. By establishing strong passwords from the beginning, you can avoid the need to change them later and mitigate the risk of forgetting about it, as there is no prompt for rule changes.

Editing Password Settings:

  • ⚠️ Some guides only change settings in /etc/pam.d/common-password. However, this doesn't affect password requirements for existing users (root, main) when changing the password (passwd username). Ensure that you edit /etc/security/pwquality.conf instead for these changes to take effect.
  • ⚠️ After editing the password aging settings in /etc/login.defs, make sure to manually apply corresponding policies to existing users using commands like chage -m 2 -M 30 -W 7 root or chage -m 2 -M 30 -W 7 username. Check password aging settings via sudo chage -l username.

Testing Password Policy Setup:

  • Test the correctness of the password policy setup by creating a user and providing a 'bad' password, e.g. by simply pressing 'Enter' when asked for one (sudo adduser test_user). Although this user will be created, attempting to log in or switch to the user should prompt for a password that was not set up, rendering the user effectively unusable. Make sure to use the '-r' flag to remove the user along with the user's home directory and its contents: sudo userdel -r test_user.

Port Setup

  • The project's specifications require the SSH service to run on port 4242 exclusively. However, there's a chance that this port is already in use on your network. In such cases, ssh username@localhost -p 4242 may not work for logging into your server ('guest') from your machine ('host'). To address this, you can use port forwarding, allowing you to log in using a different port. For example, with the setup below, ssh username@localhost -p 4243 does the trick.

  • In VirtualBox, open the settings of your virtual machine/server -> Network -> Advanced (Adapter 1) -> Port Forwarding.

    VirtualBox Port Forwarding

Editing Sudoers

  • When modifying the sudoers file, use sudo visudo instead of calling an editor (e.g., sudo nano /etc/sudoers). Visudo not only grants you access to edit the sudoers file but also performs syntax checking, reducing the risk of errors that could potentially lock you out of sudo access.

Monitoring Script

  • As with everything you find online: Take it with a grain of salt and make sure to understand and test before implementing it into your own project. I've encountered a few monitoring scripts that were not 100% correct (e.g., counting the header line of a table showing TCP connections, therefore always being "1" too much).
  • Here is my monitoring script; please do not copy and paste it before testing it yourself! ;)
#!/bin/bash

# ARCH
arc=$(uname -a)

# CPU PHYSICAL
cpu_p=$(grep "physical id" /proc/cpuinfo | sort | uniq | wc -l)

# CPU VIRTUAL
cpu_v=$(grep "^processor" /proc/cpuinfo | wc -l)

# RAM
ram_total=$(free -m | awk '$1 == "Mem:" {print $2}')
ram_use=$(free -m | awk '$1 == "Mem:" {print $3}')
ram_perc=$(free | awk '$1 == "Mem:" {printf("%.2f"), $3/$2*100}')

# DISK
disk_total=$(df -BG | grep '^/dev/' | grep -v '/boot$' | awk '{ft += $2} END {print ft}')
disk_use=$(df -BM | grep '^/dev/' | grep -v '/boot$' | awk '{ut += $3} END {print ut}')
disk_perc=$(df -BM | grep '^/dev/' | grep -v '/boot$' | awk '{ut += $3} {ft+= $2} END {printf("%d"), ut/ft*100}')

# CPU LOAD
cpu_l=$(top -bn1 | grep '^%Cpu' | cut -c 9- | xargs | awk '{printf("%.1f%%"), $1 + $3}')

# LAST BOOT
lb=$(who -b | awk '$1 == "system" {print $3 " " $4}')

# LVM USE
lvmu=$(if [ $(lsblk | grep "lvm" | wc -l) -eq 0 ]; then echo no; else echo yes; fi)

# TCP CONNECTIONS
tcpc=$(ss -neopt state established | grep -v "Recv-Q" | wc -l)

# USER LOG
ulog=$(users | wc -w)

# NETWORK
ip=$(hostname -I)
mac=$(ip link show | grep "ether" | awk '{print $2}')

# SUDO COMMANDS                             
cmds=$(journalctl _COMM=sudo | grep COMMAND | wc -l)

# Multiline message
message="
     #Architecture: $arc
     #CPU physical: $cpu_p
     #vCPU: $cpu_v
     #Memory Usage: $ram_use/${ram_total}MB ($ram_perc%)
     #Disk Usage: $disk_use/${disk_total}GB ($disk_perc%)
     #CPU load: $cpu_l
     #Last boot: $lb
     #LVM use: $lvmu
     #Connections TCP: $tcpc ESTABLISHED
     #User log: $ulog
     #Network: IP $ip ($mac)
     #Sudo: $cmds commands"

# Send the message to all logged-in users
echo "$message" | wall`
  • arc=$(uname -a)

    • Purpose: Retrieves system architecture information.
    • Explanation: The uname -a command prints system information, including the system architecture (e.g., x86_64), kernel version, and other details.
  • cpu_p=$(grep "physical id" /proc/cpuinfo | sort | uniq | wc -l)

    • Purpose: Counts the number of physical CPUs.
    • Explanation: This line uses a series of commands to extract unique physical CPU IDs from /proc/cpuinfo, sorts them, and counts the unique IDs using wc -l (number of output lines).
  • cpu_v=$(grep "^processor" /proc/cpuinfo | wc -l)

    • Purpose: Counts the number of virtual CPUs.
    • Explanation: This line counts the number of lines starting with "processor" in /proc/cpuinfo, representing virtual CPUs.
  • ram_total=$(free -m | awk '$1 == "Mem:" {print $2}')

    • Purpose: Retrieves total RAM in megabytes.
    • Explanation: Uses the free -m command to display memory information in megabytes, then uses awk to extract the total RAM (in MB) from the line starting with "Mem:".
  • ram_use=$(free -m | awk '$1 == "Mem:" {print $3}')

    • Purpose: Retrieves used RAM in megabytes.
    • Explanation: Similar to the previous line, extracts the used RAM (in MB) from the "Mem:" line using awk.
  • ram_perc=$(free | awk '$1 == "Mem:" {printf("%.2f"), $3/$2*100}')

    • Purpose: Calculates the percentage of used RAM.
    • Explanation: Uses awk to calculate the percentage of used RAM and formats the result to two decimal places. It divides used memory ($3) by total memory ($2) and multiplies by 100.
  • disk_total=$(df -BG | grep '^/dev/' | grep -v '/boot$' | awk '{ft += $2} END {print ft}')

    • Purpose: Calculates total disk space in gigabytes.
    • Explanation: Uses df -BG to display disk space information in gigabytes, filters lines starting with '/dev/' excluding '/boot', and calculates the total disk space by summing up the second field ($2).
  • disk_use=$(df -BM | grep '^/dev/' | grep -v '/boot$' | awk '{ut += $3} END {print ut}')

    • Purpose: Calculates used disk space in megabytes.
    • Explanation: Similar to the previous line, calculates the total used disk space by summing up the third field ($3).
  • disk_perc=$(df -BM | grep '^/dev/' | grep -v '/boot$' | awk '{ut += $3} {ft+= $2} END {printf("%d"), ut/ft*100}')

    • Purpose: Calculates the percentage of used disk space.
    • Explanation: Calculates the percentage of used disk space by dividing used disk space (ut) by total disk space (ft), multiplying by 100, and formatting the result as an integer.
  • cpu_l=$(top -bn1 | grep '^%Cpu' | cut -c 9- | xargs | awk '{printf("%.1f%%"), $1 + $3}')

    • Purpose: Retrieves CPU load percentage.
    • b: batch (snapshot) → not dynamic/real-time updates
    • n1: runs for a single iteration and then stops
    • Explanation: Uses top to get CPU usage information, extracts the CPU load percentage, and formats the result to one decimal place (xargs takes the output of the previous command and converts it into arguments, turning the multi-column output into a single line of space-separated values).
  • lb=$(who -b | awk '$1 == "system" {print $3 " " $4}')

    • Purpose: Retrieves the last system boot time.
    • Explanation: Uses who -b to get information about the last system boot and extracts the date and time.
  • lvmu=$(if [ $(lsblk | grep "lvm" | wc -l) -eq 0 ]; then echo no; else echo yes; fi)

    • Purpose: Checks if Logical Volume Management (LVM) is in use.
    • Explanation: Uses lsblk to check if any block devices mention "lvm." If the count is zero, it echoes "no"; otherwise, it echoes "yes".
  • tcpc=$(ss -neopt state established | grep -v "Recv-Q" | wc -l)

    • Purpose: Counts established TCP connections.
    • Explanation: Uses ss to list established TCP connections and counts the lines. grep -v "Recv-Q" excludes the header line
  • ulog=$(users | wc -w)

    • Purpose: Counts the number of logged-in users.
    • Explanation: Uses users to list logged-in users and wc -w to count the number of words (users).
  • ip=$(hostname -I)

    • Purpose: Retrieves the system's IP address.
    • Explanation: Uses hostname -I to obtain the system's IP address.
  • mac=$(ip link show | grep "ether" | awk '{print $2}')

    • Purpose: Retrieves the MAC address of the network interface.
    • Explanation: Uses ip link show to list network interfaces, filters lines containing "ether" (MAC address), and extracts the MAC address. The MAC address (Media Access Control address) is a unique identifier assigned to network interfaces for communication on a network.
  • cmds=$(journalctl _COMM=sudo | grep COMMAND | wc -l)

    • Purpose: Counts the number of sudo commands executed.
    • Explanation: Uses journalctl to query the system's journal for sudo commands, filters lines with "_COMM=sudo" and "COMMAND", and counts the lines.
  • The wall command in Linux is used to send a message to all users currently logged into the system. When you pipe a message into wall using the echo command, it broadcasts that message to all open terminal sessions.

Setting up a Cron Job

  • While the directory /usr/local/bin is commonly designated for locally installed user executables, you are free to place your custom scripts anywhere you prefer.
  • Many guides set up the cron job to execute the monitoring script like this in the crontab settings (sudo crontab -u root -e):
    */10 * * * * sh /usr/local/bin/monitor_script.sh
  • ⚠️ While this setup does execute the script every 10 minutes, it does so every full 10 minutes on the clock (e.g., 12:00, 12:10, 12:20, ...). The project's subject, however, specifies that the monitoring information is to be shown at server startup and then every 10 minutes. This means that if the server is started at 12:34:56 (HH:MM:SS), the monitoring info is expected to be displayed right away and then at 12:44:56, 12:54:56, 01:04:56, and so on.
  • To align with the project requirements, consider the crontab setting below (sudo crontab -u root -e):
    @reboot sleep 30 && /usr/local/bin/monitor_script.sh
    */10 * * * * /usr/local/bin/sleep_script.sh && /usr/local/bin/monitor_script.sh
    • @reboot sleep 30 ensures a 30-second delay to allow the user to access the encrypted partition and log in before executing the monitoring script. This delay is necessary as system information would not be displayed while the user is still accessing.

    • sleep_script.sh introduces a delay in the execution of the monitoring script, ensuring that system information is displayed 10 minutes after server startup. By employing a separate script, you can conveniently adjust the scheduling of system information messages without modifying the monitoring script itself:

      #!/bin/bash
      
        # boot time(hh:min); 12:34:56
        boot_time=$(last -FR | grep reboot | head -n 1 | awk '{print $7}')
      
        # boot time, min single digit; 12:34:56 -> 4
        single_digit_minute=$(echo "$boot_time" | awk '{printf substr($1,5,1)}')
      
        # boot time, sec; 12:34:56 -> 56
        seconds=$(echo "$boot_time" |  awk '{printf substr($1,7,2)}')
      
        sleep $(($single_digit_minute*60 + $seconds))
    • boot_time=$(last -FR | grep reboot | head -n 1 | awk '{print $7}')

      • Purpose: Retrieves the timestamp of the last system reboot.
      • Uses the 'last' command with options -FR to display the full date and seconds for the reboot history. It then filters lines containing 'reboot', takes the first line using head -n 1, and extracts the seventh column (timestamp) using awk.
    • single_digit_minute=$(echo "$boot_time" | awk '{printf substr($1,5,1)}')

      • Purpose: Extracts the single-digit minute from the boot timestamp.
      • Explanation: Takes the value stored in boot_time and uses awk to extract a substring starting from the 5th character with a length of 1 (the single-digit minute).
    • seconds=$(echo "$boot_time" | awk '{printf substr($1,7,2)}')

      • Purpose: Extracts the seconds from the boot timestamp.
      • Explanation: Similar to the previous line, it uses awk to extract a substring starting from the 7th character with a length of 2 (the seconds).
  • Ensure that the cron jobs can execute the scripts by setting the appropriate permissions. You can check the permissions with ls -l. If adjustments are needed, use the following commands:
    chmod +x /usr/local/bin/monitor_script.sh
    chmod +x /usr/local/bin/sleep_script.sh

WordPress Website Setup

  • This guide explains the setup neatly in step 1–5.
  • Access your website through your browser at http://localhost:8080, where you'll be prompted to sign up for WordPress. To edit your website, visit http://localhost:8080/wp-admin, using the credentials you've just set up. Please note that the use of port 8080 is a personal choice, as this port is often employed for development and testing purposes, and port 80 was not available on my network. By default, Lighttpd is configured to listen on port 80, the standard port for HTTP (check Port Setup to see how this is managed).

Saving the Virtual Machine State

  • Taking a snapshot in VirtualBox allows you to capture the current state of your virtual machine, making it easy to revert back to that state later. This can be especially useful before making significant changes, ensuring a stable point to return to. Reverting back to a snapshot after an evaluation ensures that the hash value of your virtual machine still aligns with the one you uploaded. Follow these steps to take a snapshot:
    • Make sure that your virtual machine is powered off before taking a snapshot.
    • Open the VirtualBox Manager and select your virtual machine. Navigate to the 'Snapshots' tab by clicking the 'list icon' (symbolized as a stack of lines) next to the virtual machine.
    • Click on the 'Take' button to create a snapshot. You'll notice the 'Current State' is 'changed' once the virtual machine starts.
    • To revert to the snapshot state, select the desired snapshot and choose 'Restore.' Screenshot from 2024-01-16 13-23-44

Acknowledgements

  • I would like to thank the authors mentioned in Guides Used for their valuable documentation.
  • The project badge used is retrieved from this repo by Ali Ogun.
  • A big thank you to the Debian and VirtualBox communities for their outstanding contributions and dedication to open-source ethos.

About

System Administration: Configuring a secure virtual machine with user authentication, firewall rules, and system monitoring.

Topics

Resources

License

Stars

Watchers

Forks