Jade Lovelace
3571817e3a
This also rewrites a lot of the command handling in the fixtures
library, since we want to more precisely control which way that the nix
store is set up in the tests, rather than the previous method of
renaming /nix/store to some temp dir (which allows builds but does not
allow any /nix/store paths or stability across runs, which is a
significant issue for snapshot testing).
It uses a builder to reduce the amount of state carelessly thrown
around.
The evil NARs are inspired by CVE-2024-45593
(https://github.com/NixOS/nix/security/advisories/GHSA-h4vv-h3jq-v493).
No bugs were found in this endeavor.
Change-Id: Iee41b055fa96529c5a3c761f680ed1d0667ba5da
208 lines
6.9 KiB
Python
208 lines
6.9 KiB
Python
import os
|
|
import json
|
|
import subprocess
|
|
from typing import Any
|
|
from pathlib import Path
|
|
from functools import partial, partialmethod
|
|
import dataclasses
|
|
|
|
|
|
@dataclasses.dataclass
|
|
class CommandResult:
|
|
cmd: list[str]
|
|
rc: int
|
|
"""Return code"""
|
|
stderr: bytes
|
|
"""Outputted stderr"""
|
|
stdout: bytes
|
|
"""Outputted stdout"""
|
|
|
|
def ok(self):
|
|
if self.rc != 0:
|
|
raise subprocess.CalledProcessError(returncode=self.rc,
|
|
cmd=self.cmd,
|
|
stderr=self.stderr,
|
|
output=self.stdout)
|
|
return self
|
|
|
|
def expect(self, rc: int):
|
|
if self.rc != rc:
|
|
raise subprocess.CalledProcessError(returncode=self.rc,
|
|
cmd=self.cmd,
|
|
stderr=self.stderr,
|
|
output=self.stdout)
|
|
return self
|
|
|
|
def json(self) -> Any:
|
|
self.ok()
|
|
return json.loads(self.stdout)
|
|
|
|
|
|
@dataclasses.dataclass
|
|
class NixSettings:
|
|
"""Settings for invoking Nix"""
|
|
experimental_features: set[str] | None = None
|
|
store: str | None = None
|
|
"""
|
|
The store to operate on (may be a path or other thing, see nix help-stores).
|
|
|
|
Note that this can be set to the test's store directory if you want to use
|
|
/nix/store paths inside that test rather than NIX_STORE_DIR renaming
|
|
/nix/store to some unstable name (assuming that no builds are invoked).
|
|
"""
|
|
nix_store_dir: Path | None = None
|
|
"""
|
|
Alternative name to use for /nix/store: breaks all references and NAR imports if
|
|
set, but does allow builds in tests (since builds do not require chroots if
|
|
the store is relocated).
|
|
"""
|
|
|
|
def feature(self, *names: str):
|
|
self.experimental_features = (self.experimental_features
|
|
or set()) | set(names)
|
|
return self
|
|
|
|
def to_config(self) -> str:
|
|
config = ''
|
|
|
|
def serialise(value):
|
|
if type(value) in {str, int}:
|
|
return str(value)
|
|
elif type(value) in {list, set}:
|
|
return ' '.join(str(e) for e in value)
|
|
else:
|
|
raise ValueError(
|
|
f'Value is unsupported in nix config: {value!r}')
|
|
|
|
def field_may(name, value, serialiser=serialise):
|
|
nonlocal config
|
|
if value is not None:
|
|
config += f'{name} = {serialiser(value)}\n'
|
|
|
|
field_may('experimental-features', self.experimental_features)
|
|
field_may('store', self.store)
|
|
assert self.store or self.nix_store_dir, 'Failing to set either nix_store_dir or store will cause accidental use of the system store.'
|
|
return config
|
|
|
|
def to_env_overlay(self) -> dict[str, str]:
|
|
ret = {'NIX_CONFIG': self.to_config()}
|
|
if self.nix_store_dir:
|
|
ret['NIX_STORE_DIR'] = str(self.nix_store_dir)
|
|
return ret
|
|
|
|
|
|
@dataclasses.dataclass
|
|
class Command:
|
|
argv: list[str]
|
|
env: dict[str, str] = dataclasses.field(default_factory=dict)
|
|
stdin: bytes | None = None
|
|
cwd: Path | None = None
|
|
|
|
def with_env(self, **kwargs) -> 'Command':
|
|
self.env.update(kwargs)
|
|
return self
|
|
|
|
def with_stdin(self, stdin: bytes) -> 'Command':
|
|
self.stdin = stdin
|
|
return self
|
|
|
|
def run(self) -> CommandResult:
|
|
proc = subprocess.Popen(
|
|
self.argv,
|
|
stdout=subprocess.PIPE,
|
|
stderr=subprocess.PIPE,
|
|
stdin=subprocess.PIPE if self.stdin else subprocess.DEVNULL,
|
|
cwd=self.cwd,
|
|
env=self.env,
|
|
)
|
|
(stdout, stderr) = proc.communicate(input=self.stdin)
|
|
rc = proc.returncode
|
|
return CommandResult(cmd=self.argv,
|
|
rc=rc,
|
|
stdout=stdout,
|
|
stderr=stderr)
|
|
|
|
|
|
@dataclasses.dataclass
|
|
class NixCommand(Command):
|
|
settings: NixSettings = dataclasses.field(default_factory=NixSettings)
|
|
|
|
def apply_nix_config(self):
|
|
self.env.update(self.settings.to_env_overlay())
|
|
|
|
def run(self) -> CommandResult:
|
|
self.apply_nix_config()
|
|
return super().run()
|
|
|
|
|
|
@dataclasses.dataclass
|
|
class Nix:
|
|
test_root: Path
|
|
|
|
def hermetic_env(self):
|
|
# mirroring vars-and-functions.sh
|
|
home = self.test_root / 'test-home'
|
|
home.mkdir(parents=True, exist_ok=True)
|
|
return {
|
|
'NIX_LOCALSTATE_DIR': self.test_root / 'var',
|
|
'NIX_LOG_DIR': self.test_root / 'var/log/nix',
|
|
'NIX_STATE_DIR': self.test_root / 'var/nix',
|
|
'NIX_CONF_DIR': self.test_root / 'etc',
|
|
'NIX_DAEMON_SOCKET_PATH': self.test_root / 'daemon-socket',
|
|
'NIX_USER_CONF_FILES': '',
|
|
'HOME': home,
|
|
}
|
|
|
|
def make_env(self):
|
|
# We conservatively assume that people might want to successfully get
|
|
# some env through to the subprocess, so we override whatever is in the
|
|
# global env.
|
|
d = os.environ.copy()
|
|
d.update(self.hermetic_env())
|
|
return d
|
|
|
|
def cmd(self, argv: list[str]):
|
|
return Command(argv=argv, cwd=self.test_root, env=self.make_env())
|
|
|
|
def settings(self, allow_builds: bool = False):
|
|
"""
|
|
Parameters:
|
|
- allow_builds: relocate the Nix store so that builds work (however, makes store paths non-reproducible across test runs!)
|
|
"""
|
|
settings = NixSettings()
|
|
store_path = self.test_root / 'store'
|
|
if allow_builds:
|
|
settings.nix_store_dir = store_path
|
|
else:
|
|
settings.store = str(store_path)
|
|
return settings
|
|
|
|
def nix_cmd(self, argv: list[str], allow_builds: bool = False):
|
|
"""
|
|
Constructs a NixCommand with the appropriate settings.
|
|
"""
|
|
return NixCommand(argv=argv,
|
|
cwd=self.test_root,
|
|
env=self.make_env(),
|
|
settings=self.settings())
|
|
|
|
def nix(self, cmd: list[str], nix_exe: str = 'nix') -> NixCommand:
|
|
return self.nix_cmd([nix_exe, *cmd])
|
|
|
|
nix_build = partialmethod(nix, nix_exe='nix-build')
|
|
nix_shell = partialmethod(nix, nix_exe='nix-shell')
|
|
nix_store = partialmethod(nix, nix_exe='nix-store')
|
|
nix_env = partialmethod(nix, nix_exe='nix-env')
|
|
nix_instantiate = partialmethod(nix, nix_exe='nix-instantiate')
|
|
nix_channel = partialmethod(nix, nix_exe='nix-channel')
|
|
nix_prefetch_url = partialmethod(nix, nix_exe='nix-prefetch-url')
|
|
|
|
def eval(
|
|
self, expr: str,
|
|
settings: NixSettings | None = None) -> CommandResult:
|
|
# clone due to reference-shenanigans
|
|
settings = dataclasses.replace(settings or self.settings()).feature('nix-command')
|
|
|
|
cmd = self.nix(['eval', '--json', '--expr', expr])
|
|
cmd.settings = settings
|
|
return cmd.run()
|