Skip to content

Commit 781a068

Browse files
committed
Fixes #6
1 parent 8f11554 commit 781a068

File tree

2 files changed

+74
-28
lines changed

2 files changed

+74
-28
lines changed

docs/api/typed_descriptors.base.rst

+5
Original file line numberDiff line numberDiff line change
@@ -29,6 +29,11 @@ TypedDescriptor
2929
:members:
3030
:special-members: __descriptor_type__, __set_name__, __get__
3131

32+
is_dict_available
33+
-----------------
34+
35+
.. autofunction:: typed_descriptors.base.is_dict_available
36+
3237
name_mangle
3338
-----------
3439

typed_descriptors/base.py

+69-28
Original file line numberDiff line numberDiff line change
@@ -24,6 +24,27 @@
2424
from typing_validation import can_validate, validate
2525

2626

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+
2748
def name_mangle(owner: type, attr_name: str) -> str:
2849
"""
2950
If the given attribute name is private and not dunder,
@@ -102,35 +123,29 @@ class DescriptorBase(TypedDescriptor[T]):
102123
103124
1. If the ``backed_by`` argument is specified in the constructor, the string
104125
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.
108128
3. Else, the backing attribute name is obtained by prepending one or
109129
two underscores to the descriptor name (one if the descriptor name starts
110130
with underscore, two if it doesn't).
111131
112132
Access logic for backing attribute:
113133
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.
124138
125139
Above, the nomenclature "``___attr`` functions" refers to :func:`getattr`,
126140
:func:`setattr`, :func:`delattr` and :func:`hasattr`.
127141
128142
If the backing attribute name starts with two underscores but does not end
129143
with two underscores, name-mangling is automatically performed.
130144
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).
134149
"""
135150

136151
# Attributes set by constructor:
@@ -317,32 +332,58 @@ def __set_name__(self, owner: Type[Any], name: str) -> None:
317332
# 2. If descriptor name is mangled, unmangle it:
318333
name = name_unmangle(owner, name)
319334
# 3. Compute backing attribute name and whether to use __dict__:
320-
__slots__ = owner.__slots__ if hasattr(owner, "__slots__") else None
321335
temp_backed_by = self.__temp_backed_by
322-
if __slots__ is None:
336+
if is_dict_available(owner):
323337
if temp_backed_by is None:
324338
temp_backed_by = name
325339
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)
335347
if temp_backed_by is None:
336348
if name.startswith("_"):
337349
temp_backed_by = f"_{name}"
338350
else:
339351
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__:
341354
raise TypeError(
342355
"When __slots__ are used and __dict__ is not available, "
343356
f"the name of the backing attribtue {temp_backed_by!r} "
344357
"must appear in __slots__."
345358
)
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+
# )
346387
backed_by = name_mangle(owner, temp_backed_by)
347388
# 4. Set owner, name (not name-mangled) and backed_by (name-mangled):
348389
self.__owner = owner

0 commit comments

Comments
 (0)