|
24 | 24 | from typing_validation import can_validate, validate
|
25 | 25 |
|
26 | 26 |
|
| 27 | +def is_dict_available(owner: type) -> bool: |
| 28 | + """ |
| 29 | + Checks whether instances of a given type have ``__dict__`` available. |
| 30 | + Returns :obj:`True` if either: |
| 31 | +
|
| 32 | + - the MRO root ``owner.__mro__[-1]`` is not :obj:`object`; |
| 33 | + - one of the classes in ``owner.__mro__[:-1]`` (i.e. excluding the MRO root) |
| 34 | + either does not define ``__slots__`` or defines ``__dict__`` in its slots. |
| 35 | +
|
| 36 | + Otherwise, returns :obj:`False`. |
| 37 | + """ |
| 38 | + mro = owner.__mro__ |
| 39 | + if mro[-1] != object: |
| 40 | + return False |
| 41 | + for cls in mro[:-1]: |
| 42 | + if not hasattr(cls, "__slots__"): |
| 43 | + return True |
| 44 | + if "__dict__" in cls.__slots__: |
| 45 | + return True |
| 46 | + return False |
| 47 | + |
27 | 48 | def name_mangle(owner: type, attr_name: str) -> str:
|
28 | 49 | """
|
29 | 50 | If the given attribute name is private and not dunder,
|
@@ -102,35 +123,29 @@ class DescriptorBase(TypedDescriptor[T]):
|
102 | 123 |
|
103 | 124 | 1. If the ``backed_by`` argument is specified in the constructor, the string
|
104 | 125 | passed to it is used as name for the backing attribute.
|
105 |
| - 2. Else, if either the owner class has no ``__slots__`` or ``__dict__`` |
106 |
| - is included in its ``__slots__``, the backing attribute name coincides |
107 |
| - with the descriptor name. |
| 126 | + 2. Else, if ``__dict__`` is available (see :func:`is_dict_available`), then |
| 127 | + the backing attribute name coincides with the descriptor name. |
108 | 128 | 3. Else, the backing attribute name is obtained by prepending one or
|
109 | 129 | two underscores to the descriptor name (one if the descriptor name starts
|
110 | 130 | with underscore, two if it doesn't).
|
111 | 131 |
|
112 | 132 | Access logic for backing attribute:
|
113 | 133 |
|
114 |
| - 1. If the owner class has no ``__slots__``, the backing attribute is |
115 |
| - accessed via ``__dict__`` if its name coincides with the descriptor name, |
116 |
| - and via ``___attr`` functions otherwise. |
117 |
| - 2. Else, if the the ``backed_by`` argument is specified in the constructor |
118 |
| - and it appears in the owner class's ``__slots__``, the backing attribute |
119 |
| - is accessed via ``___attr`` functions. |
120 |
| - 3. Else, if ``__dict__`` appears in the owner class's ``__slots__``, the |
121 |
| - backing attribute is accessed via ``__dict__`` if its name coincides with |
122 |
| - the descriptor name, and via ``___attr`` functions otherwise. |
123 |
| - 4. Else, the backing attribute is accessed via ``___attr`` functions. |
| 134 | + 1. If ``__dict__`` is available (see :func:`is_dict_available`), the backing |
| 135 | + attribute is accessed via ``__dict__`` if its name coincides with the |
| 136 | + descriptor name, and via ``___attr`` functions otherwise. |
| 137 | + 2. Else, the backing attribute is accessed via ``___attr`` functions. |
124 | 138 |
|
125 | 139 | Above, the nomenclature "``___attr`` functions" refers to :func:`getattr`,
|
126 | 140 | :func:`setattr`, :func:`delattr` and :func:`hasattr`.
|
127 | 141 |
|
128 | 142 | If the backing attribute name starts with two underscores but does not end
|
129 | 143 | with two underscores, name-mangling is automatically performed.
|
130 | 144 |
|
131 |
| - If the owner class has ``__slots__`` and ``__dict__`` is not included in its |
132 |
| - ``__slots__``, a :obj:`TypeError` is raised at the time when the descriptor |
133 |
| - is assigned if the backing attribute name does not appear in ``__slots__``. |
| 145 | + If ``__dict__`` is not available (see :func:`is_dict_available`) and the |
| 146 | + backing attribute name does not appear in ``__slots__``, a :obj:`TypeError` |
| 147 | + is raised at the time when the descriptor is assigned (or, more precisely, |
| 148 | + at the time when its ``__set_name__`` method is called). |
134 | 149 | """
|
135 | 150 |
|
136 | 151 | # Attributes set by constructor:
|
@@ -317,32 +332,58 @@ def __set_name__(self, owner: Type[Any], name: str) -> None:
|
317 | 332 | # 2. If descriptor name is mangled, unmangle it:
|
318 | 333 | name = name_unmangle(owner, name)
|
319 | 334 | # 3. Compute backing attribute name and whether to use __dict__:
|
320 |
| - __slots__ = owner.__slots__ if hasattr(owner, "__slots__") else None |
321 | 335 | temp_backed_by = self.__temp_backed_by
|
322 |
| - if __slots__ is None: |
| 336 | + if is_dict_available(owner): |
323 | 337 | if temp_backed_by is None:
|
324 | 338 | temp_backed_by = name
|
325 | 339 | use_dict = temp_backed_by == name
|
326 |
| - elif temp_backed_by in __slots__: |
327 |
| - assert temp_backed_by is not None |
328 |
| - use_dict = False |
329 |
| - elif "__dict__" in __slots__: |
330 |
| - if temp_backed_by is None: |
331 |
| - temp_backed_by = name |
332 |
| - use_dict = temp_backed_by == name |
333 |
| - else: # __slots__ is used and __dict__ is not available |
334 |
| - use_dict = False |
| 340 | + assert ( |
| 341 | + not use_dict |
| 342 | + or not hasattr(owner, "__slots__") |
| 343 | + or name not in owner.__slots__ |
| 344 | + ) |
| 345 | + else: |
| 346 | + assert hasattr(owner, "__slots__"), dir(owner) |
335 | 347 | if temp_backed_by is None:
|
336 | 348 | if name.startswith("_"):
|
337 | 349 | temp_backed_by = f"_{name}"
|
338 | 350 | else:
|
339 | 351 | temp_backed_by = f"__{name}"
|
340 |
| - if temp_backed_by not in __slots__: |
| 352 | + use_dict = False |
| 353 | + if temp_backed_by not in owner.__slots__: |
341 | 354 | raise TypeError(
|
342 | 355 | "When __slots__ are used and __dict__ is not available, "
|
343 | 356 | f"the name of the backing attribtue {temp_backed_by!r} "
|
344 | 357 | "must appear in __slots__."
|
345 | 358 | )
|
| 359 | + # if not hasattr(owner, "__slots__"): |
| 360 | + # assert hasattr(owner, "__dict__"), dir(owner) |
| 361 | + # if temp_backed_by is None: |
| 362 | + # temp_backed_by = name |
| 363 | + # use_dict = temp_backed_by == name |
| 364 | + # else: # owner has __slots__ defined |
| 365 | + # __slots__ = owner.__slots__ |
| 366 | + # if temp_backed_by in __slots__: |
| 367 | + # assert temp_backed_by is not None |
| 368 | + # use_dict = False |
| 369 | + # elif hasattr(owner, "__dict__"): |
| 370 | + # if temp_backed_by is None: |
| 371 | + # temp_backed_by = name |
| 372 | + # use_dict = temp_backed_by == name |
| 373 | + # else: |
| 374 | + # # __dict__ is not available and |
| 375 | + # use_dict = False |
| 376 | + # if temp_backed_by is None: |
| 377 | + # if name.startswith("_"): |
| 378 | + # temp_backed_by = f"_{name}" |
| 379 | + # else: |
| 380 | + # temp_backed_by = f"__{name}" |
| 381 | + # if temp_backed_by not in __slots__: |
| 382 | + # raise TypeError( |
| 383 | + # "When __slots__ are used and __dict__ is not available, " |
| 384 | + # f"the name of the backing attribtue {temp_backed_by!r} " |
| 385 | + # "must appear in __slots__." |
| 386 | + # ) |
346 | 387 | backed_by = name_mangle(owner, temp_backed_by)
|
347 | 388 | # 4. Set owner, name (not name-mangled) and backed_by (name-mangled):
|
348 | 389 | self.__owner = owner
|
|
0 commit comments