clan-core/pkgs/clan-cli/tests/fixtures_flakes.py

270 lines
9.0 KiB
Python
Raw Normal View History

import fileinput
import json
import logging
2023-10-25 11:10:30 +00:00
import os
import shutil
2023-10-25 17:23:28 +00:00
import subprocess as sp
import tempfile
2023-11-29 11:40:48 +00:00
from collections.abc import Iterator
from pathlib import Path
2023-11-29 11:40:48 +00:00
from typing import NamedTuple
import pytest
from root import CLAN_CORE
from clan_cli.dirs import nixpkgs_source
log = logging.getLogger(__name__)
# substitutes string sin a file.
# This can be used on the flake.nix or default.nix of a machine
def substitute(
file: Path,
clan_core_flake: Path | None = None,
flake: Path = Path(__file__).parent,
) -> None:
sops_key = str(flake.joinpath("sops.key"))
for line in fileinput.input(file, inplace=True):
line = line.replace("__NIXPKGS__", str(nixpkgs_source()))
if clan_core_flake:
line = line.replace("__CLAN_CORE__", str(clan_core_flake))
line = line.replace(
"git+https://git.clan.lol/clan/clan-core", str(clan_core_flake)
)
line = line.replace("__CLAN_SOPS_KEY_PATH__", sops_key)
line = line.replace("__CLAN_SOPS_KEY_DIR__", str(flake))
print(line, end="")
class FlakeForTest(NamedTuple):
path: Path
def generate_flake(
temporary_home: Path,
flake_template: Path,
substitutions: dict[str, str] = {
"__CHANGE_ME__": "_test_vm_persistence",
"git+https://git.clan.lol/clan/clan-core": "path://" + str(CLAN_CORE),
2024-06-17 10:42:28 +00:00
"https://git.clan.lol/clan/clan-core/archive/main.tar.gz": "path://"
+ str(CLAN_CORE),
},
# define the machines directly including their config
machine_configs: dict[str, dict] = {},
) -> FlakeForTest:
"""
Creates a clan flake with the given name.
Machines are fully generated from the machine_configs.
Example:
machine_configs = dict(
my_machine=dict(
2024-06-17 10:42:28 +00:00
clan=dict(
core=dict(
backups=dict(
...
)
)
)
)
)
"""
# copy the template to a new temporary location
flake = temporary_home / "flake"
shutil.copytree(flake_template, flake)
sp.run(["chmod", "+w", "-R", str(flake)], check=True)
# substitute `substitutions` in all files of the template
for file in flake.rglob("*"):
if file.is_file():
print(f"Final Content of {file}:")
for line in fileinput.input(file, inplace=True):
for key, value in substitutions.items():
line = line.replace(key, value)
print(line, end="")
# generate machines from machineConfigs
for machine_name, machine_config in machine_configs.items():
settings_path = flake / "machines" / machine_name / "settings.json"
settings_path.parent.mkdir(parents=True, exist_ok=True)
settings_path.write_text(json.dumps(machine_config, indent=2))
if "/tmp" not in str(os.environ.get("HOME")):
log.warning(
f"!! $HOME does not point to a temp directory!! HOME={os.environ['HOME']}"
)
# TODO: Find out why test_vms_api.py fails in nix build
# but works in pytest when this bottom line is commented out
sp.run(
["git", "config", "--global", "init.defaultBranch", "main"],
cwd=flake,
check=True,
)
sp.run(["git", "init"], cwd=flake, check=True)
sp.run(["git", "add", "."], cwd=flake, check=True)
sp.run(["git", "config", "user.name", "clan-tool"], cwd=flake, check=True)
sp.run(["git", "config", "user.email", "clan@example.com"], cwd=flake, check=True)
sp.run(["git", "commit", "-a", "-m", "Initial commit"], cwd=flake, check=True)
return FlakeForTest(flake)
def create_flake(
monkeypatch: pytest.MonkeyPatch,
2023-10-23 20:31:12 +00:00
temporary_home: Path,
flake_template: str | Path,
clan_core_flake: Path | None = None,
# names referring to pre-defined machines from ../machines
machines: list[str] = [],
# alternatively specify the machines directly including their config
machine_configs: dict[str, dict] = {},
remote: bool = False,
) -> Iterator[FlakeForTest]:
"""
Creates a flake with the given name and machines.
The machine names map to the machines in ./test_machines
"""
if isinstance(flake_template, Path):
template_path = flake_template
else:
template_path = Path(__file__).parent / flake_template
flake_template_name = template_path.name
# copy the template to a new temporary location
flake = temporary_home / flake_template_name
shutil.copytree(template_path, flake)
sp.run(["chmod", "+w", "-R", str(flake)], check=True)
# add the requested machines to the flake
if machines:
(flake / "machines").mkdir(parents=True, exist_ok=True)
for machine_name in machines:
machine_path = Path(__file__).parent / "machines" / machine_name
shutil.copytree(machine_path, flake / "machines" / machine_name)
substitute(flake / "machines" / machine_name / "default.nix", flake)
# generate machines from machineConfigs
for machine_name, machine_config in machine_configs.items():
settings_path = flake / "machines" / machine_name / "settings.json"
settings_path.parent.mkdir(parents=True, exist_ok=True)
settings_path.write_text(json.dumps(machine_config, indent=2))
# in the flake.nix file replace the string __CLAN_URL__ with the the clan flake
# provided by get_test_flake_toplevel
flake_nix = flake / "flake.nix"
# this is where we would install the sops key to, when updating
substitute(flake_nix, clan_core_flake, flake)
2023-10-24 14:44:54 +00:00
2023-10-25 11:15:38 +00:00
if "/tmp" not in str(os.environ.get("HOME")):
2023-10-25 17:23:28 +00:00
log.warning(
f"!! $HOME does not point to a temp directory!! HOME={os.environ['HOME']}"
)
2023-10-25 11:10:30 +00:00
# TODO: Find out why test_vms_api.py fails in nix build
# but works in pytest when this bottom line is commented out
2023-10-25 17:23:28 +00:00
sp.run(
2023-10-25 11:15:38 +00:00
["git", "config", "--global", "init.defaultBranch", "main"],
2023-10-25 17:23:28 +00:00
cwd=flake,
2023-10-25 11:15:38 +00:00
check=True,
)
2023-10-25 17:23:28 +00:00
sp.run(["git", "init"], cwd=flake, check=True)
sp.run(["git", "add", "."], cwd=flake, check=True)
sp.run(["git", "config", "user.name", "clan-tool"], cwd=flake, check=True)
sp.run(["git", "config", "user.email", "clan@example.com"], cwd=flake, check=True)
sp.run(["git", "commit", "-a", "-m", "Initial commit"], cwd=flake, check=True)
2023-10-24 14:44:54 +00:00
if remote:
2023-10-25 11:10:30 +00:00
with tempfile.TemporaryDirectory():
2023-11-15 13:28:40 +00:00
yield FlakeForTest(flake)
else:
2023-11-15 13:28:40 +00:00
yield FlakeForTest(flake)
@pytest.fixture
def test_flake(
2023-10-25 17:23:28 +00:00
monkeypatch: pytest.MonkeyPatch, temporary_home: Path
) -> Iterator[FlakeForTest]:
2023-11-15 13:28:40 +00:00
yield from create_flake(monkeypatch, temporary_home, "test_flake")
# check that git diff on ./sops is empty
if (temporary_home / "test_flake" / "sops").exists():
git_proc = sp.run(
["git", "diff", "--exit-code", "./sops"],
cwd=temporary_home / "test_flake",
stderr=sp.PIPE,
)
if git_proc.returncode != 0:
log.error(git_proc.stderr.decode())
raise Exception(
"git diff on ./sops is not empty. This should not happen as all changes should be committed"
)
@pytest.fixture
def test_flake_with_core(
2023-10-25 17:23:28 +00:00
monkeypatch: pytest.MonkeyPatch, temporary_home: Path
) -> Iterator[FlakeForTest]:
if not (CLAN_CORE / "flake.nix").exists():
raise Exception(
"clan-core flake not found. This test requires the clan-core flake to be present"
)
yield from create_flake(
2023-10-24 14:44:54 +00:00
monkeypatch,
temporary_home,
2023-11-15 13:28:40 +00:00
"test_flake_with_core",
2023-10-24 14:44:54 +00:00
CLAN_CORE,
)
@pytest.fixture
def test_local_democlan(
monkeypatch: pytest.MonkeyPatch, temporary_home: Path
) -> Iterator[FlakeForTest]:
democlan = os.getenv(key="DEMOCLAN_ROOT")
if democlan is None:
raise Exception(
"DEMOCLAN_ROOT not set. This test requires the democlan flake to be present"
)
democlan_p = Path(democlan).resolve()
if not democlan_p.is_dir():
raise Exception(
f"DEMOCLAN_ROOT ({democlan_p}) is not a directory. This test requires the democlan directory to be present"
)
yield FlakeForTest(democlan_p)
@pytest.fixture
def test_flake_with_core_and_pass(
2023-10-25 17:23:28 +00:00
monkeypatch: pytest.MonkeyPatch, temporary_home: Path
) -> Iterator[FlakeForTest]:
if not (CLAN_CORE / "flake.nix").exists():
raise Exception(
"clan-core flake not found. This test requires the clan-core flake to be present"
)
2023-10-14 13:17:58 +00:00
yield from create_flake(
monkeypatch,
2023-10-23 20:31:12 +00:00
temporary_home,
2023-11-15 13:28:40 +00:00
"test_flake_with_core_and_pass",
CLAN_CORE,
2023-10-14 13:17:58 +00:00
)
@pytest.fixture
def test_flake_minimal(
monkeypatch: pytest.MonkeyPatch, temporary_home: Path
) -> Iterator[FlakeForTest]:
if not (CLAN_CORE / "flake.nix").exists():
raise Exception(
"clan-core flake not found. This test requires the clan-core flake to be present"
)
yield from create_flake(
monkeypatch,
temporary_home,
CLAN_CORE / "templates" / "minimal",
CLAN_CORE,
)