-
-
Notifications
You must be signed in to change notification settings - Fork 15.1k
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Allow running NixOS services outside of systemd
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.
- Loading branch information
Showing
3 changed files
with
122 additions
and
2 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,114 @@ | ||
{ config, pkgs, ... }: | ||
|
||
with pkgs.lib; | ||
|
||
let | ||
|
||
makeScript = name: service: pkgs.writeScript "${name}-runner" | ||
'' | ||
#! ${pkgs.perl}/bin/perl -w -I${pkgs.perlPackages.FileSlurp}/lib/perl5/site_perl | ||
use File::Slurp; | ||
sub run { | ||
my ($cmd) = @_; | ||
my @args = split " ", $cmd; | ||
my $prog; | ||
if (substr($args[0], 0, 1) eq "@") { | ||
$prog = substr($args[0], 1); | ||
shift @args; | ||
} else { | ||
$prog = $args[0]; | ||
} | ||
my $pid = fork; | ||
if ($pid == 0) { | ||
setpgrp; # don't receive SIGINT etc. from terminal | ||
exec { $prog } @args; | ||
die "failed to exec $prog\n"; | ||
} elsif (!defined $pid) { | ||
die "failed to fork: $!\n"; | ||
} | ||
return $pid; | ||
}; | ||
sub run_wait { | ||
my ($cmd) = @_; | ||
my $pid = run $cmd; | ||
die if waitpid($pid, 0) != $pid; | ||
return $?; | ||
}; | ||
# Set the environment. FIXME: escaping. | ||
foreach my $key (keys %ENV) { | ||
next if $key eq 'LOCALE_ARCHIVE'; | ||
delete $ENV{$key}; | ||
} | ||
${concatStrings (mapAttrsToList (n: v: '' | ||
$ENV{'${n}'} = '${v}'; | ||
'') service.environment)} | ||
# Run the ExecStartPre program. FIXME: this could be a list. | ||
my $preStart = '${service.serviceConfig.ExecStartPre or ""}'; | ||
if ($preStart ne "") { | ||
print STDERR "running ExecStartPre: $preStart\n"; | ||
my $res = run_wait $preStart; | ||
die "$0: ExecStartPre failed with status $res\n" if $res; | ||
}; | ||
# Run the ExecStart program. | ||
my $cmd = '${service.serviceConfig.ExecStart}'; | ||
print STDERR "running ExecStart: $cmd\n"; | ||
my $mainPid = run $cmd; | ||
$ENV{'MAINPID'} = $mainPid; | ||
# Catch SIGINT, propagate to the main program. | ||
sub intHandler { | ||
print STDERR "got SIGINT, stopping service...\n"; | ||
kill 'INT', $mainPid; | ||
}; | ||
$SIG{'INT'} = \&intHandler; | ||
$SIG{'QUIT'} = \&intHandler; | ||
# Run the ExecStartPost program. | ||
my $postStart = '${service.serviceConfig.ExecStartPost or ""}'; | ||
if ($postStart ne "") { | ||
print STDERR "running ExecStartPost: $postStart\n"; | ||
my $res = run_wait $postStart; | ||
die "$0: ExecStartPost failed with status $res\n" if $res; | ||
} | ||
# Wait for the main program to exit. | ||
die if waitpid($mainPid, 0) != $mainPid; | ||
my $mainRes = $?; | ||
# Run the ExecStopPost program. | ||
my $postStop = '${service.serviceConfig.ExecStopPost or ""}'; | ||
if ($postStop ne "") { | ||
print STDERR "running ExecStopPost: $postStop\n"; | ||
my $res = run_wait $postStop; | ||
die "$0: ExecStopPost failed with status $res\n" if $res; | ||
} | ||
exit($mainRes & 127 ? 255 : $mainRes << 8); | ||
''; | ||
|
||
in | ||
|
||
{ | ||
options = { | ||
systemd.services = mkOption { | ||
options = | ||
{ config, name, ... }: | ||
{ options.runner = mkOption { | ||
internal = true; | ||
description = '' | ||
A script that runs the service outside of systemd, | ||
useful for testing or for using NixOS services outside | ||
of NixOS. | ||
''; | ||
}; | ||
config.runner = makeScript name config; | ||
}; | ||
}; | ||
}; | ||
} |