add factsStore modules
This commit is contained in:
@@ -44,6 +44,13 @@
|
|||||||
the directory on the deployment server where secrets are uploaded
|
the directory on the deployment server where secrets are uploaded
|
||||||
'';
|
'';
|
||||||
};
|
};
|
||||||
|
factsModule = lib.mkOption {
|
||||||
|
type = lib.types.str;
|
||||||
|
description = ''
|
||||||
|
the python import path to the facts module
|
||||||
|
'';
|
||||||
|
default = "clan_cli.facts.modules.in_repo";
|
||||||
|
};
|
||||||
secretsModule = lib.mkOption {
|
secretsModule = lib.mkOption {
|
||||||
type = lib.types.str;
|
type = lib.types.str;
|
||||||
description = ''
|
description = ''
|
||||||
@@ -84,7 +91,7 @@
|
|||||||
# optimization for faster secret generate/upload and machines update
|
# optimization for faster secret generate/upload and machines update
|
||||||
config = {
|
config = {
|
||||||
system.clan.deployment.data = {
|
system.clan.deployment.data = {
|
||||||
inherit (config.system.clan) secretsModule secretsData;
|
inherit (config.system.clan) factsModule secretsModule secretsData;
|
||||||
inherit (config.clan.networking) targetHost buildHost;
|
inherit (config.clan.networking) targetHost buildHost;
|
||||||
inherit (config.clan.deployment) requireExplicitUpdate;
|
inherit (config.clan.deployment) requireExplicitUpdate;
|
||||||
inherit (config.clanCore) secretsUploadDirectory;
|
inherit (config.clanCore) secretsUploadDirectory;
|
||||||
|
|||||||
@@ -6,7 +6,7 @@ from pathlib import Path
|
|||||||
from types import ModuleType
|
from types import ModuleType
|
||||||
from typing import Any
|
from typing import Any
|
||||||
|
|
||||||
from . import backups, config, flakes, flash, history, machines, secrets, vms
|
from . import backups, config, flakes, flash, history, machines, secrets, vms, facts
|
||||||
from .custom_logger import setup_logging
|
from .custom_logger import setup_logging
|
||||||
from .dirs import get_clan_flake_toplevel
|
from .dirs import get_clan_flake_toplevel
|
||||||
from .errors import ClanCmdError, ClanError
|
from .errors import ClanCmdError, ClanError
|
||||||
@@ -91,6 +91,9 @@ def create_parser(prog: str | None = None) -> argparse.ArgumentParser:
|
|||||||
parser_secrets = subparsers.add_parser("secrets", help="manage secrets")
|
parser_secrets = subparsers.add_parser("secrets", help="manage secrets")
|
||||||
secrets.register_parser(parser_secrets)
|
secrets.register_parser(parser_secrets)
|
||||||
|
|
||||||
|
parser_facts = subparsers.add_parser("facts", help="manage facts")
|
||||||
|
facts.register_parser(parser_facts)
|
||||||
|
|
||||||
parser_machine = subparsers.add_parser(
|
parser_machine = subparsers.add_parser(
|
||||||
"machines", help="Manage machines and their configuration"
|
"machines", help="Manage machines and their configuration"
|
||||||
)
|
)
|
||||||
|
|||||||
21
pkgs/clan-cli/clan_cli/facts/__init__.py
Normal file
21
pkgs/clan-cli/clan_cli/facts/__init__.py
Normal file
@@ -0,0 +1,21 @@
|
|||||||
|
# !/usr/bin/env python3
|
||||||
|
import argparse
|
||||||
|
|
||||||
|
from .check import register_check_parser
|
||||||
|
from .list import register_list_parser
|
||||||
|
|
||||||
|
|
||||||
|
# takes a (sub)parser and configures it
|
||||||
|
def register_parser(parser: argparse.ArgumentParser) -> None:
|
||||||
|
subparser = parser.add_subparsers(
|
||||||
|
title="command",
|
||||||
|
description="the command to run",
|
||||||
|
help="the command to run",
|
||||||
|
required=True,
|
||||||
|
)
|
||||||
|
|
||||||
|
check_parser = subparser.add_parser("check", help="check if facts are up to date")
|
||||||
|
register_check_parser(check_parser)
|
||||||
|
|
||||||
|
list_parser = subparser.add_parser("list", help="list all facts")
|
||||||
|
register_list_parser(list_parser)
|
||||||
37
pkgs/clan-cli/clan_cli/facts/check.py
Normal file
37
pkgs/clan-cli/clan_cli/facts/check.py
Normal file
@@ -0,0 +1,37 @@
|
|||||||
|
import argparse
|
||||||
|
import importlib
|
||||||
|
import logging
|
||||||
|
|
||||||
|
from ..machines.machines import Machine
|
||||||
|
|
||||||
|
log = logging.getLogger(__name__)
|
||||||
|
|
||||||
|
|
||||||
|
def check_facts(machine: Machine) -> bool:
|
||||||
|
facts_module = importlib.import_module(machine.facts_module)
|
||||||
|
fact_store = facts_module.FactStore(machine=machine)
|
||||||
|
|
||||||
|
missing_facts = []
|
||||||
|
for service in machine.secrets_data:
|
||||||
|
for fact in machine.secrets_data[service]["facts"]:
|
||||||
|
if not fact_store.get(service, fact):
|
||||||
|
log.info(f"Fact {fact} for service {service} is missing")
|
||||||
|
missing_facts.append((service, fact))
|
||||||
|
|
||||||
|
if missing_facts:
|
||||||
|
return False
|
||||||
|
return True
|
||||||
|
|
||||||
|
|
||||||
|
def check_command(args: argparse.Namespace) -> None:
|
||||||
|
machine = Machine(name=args.machine, flake=args.flake)
|
||||||
|
if check_facts(machine):
|
||||||
|
print("All facts are present")
|
||||||
|
|
||||||
|
|
||||||
|
def register_check_parser(parser: argparse.ArgumentParser) -> None:
|
||||||
|
parser.add_argument(
|
||||||
|
"machine",
|
||||||
|
help="The machine to check facts for",
|
||||||
|
)
|
||||||
|
parser.set_defaults(func=check_command)
|
||||||
36
pkgs/clan-cli/clan_cli/facts/list.py
Normal file
36
pkgs/clan-cli/clan_cli/facts/list.py
Normal file
@@ -0,0 +1,36 @@
|
|||||||
|
import json
|
||||||
|
import argparse
|
||||||
|
import importlib
|
||||||
|
import logging
|
||||||
|
|
||||||
|
from ..machines.machines import Machine
|
||||||
|
|
||||||
|
log = logging.getLogger(__name__)
|
||||||
|
|
||||||
|
|
||||||
|
def get_all_facts(machine: Machine) -> dict:
|
||||||
|
facts_module = importlib.import_module(machine.facts_module)
|
||||||
|
fact_store = facts_module.FactStore(machine=machine)
|
||||||
|
|
||||||
|
# for service in machine.secrets_data:
|
||||||
|
# facts[service] = {}
|
||||||
|
# for fact in machine.secrets_data[service]["facts"]:
|
||||||
|
# fact_content = fact_store.get(service, fact)
|
||||||
|
# if fact_content:
|
||||||
|
# facts[service][fact] = fact_content.decode()
|
||||||
|
# else:
|
||||||
|
# log.error(f"Fact {fact} for service {service} is missing")
|
||||||
|
return fact_store.get_all()
|
||||||
|
|
||||||
|
|
||||||
|
def get_command(args: argparse.Namespace) -> None:
|
||||||
|
machine = Machine(name=args.machine, flake=args.flake)
|
||||||
|
print(json.dumps(get_all_facts(machine), indent=4))
|
||||||
|
|
||||||
|
|
||||||
|
def register_list_parser(parser: argparse.ArgumentParser) -> None:
|
||||||
|
parser.add_argument(
|
||||||
|
"machine",
|
||||||
|
help="The machine to print facts for",
|
||||||
|
)
|
||||||
|
parser.set_defaults(func=get_command)
|
||||||
0
pkgs/clan-cli/clan_cli/facts/modules/__init__.py
Normal file
0
pkgs/clan-cli/clan_cli/facts/modules/__init__.py
Normal file
42
pkgs/clan-cli/clan_cli/facts/modules/in_repo.py
Normal file
42
pkgs/clan-cli/clan_cli/facts/modules/in_repo.py
Normal file
@@ -0,0 +1,42 @@
|
|||||||
|
from pathlib import Path
|
||||||
|
|
||||||
|
from clan_cli.errors import ClanError
|
||||||
|
from clan_cli.machines.machines import Machine
|
||||||
|
|
||||||
|
|
||||||
|
class FactStore:
|
||||||
|
def __init__(self, machine: Machine) -> None:
|
||||||
|
self.machine = machine
|
||||||
|
|
||||||
|
def set(self, _service: str, name: str, value: bytes) -> Path | None:
|
||||||
|
if isinstance(self.machine.flake, Path):
|
||||||
|
fact_path = (
|
||||||
|
self.machine.flake / "machines" / self.machine.name / "facts" / name
|
||||||
|
)
|
||||||
|
fact_path.parent.mkdir(parents=True, exist_ok=True)
|
||||||
|
fact_path.touch()
|
||||||
|
fact_path.write_bytes(value)
|
||||||
|
return fact_path
|
||||||
|
else:
|
||||||
|
raise ClanError(
|
||||||
|
f"in_flake fact storage is only supported for local flakes: {self.machine.flake}"
|
||||||
|
)
|
||||||
|
|
||||||
|
def exists(self, _service: str, name: str) -> bool:
|
||||||
|
fact_path = self.machine.flake_dir / "machines" / self.machine.name / "facts" / name
|
||||||
|
return fact_path.exists()
|
||||||
|
|
||||||
|
# get a single fact
|
||||||
|
def get(self, _service: str, name: str) -> bytes:
|
||||||
|
fact_path = self.machine.flake_dir / "machines" / self.machine.name / "facts" / name
|
||||||
|
return fact_path.read_bytes()
|
||||||
|
|
||||||
|
# get all facts
|
||||||
|
def get_all(self) -> dict[str, dict[str, bytes]]:
|
||||||
|
facts_folder = self.machine.flake_dir / "machines" / self.machine.name / "facts"
|
||||||
|
facts: dict[str, dict[str, bytes]] = {}
|
||||||
|
facts["TODO"] = {}
|
||||||
|
if facts_folder.exists():
|
||||||
|
for fact_path in facts_folder.iterdir():
|
||||||
|
facts["TODO"][fact_path.name] = fact_path.read_bytes()
|
||||||
|
return facts
|
||||||
@@ -96,6 +96,10 @@ class Machine:
|
|||||||
def secrets_module(self) -> str:
|
def secrets_module(self) -> str:
|
||||||
return self.deployment_info["secretsModule"]
|
return self.deployment_info["secretsModule"]
|
||||||
|
|
||||||
|
@property
|
||||||
|
def facts_module(self) -> str:
|
||||||
|
return self.deployment_info["factsModule"]
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def secrets_data(self) -> dict:
|
def secrets_data(self) -> dict:
|
||||||
if self.deployment_info["secretsData"]:
|
if self.deployment_info["secretsData"]:
|
||||||
|
|||||||
@@ -10,6 +10,8 @@ log = logging.getLogger(__name__)
|
|||||||
def check_secrets(machine: Machine) -> bool:
|
def check_secrets(machine: Machine) -> bool:
|
||||||
secrets_module = importlib.import_module(machine.secrets_module)
|
secrets_module = importlib.import_module(machine.secrets_module)
|
||||||
secret_store = secrets_module.SecretStore(machine=machine)
|
secret_store = secrets_module.SecretStore(machine=machine)
|
||||||
|
facts_module = importlib.import_module(machine.facts_module)
|
||||||
|
fact_store = facts_module.FactsStore(machine=machine)
|
||||||
|
|
||||||
missing_secrets = []
|
missing_secrets = []
|
||||||
missing_facts = []
|
missing_facts = []
|
||||||
@@ -20,7 +22,7 @@ def check_secrets(machine: Machine) -> bool:
|
|||||||
missing_secrets.append((service, secret))
|
missing_secrets.append((service, secret))
|
||||||
|
|
||||||
for fact in machine.secrets_data[service]["facts"].values():
|
for fact in machine.secrets_data[service]["facts"].values():
|
||||||
if not (machine.flake / fact).exists():
|
if not fact_store.exists(service, fact):
|
||||||
log.info(f"Fact {fact} for service {service} is missing")
|
log.info(f"Fact {fact} for service {service} is missing")
|
||||||
missing_facts.append((service, fact))
|
missing_facts.append((service, fact))
|
||||||
|
|
||||||
|
|||||||
@@ -2,7 +2,6 @@ import argparse
|
|||||||
import importlib
|
import importlib
|
||||||
import logging
|
import logging
|
||||||
import os
|
import os
|
||||||
import shutil
|
|
||||||
from pathlib import Path
|
from pathlib import Path
|
||||||
from tempfile import TemporaryDirectory
|
from tempfile import TemporaryDirectory
|
||||||
|
|
||||||
@@ -21,6 +20,9 @@ def generate_secrets(machine: Machine) -> None:
|
|||||||
secrets_module = importlib.import_module(machine.secrets_module)
|
secrets_module = importlib.import_module(machine.secrets_module)
|
||||||
secret_store = secrets_module.SecretStore(machine=machine)
|
secret_store = secrets_module.SecretStore(machine=machine)
|
||||||
|
|
||||||
|
facts_module = importlib.import_module(machine.facts_module)
|
||||||
|
fact_store = facts_module.FactStore(machine=machine)
|
||||||
|
|
||||||
with TemporaryDirectory() as d:
|
with TemporaryDirectory() as d:
|
||||||
for service in machine.secrets_data:
|
for service in machine.secrets_data:
|
||||||
tmpdir = Path(d) / service
|
tmpdir = Path(d) / service
|
||||||
@@ -84,10 +86,10 @@ def generate_secrets(machine: Machine) -> None:
|
|||||||
msg = f"did not generate a file for '{name}' when running the following command:\n"
|
msg = f"did not generate a file for '{name}' when running the following command:\n"
|
||||||
msg += machine.secrets_data[service]["generator"]
|
msg += machine.secrets_data[service]["generator"]
|
||||||
raise ClanError(msg)
|
raise ClanError(msg)
|
||||||
fact_path = machine.flake / fact_path
|
fact_file = fact_store.set(
|
||||||
fact_path.parent.mkdir(parents=True, exist_ok=True)
|
service, fact_path, fact_file.read_bytes()
|
||||||
shutil.copyfile(fact_file, fact_path)
|
)
|
||||||
files_to_commit.append(fact_path)
|
files_to_commit.append(fact_file)
|
||||||
commit_files(
|
commit_files(
|
||||||
files_to_commit,
|
files_to_commit,
|
||||||
machine.flake_dir,
|
machine.flake_dir,
|
||||||
|
|||||||
Reference in New Issue
Block a user