2024-08-19 16:16:01 +00:00
|
|
|
{
|
|
|
|
lib,
|
|
|
|
pkgs,
|
|
|
|
config,
|
|
|
|
...
|
|
|
|
}:
|
|
|
|
let
|
|
|
|
cfg = config.services.wakapi;
|
|
|
|
|
|
|
|
settingsFormat = pkgs.formats.yaml { };
|
|
|
|
settingsFile = settingsFormat.generate "wakapi-settings" cfg.settings;
|
|
|
|
|
|
|
|
inherit (lib)
|
|
|
|
getExe
|
|
|
|
mkOption
|
|
|
|
mkEnableOption
|
|
|
|
mkPackageOption
|
|
|
|
types
|
|
|
|
mkIf
|
|
|
|
optional
|
|
|
|
mkMerge
|
|
|
|
singleton
|
|
|
|
;
|
|
|
|
in
|
|
|
|
{
|
|
|
|
options.services.wakapi = {
|
|
|
|
enable = mkEnableOption "Wakapi";
|
|
|
|
package = mkPackageOption pkgs "wakapi" { };
|
|
|
|
|
|
|
|
settings = mkOption {
|
|
|
|
inherit (settingsFormat) type;
|
|
|
|
default = { };
|
|
|
|
description = ''
|
|
|
|
Settings for Wakapi.
|
|
|
|
|
|
|
|
See [config.default.yml](https://github.com/muety/wakapi/blob/master/config.default.yml) for a list of all possible options.
|
|
|
|
'';
|
|
|
|
};
|
|
|
|
|
|
|
|
passwordSalt = mkOption {
|
|
|
|
type = types.nullOr types.str;
|
|
|
|
default = null;
|
|
|
|
description = ''
|
|
|
|
The password salt to use for Wakapi.
|
|
|
|
'';
|
|
|
|
};
|
|
|
|
passwordSaltFile = mkOption {
|
|
|
|
type = types.nullOr types.path;
|
|
|
|
default = null;
|
|
|
|
description = ''
|
|
|
|
The path to a file containing the password salt to use for Wakapi.
|
|
|
|
'';
|
|
|
|
};
|
|
|
|
|
|
|
|
smtpPassword = mkOption {
|
|
|
|
type = types.nullOr types.str;
|
|
|
|
default = null;
|
|
|
|
description = ''
|
|
|
|
The password used for the smtp mailed to used by Wakapi.
|
|
|
|
'';
|
|
|
|
};
|
|
|
|
smtpPasswordFile = mkOption {
|
|
|
|
type = types.nullOr types.path;
|
|
|
|
default = null;
|
|
|
|
description = ''
|
|
|
|
The path to a file containing the password for the smtp mailer used by Wakapi.
|
|
|
|
'';
|
|
|
|
};
|
2024-09-11 10:08:22 +00:00
|
|
|
|
|
|
|
database = {
|
|
|
|
createLocally = mkEnableOption ''
|
|
|
|
automatic database configuration.
|
|
|
|
|
|
|
|
::: {.note}
|
|
|
|
Only PostgreSQL is supported for the time being.
|
|
|
|
:::
|
|
|
|
'';
|
|
|
|
|
|
|
|
dialect = mkOption {
|
|
|
|
type = types.nullOr (
|
|
|
|
types.enum [
|
|
|
|
"postgres"
|
|
|
|
"sqlite3"
|
|
|
|
"mysql"
|
|
|
|
"cockroach"
|
|
|
|
"mssql"
|
|
|
|
]
|
|
|
|
);
|
|
|
|
default = cfg.settings.db.dialect or null; # handle case where dialect is not set
|
|
|
|
defaultText = ''
|
|
|
|
Database dialect from settings if {option}`services.wakatime.settings.db.dialect`
|
|
|
|
is set, or `null` otherwise.
|
|
|
|
'';
|
|
|
|
description = ''
|
|
|
|
The database type to use for Wakapi.
|
|
|
|
'';
|
|
|
|
};
|
|
|
|
|
|
|
|
name = mkOption {
|
|
|
|
type = types.str;
|
|
|
|
default = cfg.settings.db.name or "wakapi";
|
|
|
|
defaultText = ''
|
|
|
|
Database name from settings if {option}`services.wakatime.settings.db.name`
|
|
|
|
is set, or "wakapi" otherwise.
|
|
|
|
'';
|
|
|
|
description = ''
|
|
|
|
The name of the database to use for Wakapi.
|
|
|
|
'';
|
|
|
|
};
|
|
|
|
|
|
|
|
user = mkOption {
|
|
|
|
type = types.str;
|
|
|
|
default = cfg.settings.db.user or "wakapi";
|
|
|
|
defaultText = ''
|
|
|
|
User from settings if {option}`services.wakatime.settings.db.user`
|
|
|
|
is set, or "wakapi" otherwise.
|
|
|
|
'';
|
|
|
|
description = ''
|
|
|
|
The name of the user to use for Wakapi.
|
|
|
|
'';
|
|
|
|
};
|
|
|
|
};
|
2024-08-19 16:16:01 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
config = mkIf cfg.enable {
|
|
|
|
systemd.services.wakapi = {
|
|
|
|
description = "Wakapi (self-hosted WakaTime-compatible backend)";
|
|
|
|
wants = [
|
|
|
|
"network-online.target"
|
2024-09-11 10:08:22 +00:00
|
|
|
] ++ optional (cfg.database.dialect == "postgres") "postgresql.service";
|
2024-08-19 16:16:01 +00:00
|
|
|
after = [
|
|
|
|
"network-online.target"
|
2024-09-11 10:08:22 +00:00
|
|
|
] ++ optional (cfg.database.dialect == "postgres") "postgresql.service";
|
2024-08-19 16:16:01 +00:00
|
|
|
wantedBy = [ "multi-user.target" ];
|
|
|
|
|
|
|
|
script = ''
|
|
|
|
exec ${getExe cfg.package} -config ${settingsFile}
|
|
|
|
'';
|
|
|
|
|
|
|
|
serviceConfig = {
|
|
|
|
Environment = mkMerge [
|
|
|
|
(mkIf (cfg.passwordSalt != null) "WAKAPI_PASSWORD_SALT=${cfg.passwordSalt}")
|
|
|
|
(mkIf (cfg.smtpPassword != null) "WAKAPI_MAIL_SMTP_PASS=${cfg.smtpPassword}")
|
|
|
|
];
|
|
|
|
EnvironmentFile = [
|
|
|
|
(optional (cfg.passwordSaltFile != null) cfg.passwordSaltFile)
|
|
|
|
(optional (cfg.smtpPasswordFile != null) cfg.smtpPasswordFile)
|
|
|
|
];
|
|
|
|
|
|
|
|
User = config.users.users.wakapi.name;
|
|
|
|
Group = config.users.users.wakapi.group;
|
|
|
|
|
|
|
|
DynamicUser = true;
|
|
|
|
ProtectHome = true;
|
|
|
|
ProtectHostname = true;
|
|
|
|
ProtectKernelLogs = true;
|
|
|
|
ProtectKernelModules = true;
|
|
|
|
ProtectKernelTunables = true;
|
|
|
|
ProtectProc = "invisible";
|
|
|
|
ProtectSystem = "strict";
|
|
|
|
RestrictAddressFamilies = [
|
|
|
|
"AF_INET"
|
|
|
|
"AF_INET6"
|
|
|
|
"AF_UNIX"
|
|
|
|
];
|
|
|
|
RestrictNamespaces = true;
|
|
|
|
RestrictRealtime = true;
|
|
|
|
RestrictSUIDSGID = true;
|
2024-10-09 06:54:49 +00:00
|
|
|
StateDirectory = "wakapi";
|
2024-08-19 16:16:01 +00:00
|
|
|
StateDirectoryMode = "0700";
|
|
|
|
Restart = "always";
|
|
|
|
};
|
|
|
|
};
|
|
|
|
|
|
|
|
services.wakapi.settings = {
|
|
|
|
env = lib.mkDefault "production";
|
|
|
|
};
|
|
|
|
|
|
|
|
assertions = [
|
|
|
|
{
|
|
|
|
assertion = cfg.passwordSalt != null || cfg.passwordSaltFile != null;
|
|
|
|
message = "Either `services.wakapi.passwordSalt` or `services.wakapi.passwordSaltFile` must be set.";
|
|
|
|
}
|
|
|
|
{
|
2024-10-22 09:27:24 +00:00
|
|
|
assertion = !(cfg.passwordSalt != null && cfg.passwordSaltFile != null);
|
2024-08-19 16:16:01 +00:00
|
|
|
message = "Both `services.wakapi.passwordSalt` `services.wakapi.passwordSaltFile` should not be set at the same time.";
|
|
|
|
}
|
|
|
|
{
|
2024-10-22 09:27:24 +00:00
|
|
|
assertion = !(cfg.smtpPassword != null && cfg.smtpPasswordFile != null);
|
2024-08-19 16:16:01 +00:00
|
|
|
message = "Both `services.wakapi.smtpPassword` `services.wakapi.smtpPasswordFile` should not be set at the same time.";
|
|
|
|
}
|
2024-09-11 10:08:22 +00:00
|
|
|
{
|
2024-10-22 09:05:02 +00:00
|
|
|
assertion = cfg.database.createLocally -> cfg.settings.db.dialect != null;
|
2024-09-11 10:08:22 +00:00
|
|
|
message = "`services.wakapi.database.createLocally` is true, but a database dialect is not set!";
|
|
|
|
}
|
|
|
|
];
|
|
|
|
|
|
|
|
warnings = [
|
2024-10-22 09:05:02 +00:00
|
|
|
(lib.optionalString (cfg.database.createLocally -> cfg.settings.db.dialect != "postgres") ''
|
2024-09-11 10:08:22 +00:00
|
|
|
You have enabled automatic database configuration, but the database dialect is not set to "posgres".
|
|
|
|
|
|
|
|
The Wakapi module only supports for PostgreSQL. Please set `services.wakapi.database.createLocally`
|
|
|
|
to `false`, or switch to "postgres" as your database dialect.
|
|
|
|
'')
|
2024-08-19 16:16:01 +00:00
|
|
|
];
|
|
|
|
|
|
|
|
users = {
|
|
|
|
users.wakapi = {
|
|
|
|
group = "wakapi";
|
|
|
|
createHome = false;
|
|
|
|
isSystemUser = true;
|
|
|
|
};
|
|
|
|
groups.wakapi = { };
|
|
|
|
};
|
|
|
|
|
2024-09-11 10:08:22 +00:00
|
|
|
services.postgresql = mkIf (cfg.database.createLocally && cfg.database.dialect == "postgres") {
|
2024-08-19 16:16:01 +00:00
|
|
|
enable = true;
|
|
|
|
|
2024-09-11 10:08:22 +00:00
|
|
|
ensureDatabases = singleton cfg.database.name;
|
2024-08-19 16:16:01 +00:00
|
|
|
ensureUsers = singleton {
|
|
|
|
name = cfg.settings.db.user;
|
|
|
|
ensureDBOwnership = true;
|
|
|
|
};
|
|
|
|
|
|
|
|
authentication = ''
|
|
|
|
host ${cfg.settings.db.name} ${cfg.settings.db.user} 127.0.0.1/32 trust
|
|
|
|
'';
|
|
|
|
};
|
|
|
|
};
|
|
|
|
|
2024-09-11 10:08:22 +00:00
|
|
|
meta.maintainers = with lib.maintainers; [
|
|
|
|
isabelroses
|
|
|
|
NotAShelf
|
|
|
|
];
|
2024-08-19 16:16:01 +00:00
|
|
|
}
|