2b0aea1793
The attribute ‘config.systemd.services.<service-name>.runner’ generates a script that runs the service outside of systemd. This is useful for testing, and also allows NixOS services to be used outside of NixOS. For instance, given a configuration file foo.nix: { config, pkgs, ... }: { services.postgresql.enable = true; services.postgresql.package = pkgs.postgresql92; services.postgresql.dataDir = "/tmp/postgres"; } you can build and run PostgreSQL as follows: $ nix-build -A config.systemd.services.postgresql.runner -I nixos-config=./foo.nix $ ./result This will run the service's ExecStartPre, ExecStart, ExecStartPost and ExecStopPost commands in an appropriate environment. It doesn't work well yet for "forking" services, since it can't track the main process. It also doesn't work for services that assume they're always executed by root.
239 lines
6.5 KiB
Nix
239 lines
6.5 KiB
Nix
{ config, pkgs, ... }:
|
|
|
|
with pkgs.lib;
|
|
|
|
let
|
|
|
|
cfg = config.services.postgresql;
|
|
|
|
# see description of extraPlugins
|
|
postgresqlAndPlugins = pg:
|
|
if cfg.extraPlugins == [] then pg
|
|
else pkgs.buildEnv {
|
|
name = "postgresql-and-plugins-${(builtins.parseDrvName pg.name).version}";
|
|
paths = [ pg ] ++ cfg.extraPlugins;
|
|
postBuild =
|
|
''
|
|
mkdir -p $out/bin
|
|
rm $out/bin/{pg_config,postgres,pg_ctl}
|
|
cp --target-directory=$out/bin ${pg}/bin/{postgres,pg_config,pg_ctl}
|
|
'';
|
|
};
|
|
|
|
postgresql = postgresqlAndPlugins cfg.package;
|
|
|
|
flags = optional cfg.enableTCPIP "-i";
|
|
|
|
# The main PostgreSQL configuration file.
|
|
configFile = pkgs.writeText "postgresql.conf"
|
|
''
|
|
hba_file = '${pkgs.writeText "pg_hba.conf" cfg.authentication}'
|
|
ident_file = '${pkgs.writeText "pg_ident.conf" cfg.identMap}'
|
|
log_destination = 'stderr'
|
|
${cfg.extraConfig}
|
|
'';
|
|
|
|
pre84 = versionOlder (builtins.parseDrvName postgresql.name).version "8.4";
|
|
|
|
in
|
|
|
|
{
|
|
|
|
###### interface
|
|
|
|
options = {
|
|
|
|
services.postgresql = {
|
|
|
|
enable = mkOption {
|
|
type = types.bool;
|
|
default = false;
|
|
description = ''
|
|
Whether to run PostgreSQL.
|
|
'';
|
|
};
|
|
|
|
package = mkOption {
|
|
type = types.path;
|
|
example = literalExample "pkgs.postgresql92";
|
|
description = ''
|
|
PostgreSQL package to use.
|
|
'';
|
|
};
|
|
|
|
port = mkOption {
|
|
type = types.int;
|
|
default = "5432";
|
|
description = ''
|
|
Port for PostgreSQL.
|
|
'';
|
|
};
|
|
|
|
dataDir = mkOption {
|
|
type = types.path;
|
|
default = "/var/db/postgresql";
|
|
description = ''
|
|
Data directory for PostgreSQL.
|
|
'';
|
|
};
|
|
|
|
authentication = mkOption {
|
|
type = types.lines;
|
|
default = "";
|
|
description = ''
|
|
Defines how users authenticate themselves to the server.
|
|
'';
|
|
};
|
|
|
|
identMap = mkOption {
|
|
type = types.lines;
|
|
default = "";
|
|
description = ''
|
|
Defines the mapping from system users to database users.
|
|
'';
|
|
};
|
|
|
|
initialScript = mkOption {
|
|
type = types.nullOr types.path;
|
|
default = null;
|
|
description = ''
|
|
A file containing SQL statements to execute on first startup.
|
|
'';
|
|
};
|
|
|
|
enableTCPIP = mkOption {
|
|
type = types.bool;
|
|
default = false;
|
|
description = ''
|
|
Whether to run PostgreSQL with -i flag to enable TCP/IP connections.
|
|
'';
|
|
};
|
|
|
|
extraPlugins = mkOption {
|
|
type = types.listOf types.path;
|
|
default = [];
|
|
example = literalExample "pkgs.postgis";
|
|
description = ''
|
|
When this list contains elements a new store path is created.
|
|
PostgreSQL and the elments are symlinked into it. Then pg_config,
|
|
postgres and pc_ctl are copied to make them use the new
|
|
$out/lib directory as pkglibdir. This makes it possible to use postgis
|
|
without patching the .sql files which reference $libdir/postgis-1.5.
|
|
'';
|
|
# Note: the duplication of executables is about 4MB size.
|
|
# So a nicer solution was patching postgresql to allow setting the
|
|
# libdir explicitely.
|
|
};
|
|
|
|
extraConfig = mkOption {
|
|
type = types.lines;
|
|
default = "";
|
|
description = "Additional text to be appended to <filename>postgresql.conf</filename>.";
|
|
};
|
|
|
|
recoveryConfig = mkOption {
|
|
type = types.nullOr types.lines;
|
|
default = null;
|
|
description = ''
|
|
Contents of the <filename>recovery.conf</filename> file.
|
|
'';
|
|
};
|
|
};
|
|
|
|
};
|
|
|
|
|
|
###### implementation
|
|
|
|
config = mkIf config.services.postgresql.enable {
|
|
|
|
services.postgresql.authentication =
|
|
''
|
|
# Generated file; do not edit!
|
|
local all all ident ${optionalString pre84 "sameuser"}
|
|
host all all 127.0.0.1/32 md5
|
|
host all all ::1/128 md5
|
|
'';
|
|
|
|
users.extraUsers.postgres =
|
|
{ name = "postgres";
|
|
uid = config.ids.uids.postgres;
|
|
group = "postgres";
|
|
description = "PostgreSQL server user";
|
|
};
|
|
|
|
users.extraGroups.postgres.gid = config.ids.gids.postgres;
|
|
|
|
environment.systemPackages = [postgresql];
|
|
|
|
systemd.services.postgresql =
|
|
{ description = "PostgreSQL Server";
|
|
|
|
wantedBy = [ "multi-user.target" ];
|
|
after = [ "network.target" ];
|
|
|
|
environment.PGDATA = cfg.dataDir;
|
|
|
|
path = [ pkgs.su postgresql ];
|
|
|
|
preStart =
|
|
''
|
|
# Initialise the database.
|
|
if ! test -e ${cfg.dataDir}; then
|
|
mkdir -m 0700 -p ${cfg.dataDir}
|
|
if [ "$(id -u)" = 0 ]; then
|
|
chown -R postgres ${cfg.dataDir}
|
|
su -s ${pkgs.stdenv.shell} postgres -c 'initdb -U root'
|
|
else
|
|
# For non-root operation.
|
|
initdb
|
|
fi
|
|
rm -f ${cfg.dataDir}/*.conf
|
|
touch "${cfg.dataDir}/.first_startup"
|
|
fi
|
|
|
|
ln -sfn "${configFile}" "${cfg.dataDir}/postgresql.conf"
|
|
${optionalString (cfg.recoveryConfig != null) ''
|
|
ln -sfn "${pkgs.writeText "recovery.conf" cfg.recoveryConfig}" \
|
|
"${cfg.dataDir}/recovery.conf"
|
|
''}
|
|
''; # */
|
|
|
|
serviceConfig =
|
|
{ ExecStart = "@${postgresql}/bin/postgres postgres ${toString flags}";
|
|
User = "postgres";
|
|
Group = "postgres";
|
|
PermissionsStartOnly = true;
|
|
|
|
# Shut down Postgres using SIGINT ("Fast Shutdown mode"). See
|
|
# http://www.postgresql.org/docs/current/static/server-shutdown.html
|
|
KillSignal = "SIGINT";
|
|
|
|
# Give Postgres a decent amount of time to clean up after
|
|
# receiving systemd's SIGINT.
|
|
TimeoutSec = 120;
|
|
};
|
|
|
|
# Wait for PostgreSQL to be ready to accept connections.
|
|
postStart =
|
|
''
|
|
while ! psql postgres -c "" 2> /dev/null; do
|
|
if ! kill -0 "$MAINPID"; then exit 1; fi
|
|
sleep 0.1
|
|
done
|
|
|
|
if test -e "${cfg.dataDir}/.first_startup"; then
|
|
${optionalString (cfg.initialScript != null) ''
|
|
cat "${cfg.initialScript}" | psql postgres
|
|
''}
|
|
rm -f "${cfg.dataDir}/.first_startup"
|
|
fi
|
|
'';
|
|
|
|
unitConfig.RequiresMountsFor = "${cfg.dataDir}";
|
|
};
|
|
|
|
};
|
|
|
|
}
|