Skip to content

Commit a7058f2

Browse files
committed
Added use_dict to Attr and Prop
1 parent 2d4179d commit a7058f2

File tree

5 files changed

+55
-13
lines changed

5 files changed

+55
-13
lines changed

mypy.ini

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,3 +1,3 @@
11
[mypy]
22
strict = True
3-
plugins = typed_descriptors.mypy_plugin, mypy.plugins.proper_plugin
3+
plugins = .\typed_descriptors\mypy_plugin.py, mypy.plugins.proper_plugin

typed_descriptors/attr.py

+12-4
Original file line numberDiff line numberDiff line change
@@ -141,6 +141,7 @@ def validator(
141141
*,
142142
readonly: bool = False,
143143
backed_by: Optional[str] = None,
144+
use_dict: Optional[bool] = None,
144145
) -> Attr[T]: ...
145146

146147
@staticmethod
@@ -151,6 +152,7 @@ def validator(
151152
*,
152153
readonly: bool = False,
153154
backed_by: Optional[str] = None,
155+
use_dict: Optional[bool] = None,
154156
) -> ValidatedAttrFactory: ...
155157

156158
@staticmethod
@@ -160,6 +162,7 @@ def validator(
160162
*,
161163
readonly: bool = False,
162164
backed_by: Optional[str] = None,
165+
use_dict: Optional[bool] = None,
163166
) -> ValidatedAttrFactory | Attr[T]:
164167
"""
165168
Decorator used to create an :class:`Attr` from a validator function,
@@ -208,11 +211,15 @@ def w(self, value: Sequence[int]) -> bool:
208211
validator=validator_fun,
209212
readonly=readonly,
210213
backed_by=backed_by,
214+
use_dict=use_dict,
211215
)
212216

213217
def _validated_attr(validator_fun: ValidatorFunction[_T]) -> Attr[_T]:
214218
return Attr.validator(
215-
validator_fun, readonly=readonly, backed_by=backed_by
219+
validator_fun,
220+
readonly=readonly,
221+
backed_by=backed_by,
222+
use_dict=use_dict,
216223
)
217224

218225
return _validated_attr
@@ -229,6 +236,7 @@ def __init__(
229236
*,
230237
readonly: bool = False,
231238
backed_by: Optional[str] = None,
239+
use_dict: Optional[bool] = None,
232240
) -> None:
233241
# pylint: disable = redefined-builtin
234242
...
@@ -242,6 +250,7 @@ def __init__(
242250
*,
243251
readonly: bool = False,
244252
backed_by: Optional[str] = None,
253+
use_dict: Optional[bool] = None,
245254
) -> None:
246255
# pylint: disable = redefined-builtin
247256
...
@@ -254,23 +263,22 @@ def __init__(
254263
*,
255264
readonly: bool = False,
256265
backed_by: Optional[str] = None,
266+
use_dict: Optional[bool] = None,
257267
) -> None:
258268
"""
259269
Creates a new attribute with the given type and optional validator.
260270
261271
:param ty: the type of the attribute
262272
:param validator: an optional validator function for the attribute
263273
:param readonly: whether the attribute is read-only
264-
:param backed_by: the name of the backing attribute for the
265-
attribute, or :obj:`None` to use a default name
266274
267275
:raises TypeError: if the type is not a valid type
268276
:raises TypeError: if the validator is not callable
269277
270278
:meta public:
271279
"""
272280
# pylint: disable = redefined-builtin
273-
super().__init__(type, backed_by=backed_by)
281+
super().__init__(type, backed_by=backed_by, use_dict=use_dict)
274282
if validator is not None:
275283
validate_validator_fun(validator)
276284
self.__doc__ = validator.__doc__

typed_descriptors/base.py

+3
Original file line numberDiff line numberDiff line change
@@ -72,6 +72,7 @@ class instances, resulting in incorrect behaviour.
7272
return True
7373
return False
7474

75+
7576
def class_slots(cls: type) -> tuple[str, ...] | None:
7677
"""
7778
Returns a tuple consisting of all slots for the given class and all
@@ -89,6 +90,7 @@ def class_slots(cls: type) -> tuple[str, ...] | None:
8990
slots.append(slot)
9091
return tuple(slots)
9192

93+
9294
def name_mangle(owner: type, attr_name: str) -> str:
9395
"""
9496
If the given attribute name is private and not dunder,
@@ -157,6 +159,7 @@ def __get__(self, instance: Any, _: Type[Any]) -> T_co | Self:
157159
:meta public:
158160
"""
159161

162+
160163
class DescriptorBase(TypedDescriptor[T]):
161164
"""
162165
Base class for descriptors backed by an attribute whose name and access mode

typed_descriptors/mypy_plugin.py

+4
Original file line numberDiff line numberDiff line change
@@ -9,6 +9,7 @@
99
from mypy.plugin import FunctionContext, Plugin
1010
from mypy.types import CallableType, Instance, get_proper_type, Type
1111

12+
1213
def typed_descriptor_hook(ctx: FunctionContext) -> Type:
1314
"""
1415
Extracts the descriptor type and assigns it to the generic type parameter,
@@ -26,11 +27,13 @@ def typed_descriptor_hook(ctx: FunctionContext) -> Type:
2627
return ret_t
2728
return ret_t
2829

