2026-04-17 15:49:25 +02:00
|
|
|
# -*- coding: utf-8 -*-
|
|
|
|
|
|
|
|
|
|
from __future__ import annotations
|
|
|
|
|
|
|
|
|
|
import re
|
|
|
|
|
|
|
|
|
|
from pathlib import Path
|
|
|
|
|
|
|
|
|
|
from typing import TYPE_CHECKING
|
|
|
|
|
|
|
|
|
|
if TYPE_CHECKING:
|
|
|
|
|
from typing import Iterable
|
|
|
|
|
from ....lib.FileContext import FileContext
|
|
|
|
|
|
|
|
|
|
from ....lib.log import *
|
|
|
|
|
from ....lib.util import run_cmd
|
2026-04-24 16:29:14 +02:00
|
|
|
from ....lib.TarIo import TarIo
|
|
|
|
|
from ....lib.ProcFilterGpg import ProcFilterGpg
|
2026-04-17 15:49:25 +02:00
|
|
|
|
|
|
|
|
from .base import Attrs
|
|
|
|
|
from .FilesContext import FilesContext
|
|
|
|
|
|
|
|
|
|
class DistroContext(FilesContext):
|
|
|
|
|
|
|
|
|
|
def __init__(self, distro: Distro) -> None:
|
|
|
|
|
super().__init__(distro.ctx)
|
|
|
|
|
self.__distro = distro
|
|
|
|
|
|
2026-04-21 11:22:51 +02:00
|
|
|
async def match_files(self, pkg_names: Iterable[str], pattern: str) -> list[str]:
|
2026-04-17 15:49:25 +02:00
|
|
|
ret: list[str] = []
|
2026-04-21 11:22:51 +02:00
|
|
|
for pkg_name in pkg_names:
|
|
|
|
|
for path in await self.__distro.pkg_files(pkg_name):
|
2026-04-17 15:49:25 +02:00
|
|
|
if re.match(pattern, path):
|
|
|
|
|
ret.append(path)
|
|
|
|
|
return ret
|
|
|
|
|
|
2026-04-21 11:22:51 +02:00
|
|
|
async def list_template_files(self, pkg_names: Iterable[str]) -> list[str]:
|
2026-04-21 11:20:42 +02:00
|
|
|
if not pkg_names:
|
|
|
|
|
pkg_names = [p.name for p in await self.__distro.select()]
|
2026-04-21 11:22:51 +02:00
|
|
|
return await self.match_files(pkg_names, pattern=r'.*\.jw-tmpl$')
|
2026-04-17 15:49:25 +02:00
|
|
|
|
2026-04-21 11:22:51 +02:00
|
|
|
async def list_secret_paths(self, pkg_names: Iterable[str], ignore_missing: bool=False) -> list[str]:
|
2026-04-17 15:49:25 +02:00
|
|
|
ret = []
|
2026-04-21 11:22:51 +02:00
|
|
|
for tmpl in await self.list_template_files(pkg_names):
|
2026-04-17 15:49:25 +02:00
|
|
|
path = str(Path(tmpl).with_suffix(".jw-secret"))
|
2026-04-21 10:21:22 +02:00
|
|
|
if ignore_missing and not await self.ctx.file_exists(path):
|
2026-04-17 15:49:25 +02:00
|
|
|
continue
|
|
|
|
|
ret.append(path)
|
|
|
|
|
return ret
|
|
|
|
|
|
2026-04-21 11:22:51 +02:00
|
|
|
async def list_compilation_targets(self, pkg_names: Iterable[str], ignore_missing: bool=False) -> list[str]:
|
2026-04-17 15:49:25 +02:00
|
|
|
ret = []
|
2026-04-21 11:22:51 +02:00
|
|
|
for tmpl in await self.list_template_files(pkg_names):
|
2026-04-17 15:49:25 +02:00
|
|
|
path = tmpl.removesuffix('.jw-tmpl')
|
2026-04-21 10:21:22 +02:00
|
|
|
if ignore_missing and not await self.ctx.file_exists(path):
|
2026-04-17 15:49:25 +02:00
|
|
|
continue
|
|
|
|
|
ret.append(path)
|
|
|
|
|
return ret
|
|
|
|
|
|
2026-04-21 11:22:51 +02:00
|
|
|
async def remove_compilation_targets(self, pkg_names: Iterable[str]) -> list[str]:
|
|
|
|
|
for path in await self.list_compilation_targets(pkg_names):
|
2026-04-17 15:49:25 +02:00
|
|
|
try:
|
|
|
|
|
self.ctx.stat(path)
|
|
|
|
|
log(NOTICE, f'Removing {path}')
|
|
|
|
|
await self.ctx.unlink(path)
|
|
|
|
|
except FileNotFoundError as e:
|
|
|
|
|
log(DEBUG, f'Compilation target {path} doesn\'t exist (ignored)')
|
|
|
|
|
continue
|
|
|
|
|
|
2026-04-21 11:22:51 +02:00
|
|
|
async def compile_template_files(self, pkg_names: Iterable[str], default_attrs: Attrs) -> list[str]:
|
2026-04-17 15:49:25 +02:00
|
|
|
missing = 0
|
2026-04-21 11:22:51 +02:00
|
|
|
for target in await self.list_compilation_targets(pkg_names):
|
2026-04-17 15:49:25 +02:00
|
|
|
if not await self.compile_template_file(target, default_attrs):
|
|
|
|
|
missing += 1
|
|
|
|
|
if missing > 0:
|
|
|
|
|
log(WARNING, f'{missing} missing secrets found. You might want to add them and run sudo {app.cmdline} again')
|
2026-04-24 16:29:14 +02:00
|
|
|
|
|
|
|
|
async def install(self, src_uri: str, pkg_names: Iterable[str], only_missing: bool=False) -> None:
|
|
|
|
|
if only_missing:
|
|
|
|
|
raise NotImplementedError('--only-missing is not yet implemented')
|
|
|
|
|
secret_paths = await self.list_secret_paths(pkg_names)
|
|
|
|
|
ec = self.ctx
|
|
|
|
|
from ....lib.ec.Local import Local
|
|
|
|
|
from ....lib.FileContext import FileContext
|
|
|
|
|
if not isinstance(ec, Local):
|
|
|
|
|
ec = Local() # Security: Use a local exec context for decrypting and filtering secrets
|
|
|
|
|
async with TarIo.create(src=src_uri, dst=self.ctx) as tio:
|
|
|
|
|
tio.src.add_proc_filter(FileContext.Direction.In, ProcFilterGpg(ec=ec))
|
|
|
|
|
extracted_paths = await tio.extract(path_filter=secret_paths)
|
|
|
|
|
for path in secret_paths:
|
|
|
|
|
if not path in extracted_paths:
|
|
|
|
|
log(NOTICE, f'not extracted: {path}')
|
|
|
|
|
else:
|
|
|
|
|
log(NOTICE, f'extracted: {path}')
|