Nginx: Difference between revisions
imported>Makefu No edit summary |
Extra Config section added |
||
(29 intermediate revisions by 18 users not shown) | |||
Line 1: | Line 1: | ||
[https://nginx.org/ {{PAGENAME}}] ([[wikipedia:en:{{PAGENAME}}]]) is a lightweight webserver. | |||
== | == Installation == | ||
To install Nginx, add the following to your NixOS configuration:{{file|/etc/nixos/configuration.nix|nix|3=services.nginx.enable = true;}} | |||
More options are available: {{nixos:option|services.nginx.}} | |||
== Sample setups == | |||
== | ==== Minimal raw html dummy page for testing <code>configuration.nix</code> ==== | ||
<syntaxhighlight lang="nix"> | |||
services.nginx = { | |||
enable = true; | |||
virtualHosts.localhost = { | |||
services.nginx | locations."/" = { | ||
return = "200 '<html><body>It works</body></html>'"; | |||
extraConfig = '' | |||
default_type text/html; | |||
''; | |||
}; | |||
}; | |||
}; | }; | ||
</syntaxhighlight> | |||
==== Static blog with ssl enforced in <code>configuration.nix</code> ==== | ==== Static blog with ssl enforced in <code>configuration.nix</code> ==== | ||
Line 90: | Line 54: | ||
forceSSL = true; | forceSSL = true; | ||
root = "/var/www/blog"; | root = "/var/www/blog"; | ||
locations."~ \.php$".extraConfig = '' | locations."~ \\.php$".extraConfig = '' | ||
fastcgi_pass unix:${config.services.phpfpm.pools.mypool.socket}; | fastcgi_pass unix:${config.services.phpfpm.pools.mypool.socket}; | ||
fastcgi_index index.php; | fastcgi_index index.php; | ||
Line 103: | Line 67: | ||
user = "nobody"; | user = "nobody"; | ||
settings = { | settings = { | ||
pm = "dynamic"; | "pm" = "dynamic"; | ||
"listen.owner" = config.services.nginx.user; | "listen.owner" = config.services.nginx.user; | ||
"pm.max_children" = 5; | "pm.max_children" = 5; | ||
Line 112: | Line 76: | ||
}; | }; | ||
}; | }; | ||
</syntaxhighlight> | |||
'''Robots.txt''' | |||
If you want to set a robots.txt for your domain (or any subdomains), add this: | |||
<syntaxhighlight lang="nix"> | |||
locations."/robots.txt" = { | |||
extraConfig = '' | |||
rewrite ^/(.*) $1; | |||
return 200 "User-agent: *\nDisallow: /"; | |||
''; | |||
}; | |||
</syntaxhighlight> | </syntaxhighlight> | ||
==== HTTP Authentication ==== | ==== HTTP Authentication ==== | ||
===== Basic Authentication ===== | ===== Basic Authentication ===== | ||
Nginx can require users to login using HTTP Basic Authentication. In NixOS, this is set using the `basicAuth` option: | Nginx can require users to login using HTTP Basic Authentication. In NixOS, this is set using the `basicAuth` option: | ||
Line 126: | Line 106: | ||
}; | }; | ||
</syntaxhighlight> | </syntaxhighlight> | ||
===== Authentication via PAM ===== | ===== Authentication via PAM ===== | ||
Line 137: | Line 116: | ||
systemd.services.nginx.serviceConfig = { | systemd.services.nginx.serviceConfig = { | ||
SupplementaryGroups = [ "shadow" ]; | SupplementaryGroups = [ "shadow" ]; | ||
}; | }; | ||
Line 194: | Line 158: | ||
}; | }; | ||
</syntaxhighlight> | </syntaxhighlight> | ||
{{Note|ACME won't be able to authenticate your domain if ports 80 & 443 aren't open in your firewall.}} | |||
==== Hardened setup with TLS and HSTS preloading ==== | ==== Hardened setup with TLS and HSTS preloading ==== | ||
Line 214: | Line 179: | ||
sslCiphers = "AES256+EECDH:AES256+EDH:!aNULL"; | sslCiphers = "AES256+EECDH:AES256+EDH:!aNULL"; | ||
appendHttpConfig = '' | |||
# Add HSTS header with preloading to HTTPS requests. | # Add HSTS header with preloading to HTTPS requests. | ||
# Adding this header to HTTP requests is discouraged | # Adding this header to HTTP requests is discouraged | ||
Line 233: | Line 198: | ||
# Prevent injection of code in other mime types (XSS Attacks) | # Prevent injection of code in other mime types (XSS Attacks) | ||
add_header X-Content-Type-Options nosniff; | add_header X-Content-Type-Options nosniff; | ||
# This might create errors | # This might create errors | ||
Line 260: | Line 221: | ||
</syntaxhighlight> | </syntaxhighlight> | ||
< | ==== Using realIP when behind CloudFlare or other CDN ==== | ||
When Nginx is behind another proxy it won't know the true IP address of clients hitting it. It will then pass down those the proxy's IP address instead of the client IP address. By using the nginx realip module, we can ensure nginx knows the real client IP, and we can further inform nginx to only trust the HTTP header from valid upstream proxies. | |||
In the following example, we are fetching the list of IPs directly from cloudflare and including a hash. This has some pros and cons. Nix will not attempt to download or update that file while it is in a nix store it trusts, but after a nix garbage collection, it will error if the list of proxies has changed informing you of that when you apply the config. | |||
<syntaxhighlight lang="nix"> | |||
services.nginx.commonHttpConfig = | |||
let | |||
realIpsFromList = lib.strings.concatMapStringsSep "\n" (x: "set_real_ip_from ${x};"); | |||
fileToList = x: lib.strings.splitString "\n" (builtins.readFile x); | |||
cfipv4 = fileToList (pkgs.fetchurl { | |||
url = "https://www.cloudflare.com/ips-v4"; | |||
sha256 = "0ywy9sg7spafi3gm9q5wb59lbiq0swvf0q3iazl0maq1pj1nsb7h"; | |||
}); | |||
cfipv6 = fileToList (pkgs.fetchurl { | |||
url = "https://www.cloudflare.com/ips-v6"; | |||
sha256 = "1ad09hijignj6zlqvdjxv7rjj8567z357zfavv201b9vx3ikk7cy"; | |||
}); | |||
in | |||
'' | |||
${realIpsFromList cfipv4} | |||
${realIpsFromList cfipv6} | |||
real_ip_header CF-Connecting-IP; | |||
''; | |||
</syntaxhighlight> | |||
==== UNIX socket reverse proxy ==== | |||
In order for nginx to be able to access UNIX sockets, you have to do some permission modifications. | |||
<syntaxhighlight lang="nix"> | |||
# Example service that supports listening to UNIX sockets | |||
services.hedgedoc = { | |||
enable = true; | |||
settings.path = "/run/hedgedoc/hedgedoc.sock" | |||
}; | |||
services.nginx = { | |||
enable = true; | |||
virtualHosts."example.com" = { | |||
enableACME = true; | |||
forceSSL = true; | |||
locations."/".proxyPass = "http://unix:/run/hedgedoc/hedgedoc.sock"; | |||
}; | |||
}; | |||
# This is needed for nginx to be able to read other processes | |||
# directories in `/run`. Else it will fail with (13: Permission denied) | |||
systemd.services.nginx.serviceConfig.ProtectHome = false; | |||
# Most services will create sockets with 660 permissions. | |||
# This means you have to add nginx to their group. | |||
users.groups.hedgedoc.members = [ "nginx" ]; | |||
# Alternatively, you can try to force the unit to create the socket with | |||
# different permissions, if you have a reason for not wanting to add nginx | |||
# to their group. This might not work, depending on how the program sets | |||
# its permissions for the socket. | |||
systemd.services.hedgedoc.serviceConfig.UMask = "0000"; | |||
</syntaxhighlight> | |||
== Let's Encrypt certificates == | |||
The nginx module for NixOS has native support for Let's Encrypt certificates; {{nixos:option|services.nginx.+acme}}. The {{manual:nixos|sec=#module-security-acme-nginx|chapter=Chapter 20. SSL/TLS Certificates with ACME}} explains it in detail. | |||
=== Minimal Internet Exposed Server Example === | |||
Assuming that <code>myhost.org</code> resolves to the IP address of your host and port 80 and 443 have been opened. | |||
<syntaxhighlight lang="nix"> | |||
services.nginx.enable = true; | |||
services.nginx.virtualHosts."myhost.org" = { | |||
addSSL = true; | |||
enableACME = true; | |||
root = "/var/www/myhost.org"; | |||
}; | |||
security.acme = { | |||
acceptTerms = true; | |||
defaults.email = "foo@bar.com"; | |||
}; | |||
</syntaxhighlight> | |||
This will set up nginx to serve files for <code>myhost.org</code>, automatically request an ACME SSL Certificate using a "'''HTTP-01'''" challenge (meaning your server must be exposed to the internet) and will configure systemd timers to renew the certificate if required. | |||
=== Minimal Private Local LAN Server Example === | |||
We can also have a private server running in our local network (including VPN), that isn't reachable from the internet, but that still can get valid Let's Encrypt certificates that are accepted in a browser. | |||
1. We have to '''modify DNS such that our domain''' like <code>myhost.org</code> '''resolves to the local IP address of our private server''' and port 80 and 443 have been opened. [https://www.youtube.com/watch?v=qlcVx-k-02E See this video tutorial] for an example on how to do that. Hint: You might need to '''add an exception to your router''' (definitely on Fritzboxes), because resolving to local IP address is usually blocked to prevent '''"DNS rebind attacks"'''. | |||
2. We have to setup the Let's Encrypt NixOS ACME services such that it uses an '''API token in a secrets file''' ([https://github.com/ryantm/agenix secrets for a server can be conveniently and securely deployed in NixOS with agenix]; just follow the tutorial) against our DNS provider to '''prove from our server that we own the domain'''. This way our server doesn't need to be exposed and reachable from the internet. NixOS ACME uses the [https://go-acme.github.io/lego/ LEGO library] to communicate to DNS providers (it supports a lot) and therefore we have to provide the token(s) in that library's secrets file format. | |||
In the example we use Hetzner as our "dnsProvider" that only needs a single API token environment in our secrets file: | |||
<syntaxhighlight lang="nix"> | |||
HETZNER_API_KEY=xxxxxxxx-xxxx-xxxx-xxxx-xxxxxxxxxxxx | |||
</syntaxhighlight>Other [https://carjorvaz.com/posts/setting-up-wildcard-lets-encrypt-certificates-on-nixos/ DNS providers need like OVH] require more environment variables. | |||
See the section "'''Credentials'''" on what you have to specify in the secrets file: https://go-acme.github.io/lego/dns/hetzner/ | |||
See Hetzner guide on how to get an API token for its "DNS console": https://docs.hetzner.com/dns-console/dns/general/api-access-token/ | |||
3. Point our virtualHost to the ACME entry. | |||
<syntaxhighlight lang="nix"> | |||
{ | |||
services.nginx = { | |||
enable = true; | |||
# we can use the main domain or any subdomain that's mentioned by | |||
# "extraDomainNames" in the acme certificate. | |||
virtualHosts."subdomain.example.org" = { | |||
# 3. Instead of "enableACME = true;" we | |||
# reuse the certificate from "security.acme.certs."example.org" | |||
# down below | |||
useACMEHost = "example.org"; | |||
forceSSL = true; | |||
locations."/" = { | |||
return = "200 '<html><body>It works</body></html>'"; | |||
extraConfig = '' | |||
default_type text/html; | |||
''; | |||
}; | |||
}; | |||
}; | |||
security.acme.acceptTerms = true; | |||
security.acme.defaults.email = "info@example.org"; | |||
# 2. Let NixOS generate a Let's Encrypt certificate that we can reuse | |||
# above for several virtualhosts above. | |||
security.acme.certs."example.org" = { | |||
domain = "example.org"; | |||
extraDomainNames = [ "subdomain.example.org" ]; | |||
# The LEGO DNS provider name. Depending on the provider, need different | |||
# contents in the credentialsFile below. | |||
dnsProvider = "hetzner"; | |||
dnsPropagationCheck = true; | |||
# agenix will decrypt our secrets file (below) on the server and make it available | |||
# under /run/agenix/secrets/hetzner-dns-token (by default): | |||
# credentialsFile = "/run/agenix/secrets/hetzner-dns-token"; | |||
credentialsFile = config.age.secrets."hetzner-dns-token.age".path; | |||
}; | |||
# Let agenix know about and copy our (encrypted) DNS API token secrets file | |||
# (containing "HETZNER_API_KEY=...") to the server an decrypt it there. | |||
# Follow the agenix tutorial on how to encrypt a secrets file | |||
# to a .age file and how to setup your Nix flake to use it. | |||
age.secrets."hetzner-dns-token.age".file = .../hetzner-dns-token.age; | |||
users.users.nginx.extraGroups = [ "acme" ]; | |||
} | |||
</syntaxhighlight> | |||
This will set up nginx to serve files for example.org, automatically request an ACME SSL Certificate using a "DNS-01" challenge (meaning '''your server doesn't need to be exposed to the internet''', which is great for self-hosting) and will configure systemd timers to renew the certificate if required. | |||
== Troubleshooting == | |||
=== Read-only Filesystem for nginx upgrade to 20.09 === | |||
With the upgrade to nixos-20.09 the nginx comes with extra hardening parameters, most prominently the restriction of write access to the Operating System Disk. | |||
When you see errors like <code>[emerg] open() "/var/spool/nginx/logs/binaergewitter.access.log" failed (30: Read-only file system)</code> you can add extra paths to nginx service like this: | |||
<syntaxhighlight lang="nix"> | |||
systemd.services.nginx.serviceConfig.ReadWritePaths = [ "/var/spool/nginx/logs/" ]; | |||
</syntaxhighlight> | |||
=== SIGTERM received from 1 === | |||
If you turn debug logging on: | |||
<pre> | |||
services.nginx.logError = "stderr debug"; | |||
</pre> | |||
You may see this: | |||
<pre> | |||
[notice] 12383#12383: signal 15 (SIGTERM) received from 1, exiting | |||
</pre> | |||
This means systemd is killing nginx for you, but systemd (in nixOS 20.09) isn't nice enough to tell you why it's happening. Chances are it's because your nginx config has daemon mode turned on, turn off daemon mode in your nginx config like so: | |||
<pre> | |||
daemon off; | |||
</pre> | |||
And it should fix nginx so systemd won't go killing your nginx anymore. | |||
=== Escape special chars in Regular Expressions === | |||
Some nginx configuration options like <code>locations</code> allows the use of Regular Expressions. | |||
Be ware that you [https://nixos.org/manual/nix/stable/language/values.html#type-string need to escape some special chars] like <code>\</code>, if provided by a double quoted <code>" "</code> string. | |||
A common example found on the internet is: | |||
<syntaxhighlight lang="nix"> | |||
locations."~ ^(.+\.php)(.*)$" = { | |||
... | |||
}; | |||
</syntaxhighlight> | |||
But in this case the <code>\.php</code> part will be parsed by Nix to <code>.php</code>. In RegEx the dot represents any character instead of the dot character itself. | |||
Thus the path /gly'''php'''ro.css will be matched, too. Additionaly to the intended match of <code>/somephpfile.php?param=value</code>. | |||
To circumvent this error <code>\.php</code> has to be double escaped as <code>\\.php</code> | |||
<syntaxhighlight lang="nix"> | |||
locations."~ ^(.+\\.php)(.*)$" = { | |||
... | |||
}; | |||
</syntaxhighlight> | |||
=== General === | |||
Nginx is run as the systemd service nginx, so <code>systemctl status nginx</code> may say something useful. If you have a problem with configuration, you can find the configuration location in the <code>systemctl status</code>, it should be at <code>/nix/store/*-nginx.conf</code>. | |||
== Replace dependencies like openssl == | |||
In wake of the 2022 OpenSSL library, Nix can support in mitigating the library by downgrading (or replacing) the SSL library. For this, the [[Overlay|overlay]] facility of nixpkgs can be used: | |||
<syntaxHighlight lang=nix> | |||
nixpkgs.overlays = [ | |||
(final: super: { | |||
nginxStable = super.nginxStable.override { openssl = super.pkgs.libressl; }; | |||
} ) | |||
]; | |||
</syntaxHighlight> | |||
When utilizing NixOS options the following configuration will also work: | |||
<syntaxHighlight lang=nix> | |||
services.nginx.package = pkgs.nginxStable.override { openssl = pkgs.libressl; }; | |||
</syntaxHighlight> | |||
== Extra config == | |||
Appart native options, Nix allows to specify verbatim Nginx configuration. Some options are mutually exclusive. | |||
Below table assumes "services.nginx." prefix for all options. These options allows to keep using Nix configuration file while taking advantage of Nginx features which are not representend in options. | |||
{| class="wikitable" | |||
|+ | |||
!Options | |||
!Block | |||
!Behaviour | |||
|- | |||
|config | |||
|nginx.conf | |||
|Verbatim <code>nginx.conf</code> configuration | |||
|- | |||
|appendConfig | |||
|nginx.conf | |||
|Lines appended to the generated Nginx configuration file | |||
|- | |||
|httpConfig | |||
|http block | |||
|exclusive with the structured configuration via virtualHosts | |||
|- | |||
|appendHttpConfig | |||
|http block | |||
|lines appended. exclusive with using config and httpConfig | |||
|- | |||
|virtualHosts.<name>.extraConfig | |||
|server | |||
|These lines go to the end of the vhost verbatim. | |||
|- | |||
|virtualHosts.<name>.locations.<name>.extraConfig | |||
|server | |||
|These lines go to the end of the location verbatim | |||
|} | |||
== See more == | == See more == | ||
* [http://nginx.org/en/docs/ Official Documentation] | * [http://nginx.org/en/docs/ Official Documentation] | ||
[[Category:Applications]] | [[Category:Applications]] | ||
[[Category: | [[Category:Server]] | ||
[[Category:Networking]] |