When the users of a secret change, when for example a new admin user is added, an error will be thrown when generating vars, prompting the user to pass --fix to re-encrypt the secrets
87 lines
3.4 KiB
Python
87 lines
3.4 KiB
Python
import argparse
|
|
import importlib
|
|
import logging
|
|
|
|
from clan_cli.completions import add_dynamic_completer, complete_machines
|
|
from clan_cli.machines.machines import Machine
|
|
|
|
log = logging.getLogger(__name__)
|
|
|
|
|
|
def vars_status(
|
|
machine: Machine, generator_name: None | str = None
|
|
) -> tuple[list[tuple[str, str]], list[tuple[str, str]], list[tuple[str, str]]]:
|
|
secret_vars_module = importlib.import_module(machine.secret_vars_module)
|
|
secret_vars_store = secret_vars_module.SecretStore(machine=machine)
|
|
public_vars_module = importlib.import_module(machine.public_vars_module)
|
|
public_vars_store = public_vars_module.FactStore(machine=machine)
|
|
|
|
missing_secret_vars = []
|
|
missing_public_vars = []
|
|
# signals if a var needs to be updated (eg. needs re-encryption due to new users added)
|
|
outdated_secret_vars = []
|
|
if generator_name:
|
|
generators = [generator_name]
|
|
else:
|
|
generators = list(machine.vars_generators.keys())
|
|
for generator_name in generators:
|
|
shared = machine.vars_generators[generator_name]["share"]
|
|
for name, file in machine.vars_generators[generator_name]["files"].items():
|
|
if file["secret"]:
|
|
if not secret_vars_store.exists(generator_name, name, shared=shared):
|
|
log.info(
|
|
f"Secret var '{name}' for service '{generator_name}' in machine {machine.name} is missing."
|
|
)
|
|
missing_secret_vars.append((generator_name, name))
|
|
else:
|
|
needs_update, msg = secret_vars_store.needs_fix(
|
|
generator_name, name, shared=shared
|
|
)
|
|
if needs_update:
|
|
log.info(
|
|
f"Secret var '{name}' for service '{generator_name}' in machine {machine.name} needs update: {msg}"
|
|
)
|
|
outdated_secret_vars.append((generator_name, name))
|
|
|
|
elif not public_vars_store.exists(generator_name, name, shared=shared):
|
|
log.info(
|
|
f"Public var '{name}' for service '{generator_name}' in machine {machine.name} is missing."
|
|
)
|
|
missing_public_vars.append((generator_name, name))
|
|
|
|
log.debug(f"missing_secret_vars: {missing_secret_vars}")
|
|
log.debug(f"missing_public_vars: {missing_public_vars}")
|
|
log.debug(f"outdated_secret_vars: {outdated_secret_vars}")
|
|
return missing_secret_vars, missing_public_vars, outdated_secret_vars
|
|
|
|
|
|
def check_vars(machine: Machine, generator_name: None | str = None) -> bool:
|
|
missing_secret_vars, missing_public_vars, outdated_secret_vars = vars_status(
|
|
machine, generator_name=generator_name
|
|
)
|
|
return not (missing_secret_vars or missing_public_vars or outdated_secret_vars)
|
|
|
|
|
|
def check_command(args: argparse.Namespace) -> None:
|
|
machine = Machine(
|
|
name=args.machine,
|
|
flake=args.flake,
|
|
)
|
|
ok = check_vars(machine, generator_name=args.service)
|
|
if not ok:
|
|
raise SystemExit(1)
|
|
|
|
|
|
def register_check_parser(parser: argparse.ArgumentParser) -> None:
|
|
machines_parser = parser.add_argument(
|
|
"machine",
|
|
help="The machine to check secrets for",
|
|
)
|
|
add_dynamic_completer(machines_parser, complete_machines)
|
|
|
|
parser.add_argument(
|
|
"--service",
|
|
help="the service to check",
|
|
)
|
|
parser.set_defaults(func=check_command)
|