Source code for pyaml.lattice.lattice_elements_linker

from abc import ABCMeta, abstractmethod
from typing import Iterable

import at
from at import Lattice
from pydantic import BaseModel, ConfigDict

from pyaml import PyAMLException
from pyaml.common.element import Element


[docs] class LinkerConfigModel(BaseModel): """Base configuration model for linker definitions. This class defines the configuration structure used to instantiate a specific linking strategy. Each concrete implementation of a `LatticeElementsLinker` may define its own subclass extending this model to include additional configuration parameters. Attributes ---------- model_config : ConfigDict Pydantic configuration allowing arbitrary field types and forbidding unexpected extra keys. """ model_config = ConfigDict(arbitrary_types_allowed=True, extra="forbid")
[docs] class LinkerIdentifier(metaclass=ABCMeta): """Abstract base class for identifiers used to match PyAML and PyAT elements. The identifier acts as an intermediate representation between the PyAML configuration and the PyAT lattice. Its exact structure depends on the linking strategy (e.g., family name, element index, or user-defined tag). Subclasses should define the fields and logic necessary to represent a unique reference to one or more PyAT elements. """ pass
[docs] class LatticeElementsLinker(metaclass=ABCMeta): """Abstract base class defining the interface for PyAT–PyAML element linking. Implementations of this class define how PyAML elements are matched to PyAT elements based on a given linking strategy (e.g., by family name, by index, or by a custom attribute). Parameters ---------- linker_config_model : LinkerConfigModel The configuration model for the linking strategy. Attributes ---------- lattice : at.Lattice Reference to the PyAT lattice handled by this linker. """ def __init__(self, linker_config_model: LinkerConfigModel): self.linker_config_model = linker_config_model self.lattice: Lattice = None
[docs] def set_lattice(self, lattice: Lattice): """ Set the lattice for element linking. Parameters ---------- lattice : Lattice The lattice to link elements with """ self.lattice = lattice
@abstractmethod def _test_at_element(self, identifier: LinkerIdentifier, element: at.Element) -> bool: pass
[docs] @abstractmethod def get_element_identifier(self, element: Element) -> LinkerIdentifier: """ Get the identifier for linking an element. Parameters ---------- element : Element The element to get the identifier for Returns ------- LinkerIdentifier The identifier for the element """ pass
def _iter_matches(self, identifier: LinkerIdentifier) -> Iterable[at.Element]: """Yield all elements in the lattice whose matches the identifier.""" for elem in self.lattice: if self._test_at_element(identifier, elem): yield elem
[docs] def get_at_elements(self, element_id: LinkerIdentifier | list[LinkerIdentifier]) -> list[at.Element]: """Return a list of PyAT elements matching the given identifiers. This method should resolve one or multiple PyAML identifiers into their corresponding PyAT elements according to the specific linking strategy implemented. Parameters ---------- element_id : LinkerIdentifier or list of LinkerIdentifier One or several identifiers describing which PyAT elements to retrieve. Returns ------- list of at.Element The list of matching PyAT elements found in the lattice. Raises ------ PyAMLException If no element matches the given identifier(s). """ if isinstance(element_id, LinkerIdentifier): identifiers = [element_id] else: identifiers = element_id results: list[at.Element] = [] for ident in identifiers: results.extend(self._iter_matches(ident)) if not results: raise PyAMLException(f"No PyAT elements found for identifier(s): {', '.join(i.__repr__() for i in identifiers)}") return results
[docs] def get_at_element(self, element_id: LinkerIdentifier) -> at.Element: """Return a single PyAT element matching the given identifier. Parameters ---------- element_id : LinkerIdentifier Identifier describing the PyAT element to retrieve. Returns ------- at.Element The PyAT element matching the identifier. Raises ------ PyAMLException If no element matches the identifier. """ for elem in self._iter_matches(element_id): return elem raise PyAMLException(f"No PyAT element found for FamName: {element_id.__repr__()}")