Source code for pylasu.model.model

import dataclasses
import inspect
import sys
import typing
from abc import ABC, abstractmethod, ABCMeta
from dataclasses import Field, MISSING, dataclass, field
from typing import Optional, Callable, List, Union

from .naming import ReferenceByName
from .position import Position, Source
from .reflection import Multiplicity, PropertyDescription
from ..reflection import get_type_annotations, get_type_arguments, is_sequence_type
from ..reflection.reflection import get_type_origin

PYLASU_FEATURE = "pylasu_feature"


[docs] class internal_property(property): pass
[docs] def internal_properties(*props: str): def decorate(cls: type): cls.__internal_properties__ = ( getattr(cls, "__internal_properties__", []) + [*Node.__internal_properties__, *props]) return cls return decorate
[docs] class InternalField(Field): pass
[docs] def internal_field( *, default=MISSING, default_factory=MISSING, init=True, repr=True, hash=None, compare=True, metadata=None, kw_only=False): """Return an object to identify internal dataclass fields. The arguments are the same as dataclasses.field.""" if default is not MISSING and default_factory is not MISSING: raise ValueError('cannot specify both default and default_factory') try: # Python 3.10+ return InternalField(default, default_factory, init, repr, hash, compare, metadata, kw_only) except TypeError: return InternalField(default, default_factory, init, repr, hash, compare, metadata)
[docs] def node_property(default=MISSING): description = PropertyDescription( "", None, multiplicity=Multiplicity.OPTIONAL if default is None else Multiplicity.SINGULAR) return field(default=default, metadata={PYLASU_FEATURE: description})
[docs] def node_containment(multiplicity: Multiplicity = Multiplicity.SINGULAR): description = PropertyDescription("", None, is_containment=True, multiplicity=multiplicity) if multiplicity == Multiplicity.SINGULAR: return field(metadata={PYLASU_FEATURE: description}) elif multiplicity == Multiplicity.OPTIONAL: return field(default=None, metadata={PYLASU_FEATURE: description}) elif multiplicity == Multiplicity.MANY: return field(default_factory=list, metadata={PYLASU_FEATURE: description})
[docs] class Origin(ABC):
[docs] @internal_property @abstractmethod def position(self) -> Optional[Position]: pass
[docs] @internal_property def source_text(self) -> Optional[str]: return None
[docs] @internal_property def source(self) -> Optional[Source]: return self.position.source if self.position is not None else None
[docs] @dataclass class CompositeOrigin(Origin): elements: List[Origin] = field(default_factory=list) position: Optional[Position] = None source_text: Optional[str] = None
[docs] class Destination(ABC): pass
[docs] @dataclass class CompositeDestination(Destination): elements: List[Destination] = field(default_factory=list)
[docs] @dataclass class TextFileDestination(Destination): position: Optional[Position] = None
[docs] def is_internal_property_or_method(value): return isinstance(value, internal_property) or isinstance(value, InternalField) or isinstance(value, Callable)
[docs] def provides_nodes(decl_type): if get_type_origin(decl_type) is Union: provides = None for tp in get_type_arguments(decl_type): if tp is type(None): continue arg_provides = provides_nodes(tp) if provides is None: provides = arg_provides elif provides != arg_provides: raise Exception(f"Type {decl_type} mixes nodes and non-nodes") return provides else: return isinstance(decl_type, type) and issubclass(decl_type, Node)
[docs] def get_only_type_arg(decl_type): """If decl_type has a single type argument, return it, otherwise return None""" type_args = get_type_arguments(decl_type) if len(type_args) == 1: return type_args[0] else: return None
[docs] def process_annotated_property(cl: type, name: str, decl_type): try: fields = dataclasses.fields(cl) except TypeError: fields = tuple() for fld in fields: if fld.name == name and PYLASU_FEATURE in fld.metadata: feature = fld.metadata[PYLASU_FEATURE] feature.name = name if isinstance(decl_type, type): feature.type = decl_type elif type(fld.type) is str: feature.type = try_to_resolve_string_type(fld.type, name, cl) return feature return compute_feature_from_annotation(cl, name, decl_type)
[docs] def compute_feature_from_annotation(cl, name, decl_type): feature = PropertyDescription(name, None, False, False, Multiplicity.SINGULAR) decl_type = try_to_resolve_type(decl_type, feature) if not isinstance(decl_type, type): fwref = None if hasattr(typing, "ForwardRef"): fwref = typing.ForwardRef if fwref and isinstance(decl_type, fwref): raise Exception(f"Feature {name}'s type is unresolved forward reference {decl_type}, " f"please use node_containment or node_property") elif type(decl_type) is str: decl_type = try_to_resolve_string_type(decl_type, name, cl) if not isinstance(decl_type, type): raise Exception(f"Unsupported feature {name} of type {decl_type} in {cl}") feature.type = decl_type feature.is_containment = provides_nodes(decl_type) and not feature.is_reference return feature
[docs] def try_to_resolve_string_type(decl_type, name, cl): try: ns = getattr(sys.modules.get(cl.__module__, None), '__dict__', globals()) decl_type = ns[decl_type] except KeyError: raise Exception(f"Unsupported feature {name} of unknown type {decl_type} in {cl}") return decl_type
[docs] def try_to_resolve_type(decl_type, feature): if get_type_origin(decl_type) is ReferenceByName: decl_type = get_only_type_arg(decl_type) or decl_type feature.is_reference = True if is_sequence_type(decl_type): decl_type = get_only_type_arg(decl_type) or decl_type feature.multiplicity = Multiplicity.MANY if get_type_origin(decl_type) is Union: type_args = get_type_arguments(decl_type) if len(type_args) == 1: decl_type = type_args[0] elif len(type_args) == 2: if type_args[0] is type(None): decl_type = type_args[1] elif type_args[1] is type(None): decl_type = type_args[0] else: raise Exception(f"Unsupported feature {feature.name} of union type {decl_type}") if feature.multiplicity == Multiplicity.SINGULAR: feature.multiplicity = Multiplicity.OPTIONAL else: raise Exception(f"Unsupported feature {feature.name} of union type {decl_type}") decl_type = try_to_resolve_type(decl_type, feature) return decl_type
[docs] class Concept(ABCMeta): def __init__(cls, what, bases=None, dict=None): super().__init__(what, bases, dict) cls.__internal_properties__ = [] for base in bases: if hasattr(base, "__internal_properties__"): cls.__internal_properties__.extend(base.__internal_properties__) if not cls.__internal_properties__: cls.__internal_properties__ = ["origin", "destination", "parent", "position", "position_override"] cls.__internal_properties__.extend([n for n, v in inspect.getmembers(cls, is_internal_property_or_method)]) @property def node_properties(cls): names = set() for cl in cls.__mro__: yield from cls._direct_node_properties(cl, names) def _direct_node_properties(cls, cl, known_property_names): if not isinstance(cls, Concept): return anns = get_type_annotations(cl) if not anns: return for name in anns: if name not in known_property_names and cls.is_node_property(name): feature = process_annotated_property(cl, name, anns[name]) known_property_names.add(name) yield feature for name in dir(cl): if name not in known_property_names and cls.is_node_property(name): feature = PropertyDescription(name, None, False, False) known_property_names.add(name) yield feature
[docs] def is_node_property(cls, name): return not name.startswith('_') and name not in cls.__internal_properties__
[docs] class Node(Origin, Destination, metaclass=Concept): origin: Optional[Origin] = None destination: Optional[Destination] = None parent: Optional["Node"] = None position_override: Optional[Position] = None def __init__(self, origin: Optional[Origin] = None, parent: Optional["Node"] = None, position_override: Optional[Position] = None): self.origin = origin self.parent = parent self.position_override = position_override
[docs] def with_origin(self, origin: Optional[Origin]): self.origin = origin return self
[docs] def with_parent(self, parent: Optional["Node"]): self.parent = parent return self
[docs] def with_position(self, position: Optional[Position]): self.position = position return self
@internal_property def position(self) -> Optional[Position]: return self.position_override if self.position_override is not None\ else self.origin.position if self.origin is not None else None
[docs] @position.setter def position(self, position: Optional[Position]): self.position_override = position
[docs] @internal_property def source_text(self) -> Optional[str]: return self.origin.source_text if self.origin is not None else None
[docs] @internal_property def source(self) -> Optional[Source]: return self.origin.source if self.origin is not None else None
[docs] @internal_property def properties(self): return (PropertyDescription(p.name, p.type, is_containment=p.is_containment, is_reference=p.is_reference, multiplicity=p.multiplicity, value=getattr(self, p.name)) for p in self.__class__.node_properties)
@internal_property def _fields(self): yield from (name for name, _ in self.properties)
[docs] @internal_property def node_type(self): return type(self)
[docs] def concept_of(node): properties = dir(node) if "__concept__" in properties: node_type = node.__concept__ elif "node_type" in properties: node_type = node.node_type else: node_type = type(node) if isinstance(node_type, Concept): return node_type else: raise Exception(f"Not a concept: {node_type} of {node}")