|
7 | 7 |
|
8 | 8 | from __future__ import annotations
|
9 | 9 | from abc import abstractmethod
|
| 10 | +from collections.abc import Container, Iterable |
10 | 11 | import sys
|
| 12 | +from types import get_original_bases |
11 | 13 | from typing import (
|
12 | 14 | Any,
|
13 | 15 | Literal,
|
|
24 | 26 | from typing_extensions import Self
|
25 | 27 | from typing_validation import can_validate, validate
|
26 | 28 |
|
27 |
| - |
28 |
| -def is_dict_available(owner: type) -> bool: |
| 29 | +def is_dict_available(cls: Any) -> bool: |
29 | 30 | """
|
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. |
38 | 33 | """
|
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 |
48 | 52 | return False
|
49 | 53 |
|
50 | 54 |
|
|
0 commit comments