Skip to content

Commit

Permalink
feat: Improve class selection during insert without type hint
Browse files Browse the repository at this point in the history
  • Loading branch information
Wuestengecko committed Dec 23, 2024
1 parent 4cb479d commit 82c2f5b
Showing 1 changed file with 59 additions and 4 deletions.
63 changes: 59 additions & 4 deletions src/capellambse/model/_descriptors.py
Original file line number Diff line number Diff line change
Expand Up @@ -51,7 +51,7 @@
import capellambse
from capellambse import helpers

from . import T, T_co, U_co
from . import T, T_co, U, U_co

_NotSpecifiedType = t.NewType("_NotSpecifiedType", object)
_NOT_SPECIFIED = _NotSpecifiedType(object())
Expand Down Expand Up @@ -464,6 +464,8 @@ def insert(
elmlist: _obj.ElementListCouplingMixin,
index: int,
value: T_co | NewObject,
*,
bounds: tuple[_obj.ClassName, ...] = (),
) -> T_co:
"""Insert the ``value`` object into the model.
Expand Down Expand Up @@ -2876,6 +2878,8 @@ def insert(
elmlist: _obj.ElementListCouplingMixin,
index: int,
value: T_co | NewObject,
*,
bounds: tuple[_obj.ClassName, ...] = (),
) -> T_co:
if self.role_tag is None:
raise RuntimeError(
Expand Down Expand Up @@ -2906,9 +2910,18 @@ def insert(

if isinstance(value, NewObject):
if not value._type_hint:
cls = self._guess_xtype(elmlist._model)
cls = self._guess_xtype(elmlist._model, bounds)
else:
cls = self._match_xtype(elmlist._model, value._type_hint)

for bound in bounds:
bound_cls = elmlist._model.resolve_class(bound)
if not isinstance(cls, bound_cls):
raise InvalidModificationError(
f"Requested class {cls!r}"
f" does not satisfy bound {bound}"
)

attrs = dict(value._kw)
if not hasattr(cls, "__capella_namespace__"):
raise TypeError(
Expand Down Expand Up @@ -2981,8 +2994,43 @@ def _match_xtype(
raise TypeError(f"Cannot insert elements into {self}")
return t.cast(type[T_co], cls)

def _guess_xtype(self, model: capellambse.MelodyModel) -> type[T_co]:
return t.cast(type[T_co], model.resolve_class(self.class_))
def _guess_xtype(
self,
model: capellambse.MelodyModel,
bounds: tuple[_obj.ClassName, ...],
) -> type[T_co]:
cls = t.cast(type[T_co], model.resolve_class(self.class_))
if not bounds:
return cls

candidates = _find_all_subclasses(cls)

for bound in bounds:
candidates &= _find_all_subclasses(model.resolve_class(bound))

if not candidates:
strbounds = ", ".join(f"{b[0].alias}:{b[1]}" for b in bounds)
raise InvalidModificationError(
f"No subclass of {self.class_[0].alias}:{self.class_[1]}"
f" satisfies all bounds: {strbounds}"
)

filtered_candidates: set[type[T_co]] = set()
for c in candidates:
if c.__capella_abstract__ or any(
c != o and issubclass(c, o) for o in candidates
):
filtered_candidates.add(c)
candidates -= filtered_candidates
assert candidates

if len(candidates) > 1:
raise RuntimeError(
"Multiple eligible classes found, specify a type hint:"
f" {candidates}"
)

return candidates.pop()

def _resolve_super_attributes(
self, super_acc: Accessor[t.Any] | None
Expand Down Expand Up @@ -3035,6 +3083,13 @@ def make_coupled_list_type(
return list_type


def _find_all_subclasses(cls: type[U]) -> set[type[U]]:
classes = set(cls.__subclasses__())
for scls in classes.copy():
classes.update(_find_all_subclasses(scls))
return classes


from . import _obj
from ._obj import Namespace # noqa: F401 # needed for Sphinx

Expand Down

0 comments on commit 82c2f5b

Please sign in to comment.