Skip to content

Commit 0c6d94a

Browse files
committed
Fixed dict availability logic. Introduced use_slots=True in constructor instead of used_dict=False.
1 parent 7b7177e commit 0c6d94a

File tree

3 files changed

+87
-63
lines changed

3 files changed

+87
-63
lines changed

typed_descriptors/attr.py

+21-7
Original file line numberDiff line numberDiff line change
@@ -9,6 +9,7 @@
99
from inspect import signature
1010
from typing import (
1111
Any,
12+
Literal,
1213
Optional,
1314
Protocol,
1415
Type,
@@ -141,7 +142,8 @@ def validator(
141142
*,
142143
readonly: bool = False,
143144
backed_by: Optional[str] = None,
144-
use_dict: Optional[bool] = None,
145+
use_dict: Optional[Literal[True]] = None,
146+
use_slots: Optional[Literal[True]] = None,
145147
) -> Attr[T]: ...
146148

147149
@staticmethod
@@ -152,7 +154,8 @@ def validator(
152154
*,
153155
readonly: bool = False,
154156
backed_by: Optional[str] = None,
155-
use_dict: Optional[bool] = None,
157+
use_dict: Optional[Literal[True]] = None,
158+
use_slots: Optional[Literal[True]] = None,
156159
) -> ValidatedAttrFactory: ...
157160

158161
@staticmethod
@@ -162,7 +165,8 @@ def validator(
162165
*,
163166
readonly: bool = False,
164167
backed_by: Optional[str] = None,
165-
use_dict: Optional[bool] = None,
168+
use_dict: Optional[Literal[True]] = None,
169+
use_slots: Optional[Literal[True]] = None,
166170
) -> ValidatedAttrFactory | Attr[T]:
167171
"""
168172
Decorator used to create an :class:`Attr` from a validator function,
@@ -212,6 +216,7 @@ def w(self, value: Sequence[int]) -> bool:
212216
readonly=readonly,
213217
backed_by=backed_by,
214218
use_dict=use_dict,
219+
use_slots=use_slots,
215220
)
216221

217222
def _validated_attr(validator_fun: ValidatorFunction[_T]) -> Attr[_T]:
@@ -220,6 +225,7 @@ def _validated_attr(validator_fun: ValidatorFunction[_T]) -> Attr[_T]:
220225
readonly=readonly,
221226
backed_by=backed_by,
222227
use_dict=use_dict,
228+
use_slots=use_slots,
223229
)
224230

225231
return _validated_attr
@@ -236,7 +242,8 @@ def __init__(
236242
*,
237243
readonly: bool = False,
238244
backed_by: Optional[str] = None,
239-
use_dict: Optional[bool] = None,
245+
use_dict: Optional[Literal[True]] = None,
246+
use_slots: Optional[Literal[True]] = None,
240247
) -> None:
241248
# pylint: disable = redefined-builtin
242249
...
@@ -250,7 +257,8 @@ def __init__(
250257
*,
251258
readonly: bool = False,
252259
backed_by: Optional[str] = None,
253-
use_dict: Optional[bool] = None,
260+
use_dict: Optional[Literal[True]] = None,
261+
use_slots: Optional[Literal[True]] = None,
254262
) -> None:
255263
# pylint: disable = redefined-builtin
256264
...
@@ -263,7 +271,8 @@ def __init__(
263271
*,
264272
readonly: bool = False,
265273
backed_by: Optional[str] = None,
266-
use_dict: Optional[bool] = None,
274+
use_dict: Optional[Literal[True]] = None,
275+
use_slots: Optional[Literal[True]] = None,
267276
) -> None:
268277
"""
269278
Creates a new attribute with the given type and optional validator.
@@ -278,7 +287,12 @@ def __init__(
278287
:meta public:
279288
"""
280289
# pylint: disable = redefined-builtin
281-
super().__init__(type, backed_by=backed_by, use_dict=use_dict)
290+
super().__init__(
291+
type,
292+
backed_by=backed_by,
293+
use_dict=use_dict,
294+
use_slots=use_slots,
295+
)
282296
if validator is not None:
283297
validate_validator_fun(validator)
284298
self.__doc__ = validator.__doc__

typed_descriptors/base.py

+28-43
Original file line numberDiff line numberDiff line change
@@ -10,6 +10,7 @@
1010
import sys
1111
from typing import (
1212
Any,
13+
Literal,
1314
Optional,
1415
Protocol,
1516
Type,
@@ -31,42 +32,16 @@ def is_dict_available(owner: type) -> bool:
3132
:obj:`object`, or if the following is true for any class ``cls`` in
3233
``owner.__mro__[:-1]`` (i.e. excluding the MRO root):
3334
34-
1. ``cls`` has no ``__slots__`` attribute
35-
2. the ``__slots__`` attribute of ``cls`` is empty
36-
3. ``__dict__`` appears in the the ``__slots__`` attribute of ``cls``
37-
38-
Otherwise, returns :obj:`False`.
39-
40-
.. warning ::
41-
42-
If the function returns :obj:`False`, then ``__dict__`` is certainly
43-
not available. However, it is possible for a class ``cls`` in the MRO to
44-
satisfy one of the three conditions above and for ``__dict__`` to not be
45-
available on instances of the descriptor owner class. For example:
46-
47-
1. The ``__slots__`` attribute might have been deleted at some point
48-
after the slots creation process.
49-
2. The ``__slots__`` attribute contents might have been cleared, or it
50-
might have been an iterable which as been fully consumed as part of
51-
the slots creation process.
52-
3. A ``__dict__`` entry might have been added to ``__slots__`` at some
53-
point after the slots creation process.
54-
55-
If this happens somewhere in the MRO of the owner class being inspected,
56-
the library will incorrectly infer that ``__dict__`` is available on
57-
class instances, resulting in incorrect behaviour.
58-
In such situations, please manually set ``use_dict`` to :obj:`False` in
59-
attribute constructors to ensure that the ``__dict__`` mechanism is not
60-
used to back the descriptor.
35+
1. ``cls`` does not define ``__slots__``, or
36+
2. ``__dict__`` appears in the ``__slots__`` for ``cls``
6137
6238
"""
6339
mro = owner.__mro__
64-
if mro[-1] != object:
65-
return False
40+
assert mro[-1] == object, "All classes should inherit from object."
6641
for cls in mro[:-1]:
6742
if not hasattr(cls, "__slots__"):
6843
return True
69-
if not cls.__slots__:
44+
if "__slots__" not in cls.__dict__:
7045
return True
7146
if "__dict__" in cls.__slots__:
7247
return True
@@ -173,17 +148,18 @@ class DescriptorBase(TypedDescriptor[T]):
173148
that ``__dict__`` is not available on instances of the descriptor owner
174149
class (cf. :func:`is_dict_available`), then a :obj:`TypeError` is raised
175150
at the time when ``__set_name__`` is called.
176-
2. If the ``use_dict`` argument is set to :obj:`False` in the descriptor
151+
2. If the ``use_slots`` argument is set to :obj:`True` in the descriptor
177152
constructor, then the "attr" functions :func:`getattr`, :func:`setattr`,
178153
:func:`delattr` and :func:`hasattr` will be used. If the library is
179154
certain that ``__dict__`` is not available on instances of the descriptor
180155
owner class (cf. :func:`is_dict_available`) and the backing attribute
181156
name is not present in the class slots (cf. :func:`class_slots`), then a
182157
:obj:`TypeError` is raised at the time when ``__set_name__`` is called.
183-
3. If ``use_dict`` is set to :obj:`None` (default value) in the descriptor
184-
constructor, then :func:`is_dict_available` is called and ``use_dict`` is
185-
set to the resulting value. Further validation is then performed as
186-
described in points 1. and 2. above.
158+
3. If neither ``use_dict`` nor ``use_slots__`` is set to :obj:`True` in the
159+
descriptor constructor (the default case), then :func:`is_dict_available`
160+
is called and the result is used to determine whether to use ``__dict__``
161+
or slots for the backing attribute. Further validation is then performed,
162+
as described in points 1 and 2 above.
187163
188164
Naming logic for the backing attribute:
189165
@@ -230,7 +206,8 @@ def __init__(
230206
/,
231207
*,
232208
backed_by: Optional[str] = None,
233-
use_dict: Optional[bool] = None,
209+
use_dict: Optional[Literal[True]] = None,
210+
use_slots: Optional[Literal[True]] = None,
234211
) -> None:
235212
# pylint: disable = redefined-builtin
236213
...
@@ -242,7 +219,8 @@ def __init__(
242219
/,
243220
*,
244221
backed_by: Optional[str] = None,
245-
use_dict: Optional[bool] = None,
222+
use_dict: Optional[Literal[True]] = None,
223+
use_slots: Optional[Literal[True]] = None,
246224
) -> None:
247225
# pylint: disable = redefined-builtin
248226
...
@@ -253,18 +231,19 @@ def __init__(
253231
/,
254232
*,
255233
backed_by: Optional[str] = None,
256-
use_dict: Optional[bool] = None,
234+
use_dict: Optional[Literal[True]] = None,
235+
use_slots: Optional[Literal[True]] = None,
257236
) -> None:
258237
"""
259238
Creates a new descriptor with the given type and optional validator.
260239
261240
:param type: the type of the descriptor.
262241
:param backed_by: name for the backing attribute (optional, default name
263242
used if not specified).
264-
:param use_dict: whether to use ``__dict__`` or slots for access to the
265-
backing attribute (optional, automatically determined
266-
if not specified).
267-
243+
:param use_dict: if set to :obj:`True`, ``__dict__`` will be used to
244+
store the the backing attribute.
245+
:param use_dict: if set to :obj:`True`, ``__slots__`` will be used to
246+
store the the backing attribute.
268247
:raises TypeError: if the type cannot be validated by the
269248
:mod:`typing_validation` library.
270249
@@ -274,9 +253,15 @@ def __init__(
274253
if not can_validate(type):
275254
raise TypeError(f"Cannot validate type {type!r}.")
276255
validate(backed_by, Optional[str])
256+
if use_dict and use_slots:
257+
raise ValueError(
258+
"Cannot set both use_dict=True and use_slots=True."
259+
)
277260
self.__type = type
278261
self.__temp_backed_by = backed_by
279-
self.__temp_use_dict = use_dict
262+
self.__temp_use_dict = (
263+
True if use_dict else False if use_slots else None
264+
)
280265
self.__descriptor_type__ = type
281266

282267
@final

typed_descriptors/prop.py

+38-13
Original file line numberDiff line numberDiff line change
@@ -10,6 +10,7 @@
1010
import sys
1111
from typing import (
1212
Any,
13+
Literal,
1314
Optional,
1415
Protocol,
1516
Type,
@@ -107,7 +108,8 @@ def value(
107108
/,
108109
*,
109110
backed_by: Optional[str] = None,
110-
use_dict: Optional[bool] = None,
111+
use_dict: Optional[Literal[True]] = None,
112+
use_slots: Optional[Literal[True]] = None,
111113
) -> Prop[T]: ...
112114

113115
@staticmethod
@@ -117,7 +119,8 @@ def value(
117119
/,
118120
*,
119121
backed_by: Optional[str] = None,
120-
use_dict: Optional[bool] = None,
122+
use_dict: Optional[Literal[True]] = None,
123+
use_slots: Optional[Literal[True]] = None,
121124
) -> PropFactory: ...
122125

123126
@staticmethod
@@ -126,7 +129,8 @@ def value(
126129
/,
127130
*,
128131
backed_by: Optional[str] = None,
129-
use_dict: Optional[bool] = None,
132+
use_dict: Optional[Literal[True]] = None,
133+
use_slots: Optional[Literal[True]] = None,
130134
) -> PropFactory | Prop[T]:
131135
"""
132136
An alias for :func:`cached_property`.
@@ -136,7 +140,10 @@ def value(
136140
decorator for attributes.
137141
"""
138142
return cached_property(
139-
value_fun, backed_by=backed_by, use_dict=use_dict
143+
value_fun,
144+
backed_by=backed_by,
145+
use_dict=use_dict,
146+
use_slots=use_slots,
140147
)
141148

142149
__value_fun: ValueFunction[T]
@@ -149,7 +156,8 @@ def __init__(
149156
/,
150157
*,
151158
backed_by: Optional[str] = None,
152-
use_dict: Optional[bool] = None,
159+
use_dict: Optional[Literal[True]] = None,
160+
use_slots: Optional[Literal[True]] = None,
153161
) -> None:
154162
# pylint: disable = redefined-builtin
155163
...
@@ -162,7 +170,8 @@ def __init__(
162170
/,
163171
*,
164172
backed_by: Optional[str] = None,
165-
use_dict: Optional[bool] = None,
173+
use_dict: Optional[Literal[True]] = None,
174+
use_slots: Optional[Literal[True]] = None,
166175
) -> None:
167176
# pylint: disable = redefined-builtin
168177
...
@@ -174,7 +183,8 @@ def __init__(
174183
/,
175184
*,
176185
backed_by: Optional[str] = None,
177-
use_dict: Optional[bool] = None,
186+
use_dict: Optional[Literal[True]] = None,
187+
use_slots: Optional[Literal[True]] = None,
178188
) -> None:
179189
"""
180190
Creates a new property with the given type and value function.
@@ -191,7 +201,12 @@ def __init__(
191201
"""
192202
# pylint: disable = redefined-builtin
193203
validate_value_fun(value)
194-
super().__init__(type, backed_by=backed_by, use_dict=use_dict)
204+
super().__init__(
205+
type,
206+
backed_by=backed_by,
207+
use_dict=use_dict,
208+
use_slots=use_slots,
209+
)
195210
if not callable(value):
196211
raise TypeError(f"Expected callable 'value', got {value!r}.")
197212
self.__value_fun = value
@@ -328,7 +343,8 @@ def cached_property(
328343
/,
329344
*,
330345
backed_by: Optional[str] = None,
331-
use_dict: Optional[bool] = None,
346+
use_dict: Optional[Literal[True]] = None,
347+
use_slots: Optional[Literal[True]] = None,
332348
) -> Prop[T]: ...
333349

334350

@@ -338,7 +354,8 @@ def cached_property(
338354
/,
339355
*,
340356
backed_by: Optional[str] = None,
341-
use_dict: Optional[bool] = None,
357+
use_dict: Optional[Literal[True]] = None,
358+
use_slots: Optional[Literal[True]] = None,
342359
) -> PropFactory: ...
343360

344361

@@ -347,7 +364,8 @@ def cached_property(
347364
/,
348365
*,
349366
backed_by: Optional[str] = None,
350-
use_dict: Optional[bool] = None,
367+
use_dict: Optional[Literal[True]] = None,
368+
use_slots: Optional[Literal[True]] = None,
351369
) -> PropFactory | Prop[T]:
352370
"""
353371
Decorator used to create a cached property from a value function,
@@ -392,12 +410,19 @@ def x(self) -> Sequence[str]:
392410
if value_fun is not None:
393411
prop_type = value_fun_return_type(value_fun)
394412
return Prop(
395-
prop_type, value_fun, backed_by=backed_by, use_dict=use_dict
413+
prop_type,
414+
value_fun,
415+
backed_by=backed_by,
416+
use_dict=use_dict,
417+
use_slots=use_slots,
396418
)
397419

398420
def _cached_property(value_fun: ValueFunction[_T]) -> Prop[_T]:
399421
return cached_property(
400-
value_fun, backed_by=backed_by, use_dict=use_dict
422+
value_fun,
423+
backed_by=backed_by,
424+
use_dict=use_dict,
425+
use_slots=use_slots,
401426
)
402427

403428
return _cached_property

0 commit comments

Comments
 (0)