2022-03-05 16:20:37 +00:00
|
|
|
{ config, lib, pkgs, ... }:
|
|
|
|
let
|
|
|
|
cfg = config.services.pgadmin;
|
|
|
|
|
2024-09-19 14:19:46 +00:00
|
|
|
_base = with lib.types; [ int bool str ];
|
|
|
|
base = with lib.types; oneOf ([ (listOf (oneOf _base)) (attrsOf (oneOf _base)) ] ++ _base);
|
2022-03-05 16:20:37 +00:00
|
|
|
|
|
|
|
formatAttrset = attr:
|
2024-09-19 14:19:46 +00:00
|
|
|
"{${lib.concatStringsSep "\n" (lib.mapAttrsToList (key: value: "${builtins.toJSON key}: ${formatPyValue value},") attr)}}";
|
2022-03-05 16:20:37 +00:00
|
|
|
|
|
|
|
formatPyValue = value:
|
|
|
|
if builtins.isString value then builtins.toJSON value
|
|
|
|
else if value ? _expr then value._expr
|
|
|
|
else if builtins.isInt value then toString value
|
|
|
|
else if builtins.isBool value then (if value then "True" else "False")
|
|
|
|
else if builtins.isAttrs value then (formatAttrset value)
|
2024-09-19 14:19:46 +00:00
|
|
|
else if builtins.isList value then "[${lib.concatStringsSep "\n" (map (v: "${formatPyValue v},") value)}]"
|
2022-03-05 16:20:37 +00:00
|
|
|
else throw "Unrecognized type";
|
|
|
|
|
|
|
|
formatPy = attrs:
|
2024-09-19 14:19:46 +00:00
|
|
|
lib.concatStringsSep "\n" (lib.mapAttrsToList (key: value: "${key} = ${formatPyValue value}") attrs);
|
2022-03-05 16:20:37 +00:00
|
|
|
|
2024-09-19 14:19:46 +00:00
|
|
|
pyType = with lib.types; attrsOf (oneOf [ (attrsOf base) (listOf base) base ]);
|
2022-03-05 16:20:37 +00:00
|
|
|
in
|
|
|
|
{
|
|
|
|
options.services.pgadmin = {
|
2024-09-19 14:19:46 +00:00
|
|
|
enable = lib.mkEnableOption "PostgreSQL Admin 4";
|
2022-03-05 16:20:37 +00:00
|
|
|
|
2024-09-19 14:19:46 +00:00
|
|
|
port = lib.mkOption {
|
2024-04-21 15:54:59 +00:00
|
|
|
description = "Port for pgadmin4 to run on";
|
2024-09-19 14:19:46 +00:00
|
|
|
type = lib.types.port;
|
2022-03-05 16:20:37 +00:00
|
|
|
default = 5050;
|
|
|
|
};
|
|
|
|
|
2024-09-19 14:19:46 +00:00
|
|
|
package = lib.mkPackageOption pkgs "pgadmin4" { };
|
2024-01-02 11:29:13 +00:00
|
|
|
|
2024-09-19 14:19:46 +00:00
|
|
|
initialEmail = lib.mkOption {
|
2024-04-21 15:54:59 +00:00
|
|
|
description = "Initial email for the pgAdmin account";
|
2024-09-19 14:19:46 +00:00
|
|
|
type = lib.types.str;
|
2022-03-05 16:20:37 +00:00
|
|
|
};
|
|
|
|
|
2024-09-19 14:19:46 +00:00
|
|
|
initialPasswordFile = lib.mkOption {
|
2024-04-21 15:54:59 +00:00
|
|
|
description = ''
|
2024-01-13 08:15:51 +00:00
|
|
|
Initial password file for the pgAdmin account. Minimum length by default is 6.
|
|
|
|
Please see `services.pgadmin.minimumPasswordLength`.
|
2022-09-30 11:47:45 +00:00
|
|
|
NOTE: Should be string not a store path, to prevent the password from being world readable
|
2022-03-05 16:20:37 +00:00
|
|
|
'';
|
2024-09-19 14:19:46 +00:00
|
|
|
type = lib.types.path;
|
2022-03-05 16:20:37 +00:00
|
|
|
};
|
|
|
|
|
2024-09-19 14:19:46 +00:00
|
|
|
minimumPasswordLength = lib.mkOption {
|
2024-04-21 15:54:59 +00:00
|
|
|
description = "Minimum length of the password";
|
2024-09-19 14:19:46 +00:00
|
|
|
type = lib.types.int;
|
2024-01-13 08:15:51 +00:00
|
|
|
default = 6;
|
|
|
|
};
|
|
|
|
|
2022-09-30 11:47:45 +00:00
|
|
|
emailServer = {
|
2024-09-19 14:19:46 +00:00
|
|
|
enable = lib.mkOption {
|
2024-04-21 15:54:59 +00:00
|
|
|
description = ''
|
2022-09-30 11:47:45 +00:00
|
|
|
Enable SMTP email server. This is necessary, if you want to use password recovery or change your own password
|
|
|
|
'';
|
2024-09-19 14:19:46 +00:00
|
|
|
type = lib.types.bool;
|
2022-09-30 11:47:45 +00:00
|
|
|
default = false;
|
|
|
|
};
|
2024-09-19 14:19:46 +00:00
|
|
|
address = lib.mkOption {
|
2024-04-21 15:54:59 +00:00
|
|
|
description = "SMTP server for email delivery";
|
2024-09-19 14:19:46 +00:00
|
|
|
type = lib.types.str;
|
2022-09-30 11:47:45 +00:00
|
|
|
default = "localhost";
|
|
|
|
};
|
2024-09-19 14:19:46 +00:00
|
|
|
port = lib.mkOption {
|
2024-04-21 15:54:59 +00:00
|
|
|
description = "SMTP server port for email delivery";
|
2024-09-19 14:19:46 +00:00
|
|
|
type = lib.types.port;
|
2022-09-30 11:47:45 +00:00
|
|
|
default = 25;
|
|
|
|
};
|
2024-09-19 14:19:46 +00:00
|
|
|
useSSL = lib.mkOption {
|
2024-04-21 15:54:59 +00:00
|
|
|
description = "SMTP server should use SSL";
|
2024-09-19 14:19:46 +00:00
|
|
|
type = lib.types.bool;
|
2022-09-30 11:47:45 +00:00
|
|
|
default = false;
|
|
|
|
};
|
2024-09-19 14:19:46 +00:00
|
|
|
useTLS = lib.mkOption {
|
2024-04-21 15:54:59 +00:00
|
|
|
description = "SMTP server should use TLS";
|
2024-09-19 14:19:46 +00:00
|
|
|
type = lib.types.bool;
|
2022-09-30 11:47:45 +00:00
|
|
|
default = false;
|
|
|
|
};
|
2024-09-19 14:19:46 +00:00
|
|
|
username = lib.mkOption {
|
2024-04-21 15:54:59 +00:00
|
|
|
description = "SMTP server username for email delivery";
|
2024-09-19 14:19:46 +00:00
|
|
|
type = lib.types.nullOr lib.types.str;
|
2022-09-30 11:47:45 +00:00
|
|
|
default = null;
|
|
|
|
};
|
2024-09-19 14:19:46 +00:00
|
|
|
sender = lib.mkOption {
|
2024-04-21 15:54:59 +00:00
|
|
|
description = ''
|
2022-09-30 11:47:45 +00:00
|
|
|
SMTP server sender email for email delivery. Some servers require this to be a valid email address from that server
|
|
|
|
'';
|
2024-09-19 14:19:46 +00:00
|
|
|
type = lib.types.str;
|
2022-09-30 11:47:45 +00:00
|
|
|
example = "noreply@example.com";
|
|
|
|
};
|
2024-09-19 14:19:46 +00:00
|
|
|
passwordFile = lib.mkOption {
|
2024-04-21 15:54:59 +00:00
|
|
|
description = ''
|
2022-09-30 11:47:45 +00:00
|
|
|
Password for SMTP email account.
|
|
|
|
NOTE: Should be string not a store path, to prevent the password from being world readable
|
|
|
|
'';
|
2024-09-19 14:19:46 +00:00
|
|
|
type = lib.types.path;
|
2022-09-30 11:47:45 +00:00
|
|
|
};
|
|
|
|
};
|
|
|
|
|
2024-09-19 14:19:46 +00:00
|
|
|
openFirewall = lib.mkEnableOption "firewall passthrough for pgadmin4";
|
2022-03-05 16:20:37 +00:00
|
|
|
|
2024-09-19 14:19:46 +00:00
|
|
|
settings = lib.mkOption {
|
2024-04-21 15:54:59 +00:00
|
|
|
description = ''
|
2022-03-05 16:20:37 +00:00
|
|
|
Settings for pgadmin4.
|
2022-09-30 11:47:45 +00:00
|
|
|
[Documentation](https://www.pgadmin.org/docs/pgadmin4/development/config_py.html)
|
2022-03-05 16:20:37 +00:00
|
|
|
'';
|
|
|
|
type = pyType;
|
2022-09-30 11:47:45 +00:00
|
|
|
default = { };
|
2022-03-05 16:20:37 +00:00
|
|
|
};
|
|
|
|
};
|
|
|
|
|
2024-09-19 14:19:46 +00:00
|
|
|
config = lib.mkIf (cfg.enable) {
|
|
|
|
networking.firewall.allowedTCPPorts = lib.mkIf (cfg.openFirewall) [ cfg.port ];
|
2022-03-05 16:20:37 +00:00
|
|
|
|
|
|
|
services.pgadmin.settings = {
|
|
|
|
DEFAULT_SERVER_PORT = cfg.port;
|
2024-01-13 08:15:51 +00:00
|
|
|
PASSWORD_LENGTH_MIN = cfg.minimumPasswordLength;
|
2022-03-05 16:20:37 +00:00
|
|
|
SERVER_MODE = true;
|
2024-01-13 08:15:51 +00:00
|
|
|
UPGRADE_CHECK_ENABLED = false;
|
2024-09-19 14:19:46 +00:00
|
|
|
} // (lib.optionalAttrs cfg.openFirewall {
|
|
|
|
DEFAULT_SERVER = lib.mkDefault "::";
|
|
|
|
}) // (lib.optionalAttrs cfg.emailServer.enable {
|
2022-09-30 11:47:45 +00:00
|
|
|
MAIL_SERVER = cfg.emailServer.address;
|
|
|
|
MAIL_PORT = cfg.emailServer.port;
|
|
|
|
MAIL_USE_SSL = cfg.emailServer.useSSL;
|
|
|
|
MAIL_USE_TLS = cfg.emailServer.useTLS;
|
|
|
|
MAIL_USERNAME = cfg.emailServer.username;
|
|
|
|
SECURITY_EMAIL_SENDER = cfg.emailServer.sender;
|
2022-03-05 16:20:37 +00:00
|
|
|
});
|
|
|
|
|
|
|
|
systemd.services.pgadmin = {
|
|
|
|
wantedBy = [ "multi-user.target" ];
|
|
|
|
after = [ "network.target" ];
|
|
|
|
requires = [ "network.target" ];
|
|
|
|
# we're adding this optionally so just in case there's any race it'll be caught
|
|
|
|
# in case postgres doesn't start, pgadmin will just start normally
|
|
|
|
wants = [ "postgresql.service" ];
|
|
|
|
|
|
|
|
path = [ config.services.postgresql.package pkgs.coreutils pkgs.bash ];
|
|
|
|
|
|
|
|
preStart = ''
|
|
|
|
# NOTE: this is idempotent (aka running it twice has no effect)
|
2024-01-13 08:15:51 +00:00
|
|
|
# Check here for password length to prevent pgadmin from starting
|
|
|
|
# and presenting a hard to find error message
|
|
|
|
# see https://github.com/NixOS/nixpkgs/issues/270624
|
2024-06-05 15:53:02 +00:00
|
|
|
PW_FILE="$CREDENTIALS_DIRECTORY/initial_password"
|
|
|
|
PW_LENGTH=$(wc -m < "$PW_FILE")
|
2024-01-13 08:15:51 +00:00
|
|
|
if [ $PW_LENGTH -lt ${toString cfg.minimumPasswordLength} ]; then
|
|
|
|
echo "Password must be at least ${toString cfg.minimumPasswordLength} characters long"
|
|
|
|
exit 1
|
|
|
|
fi
|
2022-03-05 16:20:37 +00:00
|
|
|
(
|
|
|
|
# Email address:
|
2024-09-19 14:19:46 +00:00
|
|
|
echo ${lib.escapeShellArg cfg.initialEmail}
|
2022-03-05 16:20:37 +00:00
|
|
|
|
|
|
|
# file might not contain newline. echo hack fixes that.
|
2024-06-05 15:53:02 +00:00
|
|
|
PW=$(cat "$PW_FILE")
|
2022-03-05 16:20:37 +00:00
|
|
|
|
|
|
|
# Password:
|
|
|
|
echo "$PW"
|
|
|
|
# Retype password:
|
|
|
|
echo "$PW"
|
2024-01-25 14:12:00 +00:00
|
|
|
) | ${cfg.package}/bin/pgadmin4-cli setup-db
|
2022-03-05 16:20:37 +00:00
|
|
|
'';
|
|
|
|
|
|
|
|
restartTriggers = [
|
|
|
|
"/etc/pgadmin/config_system.py"
|
|
|
|
];
|
|
|
|
|
|
|
|
serviceConfig = {
|
|
|
|
User = "pgadmin";
|
|
|
|
DynamicUser = true;
|
|
|
|
LogsDirectory = "pgadmin";
|
|
|
|
StateDirectory = "pgadmin";
|
2024-01-02 11:29:13 +00:00
|
|
|
ExecStart = "${cfg.package}/bin/pgadmin4";
|
2024-06-05 15:53:02 +00:00
|
|
|
LoadCredential = [ "initial_password:${cfg.initialPasswordFile}" ]
|
2024-09-19 14:19:46 +00:00
|
|
|
++ lib.optional cfg.emailServer.enable "email_password:${cfg.emailServer.passwordFile}";
|
2022-03-05 16:20:37 +00:00
|
|
|
};
|
|
|
|
};
|
|
|
|
|
|
|
|
users.users.pgadmin = {
|
|
|
|
isSystemUser = true;
|
|
|
|
group = "pgadmin";
|
|
|
|
};
|
|
|
|
|
2022-09-30 11:47:45 +00:00
|
|
|
users.groups.pgadmin = { };
|
2022-03-05 16:20:37 +00:00
|
|
|
|
|
|
|
environment.etc."pgadmin/config_system.py" = {
|
2022-09-30 11:47:45 +00:00
|
|
|
text = lib.optionalString cfg.emailServer.enable ''
|
2024-06-05 15:53:02 +00:00
|
|
|
import os
|
|
|
|
with open(os.path.join(os.environ['CREDENTIALS_DIRECTORY'], 'email_password')) as f:
|
2022-09-30 11:47:45 +00:00
|
|
|
pw = f.read()
|
|
|
|
MAIL_PASSWORD = pw
|
|
|
|
'' + formatPy cfg.settings;
|
2022-03-05 16:20:37 +00:00
|
|
|
mode = "0600";
|
|
|
|
user = "pgadmin";
|
|
|
|
group = "pgadmin";
|
|
|
|
};
|
|
|
|
};
|
|
|
|
}
|