post: 'Using Caddy as a reverse proxy on NixOS (Part 1)'

This commit is contained in:
MDLeom 2020-02-22 21:12:19 +00:00
parent deeec3eaf9
commit bf297d7c79
No known key found for this signature in database
GPG Key ID: 5D9DB57A25D34EE3
1 changed files with 261 additions and 0 deletions

View File

@ -0,0 +1,261 @@
---
title: Using Caddy as a reverse proxy on NixOS (Part 1)
excerpt: "Part 1: Installing NixOS"
date: 2020-02-23
tags:
- web
- linux
---
## Background
> Skip to [Installation](#Installation) part.
I set up this website with JAMstack architecture. Before moving to the current domain, I set four requirements:
1. Retain current JAMstack workflow
2. Ability to add and _remove_ HTTP headers
3. Available on Tor .onion
4. Cloudflare CDN
This website's JAMstack workflow goes like this:
1. Pages are written in Markdown.
2. Markdown files are Git-hosted on GitLab
3. Any commit to GitLab repository triggers a build on Netlify CI
4. Markdown files are processed into HTML pages using Nodejs-powered Hexo.
5. Generated pages are hosted on curben.netlify.com
Right off the bat I can already see the need of setting up a private server due to the second requirement (ability to remove HTTP header). I had an option to drop Netlify by building the pages on my workstation and deploy to the web server (using a Hexo deployer plugin). So far I do find Netlify service to be reliable and it offers features like adding headers and reverse proxy which are easy to setup. Speaking of Netlify's features, I then had an idea of setting up a web server which reverse proxy to Netlify. This approach meets all the four requirements; a side-benefit is that if I screw up the web server, at least my website is still up on curben.netlify.com and I can easily migrate this domain to Netlify.
As for web server, I went with Caddy, which has the most secured defaults. It is installed in NixOS, which is attractive for its centralised configuration. I initially planned to use Ubuntu, and then I noticed [NixOS](https://nixos.org/); unlike most other Linux servers which has configs scattered around, NixOS is configured through a single "configuration.nix" file. It is [declarative](https://en.wikipedia.org/wiki/Declarative_programming), meaning you simply supply the desired configuration and NixOS would figure how to achieve that. For example, to open port 80, you just need `networking.firewall.allowedTCPPorts = [ 80 ]`, instead of mucking around with iptables. This significantly helps reproducibility, making server migration much easier; simply supply the "configuration.nix" used in the previous server and the new server would have the same state. Having Caddy in the repo is the tipping point that finally made me dive into NixOS.
## Installation
NixOS has a detailed installation [guide](https://nixos.org/nixos/manual/index.html#sec-installation), anyhow this is how I installed it.
1. The LiveCD automatically login as "nixos". Simply switch to root shell by,
``` sh
sudo -s
```
2. Create the necessary partitions. I went with the classic MBR since my VPS provider still supports it and I don't need 2 TB partition. I set up a "swap" partition due to having a tiny RAM; if you have less than 2 GB RAM, it's better to have it, otherwise the `nixos-install` step would fail.
``` sh
# Most KVM-powered VPS use "/dev/vda" naming scheme (instead of "/dev/sda")
# Check the output of `ls /dev/` to make sure
parted /dev/vda -- mklabel msdos
# Create a new partition that fill the disk but
# leaves 1 GB space for the swap
parted /dev/vda -- mkpart primary 1MiB -1GiB
# Swap partition
parted /dev/vda -- mkpart primary linux-swap -1GiB 100%
```
3. Format the partitions.
``` sh
mkfs.btrfs -L nixos /dev/vda1
# Or "mkfs.ext4" if preferred
mkswap -L swap /dev/vda2
```
4. Mount the partitions.
``` sh
mount /dev/disk/by-label/nixos /mnt
swapon /dev/vda2
```
5. Generate the configs. This generates "configuration.nix" and "hardware-configuration.nix".
``` sh
nixos-generate-config --root /mnt
```
6. I replaced the generated "configuration.nix" with my own "configuration.nix". Before uploading the config to the server, I did the following change,
1. Replace "/dev/sda" with "/dev/vda" in `boot.loader.grub.device`
2. Replace "eth0" to "ens3" in firewall config (check output of `ifconfig`)
3. Encrypt the file using 7zip before upload.
``` sh
# This is much less memory-intensive than `nix-env -i package`
# ffsend (unofficial CLI client of Firefox Send) is a good alternative to magic-wormhole,
# but it has a long URL so it's only usable in ssh where you can copy-paste.
nix-env -f '<nixpkgs>' -iA magic-wormhole p7zip
cd /tmp
wormhole receive configurations.7z
7z x configurations.7z
cp configuration.nix /mnt/etc/nixos/
```
7. Install it without setting root password (so that it remains disabled)
```
nixos-install --no-root-passwd
```
8. In my setup, the installation downloaded around 1 GB of packages. Once installed, shutdown now, unmount the live cd and boot.
Following is my "configuration.nix". I show you how to secure NixOS using hashed password, firewall, DNS-over-TLS and USBGuard in my next post. After that, I show you how to setup Caddy and Tor (they are disabled for now).
```
{ config, pkgs, ... }:
{
imports =
[ # Include the results of the hardware scan.
./hardware-configuration.nix
];
system.stateVersion = "19.09";
# Use the GRUB 2 boot loader.
boot.loader.grub.enable = true;
boot.loader.grub.version = 2;
boot.loader.grub.device = "/dev/vda";
networking.useDHCP = false;
networking.interfaces.ens3.useDHCP = true;
environment.systemPackages = with pkgs; [
dnsutils usbguard magic-wormhole p7zip
];
time.timeZone = "UTC";
## Unattended upgrade
system.autoUpgrade = {
enable = false;
allowReboot = true;
dates = "00:00";
};
## USBGuard
services.usbguard = {
enable = false;
ruleFile = "/var/lib/usbguard/rules.conf";
};
## DNS-over-TLS
services.stubby = {
enable = true;
listenAddresses = [ "0::1" "127.0.0.1" ];
roundRobinUpstreams = false;
upstreamServers =
''
## Cloudflare DNS
- address_data: 2606:4700:4700::1111
tls_auth_name: "cloudflare-dns.com"
- address_data: 2606:4700:4700::1001
tls_auth_name: "cloudflare-dns.com"
- address_data: 1.1.1.1
tls_auth_name: "cloudflare-dns.com"
- address_data: 1.0.0.1
tls_auth_name: "cloudflare-dns.com"
'';
};
networking.nameservers = [ "::1" "127.0.0.1" ];
services.resolved = {
enable = true;
fallbackDns = [ "2606:4700:4700::1111" "2606:4700:4700::1001" "1.1.1.1" "1.0.0.1" ];
};
## Port forwarding
networking.firewall = {
enable = true;
interfaces.ens3 = {
allowedTCPPorts = [ 443 4430 ];
};
extraCommands =
''
ip6tables -t nat -I PREROUTING -i ens3 -p tcp -m tcp --dport 443 -j REDIRECT --to-ports 4430
'';
};
## Create service users
users = {
mutableUsers = false; # Disable useradd & passwd
users = {
root = {
hashedPassword = "*"; # Disable root password
};
nixos = {
hashedPassword = "xxxx";
isNormalUser = true;
extraGroups = [ "wheel" ]; # Enable sudo for the user.
};
caddyProxy = {
home = "/var/lib/caddyProxy";
createHome = true;
};
caddyTor = {
home = "/var/lib/caddyTor";
createHome = true;
};
tor = {
home = "/var/lib/tor";
createHome = true;
};
};
groups = {
caddyProxy = {
members = [ "caddyProxy" ];
};
caddyTor = {
members = [ "caddyTor" ];
};
tor = {
members = [ "tor" ];
};
};
};
## Caddy web server
# require = [ /etc/caddy/caddyProxy.nix /etc/caddy/caddyTor.nix ];
# services.caddyProxy = {
# enable = true;
# config = "/etc/caddy/caddyProxy.conf";
# };
# services.caddyTor = {
# enable = true;
# config = "/etc/caddy/caddyTor.conf";
# };
## Tor onion
services.tor = {
enable = false;
enableGeoIP = false;
hiddenServices = {
proxy = {
name = "proxy";
version = 3;
map = [
{
port = "80";
toHost = "[::1]";
toPort = "8080";
}
];
};
};
extraConfig =
''
ClientUseIPv4 0
ClientUseIPv6 1
ClientPreferIPv6ORPort 1
'';
};
}
```