Taco de Wolff
The perfect webserver
To run a secure and performant webserver, you need to think-through the layout and configuration of your services. You want to enable HTTP/2 and set secure SSL settings while setting the file permissions in a way that even in case a website gets hacked, no other part of the webserver is vulnerable. This is what I learned when I replaced my second VPS with a better one.
My VPS runs on CentOS 7 with 4GB RAM and a dual core.
Software
NGINX - the webserver that will process requests and return responses. It also does SSL termination and deferral to PHP-FPM.
PHP-FPM - execute PHP scripts in worker threads. This is a performant way to process PHP scripts, as the worker threads keep running to process new requests. This is unlike the historically more common mod_php, which spawns a worker for every request. PHP code should be cached in socache to reduce the overhead of parsing the script. For sessions we will use an in-memory cache.
Let’s Encrypt and lego - for our SSL certificates we will use the Let’s Encrypt service that provides free SSL certificates. The ACME client we use to obtain the certificates is github.com/xenolf/lego, which is a Go library with a CLI with which we can automate the creation and renewal of certificates.
Users and file permissions
We will use different user accounts to separate and secure the different websites that are hosted. That is to say, each website will have its own system account that owns the files. PHP-FPM will have a pool for each website to run using that user. All the files will be restricted so that only the user can edit the files and the group (NGINX) has read access.
The websites will be located under /var/www
. Each website will have subdirectories to store the websites, the temporary files, logs, and emails for example. The website files will have subdirectories for all subdomains so that we can add subdomains automatically by adding a new subdirectory. The subdirectory _
refers to when no subdomain is given and www
for example is the common www subdomain. The file layout is thus as follows:
/var/www/
├── default/ # HTML files displayed if no domain matches
├── jaap.nl/
│ ├── tmp/
│ ├── log/
│ └── html/
│ ├── _/
│ ├── www/
│ └── [more subdomains]
├── piet.nl/
│ ├── tmp/
│ ├── log/
│ └── html/
│ ├── _/
│ ├── zwarte/
│ └── [more subdomains]
└── [more websites]
To continue with the jaap.nl
example, we create a new website and system user as follows:
mkdir -p /var/www/jaap.nl/html/_
mkdir /var/www/jaap.nl/tmp
mkdir /var/www/jaap.nl/log
# make sure the user is a system user and cannot login
useradd -d /var/www/jaap.nl -s /sbin/nologin jaap >/dev/null 2>&1
usermod -a -G jaap nginx
# set all permissions correctly, run whenever to make sure all is correct
chown -R jaap:jaap /var/www/jaap.nl
find /var/www/jaap.nl -type d -exec chmod 750 {} \;
find /var/www/jaap.nl -type d -exec chmod g+s {} \;
find /var/www/jaap.nl -type f -exec chmod 640 {} \;
Add server-wide account with write access
In order to enable a system-wide SSH account to be able to access and modify all websites, we’ll use ACLs to add another user with write access. This is useful if you need to manage a handful of websites but would like to use scp
or rsync
to transfer files. If this SSH account is called admin
, we can give it write permissions as follows:
setfacl -m u:admin:rwx /var/www/jaap.nl
find /var/www/jaap.nl/html -type d -exec setfacl -m u:admin:rwx,d:u:admin:rwx,m:rwx {} \;
find /var/www/jaap.nl/html -type f -exec setfacl -m u:admin:rw,m:rw {} \;
This will give directories rwx
access to admin
, and rw
access to files. Additionally, newly created files and directories will automatically be given rwx
permissions, which in combination with the standard file mask will result in rw
for files. Creating new files with this account will ofcourse change the ownership of those files to admin
, but since we’ve set chmod g+s
to all directories, the group will still be jaap
so that jaap
and NGINX still have read access.
NGINX
The brunt of the webserver will be done by NGINX: serving the static files, redirecting to PHP-FPM for the dynamic files, termination of SSL, handling of HTTP/2, redirection of automatic certificate renewal, etc.
In /etc/nginx/nginx.conf
we set the following:
user nginx;
worker_processes 2; # one per CPU core
worker_rlimit_nofile 8192;
error_log /var/log/nginx/error.log warn;
pid /var/run/nginx.pid;
events {
worker_connections 4096;
multi_accept on;
use epoll;
}
http {
include /etc/nginx/mime.types;
default_type application/octet-stream;
# set secure SSL settings
ssl_protocols TLSv1.2 TLSv1.3;
ssl_prefer_server_ciphers on;
ssl_dhparam /etc/nginx/dhparam.pem;
ssl_ciphers ECDHE-RSA-AES256-GCM-SHA512:DHE-RSA-AES256-GCM-SHA512:ECDHE-RSA-AES256-GCM-SHA384:DHE-RSA-AES256-GCM-SHA384:ECDHE-RSA-AES256-SHA384;
ssl_ecdh_curve secp384r1;
ssl_session_timeout 10m;
ssl_session_cache shared:SSL:10m;
ssl_session_tickets off;
ssl_stapling on;
ssl_stapling_verify on;
resolver 1.1.1.1 1.0.0.1 valid=300s ipv6=off;
resolver_timeout 5s;
ssl_certificate /etc/pki/lego/certificates/fqdn.server.nl.crt;
ssl_certificate_key /etc/pki/lego/certificates/fqdn.server.nl.key;
server_names_hash_bucket_size 128;
log_format main '$remote_addr - $remote_user [$time_local] "$request" '
'$status $body_bytes_sent "$http_referer" '
'"$http_user_agent" '
'req=$request_time '
'resp=$upstream_response_time';
access_log off;
sendfile on;
tcp_nopush on;
tcp_nodelay on;
keepalive_timeout 30;
# enable automatic GZIP compression
gzip on;
gzip_disable msie6;
gzip_vary on;
gzip_proxied any;
gzip_comp_level 1;
gzip_min_length 10240;
gzip_types
text/css
text/plain
text/javascript
application/javascript
application/json
application/x-javascript
application/xml
application/rss+xml
application/atom+xml
application/xhtml+xml
application/x-font-ttf
application/x-font-opentype
application/vnd.ms-fontobject
image/svg+xml
image/x-icon
font/truetype
font/opentype;
# improve security
server_tokens off;
charset UTF-8;
add_header X-Frame-Options "SAMEORIGIN";
add_header X-XSS-Protection "1; mode=block";
add_header X-Content-Type-Options nosniff;
add_header Strict-Transport-Security "max-age=31536000; includeSubDomains" always;
client_max_body_size 1024m;
include /etc/nginx/conf.d/default.conf; # load in the default handling
include /etc/nginx/sites-available/*.conf; # load in our websites
}
Note that you need to have a fqdn.server.nl.crt
for your whole server. This is when someone tries to contact your server for an unknown domain name through HTTPS, in which case the default certificate will be used (which is unlikely to match with the requested domain name and will fail the security test in the browser).
In /etc/nginx/conf.d/default.conf
we will set a couple of default handlers:
# Default server when nothing matched
server {
listen 80 default_server;
listen 443 ssl http2 default_server;
server_name _;
root /var/www/default;
# accept certificate verification for (yet) unavailable websites
location ^~ /.well-known/acme-challenge/ {
proxy_pass http://127.0.0.1:8081;
proxy_set_header Host $host;
}
location / {
error_log off;
return 404;
}
}
# IP address access
server {
listen 80;
server_name 83.96.200.140;
root /var/www/default;
error_log off;
}
server {
listen 443 ssl http2;
server_name 83.96.200.140;
root /var/www/default;
error_log off;
return 301 http://$host$request_uri;
}
# FQDN
server {
listen 80;
server_name fqdn.server.nl;
root /var/www/default;
location ^~ /.well-known/acme-challenge/ {
proxy_pass http://127.0.0.1:8081;
proxy_set_header Host $host;
}
location / {
error_log off;
return 301 https://$host$request_uri;
}
}
server {
listen 443 ssl http2;
server_name fqdn.server.nl;
root /var/www/default;
# set up routes for server stats or other utilities admins need
}
Note that we use /var/www/default
as the default website without PHP support to load when accessing the IP or the FQDN. Also, some routes will accept the /.well-known/acme-challenge/
route to confirm domain names for SSL creation using lego
.
Website configuration
To add a new website we could use the following configuration:
server {
listen 80;
listen 443 ssl http2;
server_name jaap.nl; # this is the domain name NGINX will use this for
ssl_certificate /etc/pki/lego/certificates/jaap.nl.crt;
ssl_certificate_key /etc/pki/lego/certificates/jaap.nl.key;
error_log /var/www/jaap.nl/log/error.log;
root /var/www/jaap.nl/html/_;
index index.php index.html index.htm;
# accept SSL verification
location ^~ /.well-known/acme-challenge/ {
proxy_pass http://127.0.0.1:8081;
proxy_set_header Host $host;
}
location / {
# refuse access to non-existing subdomains
if (!-d /var/www/jaap.nl/html/_) {
return 404;
}
# force HTTPS scheme
if ($scheme = "http") {
return 301 https://$host$request_uri;
}
try_files $uri $uri.html $uri/ /index.php?$args;
http2_push_preload on;
}
# handle dynamic pages
location ~ \.php$ {
try_files $uri =404;
http2_push_preload on;
fastcgi_pass unix:/var/run/php-fpm/jaap.nl.sock;
fastcgi_index index.php;
fastcgi_split_path_info ^(.+\.php)(/.+)$;
include fastcgi_params;
}
# use browser cache for images and other resources
location ~* \.(jpg|jpeg|png|gif|ico|css|js)$ {
expires 7d;
add_header Cache-Control "public";
}
}
Website redirection
For a subdomain you obviously have to add the subdomain in the server_name
and change the root
as well. To redirect the www
subdomain to the previous handler, we’d use:
listen 80;
listen 443 ssl http2;
server_name www.jaap.nl;
ssl_certificate /etc/pki/lego/certificates/jaap.nl.crt;
ssl_certificate_key /etc/pki/lego/certificates/jaap.nl.key;
error_log /var/www/jaap.nl/log/error.log;
location ^~ /.well-known/acme-challenge/ {
proxy_pass http://127.0.0.1:8081;
proxy_set_header Host $host;
}
location / {
error_log off;
return 301 https://jaap.nl$request_uri;
}
Lego ACME
By using lego and its CLI we can automatically create SSL certificates and do auto-renewal through a cron job.
Certificate creation
We will save our certificates to /etc/pki/lego
and disable the tls-sni-01
and dns-01
methods, as we won’t use those.
lego -m admin@example.nl -k rsa4096 --http ":8081" -x "tls-alpn-01" -x "dns-01" --path /etc/pki/lego -d jaap.nl run
You can add alternative names to the certificate by appending more domain names such as -d www.jaap.nl
. The certificates are put into /etc/pki/lego/certificates/
and consist of a key and certificate file.
To create one for all subdomains, you could write a bash script such as:
DOMAIN="jaap.nl"
DOMAINS="-d $DOMAIN -d www.$DOMAIN"
for d in /var/www/$DOMAIN/html/*/; do
d=$(basename "$d")
if [ $d != "_" ] && [ $d != "www" ]; then
DOMAINS="$DOMAINS -d $d.$DOMAIN"
fi
done
echo "Generating certificates: $DOMAINS"
lego -m admin@example.nl -k rsa4096 --http ":8081" -x "tls-alpn-01" -x "dns-01" --path /etc/pki/lego $DOMAINS run
Certificate renewal
For automatic renewal we’ll create a script that is to be called by a cron job daily and renews certificates that are less then 30 days valid (recommended by Let’s Encrypt).
Create /etc/cron.daily/tls-renew-all.sh
that contains:
echo "Renewing all certificates"
for d in /etc/pki/lego/certificates/*.key; do
echo "Checking $d..."
d=$(basename "$d")
d=${d::-4}
lego -m admin@example.nl -k rsa4096 --http ":8081" -x "tls-alpn-01" -x "dns-01" --path /etc/pki/lego -d $d renew --days 30
done
echo "Restarting NGINX..."
systemctl reload nginx
PHP-FPM
Create /etc/php-fpm.d/jaap.nl.conf
with the following content. Note that here we ensure that PHP is being run using jaap:jaap
as the user and group, as well as that temporary files and logs are saved under /var/www/jaap.nl/
.
[jaap.nl]
listen = /var/run/php-fpm/jaap.nl.sock
listen.owner = nginx
listen.group = nginx
user = jaap
group = jaap
pm = ondemand
pm.process_idle_timeout = 10s
pm.max_children = 5
pm.max_requests = 4096
catch_workers_output = yes
env[TMP] = /var/www/jaap.nl/tmp
env[TMPDIR] = /var/www/jaap.nl/tmp
env[TEMP] = /var/www/jaap.nl/tmp
php_admin_value[open_basedir] = /var/www/jaap.nl/public_html/:/var/www/jaap.nl/tmp/
php_admin_value[upload_tmp_dir] = /var/www/jaap.nl/tmp
php_admin_value[error_log] = /var/www/jaap.nl/log/php_errors.log
php_admin_value[session.name] = jaap
php_admin_value[session.cookie_domain] = ".jaap.nl"
php_admin_value[sendmail_path] = "/usr/sbin/sendmail -t -i -f noreply@fqdn.server.nl"
Closing notes
This is an example setup with NGINX + PHP-FPM and secure file permissions. You’ll obviously need to install MariaDB for most websites. I also recommand installing fail2ban
and enable it on all ports. If your websites run WordPress make sure to install the fail2ban plugin and enable the filter in fail2ban, this will prevent people bruteforcing their way into your client’s WordPress.
Some of the scripts displayed above could benefit from using a template file with variables such as $domain
or $subdomain
. That way you can easily replace all occurrences of these variables by new values using sed
for example.