109 lines
4.2 KiB
Python
109 lines
4.2 KiB
Python
import json
|
|
import os
|
|
import re
|
|
from pathlib import Path
|
|
from tempfile import NamedTemporaryFile
|
|
|
|
from clan_cli.cmd import Log, run
|
|
from clan_cli.dirs import machine_settings_file, nixpkgs_source, specific_machine_dir
|
|
from clan_cli.errors import ClanError, ClanHttpError
|
|
from clan_cli.git import commit_file
|
|
from clan_cli.nix import nix_eval
|
|
|
|
|
|
def verify_machine_config(
|
|
flake_dir: Path,
|
|
machine_name: str,
|
|
config: dict | None = None,
|
|
) -> str | None:
|
|
"""
|
|
Verify that the machine evaluates successfully
|
|
Returns a tuple of (success, error_message)
|
|
"""
|
|
if config is None:
|
|
config = config_for_machine(flake_dir, machine_name)
|
|
flake = flake_dir
|
|
with NamedTemporaryFile(mode="w", dir=flake) as clan_machine_settings_file:
|
|
json.dump(config, clan_machine_settings_file, indent=2)
|
|
clan_machine_settings_file.seek(0)
|
|
env = os.environ.copy()
|
|
env["CLAN_MACHINE_SETTINGS_FILE"] = clan_machine_settings_file.name
|
|
cmd = nix_eval(
|
|
flags=[
|
|
"--show-trace",
|
|
"--impure", # needed to access CLAN_MACHINE_SETTINGS_FILE
|
|
"--expr",
|
|
f"""
|
|
let
|
|
# hardcoding system for now, not sure where to get it from
|
|
system = "x86_64-linux";
|
|
flake = builtins.getFlake (toString {flake});
|
|
clan-core = flake.inputs.clan-core;
|
|
nixpkgsSrc = flake.inputs.nixpkgs or {nixpkgs_source()};
|
|
lib = import (nixpkgsSrc + /lib);
|
|
pkgs = import nixpkgsSrc {{ inherit system; }};
|
|
config = lib.importJSON (builtins.getEnv "CLAN_MACHINE_SETTINGS_FILE");
|
|
fakeMachine = pkgs.nixos {{
|
|
imports =
|
|
[
|
|
clan-core.nixosModules.clanCore
|
|
# potentially the config might affect submodule options,
|
|
# therefore we need to import it
|
|
config
|
|
{{clan.core.clanDir = {flake};}}
|
|
]
|
|
# add all clan modules specified via clanImports
|
|
++ (map (name: clan-core.clanModules.${{name}}) config.clanImports or []);
|
|
}};
|
|
in
|
|
fakeMachine.config.system.build.vm.outPath
|
|
""",
|
|
],
|
|
)
|
|
|
|
proc = run(
|
|
cmd,
|
|
cwd=flake,
|
|
env=env,
|
|
log=Log.BOTH,
|
|
)
|
|
if proc.returncode != 0:
|
|
return proc.stderr
|
|
return None
|
|
|
|
|
|
def config_for_machine(flake_dir: Path, machine_name: str) -> dict:
|
|
# read the config from a json file located at {flake}/machines/{machine_name}/settings.json
|
|
if not specific_machine_dir(flake_dir, machine_name).exists():
|
|
raise ClanHttpError(
|
|
msg=f"Machine {machine_name} not found. Create the machine first`",
|
|
status_code=404,
|
|
)
|
|
settings_path = machine_settings_file(flake_dir, machine_name)
|
|
if not settings_path.exists():
|
|
return {}
|
|
with open(settings_path) as f:
|
|
return json.load(f)
|
|
|
|
|
|
def set_config_for_machine(flake_dir: Path, machine_name: str, config: dict) -> None:
|
|
hostname_regex = r"^(?!-)[A-Za-z0-9-]{1,63}(?<!-)$"
|
|
if not re.match(hostname_regex, machine_name):
|
|
raise ClanError("Machine name must be a valid hostname")
|
|
if "networking" in config and "hostName" in config["networking"]:
|
|
if machine_name != config["networking"]["hostName"]:
|
|
raise ClanHttpError(
|
|
msg="Machine name does not match the 'networking.hostName' setting in the config",
|
|
status_code=400,
|
|
)
|
|
config["networking"]["hostName"] = machine_name
|
|
# create machine folder if it doesn't exist
|
|
# write the config to a json file located at {flake}/machines/{machine_name}/settings.json
|
|
settings_path = machine_settings_file(flake_dir, machine_name)
|
|
settings_path.parent.mkdir(parents=True, exist_ok=True)
|
|
with open(settings_path, "w") as f:
|
|
json.dump(config, f)
|
|
|
|
if flake_dir is not None:
|
|
commit_file(settings_path, flake_dir)
|