working on harmonia

This commit is contained in:
Qyriad 2026-02-10 14:59:44 +01:00
parent 1f466b63d3
commit 8dba8e7ce8
20 changed files with 556 additions and 90 deletions

View file

@ -19,17 +19,14 @@ in stdenv.mkDerivation (self: {
]; ];
}; };
phases = [ "unpackPhase" "patchPhase" "installPhase "]; phases = [ "unpackPhase" "patchPhase" "installPhase" ];
installPhase = lib.dedent '' installPhase = lib.dedent ''
mkdir -vp "$out" mkdir -p "$out"
cp -rv * "$out/" cp -r * "$out/"
#mkdir -vp "$modules/share/nix/modules/dynix" mkdir -p "$modules/share/nixos/modules/dynix"
#cp --reflink=auto -rv * "$modules/share/nix/modules/dynix/" cp --reflink=auto -r "$out/"* "$modules/share/nixos/modules/dynix/"
mkdir -vp "$modules/share/nixos/modules/dynix"
cp --reflink=auto -rv * "$modules/share/nixos/modules/dynix/"
''; '';
passthru.mkDevShell = { passthru.mkDevShell = {
@ -51,6 +48,8 @@ in stdenv.mkDerivation (self: {
] |> lib.concatStringsSep ":"; ] |> lib.concatStringsSep ":";
}; };
passthru.modulesPath = self.finalPackage.modules + "/share/nixos/modules";
passthru.tests = lib.fix (callPackage ./tests { passthru.tests = lib.fix (callPackage ./tests {
dynix = self.finalPackage; dynix = self.finalPackage;
}).packages; }).packages;

View file

