10
10
import sys
11
11
from typing import (
12
12
Any ,
13
+ Literal ,
13
14
Optional ,
14
15
Protocol ,
15
16
Type ,
@@ -31,42 +32,16 @@ def is_dict_available(owner: type) -> bool:
31
32
:obj:`object`, or if the following is true for any class ``cls`` in
32
33
``owner.__mro__[:-1]`` (i.e. excluding the MRO root):
33
34
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``
61
37
62
38
"""
63
39
mro = owner .__mro__
64
- if mro [- 1 ] != object :
65
- return False
40
+ assert mro [- 1 ] == object , "All classes should inherit from object."
66
41
for cls in mro [:- 1 ]:
67
42
if not hasattr (cls , "__slots__" ):
68
43
return True
69
- if not cls .__slots__ :
44
+ if "__slots__" not in cls .__dict__ :
70
45
return True
71
46
if "__dict__" in cls .__slots__ :
72
47
return True
@@ -173,17 +148,18 @@ class DescriptorBase(TypedDescriptor[T]):
173
148
that ``__dict__`` is not available on instances of the descriptor owner
174
149
class (cf. :func:`is_dict_available`), then a :obj:`TypeError` is raised
175
150
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
177
152
constructor, then the "attr" functions :func:`getattr`, :func:`setattr`,
178
153
:func:`delattr` and :func:`hasattr` will be used. If the library is
179
154
certain that ``__dict__`` is not available on instances of the descriptor
180
155
owner class (cf. :func:`is_dict_available`) and the backing attribute
181
156
name is not present in the class slots (cf. :func:`class_slots`), then a
182
157
: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.
187
163
188
164
Naming logic for the backing attribute:
189
165
@@ -230,7 +206,8 @@ def __init__(
230
206
/ ,
231
207
* ,
232
208
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 ,
234
211
) -> None :
235
212
# pylint: disable = redefined-builtin
236
213
...
@@ -242,7 +219,8 @@ def __init__(
242
219
/ ,
243
220
* ,
244
221
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 ,
246
224
) -> None :
247
225
# pylint: disable = redefined-builtin
248
226
...
@@ -253,18 +231,19 @@ def __init__(
253
231
/ ,
254
232
* ,
255
233
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 ,
257
236
) -> None :
258
237
"""
259
238
Creates a new descriptor with the given type and optional validator.
260
239
261
240
:param type: the type of the descriptor.
262
241
:param backed_by: name for the backing attribute (optional, default name
263
242
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.
268
247
:raises TypeError: if the type cannot be validated by the
269
248
:mod:`typing_validation` library.
270
249
@@ -274,9 +253,15 @@ def __init__(
274
253
if not can_validate (type ):
275
254
raise TypeError (f"Cannot validate type { type !r} ." )
276
255
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
+ )
277
260
self .__type = type
278
261
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
+ )
280
265
self .__descriptor_type__ = type
281
266
282
267
@final
0 commit comments