Source code for bacommon.langstr._wrapper

# Released under the MIT License. See LICENSE for details.
#
"""Runtime accessor for generated language-string wrapper packages.

A generated wrapper module exposes a ``strings`` object built from a
compact nested param-tree; its precise types live in the module's
``if TYPE_CHECKING`` shadow (decision #28 -- bare annotations, no per-entry
runtime class). This drives the *runtime* side: a no-arg string reads as a
property yielding an :class:`Lstr`, a parameterized one is a callable that
builds an :class:`Lstr` from keyword substitutions, and a subdir is a
nested :class:`LangStrDir`.
"""

from __future__ import annotations  # Docs-generation hack.

from bacommon.langstr._core import Lstr, PackageStructure

#: A wrapper's compact runtime tree: a leaf is its ordered param-keyword
#: tuple (``()`` for a no-arg string); a subdir is a nested tree.
type WrapperTree = dict[str, 'tuple[str, ...] | WrapperTree']


[docs] def package_structure(apverid: str, tree: WrapperTree) -> PackageStructure: """Build a :class:`PackageStructure` from a wrapper's runtime ``_TREE``. Flattens the nested tree into the ``{logical-path: param-keywords}`` map the encode/decode contexts need -- so a consumer of a vendored package just passes ``module.APVERID, module._TREE`` (both module-level). """ flat: dict[str, tuple[str, ...]] = {} def _walk(node: WrapperTree, prefix: str) -> None: for name, value in node.items(): full = f'{prefix}/{name}' if prefix else name if isinstance(value, dict): _walk(value, full) else: flat[full] = value _walk(tree, '') return PackageStructure(apverid, flat)
class _LstrMaker: """Callable leaf: builds an :class:`Lstr` from keyword substitutions.""" __slots__ = ('_apverid', '_name') def __init__(self, apverid: str, name: str) -> None: self._apverid = apverid self._name = name def __call__(self, **subs: 'str | int | Lstr') -> Lstr: return Lstr(self._apverid, self._name, dict(subs))
[docs] class LangStrDir: """Runtime root/subdir accessor for a generated wrapper package.""" __slots__ = ('_apverid', '_tree', '_prefix') def __init__( self, apverid: str, tree: WrapperTree, prefix: str = '' ) -> None: self._apverid = apverid self._tree = tree self._prefix = prefix def __getattr__(self, name: str) -> 'Lstr | _LstrMaker | LangStrDir': try: child = self._tree[name] except KeyError: raise AttributeError(name) from None full = f'{self._prefix}/{name}' if self._prefix else name if isinstance(child, dict): return LangStrDir(self._apverid, child, full) # A leaf: its param-keyword tuple. Empty -> a no-arg string, read # as a property yielding the Lstr directly; otherwise a maker. if not child: return Lstr(self._apverid, full) return _LstrMaker(self._apverid, full)
# Docs-generation hack; import some stuff that we likely only forward-declared # in our actual source code so that docs tools can find it. from typing import (Coroutine, Any, Literal, Callable, Generator, Awaitable, Sequence, Self) import asyncio from concurrent.futures import Future from pathlib import Path from enum import Enum