Caddy v2 brought many major changes, particularly to the Caddyfile syntax. This [site](https://mdleom.com/) is powered by the reverse proxy feature of Caddy, so I need to make sure everything works before I finally upgrade. While v2 has been released for more than 2 weeks by now (after months of beta testing), I only managed get my feet wet last weekend, even though I should've done it during the beta releases. After testing v2 on a local server (plus some forum posts), I would say it is _mostly_ working. While v2.0 has reached feature parity with v1, Caddyfile has not; there are two TLS/HTTPS options that are not yet supported in Caddyfile (see [#3219](https://github.com/caddyserver/caddy/issues/3219), [#3334](https://github.com/caddyserver/caddy/issues/3334); planned to be released in v2.1). So, if you don't need HTTPS--like my {% post_link tor-hidden-onion-nixos 'Tor' %} and {% post_link i2p-eepsite-nixos 'I2P' %} proxies--it should be safe to upgrade.
## proxy to reverse_proxy
`proxy` directive is updated to `reverse_proxy`.
Reverse proxy the whole website:
``` plain v1
proxy / https://backend.com
```
In v2, the matcher needs `*`:
``` plain v2
reverse_proxy /* https://backend.com
```
If no matcher is specified, it defaults to `/*` which match every path:
``` plain v2
reverse_proxy https://backend.com
```
### Custom path
Reverse proxy a certain path, like `/api`:
``` plain v2
reverse_proxy /api/* https://backend.com
```
Requests to `https://example.com/api/foo/bar/` is redirected to `https://backend.com/api/foo/bar/`.
To remove the path prefix:
``` plain v1
proxy /api https://backend.com {
without /api
}
```
Requests to `https://example.com/api/foo/bar/` is redirected to `https://backend.com/foo/bar/`.
v2 doesn't have `without` directive, instead you need to use `route` the request and remove the prefix using `uri strip_prefix`:
`header` directive still keeps similar syntax, but operates a bit different. In v2, when used alongside with `reverse_proxy`, Caddy modifies the header _before_ receiving header response from the backend. This behaviour is apparent when you want to replace existing header(s); instead of replacing, Caddy adds the header and results in duplicate headers. To avoid this issue, you should use `defer`:
In v2, Caddy automatically listens on HTTP (port 80) and redirects to HTTPS, whereas in v1, you need add a separate `redir 301`. This is handy is most use cases, but doesn't apply to my {% post_link caddy-nixos-part-3 'use case' %}--listens on HTTPS only.
Client authentication adds another step to TLS connection process whereby a client needs to present a certificate (that has been signed by a CA certificate) to the server (which has the CA certificate) when it attempts to establish a TLS connection. Once the client is authenticated, the process is reversed and client authenticates the server instead. The padlock icon next to the web address indicates that the website's certificate is valid. Client authentication is only used in private web server to restrict access to authorised clients only. In my case, I restrict my origin server to [Cloudflare CDN](https://support.cloudflare.com/hc/en-us/articles/204899617-Authenticated-Origin-Pulls) only; mdleom.com is only accessible via Cloudflare, direct connection to the origin server will be dropped.
In v2.0, this can only be disabled in [JSON](https://caddyserver.com/docs/json/apps/http/servers/tls_connection_policies/#client_authentication).
v2.1 supports configuring client authentication in Caddyfile using `clients` directive in tls option:
``` plain Caddyfile
example.com {
tls cert.pem cert.key {
client_auth {
mode require_and_verify
trusted_ca_cert_file origin-pull-ca.pem
# base64 DER-encoded CA cert is also supported
# trusted_ca_cert MIIDSzCCAjOgAwIBAg
}
}
}
```
## Administration endpoint
[Admin endpoint](https://caddyserver.com/docs/api) is the highlight feature of v2.0; new config can be loaded without restarting Caddy. It is enabled by default and listens on `http://localhost:2019`.