2018-04-12 20:36:46 +02:00
|
|
|
{ lib, config, pkgs, ... }:
|
|
|
|
|
|
|
|
with lib;
|
|
|
|
|
|
|
|
let
|
|
|
|
cfg = config.services.haproxy-acme;
|
|
|
|
|
|
|
|
nginx_port = 54321;
|
2018-08-06 20:25:27 +02:00
|
|
|
|
|
|
|
haproxyConf = ''
|
|
|
|
global
|
|
|
|
log /dev/log local0
|
|
|
|
log /dev/log local1 notice
|
|
|
|
user haproxy
|
|
|
|
group haproxy
|
|
|
|
ssl-default-bind-ciphers ECDHE-ECDSA-AES256-GCM-SHA384:ECDHE-RSA-AES256-GCM-SHA384:ECDHE-ECDSA-CHACHA20-POLY1305:ECDHE-RSA-CHACHA20-POLY1305:ECDHE-ECDSA-AES128-GCM-SHA256:ECDHE-RSA-AES128-GCM-SHA256:ECDHE-ECDSA-AES256-SHA384:ECDHE-RSA-AES256-SHA384:ECDHE-ECDSA-AES128-SHA256:ECDHE-RSA-AES128-SHA256
|
|
|
|
ssl-default-bind-options no-sslv3 no-tlsv10 no-tlsv11 no-tls-tickets
|
|
|
|
ssl-default-server-ciphers ECDHE-ECDSA-AES256-GCM-SHA384:ECDHE-RSA-AES256-GCM-SHA384:ECDHE-ECDSA-CHACHA20-POLY1305:ECDHE-RSA-CHACHA20-POLY1305:ECDHE-ECDSA-AES128-GCM-SHA256:ECDHE-RSA-AES128-GCM-SHA256:ECDHE-ECDSA-AES256-SHA384:ECDHE-RSA-AES256-SHA384:ECDHE-ECDSA-AES128-SHA256:ECDHE-RSA-AES128-SHA256
|
|
|
|
ssl-default-server-options no-sslv3 no-tlsv10 no-tlsv11 no-tls-tickets
|
|
|
|
defaults
|
|
|
|
option forwardfor
|
|
|
|
option http-server-close
|
|
|
|
timeout client 10s
|
|
|
|
timeout connect 4s
|
|
|
|
timeout server 30s
|
2019-01-24 18:56:12 +01:00
|
|
|
errorfile 503 ${./errorfiles/503.html}
|
2018-08-06 20:25:27 +02:00
|
|
|
userlist LOUTRE
|
2019-01-24 09:52:58 +01:00
|
|
|
user paul password $6$YNjCpiPABu9$.iEp.3BgoswHcX3SMjz1/CiyqFQn/fjnxtT9CWBqQHBKynvK2kh/i62ije0WmCvhKRUhy9gdVbJStM3ciGXnC1
|
2018-08-06 20:25:27 +02:00
|
|
|
frontend public
|
|
|
|
bind :::80 v4v6
|
|
|
|
bind :::443 v4v6 ssl crt /var/lib/acme/${cfg.domaine}/full.pem alpn h2,http/1.1
|
|
|
|
mode http
|
|
|
|
acl letsencrypt-acl path_beg /.well-known/acme-challenge/
|
|
|
|
acl haproxy-acl path_beg /haproxy
|
|
|
|
redirect scheme https code 301 if !{ ssl_fc } !letsencrypt-acl
|
|
|
|
http-response set-header Strict-Transport-Security max-age=15768000
|
|
|
|
use_backend letsencrypt-backend if letsencrypt-acl
|
|
|
|
use_backend haproxy_stats if haproxy-acl
|
|
|
|
|
|
|
|
${concatStrings (
|
|
|
|
mapAttrsToList (name: value:
|
|
|
|
" acl ${name}-acl hdr(host) -i ${name}\n"
|
|
|
|
+ " use_backend ${name}-backend if ${name}-acl\n"
|
|
|
|
) cfg.services)}
|
|
|
|
|
|
|
|
backend letsencrypt-backend
|
|
|
|
mode http
|
|
|
|
server letsencrypt 127.0.0.1:${toString nginx_port}
|
|
|
|
backend haproxy_stats
|
|
|
|
mode http
|
|
|
|
stats enable
|
|
|
|
stats hide-version
|
|
|
|
acl AuthOK_LOUTRE http_auth(LOUTRE)
|
|
|
|
http-request auth realm LOUTRE if !AuthOK_LOUTRE
|
|
|
|
|
|
|
|
${concatStrings (
|
|
|
|
mapAttrsToList (name: value:
|
|
|
|
''
|
|
|
|
backend ${name}-backend
|
|
|
|
mode http
|
|
|
|
${(
|
|
|
|
if value.socket == "" then
|
|
|
|
''
|
|
|
|
server ${name} ${value.ip}:${toString value.port}
|
|
|
|
''
|
|
|
|
else
|
|
|
|
''
|
|
|
|
server ${name} ${value.socket}
|
|
|
|
''
|
|
|
|
)}
|
|
|
|
${(if value.auth then (
|
|
|
|
value.extraAcls
|
|
|
|
+ ''
|
|
|
|
acl AUTH_OK http_auth(LOUTRE)
|
|
|
|
http-request auth realm LOUTRE if ${value.aclBool}
|
|
|
|
''
|
|
|
|
) else "")}
|
|
|
|
''
|
|
|
|
) cfg.services)}
|
|
|
|
|
|
|
|
'';
|
2018-04-12 20:36:46 +02:00
|
|
|
in
|
|
|
|
{
|
|
|
|
options.services.haproxy-acme = {
|
|
|
|
enable = mkEnableOption "HAproxy + ACME";
|
|
|
|
|
|
|
|
domaine = mkOption {
|
|
|
|
type = types.string;
|
|
|
|
example = "example.com";
|
|
|
|
description = ''
|
|
|
|
Sous domaine à utiliser
|
|
|
|
|
|
|
|
Il est necessaire d'avoir un enregistrement pointant sur la wildcard de ce domaine vers le serveur
|
|
|
|
'';
|
|
|
|
};
|
|
|
|
|
|
|
|
services = mkOption {
|
2018-05-10 18:53:34 +02:00
|
|
|
type = with types; attrsOf (submodule { options = {
|
|
|
|
ip = mkOption { type = str; description = "IP address"; };
|
|
|
|
port = mkOption { type = int; description = "Port number"; };
|
2018-06-05 14:03:52 +02:00
|
|
|
socket = mkOption { type = str; description = "Emplacement du socket"; default = ""; };
|
2018-05-10 18:53:34 +02:00
|
|
|
auth = mkOption { type = bool; description = "Enable authentification"; default = false; };
|
2018-06-21 10:15:39 +02:00
|
|
|
extraAcls = mkOption { type = str; description = "ACL HaProxy suplémentaires"; default = ""; };
|
|
|
|
aclBool = mkOption { type = str; description = "Logique d'authentification"; default = "!AUTH_OK"; };
|
2018-05-10 18:53:34 +02:00
|
|
|
}; });
|
2018-04-12 20:36:46 +02:00
|
|
|
example = ''
|
|
|
|
haproxy_backends = {
|
|
|
|
example = { ip = "127.0.0.1"; port = 1234; auth = false; };
|
|
|
|
};
|
|
|
|
'';
|
|
|
|
description = "Liste des noms de domaines associés à leur backend";
|
|
|
|
};
|
|
|
|
};
|
|
|
|
|
|
|
|
config = mkIf cfg.enable {
|
|
|
|
|
|
|
|
services.haproxy.enable = true;
|
|
|
|
|
2018-08-06 20:25:27 +02:00
|
|
|
services.haproxy.config = haproxyConf;
|
2018-04-12 20:36:46 +02:00
|
|
|
|
|
|
|
services.nginx.enable = true;
|
|
|
|
services.nginx.virtualHosts = {
|
|
|
|
"acme" = {
|
|
|
|
listen = [ { addr = "127.0.0.1"; port = nginx_port; } ];
|
|
|
|
locations = { "/" = { root = "/var/www/challenges"; }; };
|
|
|
|
};
|
|
|
|
};
|
|
|
|
|
|
|
|
security.acme.certs = {
|
|
|
|
${cfg.domaine} = {
|
|
|
|
extraDomains = mapAttrs' (name: value:
|
2018-05-17 15:43:35 +02:00
|
|
|
nameValuePair ("${name}") (null)
|
2018-04-12 20:36:46 +02:00
|
|
|
) cfg.services;
|
2018-05-17 15:43:35 +02:00
|
|
|
webroot = "/var/www/challenges";
|
2018-04-12 20:36:46 +02:00
|
|
|
email = "paul@nyanlout.re";
|
2019-02-12 11:15:01 +01:00
|
|
|
allowKeysForGroup = true;
|
|
|
|
group = "acme";
|
2018-04-20 20:25:50 +02:00
|
|
|
postRun = ''
|
|
|
|
systemctl reload haproxy.service
|
|
|
|
'';
|
2018-04-12 20:36:46 +02:00
|
|
|
};
|
|
|
|
};
|
|
|
|
security.acme.directory = "/var/lib/acme";
|
|
|
|
|
2019-02-12 11:15:01 +01:00
|
|
|
users.groups.acme.members = [ "haproxy" ];
|
|
|
|
|
2018-04-18 21:03:20 +02:00
|
|
|
networking.firewall.allowedTCPPorts = [
|
|
|
|
80 443
|
|
|
|
];
|
|
|
|
|
2018-04-12 20:36:46 +02:00
|
|
|
};
|
|
|
|
}
|