@ -4,6 +4,9 @@ let
mkOption mkOption
showOption showOption
; ;
inherit (lib.asserts)
checkAssertWarn
;
t = lib.types; t = lib.types;
inherit (import ./lib.nix { inherit lib; }) inherit (import ./lib.nix { inherit lib; })
@ -19,7 +22,6 @@ let
opts = options.dynamicism; opts = options.dynamicism;
subOpts = lib.mapAttrs (_: metaAttr: metaAttr.configuration.options) options.dynamicism.for.valueMeta.attrs; subOpts = lib.mapAttrs (_: metaAttr: metaAttr.configuration.options) options.dynamicism.for.valueMeta.attrs;
settingsFormat = pkgs.formats.yaml { };
finalSettingsFor = { ... }@submod: recUpdateFoldl (optPath: finalSettingsFor = { ... }@submod: recUpdateFoldl (optPath:
lib.setAttrByPath optPath (lib.getAttrFromPath optPath config) lib.setAttrByPath optPath (lib.getAttrFromPath optPath config)
@ -82,7 +84,18 @@ in
# #
# Generic implementation. # Generic implementation.
# #
config.dynamicism.doChange = { config.system.activationScripts."dynamicism-reset" = {
deps = [ "etc" "stdio" "specialfs" ];
text = ''
echo "DYNIX: removing existing systemd dropins"
# FIXME: do for each enabled submodule
if [[ -d /run/systemd/system ]]; then
rm -v /run/systemd/system/*/dynix-*.conf || true
fi
'';
};
config.dynamicism = {
doChange = {
option, option,
value, value,
configuration ? builtins.getEnv "NIXOS_CONFIG", configuration ? builtins.getEnv "NIXOS_CONFIG",
@ -102,11 +115,14 @@ in
}; };
}; };
allActivations = lib.mapAttrsToList (name: submod: submod.activate) config.dynamicism.for; allActivations = config.dynamicism.for
|> lib.filterAttrs (name: submod: submod.enable)
|> lib.mapAttrsToList (name: submod: submod.activate);
allActivationScripts = pkgs.writeShellApplication { allActivationScripts = pkgs.writeShellApplication {
name = "dynamicism-activate"; name = "dynamicism-activate";
runtimeInputs = allActivations; runtimeInputs = allActivations;
text = nixosAfter.config.dynamicism.for text = nixosAfter.config.dynamicism.for
|> lib.filterAttrs (name: submod: submod.enable)
|> lib.mapAttrsToList (name: submod: '' |> lib.mapAttrsToList (name: submod: ''
echo "Activating dynamicism for ${name}" echo "Activating dynamicism for ${name}"
${lib.getExe submod.activate} ${lib.getExe submod.activate}
@ -115,12 +131,15 @@ in
}; };
in allActivationScripts; in allActivationScripts;
config.dynamicism.finalSettings = lib.asserts.checkAssertWarn ourAssertions [ ] ( finalSettings = config.dynamicism.for
recUpdateFoldlAttrs (name: { ... }@submod: finalSettingsFor submod) config.dynamicism.for |> recUpdateFoldlAttrs (name: { ... }@submod: finalSettingsFor submod)
); |> checkAssertWarn ourAssertions [ ];
};
# Implementations. # Implementations.
imports = [ imports = [
./gotosocial.nix ./gotosocial.nix
./harmonia.nix
#./tzupdate.nix
]; ];
} }

View file

@ -3,17 +3,20 @@ let
cfg = config.dynamicism.for.gotosocial; cfg = config.dynamicism.for.gotosocial;
settingsFormat = pkgs.formats.yaml { }; settingsFormat = pkgs.formats.yaml { };
configFile = settingsFormat.generate "gotosocial-override.yml" config.services.gotosocial.settings;
in in
{ {
dynamicism.for.gotosocial = { dynamicism.for.gotosocial = {
source-options = [ "services.gotosocial.settings" ]; source-options = [ "services.gotosocial.settings" ];
configFile = settingsFormat.generate "gotosocial-overrde.yml" config.services.gotosocial.settings; unitDropins."gotosocial.service" = pkgs.writeTextFile {
name = "gotosocial-override.conf";
unitDropins."gotosocial.service" = pkgs.writeText "gotosocial-override.conf" '' text = ''
[Service] [Service]
ExecStart= ExecStart=
ExecStart=${lib.getExe' pkgs.gotosocial "gotosocial"} --config-path ${cfg.configFile} start ExecStart=${lib.getExe' pkgs.gotosocial "gotosocial"} --config-path ${configFile} start
''; '';
passthru = { inherit configFile; };
};
}; };
} }

View file

@ -0,0 +1,26 @@
{ pkgs, lib, config, ... }:
let
cfg = config.dynamicism.for.harmonia;
settingsFormat = pkgs.formats.toml { };
# FIXME: referring to config.dynamicism.finalSettings.* in here
# makes lib.checkAssertWarn in the generic module cause infinite recursion.
finalSettings = config.services.harmonia.settings;
configFile = settingsFormat.generate "harmonia-override.toml" finalSettings;
in
{
dynamicism.for.harmonia = {
source-options = [ "services.harmonia.settings" ];
unitDropins."harmonia.service" = pkgs.writeTextFile {
name = "harmonia-override.conf";
text = ''
[Service]
Environment=CONFIG_FILE=${configFile}
'';
passthru = { inherit configFile; };
};
};
}

View file

@ -65,7 +65,7 @@ in
}; };
unitDropins = mkOption { unitDropins = mkOption {
type = t.attrsOf t.pathInStore; type = t.attrsOf t.package;
internal = true; internal = true;
}; };
@ -82,7 +82,7 @@ in
text = let text = let
doEdits = config.unitDropins doEdits = config.unitDropins
|> lib.mapAttrsToList (service: dropin: '' |> lib.mapAttrsToList (service: dropin: ''
cat "${dropin}" | systemctl edit "${service}" --runtime --stdin cat "${dropin}" | systemctl edit "${service}" --runtime --drop=dynix-${dropin.name} --stdin
''); '');
doReloads = config.unitDropins doReloads = config.unitDropins
|> lib.mapAttrsToList (service: _: '' |> lib.mapAttrsToList (service: _: ''

View file

@ -0,0 +1,20 @@
{ pkgs, lib, config, ... }: let
cfg = config.dynamicism.for.tzupdate;
# FIXME: referring to config.dynamicism.finalSettings.* in here
# makes lib.checkAssertWarn in the generic module cause infinite recursion.
#finalSettings = config.dynamicism.finalSettings.tzupdate;
finalSettings = config.services.tzupdate.timer;
in
{
dynamicism.for.tzupdate = {
source-options = [ "services.tzupdate.timer" ];
unitDropins."tzupdate.timer" = pkgs.writeText "tzupdate-timer-override.conf" ''
[Timer]
OnCalendar=
OnCalendar=${finalSettings.interval}
'';
};
}

View file

@ -10,22 +10,40 @@
runDynixTest = testModule: pkgs.testers.runNixOSTest { runDynixTest = testModule: pkgs.testers.runNixOSTest {
imports = [ testModule ]; imports = [ testModule ];
# Note: these are arguments to the *test* modules.
# Not the NixOS modules for the NixOS configuration the test is testing.
# Wew.
_module.args = { inherit dynix; };
# Why is this argument called "extraBaseModule**s**" but take a single module argument... # Why is this argument called "extraBaseModule**s**" but take a single module argument...
extraBaseModules = { name, ... }: { # Also note this is an extra base module for each node of the test,
#imports = [ dynixInjectionModule ]; # not an extra test module.
config.environment.systemPackages = [ dynix ]; extraBaseModules = { name, config, options, modulesPath, ... }: {
imports = (import "${modulesPath}/module-list.nix") ++ [
./module-allow-rebuild-in-vm.nix
"${modulesPath}/testing/test-instrumentation.nix"
(toString dynix)
];
environment.systemPackages = [ dynix ];
systemd.services."install-dynix" = {
enable = true;
serviceConfig.Type = "oneshot";
serviceConfig.RemainAfterExit = true;
path = [ config.system.path ];
wantedBy = [ "multi-user.target" ];
after = [ "default.target" ];
script = ''
nix profile install -vv "$(realpath /run/current-system/sw/share/nixos/modules/dynix/)"
'';
};
passthru = { inherit options; };
# Just making something in this strict in `name`, # Just making something in this strict in `name`,
# which is only present as an argument for nodes and I don't want to # which is only present as an argument for nodes and I don't want to
# confuse that with the test modules. # confuse that with the test modules.
config.warnings = builtins.seq name [ ]; warnings = builtins.seq name [ ];
}; };
}; };
in lib.makeScope lib.callPackageWith (self: { in lib.makeScope lib.callPackageWith (self: {
gotosocial = runDynixTest ./gotosocial/test.nix; gotosocial = runDynixTest ./gotosocial/test.nix;
harmonia = runDynixTest ./harmonia/test.nix;
#tzupdate = runDynixTest ./tzupdate/test.nix;
}) })

View file

@ -1,9 +1,7 @@
{ {
runCommand, runCommand,
}: runCommand "tests-basic-configuration-dot-nix" { }: runCommand "tests-gotosocial-configuration-dot-nix" {
} '' } ''
set -euo pipefail set -euo pipefail
mkdir -vp "$out/share/nixos" install -Dm a=r ${./configuration.nix} "$out/share/nixos/configuration.nix"
cp -rv ${./configuration.nix} "$out/share/nixos/configuration.nix"
#cp -rv ${../../modules/dynamicism} "$out/share/nixos/dynamicism"
'' ''

View file

@ -18,7 +18,7 @@ in
system.switch.enable = true; system.switch.enable = true;
documentation.enable = false; documentation.enable = false;
networking.hostName = "machine"; networking.hostName = "gotosocial-machine";
boot.loader.grub = { boot.loader.grub = {
enable = true; enable = true;
@ -47,7 +47,7 @@ in
enable = true; enable = true;
setupPostgresqlDB = true; setupPostgresqlDB = true;
settings = { settings = {
application-name = "gotosocial-for-${name}"; application-name = "gotosocial-for-machine";
host = "${name}.local"; host = "${name}.local";
}; };
}; };

View file

@ -34,7 +34,6 @@ def run_log(machine: Machine, *commands: str, timeout: int | None = 60) -> str:
def get_config_file() -> str: def get_config_file() -> str:
machine.wait_for_unit("gotosocial.service") machine.wait_for_unit("gotosocial.service")
gotosocial_pid = int(machine.get_unit_property("gotosocial.service", "MainPID")) gotosocial_pid = int(machine.get_unit_property("gotosocial.service", "MainPID"))
print(f"{gotosocial_pid=}")
cmdline = machine.succeed(f"cat /proc/{gotosocial_pid}/cmdline") cmdline = machine.succeed(f"cat /proc/{gotosocial_pid}/cmdline")
cmdline_args = cmdline.split("\0") cmdline_args = cmdline.split("\0")
@ -53,8 +52,7 @@ def get_config_file() -> str:
machine.wait_for_unit("default.target") machine.wait_for_unit("default.target")
assert "lix" in machine.succeed("nix --version").lower() assert "lix" in machine.succeed("nix --version").lower()
machine.log("INIT") machine.log("INIT")
run_log(machine, "journalctl --no-pager -eu install-dynix.service")
machine.succeed("nix profile install -vv $(realpath /run/current-system/sw/share/nixos/modules/dynix)")
machine.succeed("nixos-generate-config") machine.succeed("nixos-generate-config")
machine.succeed("mkdir -vp /etc/nixos") machine.succeed("mkdir -vp /etc/nixos")

View file

@ -1,7 +1,7 @@
{ dynix, ... }: { ... }:
{ {
name = "nixos-test-dynamicism-main"; name = "nixos-test-dynamicism-gotosocial";
defaults = { ... }: { }; defaults = { ... }: { };
@ -9,27 +9,12 @@
p.beartype p.beartype
]; ];
nodes.machine = { pkgs, config, ... }: { nodes.machine = { pkgs, ... }: {
# NOTE: Anything in this `nodes.machine = ` module will not be included # NOTE: Anything in this `nodes.machine = ` module will not be included
# in the VM's NixOS configuration once it does `nixos-rebuild switch`, # in the VM's NixOS configuration once it does `nixos-rebuild switch`,
# except for `./configuration.nix` which will be copied to `/etc/nixos/`. # except for `./configuration.nix` which will be copied to `/etc/nixos/`.
# dynix will also be statefully installed to root's user profile. # dynix will also be statefully installed to root's user profile.
imports = [ imports = [ ./configuration.nix ];
./configuration.nix
(toString dynix)
];
system.includeBuildDependencies = true;
system.switch.enable = true;
virtualisation.additionalPaths = [ config.system.build.toplevel ];
virtualisation = {
memorySize = 4096;
cores = 4;
writableStore = true;
mountHostNixStore = true;
installBootLoader = true;
};
environment.systemPackages = let environment.systemPackages = let
configFileTree = pkgs.callPackage ./configuration-package.nix { }; configFileTree = pkgs.callPackage ./configuration-package.nix { };
@ -38,7 +23,5 @@
]; ];
}; };
# What's a little IFD between friends? testScript = builtins.readFile ./test-script.py;
testScript = ./test-script.py
|> builtins.readFile;
} }

View file

@ -0,0 +1,72 @@
{ pkgs, lib, config, modulesPath, ... }:
let
moduleList = import "${modulesPath}/module-list.nix";
dynixFromSearchPath = let
res = builtins.tryEval <dynix>;
in lib.optional res.success res.value;
in
{
imports = [
"${modulesPath}/testing/test-instrumentation.nix"
./hardware-configuration.nix
] ++ lib.concatLists [
dynixFromSearchPath
moduleList
];
dynamicism.for.harmonia.enable = true;
services.harmonia = {
enable = true;
settings = {
# Default.
workers = 4;
# Default.
max_connection_rate = 256;
};
};
system.switch.enable = true;
documentation.enable = false;
networking.hostName = "harmonia-machine";
boot.loader.grub = {
enable = true;
device = "/dev/vda";
forceInstall = true;
};
nix = {
package = pkgs.lixPackageSets.latest.lix;
nixPath = [
"nixpkgs=${pkgs.path}"
"/nix/var/nix/profiles/per-user/root/profile/share/nixos/modules"
];
settings = {
experimental-features = [ "nix-command" "pipe-operator" ];
substituters = lib.mkForce [ ];
hashed-mirrors = null;
connect-timeout = 1;
# For my debugging purposes.
show-trace = true;
};
};
environment.pathsToLink = [ "/share" ];
environment.extraOutputsToInstall = [ "modules" ];
environment.variables = {
"NIXOS_CONFIG" = "/etc/nixos/configuration.nix";
};
environment.shellAliases = {
ls = "eza --long --header --group --group-directories-first --classify --binary";
};
environment.systemPackages = with pkgs; [
eza
fd
ripgrep
];
}

View file

@ -0,0 +1,4 @@
{ ... }:
{
}

View file

@ -0,0 +1,125 @@
import functools
from pathlib import Path
from pprint import pformat
import shlex
import textwrap
import tomllib
from typing import Any, cast, TYPE_CHECKING
from beartype import beartype
from test_driver.machine import Machine
from test_driver.errors import RequestedAssertionFailed
if TYPE_CHECKING:
global machine
machine = cast(Machine, ...)
assert machine.shell is not None
ls = "eza -lah --color=always --group-directories-first"
indent = functools.partial(textwrap.indent, prefix=' ')
@beartype
def run_log(machine: Machine, *commands: str, timeout: int | None = 60) -> str:
output = ""
for command in commands:
with machine.nested(f"must succeed: {command}"):
(status, out) = machine.execute(f"{command} | tee /dev/stderr", timeout=timeout)
if status != 0:
machine.log(f"output: {out}")
raise RequestedAssertionFailed(
f"command `{command}` failed (exit code {status})",
)
output += out
return output
@beartype
def get_config_file() -> dict[str, Any]:
machine.wait_for_unit("harmonia.service")
pid = int(machine.get_unit_property("harmonia.service", "MainPID"))
env_lines: list[str] = machine.succeed(f"cat /proc/{pid}/environ").replace("\0", "\n").splitlines()
pairs: list[list[str]] = [line.split("=", maxsplit=1) for line in env_lines]
env = dict(pairs)
config_file = Path(env["CONFIG_FILE"])
machine.log(f"copying from VM: {config_file=}")
machine.copy_from_vm(config_file.as_posix())
config_file_path = machine.out_dir / config_file.name
with open(config_file_path, "rb") as f:
config_data = tomllib.load(f)
config_file_path.unlink()
return config_data
machine.wait_for_unit("default.target")
assert "lix" in machine.succeed("nix --version").lower()
machine.log("INIT")
# Config should have our initial values.
config_toml = get_config_file()
machine.log(f"config.toml BEFORE first rebuild (initial): {indent(pformat(config_toml))}")
assert int(config_toml['workers']) == 4, f"{config_toml['workers']=} != 4"
assert int(config_toml['max_connection_rate']) == 256, f"{config_toml['max_connection_rate']=} != 256"
with machine.nested("must succeed: initial nixos-rebuild switch"):
machine.succeed("nixos-generate-config")
machine.succeed("mkdir -vp /etc/nixos")
# Dereference is required since that configuration.nix is probably a symlink to the store.
machine.succeed("cp -rv --dereference /run/current-system/sw/share/nixos/configuration.nix /etc/nixos/")
machine.succeed("env PAGER= nixos-rebuild switch --log-format raw-with-logs --fallback")
# Config should not have changed.
config_toml = get_config_file()
machine.log(f"config.toml after first rebuild: {indent(pformat(config_toml))}")
assert int(config_toml['workers']) == 4, f"{config_toml['workers']=} != 4"
assert int(config_toml['max_connection_rate']) == 256, f"{config_toml['max_connection_rate']=} != 256"
new_workers = 20
expr = textwrap.dedent(f"""
let
nixos = import <nixpkgs/nixos> {{ }};
in nixos.config.dynamicism.doChange {{
option = "services.harmonia.settings.workers";
value = {new_workers};
}}
""").strip()
machine.succeed(rf"""
nix run --show-trace --log-format raw-with-logs --impure -E {shlex.quote(expr)}
""".strip())
# Workers, but not max connection rate, should have changed.
config_toml = get_config_file()
machine.log(f"config.toml after DYNAMIC ACTIVATION: {indent(pformat(config_toml))}")
assert int(config_toml['workers']) == new_workers, f"{config_toml['workers']=} != {new_workers}"
assert int(config_toml['max_connection_rate']) == 256, f"{config_toml['max_connection_rate']=} != 256"
new_max_connection_rate = 100
expr = textwrap.dedent(f"""
let
nixos = import <nixpkgs/nixos> {{ }};
in nixos.config.dynamicism.doChange {{
option = [ "services" "harmonia" "settings" "max_connection_rate" ];
value = {new_max_connection_rate};
}}
""").strip()
machine.succeed(rf"""
nix run --show-trace --log-format raw-with-logs --impure -E {shlex.quote(expr)}
""".strip())
# Max connection rate should have changed, but workers should have reverted.
config_toml = get_config_file()
machine.log(f"config_toml after SECOND dynamic activation: {indent(pformat(config_toml))}")
assert int(config_toml['max_connection_rate']) == new_max_connection_rate, f"{config_toml['max_connection_rate']=} != {new_max_connection_rate}"
# And this should set everything back.
machine.succeed("env PAGER= nixos-rebuild switch --log-format raw-with-logs --fallback")
machine.systemctl("restart harmonia.service")
machine.wait_for_unit("harmonia.service")
config_toml = get_config_file()
machine.log(f"config_toml after NORMAL activation: {indent(pformat(config_toml))}")
assert int(config_toml['max_connection_rate']) == 256, f'{config_toml["max_connection_rate"]=} != 256'
assert int(config_toml['workers']) == 4, f'{config_toml["workers"]=} != 4'

23
tests/harmonia/test.nix Normal file
View file

@ -0,0 +1,23 @@
{ ... }:
{
name = "nixos-test-dynamicism-harmonia";
defaults = { ... }: { };
extraPythonPackages = p: [ p.beartype ];
nodes.machine = { name, pkgs, ... }: {
imports = [ ./configuration.nix ];
environment.systemPackages = let
configFileTree = pkgs.runCommand "${name}-configuration-dot-nix" { } ''
set -euo pipefail
install -Dm a=r ${./configuration.nix} "$out/share/nixos/configuration.nix"
'';
in [
configFileTree
];
};
testScript = builtins.readFile ./test-script.py;
}

View file

@ -0,0 +1,31 @@
{ name, config, ... }:
{
system.includeBuildDependencies = true;
system.switch.enable = true;
documentation.enable = false;
virtualisation = {
additionalPaths = [ config.system.build.toplevel ];
writableStore = true;
mountHostNixStore = true;
installBootLoader = true;
# With how much memory Nix eval uses, this is essentially required.
memorySize = 4096;
cores = 4;
};
boot.loader.grub = {
enable = true;
device = "/dev/vda";
forceInstall = true;
};
# Just making something in this strict in `name`,
# which is only present as an argument for nodes and I don't want to
# confuse that with the test modules.
warnings = builtins.seq name [ ];
}

View file

@ -0,0 +1,67 @@
{ pkgs, lib, config, modulesPath, ... }:
let
name = config.networking.hostName;
moduleList = import "${modulesPath}/module-list.nix";
dynixFromSearchPath = let
res = builtins.tryEval <dynix>;
in lib.optional res.success res.value;
in
{
imports = [
"${modulesPath}/testing/test-instrumentation.nix"
./hardware-configuration.nix
] ++ lib.concatLists [
moduleList
dynixFromSearchPath
];
dynamicism.for.tzupdate.enable = true;
services.tzupdate = {
enable = true;
};
system.switch.enable = true;
documentation.enable = false;
networking.hostName = "tzupdate-machine";
boot.loader.grub = {
enable = true;
device = "/dev/vda";
forceInstall = true;
};
nix = {
package = pkgs.lixPackageSets.latest.lix;
nixPath = [
"nixpkgs=${pkgs.path}"
"/nix/var/nix/profiles/per-user/root/profile/share/nixos/modules"
];
settings = {
experimental-features = [ "nix-command" "pipe-operator" ];
substituters = lib.mkForce [ ];
hashed-mirrors = null;
connect-timeout = 1;
# For my debugging purposes.
show-trace = true;
};
};
environment.pathsToLink = [ "/share" ];
environment.extraOutputsToInstall = [ "modules" ];
environment.variables = {
"NIXOS_CONFIG" = "/etc/nixos/configuration.nix";
};
environment.shellAliases = {
ls = "eza --long --header --group --group-directories-first --classify --binary";
};
environment.systemPackages = with pkgs; [
eza
fd
ripgrep
];
}

View file

@ -0,0 +1,4 @@
{ ... }:
{
}

View file

@ -0,0 +1,52 @@
#from pathlib import Path
#import shlex
#import textwrap
from typing import cast, TYPE_CHECKING
from beartype import beartype
from test_driver.machine import Machine
from test_driver.errors import RequestedAssertionFailed
if TYPE_CHECKING:
global machine
machine = cast(Machine, ...)
assert machine.shell is not None
ls = "eza -lah --color=always --group-directories-first"
@beartype
def run_log(machine: Machine, *commands: str, timeout: int | None = 60) -> str:
output = ""
for command in commands:
with machine.nested(f"must succeed: {command}"):
(status, out) = machine.execute(f"{command} | tee /dev/stderr", timeout=timeout)
if status != 0:
machine.log(f"output: {out}")
raise RequestedAssertionFailed(
f"command `{command}` failed (exit code {status})",
)
output += out
return output
machine.wait_for_unit("default.target")
assert "lix" in machine.succeed("nix --version").lower()
machine.log("INIT")
machine.succeed("nixos-generate-config")
machine.succeed("mkdir -vp /etc/nixos")
# Dereference is required since that configuration.nix is probably a symlink to the store.
machine.succeed("cp -rv --dereference /run/current-system/sw/share/nixos/configuration.nix /etc/nixos/")
machine.succeed("env PAGER= nixos-rebuild switch --log-format raw-with-logs --fallback")
@beartype
def get_interval() -> str:
prop = machine.get_unit_property("tzupdate.timer", "OnCalendar")
print(f"{prop=}")
return prop
get_interval()

24
tests/tzupdate/test.nix Normal file
View file

@ -0,0 +1,24 @@
{ ... }:
{
name = "nixos-test-dynamicism-tzupdate";
extraPythonPackages = p: [
p.beartype
];
nodes.machine = { name, pkgs, ... }: {
imports = [ ./configuration.nix ];
environment.systemPackages = let
configFileTree = pkgs.runCommand "${name}-configuration-dot-nix" { } ''
set -euo pipefail
install -Dm a=r ${./configuration.nix} "$out/share/nixos/configuration.nix"
'';
in [
configFileTree
];
};
testScript = builtins.readFile ./test-script.py;
}