2023-10-09 19:29:22 +00:00
|
|
|
{ config, lib, pkgs, ... }:
|
|
|
|
|
|
|
|
with lib;
|
|
|
|
|
|
|
|
let
|
|
|
|
cfg = config.services.stalwart-mail;
|
|
|
|
configFormat = pkgs.formats.toml { };
|
|
|
|
configFile = configFormat.generate "stalwart-mail.toml" cfg.settings;
|
|
|
|
dataDir = "/var/lib/stalwart-mail";
|
2024-06-05 15:53:02 +00:00
|
|
|
useLegacyStorage = versionOlder config.system.stateVersion "24.11";
|
2023-10-09 19:29:22 +00:00
|
|
|
|
|
|
|
in {
|
|
|
|
options.services.stalwart-mail = {
|
2024-04-21 15:54:59 +00:00
|
|
|
enable = mkEnableOption "the Stalwart all-in-one email server";
|
2024-06-05 15:53:02 +00:00
|
|
|
|
2024-01-02 11:29:13 +00:00
|
|
|
package = mkPackageOption pkgs "stalwart-mail" { };
|
2023-10-09 19:29:22 +00:00
|
|
|
|
|
|
|
settings = mkOption {
|
|
|
|
inherit (configFormat) type;
|
|
|
|
default = { };
|
2024-04-21 15:54:59 +00:00
|
|
|
description = ''
|
2023-10-09 19:29:22 +00:00
|
|
|
Configuration options for the Stalwart email server.
|
|
|
|
See <https://stalw.art/docs/category/configuration> for available options.
|
|
|
|
|
|
|
|
By default, the module is configured to store everything locally.
|
|
|
|
'';
|
|
|
|
};
|
|
|
|
};
|
|
|
|
|
|
|
|
config = mkIf cfg.enable {
|
2024-06-05 15:53:02 +00:00
|
|
|
|
2023-10-09 19:29:22 +00:00
|
|
|
# Default config: all local
|
|
|
|
services.stalwart-mail.settings = {
|
2024-06-05 15:53:02 +00:00
|
|
|
tracer.stdout = {
|
|
|
|
type = mkDefault "stdout";
|
|
|
|
level = mkDefault "info";
|
|
|
|
ansi = mkDefault false; # no colour markers to journald
|
|
|
|
enable = mkDefault true;
|
|
|
|
};
|
2023-10-09 19:29:22 +00:00
|
|
|
queue.path = mkDefault "${dataDir}/queue";
|
|
|
|
report.path = mkDefault "${dataDir}/reports";
|
2024-06-05 15:53:02 +00:00
|
|
|
store = if useLegacyStorage then {
|
|
|
|
# structured data in SQLite, blobs on filesystem
|
|
|
|
db.type = mkDefault "sqlite";
|
|
|
|
db.path = mkDefault "${dataDir}/data/index.sqlite3";
|
|
|
|
fs.type = mkDefault "fs";
|
|
|
|
fs.path = mkDefault "${dataDir}/data/blobs";
|
|
|
|
} else {
|
|
|
|
# everything in RocksDB
|
|
|
|
db.type = mkDefault "rocksdb";
|
|
|
|
db.path = mkDefault "${dataDir}/db";
|
|
|
|
db.compression = mkDefault "lz4";
|
|
|
|
};
|
2024-02-29 20:09:43 +00:00
|
|
|
storage.data = mkDefault "db";
|
|
|
|
storage.fts = mkDefault "db";
|
2024-05-15 15:35:15 +00:00
|
|
|
storage.lookup = mkDefault "db";
|
2024-06-05 15:53:02 +00:00
|
|
|
storage.blob = mkDefault (if useLegacyStorage then "fs" else "db");
|
|
|
|
directory.internal.type = mkDefault "internal";
|
|
|
|
directory.internal.store = mkDefault "db";
|
|
|
|
storage.directory = mkDefault "internal";
|
2023-10-09 19:29:22 +00:00
|
|
|
resolver.type = mkDefault "system";
|
2024-06-05 15:53:02 +00:00
|
|
|
resolver.public-suffix = lib.mkDefault [
|
|
|
|
"file://${pkgs.publicsuffix-list}/share/publicsuffix/public_suffix_list.dat"
|
|
|
|
];
|
|
|
|
};
|
|
|
|
|
|
|
|
# This service stores a potentially large amount of data.
|
|
|
|
# Running it as a dynamic user would force chown to be run everytime the
|
|
|
|
# service is restarted on a potentially large number of files.
|
|
|
|
# That would cause unnecessary and unwanted delays.
|
|
|
|
users = {
|
|
|
|
groups.stalwart-mail = { };
|
|
|
|
users.stalwart-mail = {
|
|
|
|
isSystemUser = true;
|
|
|
|
group = "stalwart-mail";
|
|
|
|
};
|
2023-10-09 19:29:22 +00:00
|
|
|
};
|
|
|
|
|
2024-06-05 15:53:02 +00:00
|
|
|
systemd = {
|
|
|
|
packages = [ cfg.package ];
|
|
|
|
services.stalwart-mail = {
|
|
|
|
wantedBy = [ "multi-user.target" ];
|
|
|
|
after = [ "local-fs.target" "network.target" ];
|
2023-10-09 19:29:22 +00:00
|
|
|
|
2024-06-05 15:53:02 +00:00
|
|
|
preStart = if useLegacyStorage then ''
|
|
|
|
mkdir -p ${dataDir}/{queue,reports,data/blobs}
|
|
|
|
'' else ''
|
|
|
|
mkdir -p ${dataDir}/{queue,reports,db}
|
|
|
|
'';
|
|
|
|
|
|
|
|
serviceConfig = {
|
|
|
|
ExecStart = [
|
|
|
|
""
|
|
|
|
"${cfg.package}/bin/stalwart-mail --config=${configFile}"
|
|
|
|
];
|
|
|
|
|
|
|
|
StandardOutput = "journal";
|
|
|
|
StandardError = "journal";
|
|
|
|
|
|
|
|
StateDirectory = "stalwart-mail";
|
|
|
|
|
|
|
|
# Bind standard privileged ports
|
|
|
|
AmbientCapabilities = [ "CAP_NET_BIND_SERVICE" ];
|
|
|
|
CapabilityBoundingSet = [ "CAP_NET_BIND_SERVICE" ];
|
2023-10-09 19:29:22 +00:00
|
|
|
|
2024-06-05 15:53:02 +00:00
|
|
|
# Hardening
|
|
|
|
DeviceAllow = [ "" ];
|
|
|
|
LockPersonality = true;
|
|
|
|
MemoryDenyWriteExecute = true;
|
|
|
|
PrivateDevices = true;
|
|
|
|
PrivateUsers = false; # incompatible with CAP_NET_BIND_SERVICE
|
|
|
|
ProcSubset = "pid";
|
|
|
|
PrivateTmp = true;
|
|
|
|
ProtectClock = true;
|
|
|
|
ProtectControlGroups = true;
|
|
|
|
ProtectHome = true;
|
|
|
|
ProtectHostname = true;
|
|
|
|
ProtectKernelLogs = true;
|
|
|
|
ProtectKernelModules = true;
|
|
|
|
ProtectKernelTunables = true;
|
|
|
|
ProtectProc = "invisible";
|
|
|
|
ProtectSystem = "strict";
|
|
|
|
RestrictAddressFamilies = [ "AF_INET" "AF_INET6" ];
|
|
|
|
RestrictNamespaces = true;
|
|
|
|
RestrictRealtime = true;
|
|
|
|
RestrictSUIDSGID = true;
|
|
|
|
SystemCallArchitectures = "native";
|
|
|
|
SystemCallFilter = [ "@system-service" "~@privileged" ];
|
|
|
|
UMask = "0077";
|
|
|
|
};
|
|
|
|
unitConfig.ConditionPathExists = [
|
|
|
|
""
|
|
|
|
"${configFile}"
|
|
|
|
];
|
2023-10-09 19:29:22 +00:00
|
|
|
};
|
|
|
|
};
|
|
|
|
|
|
|
|
# Make admin commands available in the shell
|
|
|
|
environment.systemPackages = [ cfg.package ];
|
|
|
|
};
|
|
|
|
|
|
|
|
meta = {
|
2024-06-05 15:53:02 +00:00
|
|
|
maintainers = with maintainers; [ happysalada pacien onny ];
|
2023-10-09 19:29:22 +00:00
|
|
|
};
|
|
|
|
}
|