diff options
Diffstat (limited to 'tools/cru-py/cru/template.py')
-rw-r--r-- | tools/cru-py/cru/template.py | 166 |
1 files changed, 110 insertions, 56 deletions
diff --git a/tools/cru-py/cru/template.py b/tools/cru-py/cru/template.py index 6749cab..35d68ac 100644 --- a/tools/cru-py/cru/template.py +++ b/tools/cru-py/cru/template.py @@ -1,74 +1,124 @@ -from collections.abc import Mapping -import os -import os.path +from abc import ABCMeta, abstractmethod +from collections.abc import Callable, Mapping from pathlib import Path from string import Template +from typing import Generic, TypeVar from ._iter import CruIterator from ._error import CruException +from .parsing import StrWrapperVarParser + class CruTemplateError(CruException): pass -class CruTemplate: +class CruTemplateBase(metaclass=ABCMeta): + def __init__(self, text: str): + self._text = text + self._variables: set[str] | None = None + + @abstractmethod + def _get_variables(self) -> set[str]: + raise NotImplementedError() + + @property + def text(self) -> str: + return self._text + + @property + def variables(self) -> set[str]: + if self._variables is None: + self._variables = self._get_variables() + return self._variables + + @property + def variable_count(self) -> int: + return len(self.variables) + + @property + def has_variables(self) -> bool: + return self.variable_count > 0 + + @abstractmethod + def _do_generate(self, mapping: dict[str, str]) -> str: + raise NotImplementedError() + + def generate(self, mapping: Mapping[str, str], allow_extra: bool = True) -> str: + values = dict(mapping) + if not self.variables <= set(values.keys()): + raise CruTemplateError("Missing variables.") + if not allow_extra and not set(values.keys()) <= self.variables: + raise CruTemplateError("Extra variables.") + return self._do_generate(values) + + +class CruTemplate(CruTemplateBase): def __init__(self, prefix: str, text: str): + super().__init__(text) self._prefix = prefix self._template = Template(text) - self._variables = ( + + def _get_variables(self) -> set[str]: + return ( CruIterator(self._template.get_identifiers()) - .filter(lambda i: i.startswith(self._prefix)) + .filter(lambda i: i.startswith(self.prefix)) .to_set() ) - self._all_variables = set(self._template.get_identifiers()) @property def prefix(self) -> str: return self._prefix @property - def raw_text(self) -> str: - return self._template.template - - @property def py_template(self) -> Template: return self._template @property - def variables(self) -> set[str]: - return self._variables - - @property def all_variables(self) -> set[str]: - return self._all_variables + return set(self._template.get_identifiers()) + + def _do_generate(self, mapping: dict[str, str]) -> str: + return self._template.safe_substitute(mapping) + + +class CruStrWrapperTemplate(CruTemplateBase): + def __init__(self, text: str, wrapper: str = "@@"): + super().__init__(text) + self._wrapper = wrapper + self._tokens: StrWrapperVarParser.Result @property - def has_variables(self) -> bool: - """ - If the template does not has any variables that starts with the given prefix, - it returns False. This usually indicates that the template is not a real - template and should be copied as is. Otherwise, it returns True. + def wrapper(self) -> str: + return self._wrapper + + def _get_variables(self): + self._tokens = StrWrapperVarParser(self.wrapper).parse(self.text) + return ( + self._tokens.cru_iter() + .filter(lambda t: t.is_var) + .map(lambda t: t.value) + .to_set() + ) - This can be used as a guard to prevent invalid templates created accidentally - without notice. - """ - return len(self.variables) > 0 + def _do_generate(self, mapping): + return ( + self._tokens.cru_iter() + .map(lambda t: mapping[t.value] if t.is_var else t.value) + .join_str("") + ) - def generate(self, mapping: Mapping[str, str], allow_extra: bool = True) -> str: - values = dict(mapping) - if not self.variables <= set(values.keys()): - raise CruTemplateError("Missing variables.") - if not allow_extra and not set(values.keys()) <= self.variables: - raise CruTemplateError("Extra variables.") - return self._template.safe_substitute(values) +_Template = TypeVar("_Template", bound=CruTemplateBase) -class TemplateTree: + +class TemplateTree(Generic[_Template]): def __init__( self, - prefix: str, + template_generator: Callable[[str], _Template], source: str, + *, template_file_suffix: str | None = ".template", ): """ @@ -80,18 +130,14 @@ class TemplateTree: If either case is false, it generally means whether the file is a template is wrongly handled. """ - self._prefix = prefix - self._files: list[tuple[Path, CruTemplate]] = [] + self._template_generator = template_generator + self._files: list[tuple[Path, _Template]] = [] self._source = source self._template_file_suffix = template_file_suffix self._load() @property - def prefix(self) -> str: - return self._prefix - - @property - def templates(self) -> list[tuple[Path, CruTemplate]]: + def templates(self) -> list[tuple[Path, _Template]]: return self._files @property @@ -103,13 +149,14 @@ class TemplateTree: return self._template_file_suffix @staticmethod - def _scan_files(root_path: str) -> list[Path]: + def _scan_files(root: str) -> list[Path]: + root_path = Path(root) result: list[Path] = [] - for root, _dirs, files in os.walk(root_path): - for file in files: - path = Path(root, file) - path = path.relative_to(root_path) - result.append(Path(path)) + for path in root_path.glob("**/*"): + if not path.is_file(): + continue + path = path.relative_to(root_path) + result.append(Path(path)) return result def _load(self) -> None: @@ -118,7 +165,7 @@ class TemplateTree: template_file = Path(self.source) / file_path with open(template_file, "r") as f: content = f.read() - template = CruTemplate(self.prefix, content) + template = self._template_generator(content) if self.template_file_suffix is not None: should_be_template = file_path.name.endswith(self.template_file_suffix) if should_be_template and not template.has_variables: @@ -136,18 +183,25 @@ class TemplateTree: s.update(template.variables) return s - def generate_to( - self, destination: str, variables: Mapping[str, str], dry_run: bool - ) -> None: - for file, template in self.templates: - des = Path(destination) / file - if self.template_file_suffix is not None and des.name.endswith( + def generate(self, variables: Mapping[str, str]) -> list[tuple[Path, str]]: + result: list[tuple[Path, str]] = [] + for path, template in self.templates: + if self.template_file_suffix is not None and path.name.endswith( self.template_file_suffix ): - des = des.parent / (des.name[: -len(self.template_file_suffix)]) + path = path.parent / (path.name[: -len(self.template_file_suffix)]) text = template.generate(variables) - if not dry_run: + result.append((path, text)) + return result + + def generate_to( + self, destination: str, variables: Mapping[str, str], dry_run: bool + ) -> None: + generated = self.generate(variables) + if not dry_run: + for path, text in generated: + des = Path(destination) / path des.parent.mkdir(parents=True, exist_ok=True) with open(des, "w") as f: f.write(text) |