base16-schemes: unstable-2024-06-21 -> unstable-2024-11-12
[NixPkgs.git] / nixos / modules / services / mail / opendkim.nix
blobdd29e503508930ffa4c2294a7c0f3c6fedc5ad32
1 { config, lib, pkgs, ... }:
2 let
4   cfg = config.services.opendkim;
6   defaultSock = "local:/run/opendkim/opendkim.sock";
8   keyFile = "${cfg.keyPath}/${cfg.selector}.private";
10   args = [ "-f" "-l"
11            "-p" cfg.socket
12            "-d" cfg.domains
13            "-k" keyFile
14            "-s" cfg.selector
15          ] ++ lib.optionals (cfg.configFile != null) [ "-x" cfg.configFile ];
17 in {
18   imports = [
19     (lib.mkRenamedOptionModule [ "services" "opendkim" "keyFile" ] [ "services" "opendkim" "keyPath" ])
20   ];
22   ###### interface
24   options = {
26     services.opendkim = {
28       enable = lib.mkOption {
29         type = lib.types.bool;
30         default = false;
31         description = "Whether to enable the OpenDKIM sender authentication system.";
32       };
34       socket = lib.mkOption {
35         type = lib.types.str;
36         default = defaultSock;
37         description = "Socket which is used for communication with OpenDKIM.";
38       };
40       user = lib.mkOption {
41         type = lib.types.str;
42         default = "opendkim";
43         description = "User for the daemon.";
44       };
46       group = lib.mkOption {
47         type = lib.types.str;
48         default = "opendkim";
49         description = "Group for the daemon.";
50       };
52       domains = lib.mkOption {
53         type = lib.types.str;
54         default = "csl:${config.networking.hostName}";
55         defaultText = lib.literalExpression ''"csl:''${config.networking.hostName}"'';
56         example = "csl:example.com,mydomain.net";
57         description = ''
58           Local domains set (see `opendkim(8)` for more information on datasets).
59           Messages from them are signed, not verified.
60         '';
61       };
63       keyPath = lib.mkOption {
64         type = lib.types.path;
65         description = ''
66           The path that opendkim should put its generated private keys into.
67           The DNS settings will be found in this directory with the name selector.txt.
68         '';
69         default = "/var/lib/opendkim/keys";
70       };
72       selector = lib.mkOption {
73         type = lib.types.str;
74         description = "Selector to use when signing.";
75       };
77       configFile = lib.mkOption {
78         type = lib.types.nullOr lib.types.path;
79         default = null;
80         description = "Additional opendkim configuration.";
81       };
83     };
85   };
88   ###### implementation
90   config = lib.mkIf cfg.enable {
92     users.users = lib.optionalAttrs (cfg.user == "opendkim") {
93       opendkim = {
94         group = cfg.group;
95         uid = config.ids.uids.opendkim;
96       };
97     };
99     users.groups = lib.optionalAttrs (cfg.group == "opendkim") {
100       opendkim.gid = config.ids.gids.opendkim;
101     };
103     environment.systemPackages = [ pkgs.opendkim ];
105     systemd.tmpfiles.rules = [
106       "d '${cfg.keyPath}' - ${cfg.user} ${cfg.group} - -"
107     ];
109     systemd.services.opendkim = {
110       description = "OpenDKIM signing and verification daemon";
111       after = [ "network.target" ];
112       wantedBy = [ "multi-user.target" ];
114       preStart = ''
115         cd "${cfg.keyPath}"
116         if ! test -f ${cfg.selector}.private; then
117           ${pkgs.opendkim}/bin/opendkim-genkey -s ${cfg.selector} -d all-domains-generic-key
118           echo "Generated OpenDKIM key! Please update your DNS settings:\n"
119           echo "-------------------------------------------------------------"
120           cat ${cfg.selector}.txt
121           echo "-------------------------------------------------------------"
122         fi
123       '';
125       serviceConfig = {
126         ExecStart = "${pkgs.opendkim}/bin/opendkim ${lib.escapeShellArgs args}";
127         User = cfg.user;
128         Group = cfg.group;
129         RuntimeDirectory = lib.optional (cfg.socket == defaultSock) "opendkim";
130         StateDirectory = "opendkim";
131         StateDirectoryMode = "0700";
132         ReadWritePaths = [ cfg.keyPath ];
134         AmbientCapabilities = [];
135         CapabilityBoundingSet = "";
136         DevicePolicy = "closed";
137         LockPersonality = true;
138         MemoryDenyWriteExecute = true;
139         NoNewPrivileges = true;
140         PrivateDevices = true;
141         PrivateMounts = true;
142         PrivateTmp = true;
143         PrivateUsers = true;
144         ProtectClock = true;
145         ProtectControlGroups = true;
146         ProtectHome = true;
147         ProtectHostname = true;
148         ProtectKernelLogs = true;
149         ProtectKernelModules = true;
150         ProtectKernelTunables = true;
151         ProtectSystem = "strict";
152         RemoveIPC = true;
153         RestrictAddressFamilies = [ "AF_INET" "AF_INET6 AF_UNIX" ];
154         RestrictNamespaces = true;
155         RestrictRealtime = true;
156         RestrictSUIDSGID = true;
157         SystemCallArchitectures = "native";
158         SystemCallFilter = [ "@system-service" "~@privileged @resources" ];
159         UMask = "0077";
160       };
161     };
163   };