base16-schemes: unstable-2024-06-21 -> unstable-2024-11-12
[NixPkgs.git] / nixos / modules / services / mail / schleuder.nix
blobca07b879071cadf6b5161230131e7fbf27468440
1 { config, pkgs, lib, ... }:
2 let
3   cfg = config.services.schleuder;
4   settingsFormat = pkgs.formats.yaml { };
5   postfixMap = entries: lib.concatStringsSep "\n" (lib.mapAttrsToList (name: value: "${name} ${value}") entries);
6   writePostfixMap = name: entries: pkgs.writeText name (postfixMap entries);
7   configScript = pkgs.writeScript "schleuder-cfg" ''
8     #!${pkgs.runtimeShell}
9     set -exuo pipefail
10     umask 0077
11     ${pkgs.yq}/bin/yq \
12       --slurpfile overrides <(${pkgs.yq}/bin/yq . <${lib.escapeShellArg cfg.extraSettingsFile}) \
13       < ${settingsFormat.generate "schleuder.yml" cfg.settings} \
14       '. * $overrides[0]' \
15       > /etc/schleuder/schleuder.yml
16     chown schleuder: /etc/schleuder/schleuder.yml
17   '';
20   options.services.schleuder = {
21     enable = lib.mkEnableOption "Schleuder secure remailer";
22     enablePostfix = lib.mkEnableOption "automatic postfix integration" // { default = true; };
23     lists = lib.mkOption {
24       description = ''
25         List of list addresses that should be handled by Schleuder.
27         Note that this is only handled by the postfix integration, and
28         the setup of the lists, their members and their keys has to be
29         performed separately via schleuder's API, using a tool such as
30         schleuder-cli.
31       '';
32       type = lib.types.listOf lib.types.str;
33       default = [ ];
34       example = [ "widget-team@example.com" "security@example.com" ];
35     };
36     /* maybe one day....
37       domains = lib.mkOption {
38       description = "Domains for which all mail should be handled by Schleuder.";
39       type = lib.types.listOf lib.types.str;
40       default = [];
41       example = ["securelists.example.com"];
42       };
43     */
44     settings = lib.mkOption {
45       description = ''
46         Settings for schleuder.yml.
48         Check the [example configuration](https://0xacab.org/schleuder/schleuder/blob/master/etc/schleuder.yml) for possible values.
49       '';
50       type = lib.types.submodule {
51         freeformType = settingsFormat.type;
52         options.keyserver = lib.mkOption {
53           type = lib.types.str;
54           description = ''
55             Key server from which to fetch and update keys.
57             Note that NixOS uses a different default from upstream, since the upstream default sks-keyservers.net is deprecated.
58           '';
59           default = "keys.openpgp.org";
60         };
61       };
62       default = { };
63     };
64     extraSettingsFile = lib.mkOption {
65       description = "YAML file to merge into the schleuder config at runtime. This can be used for secrets such as API keys.";
66       type = lib.types.nullOr lib.types.path;
67       default = null;
68     };
69     listDefaults = lib.mkOption {
70       description = ''
71         Default settings for lists (list-defaults.yml).
73         Check the [example configuration](https://0xacab.org/schleuder/schleuder/-/blob/master/etc/list-defaults.yml) for possible values.
74       '';
75       type = settingsFormat.type;
76       default = { };
77     };
78   };
79   config = lib.mkIf cfg.enable {
80     assertions = [
81       {
82         assertion = !(cfg.settings.api ? valid_api_keys);
83         message = ''
84           services.schleuder.settings.api.valid_api_keys is set. Defining API keys via NixOS config results in them being copied to the world-readable Nix store. Please use the extraSettingsFile option to store API keys in a non-public location.
85         '';
86       }
87       {
88         assertion = !(lib.any (db: db ? password) (lib.attrValues cfg.settings.database or {}));
89         message = ''
90           A password is defined for at least one database in services.schleuder.settings.database. Defining passwords via NixOS config results in them being copied to the world-readable Nix store. Please use the extraSettingsFile option to store database passwords in a non-public location.
91         '';
92       }
93     ];
94     users.users.schleuder.isSystemUser = true;
95     users.users.schleuder.group = "schleuder";
96     users.groups.schleuder = {};
97     environment.systemPackages = [
98       pkgs.schleuder-cli
99     ];
100     services.postfix = lib.mkIf cfg.enablePostfix {
101       extraMasterConf = ''
102         schleuder  unix  -       n       n       -       -       pipe
103           flags=DRhu user=schleuder argv=/${pkgs.schleuder}/bin/schleuder work ''${recipient}
104       '';
105       transport = lib.mkIf (cfg.lists != [ ]) (postfixMap (lib.genAttrs cfg.lists (_: "schleuder:")));
106       extraConfig = ''
107         schleuder_destination_recipient_limit = 1
108       '';
109       # review: does this make sense?
110       localRecipients = lib.mkIf (cfg.lists != [ ]) cfg.lists;
111     };
112     systemd.services = let commonServiceConfig = {
113       # We would have liked to use DynamicUser, but since the default
114       # database is SQLite and lives in StateDirectory, and that same
115       # database needs to be readable from the postfix service, this
116       # isn't trivial to do.
117       User = "schleuder";
118       StateDirectory = "schleuder";
119       StateDirectoryMode = "0700";
120     }; in
121       {
122         schleuder-init = {
123           serviceConfig = commonServiceConfig // {
124             ExecStartPre = lib.mkIf (cfg.extraSettingsFile != null) [
125               "+${configScript}"
126             ];
127             ExecStart = [ "${pkgs.schleuder}/bin/schleuder install" ];
128             Type = "oneshot";
129           };
130         };
131         schleuder-api-daemon = {
132           after = [ "local-fs.target" "network.target" "schleuder-init.service" ];
133           wantedBy = [ "multi-user.target" ];
134           requires = [ "schleuder-init.service" ];
135           serviceConfig = commonServiceConfig // {
136             ExecStart = [ "${pkgs.schleuder}/bin/schleuder-api-daemon" ];
137           };
138         };
139         schleuder-weekly-key-maintenance = {
140           after = [ "local-fs.target" "network.target" ];
141           startAt = "weekly";
142           serviceConfig = commonServiceConfig // {
143             ExecStart = [
144               "${pkgs.schleuder}/bin/schleuder refresh_keys"
145               "${pkgs.schleuder}/bin/schleuder check_keys"
146             ];
147           };
148         };
149       };
151     environment.etc."schleuder/schleuder.yml" = lib.mkIf (cfg.extraSettingsFile == null) {
152       source = settingsFormat.generate "schleuder.yml" cfg.settings;
153     };
154     environment.etc."schleuder/list-defaults.yml".source = settingsFormat.generate "list-defaults.yml" cfg.listDefaults;
156     services.schleuder = {
157       #lists_dir = "/var/lib/schleuder.lists";
158       settings.filters_dir = lib.mkDefault "/var/lib/schleuder/filters";
159       settings.keyword_handlers_dir = lib.mkDefault "/var/lib/schleuder/keyword_handlers";
160     };
161   };