30+
2931
_function_hooks = {
3032
"typed_descriptors.attr.Attr": typed_descriptor_hook,
3133
"typed_descriptors.prop.Prop": typed_descriptor_hook,
3234
}
3335

36+
3437
class TypedDescriptorsPlugin(Plugin):
3538
"""
3639
Mypy plugin which expands type inference for typed descriptor,
@@ -45,6 +48,7 @@ def get_function_hook(
4548
return hook
4649
return super().get_function_hook(fullname)
4750

51+
4852
def plugin(version: str) -> typing.Type[TypedDescriptorsPlugin]:
4953
"""
5054
Entry point for the Mypy plugin.

typed_descriptors/prop.py

+35-8
Original file line numberDiff line numberDiff line change
@@ -103,13 +103,21 @@ class Prop(DescriptorBase[T]):
103103
@staticmethod
104104
@overload
105105
def value(
106-
value_fun: ValueFunction[T], /, *, backed_by: Optional[str] = None
106+
value_fun: ValueFunction[T],
107+
/,
108+
*,
109+
backed_by: Optional[str] = None,
110+
use_dict: Optional[bool] = None,
107111
) -> Prop[T]: ...
108112

109113
@staticmethod
110114
@overload
111115
def value(
112-
value_fun: None = None, /, *, backed_by: Optional[str] = None
116+
value_fun: None = None,
117+
/,
118+
*,
119+
backed_by: Optional[str] = None,
120+
use_dict: Optional[bool] = None,
113121
) -> PropFactory: ...
114122

115123
@staticmethod
@@ -118,6 +126,7 @@ def value(
118126
/,
119127
*,
120128
backed_by: Optional[str] = None,
129+
use_dict: Optional[bool] = None,
121130
) -> PropFactory | Prop[T]:
122131
"""
123132
An alias for :func:`cached_property`.
@@ -126,7 +135,9 @@ def value(
126135
aligned to the :meth:`Attr.validator<typed_descriptors.attr.Attr.validator>`
127136
decorator for attributes.
128137
"""
129-
return cached_property(value_fun, backed_by=backed_by)
138+
return cached_property(
139+
value_fun, backed_by=backed_by, use_dict=use_dict
140+
)
130141

131142
__value_fun: ValueFunction[T]
132143

@@ -138,6 +149,7 @@ def __init__(
138149
/,
139150
*,
140151
backed_by: Optional[str] = None,
152+
use_dict: Optional[bool] = None,
141153
) -> None:
142154
# pylint: disable = redefined-builtin
143155
...
@@ -150,6 +162,7 @@ def __init__(
150162
/,
151163
*,
152164
backed_by: Optional[str] = None,
165+
use_dict: Optional[bool] = None,
153166
) -> None:
154167
# pylint: disable = redefined-builtin
155168
...
@@ -161,6 +174,7 @@ def __init__(
161174
/,
162175
*,
163176
backed_by: Optional[str] = None,
177+
use_dict: Optional[bool] = None,
164178
) -> None:
165179
"""
166180
Creates a new property with the given type and value function.
@@ -177,7 +191,7 @@ def __init__(
177191
"""
178192
# pylint: disable = redefined-builtin
179193
validate_value_fun(value)
180-
super().__init__(type, backed_by=backed_by)
194+
super().__init__(type, backed_by=backed_by, use_dict=use_dict)
181195
if not callable(value):
182196
raise TypeError(f"Expected callable 'value', got {value!r}.")
183197
self.__value_fun = value
@@ -310,13 +324,21 @@ def __repr__(self) -> str:
310324

311325
@overload
312326
def cached_property(
313-
value_fun: ValueFunction[T], /, *, backed_by: Optional[str] = None
327+
value_fun: ValueFunction[T],
328+
/,
329+
*,
330+
backed_by: Optional[str] = None,
331+
use_dict: Optional[bool] = None,
314332
) -> Prop[T]: ...
315333

316334

317335
@overload
318336
def cached_property(
319-
value_fun: None = None, /, *, backed_by: Optional[str] = None
337+
value_fun: None = None,
338+
/,
339+
*,
340+
backed_by: Optional[str] = None,
341+
use_dict: Optional[bool] = None,
320342
) -> PropFactory: ...
321343

322344

@@ -325,6 +347,7 @@ def cached_property(
325347
/,
326348
*,
327349
backed_by: Optional[str] = None,
350+
use_dict: Optional[bool] = None,
328351
) -> PropFactory | Prop[T]:
329352
"""
330353
Decorator used to create a cached property from a value function,
@@ -368,9 +391,13 @@ def x(self) -> Sequence[str]:
368391
"""
369392
if value_fun is not None:
370393
prop_type = value_fun_return_type(value_fun)
371-
return Prop(prop_type, value_fun, backed_by=backed_by)
394+
return Prop(
395+
prop_type, value_fun, backed_by=backed_by, use_dict=use_dict
396+
)
372397

373398
def _cached_property(value_fun: ValueFunction[_T]) -> Prop[_T]:
374-
return cached_property(value_fun, backed_by=backed_by)
399+
return cached_property(
400+
value_fun, backed_by=backed_by, use_dict=use_dict
401+
)
375402

376403
return _cached_property

0 commit comments

Comments
 (0)