Skip to content

Commit 32118cc

Browse files
committed
Closes #8
1 parent 0c6d94a commit 32118cc

File tree

1 file changed

+23
-19
lines changed

1 file changed

+23
-19
lines changed

typed_descriptors/base.py

+23-19
Original file line numberDiff line numberDiff line change
@@ -7,7 +7,9 @@
77

88
from __future__ import annotations
99
from abc import abstractmethod
10+
from collections.abc import Container, Iterable
1011
import sys
12+
from types import get_original_bases
1113
from typing import (
1214
Any,
1315
Literal,
@@ -24,27 +26,29 @@
2426
from typing_extensions import Self
2527
from typing_validation import can_validate, validate
2628

27-
28-
def is_dict_available(owner: type) -> bool:
29+
def is_dict_available(cls: Any) -> bool:
2930
"""
30-
Checks whether instances of a descriptor owner class have ``__dict__``.
31-
Returns :obj:`True` if the MRO root ``owner.__mro__[-1]`` is not
32-
:obj:`object`, or if the following is true for any class ``cls`` in
33-
``owner.__mro__[:-1]`` (i.e. excluding the MRO root):
34-
35-
1. ``cls`` does not define ``__slots__``, or
36-
2. ``__dict__`` appears in the ``__slots__`` for ``cls``
37-
31+
Checks whether instances of a descriptor owner class have ``__dict__``
32+
available on them.
3833
"""
39-
mro = owner.__mro__
40-
assert mro[-1] == object, "All classes should inherit from object."
41-
for cls in mro[:-1]:
42-
if not hasattr(cls, "__slots__"):
43-
return True
44-
if "__slots__" not in cls.__dict__:
45-
return True
46-
if "__dict__" in cls.__slots__:
47-
return True
34+
if cls is object:
35+
return False
36+
if not hasattr(cls, "__slots__"):
37+
return True
38+
if hasattr(cls, "__dict__") and "__slots__" not in cls.__dict__:
39+
return True
40+
if isinstance(cls.__slots__, Container) and "__dict__" in cls.__slots__:
41+
return True
42+
if isinstance(cls, type):
43+
# See:
44+
# - peps.python.org/pep-0560/
45+
# - docs.python.org/3/reference/datamodel.html#object.__mro_entries__
46+
# - docs.python.org/3/library/types.html#types.get_original_bases
47+
bases = get_original_bases(cls)
48+
print(cls, bases)
49+
for base in bases:
50+
if is_dict_available(base):
51+
return True
4852
return False
4953

5054

0 commit comments

Comments
 (0)