5
5
import sys
6
6
import types
7
7
from collections .abc import Generator , Sequence
8
+ from dataclasses import InitVar
8
9
from enum import Enum , IntEnum , auto
9
- from typing import Any , Literal , NamedTuple
10
+ from typing import Any , Literal , NamedTuple , cast
10
11
11
- from typing_extensions import TypeAlias , assert_never , get_origin
12
+ from typing_extensions import TypeAlias , assert_never , get_args , get_origin
12
13
13
14
from . import typing_objects
14
15
@@ -172,9 +173,11 @@ def get_literal_values(
172
173
yield from (p for p , _ in dct )
173
174
174
175
175
- Qualifier : TypeAlias = Literal ['required' , 'not_required' , 'read_only' , 'class_var' , 'final' ]
176
+ Qualifier : TypeAlias = Literal ['required' , 'not_required' , 'read_only' , 'class_var' , 'init_var' , ' final' ]
176
177
"""A [type qualifier][]."""
177
178
179
+ _all_qualifiers : set [Qualifier ] = set (get_args (Qualifier ))
180
+
178
181
179
182
# TODO at some point, we could switch to an enum flag, so that multiple sources
180
183
# can be combined. However, is there a need for this?
@@ -187,7 +190,7 @@ class AnnotationSource(IntEnum):
187
190
Depending on the source, different [type qualifiers][type qualifier] may be (dis)allowed.
188
191
"""
189
192
190
- ASSIGNMENT_OR_VARIABLE = 1
193
+ ASSIGNMENT_OR_VARIABLE = auto ()
191
194
"""An annotation used in an assignment or variable annotation:
192
195
193
196
```python
@@ -198,7 +201,7 @@ class AnnotationSource(IntEnum):
198
201
**Allowed type qualifiers:** [`Final`][typing.Final].
199
202
"""
200
203
201
- CLASS = 2
204
+ CLASS = auto ()
202
205
"""An annotation used in the body of a class:
203
206
204
207
```python
@@ -210,7 +213,20 @@ class Test:
210
213
**Allowed type qualifiers:** [`ClassVar`][typing.ClassVar], [`Final`][typing.Final].
211
214
"""
212
215
213
- TYPED_DICT = 3
216
+ DATACLASS = auto ()
217
+ """An annotation used in the body of a dataclass:
218
+
219
+ ```python
220
+ @dataclass
221
+ class Test:
222
+ x: Final[int] = 1
223
+ y: InitVar[str] = 'test'
224
+ ```
225
+
226
+ **Allowed type qualifiers:** [`ClassVar`][typing.ClassVar], [`Final`][typing.Final], [`InitVar`][dataclasses-init-only-variables].
227
+ """ # noqa: E501
228
+
229
+ TYPED_DICT = auto ()
214
230
"""An annotation used in the body of a [`TypedDict`][typing.TypedDict]:
215
231
216
232
```python
@@ -223,7 +239,7 @@ class TD(TypedDict):
223
239
[`NotRequired`][typing.NotRequired].
224
240
"""
225
241
226
- NAMED_TUPLE = 4
242
+ NAMED_TUPLE = auto ()
227
243
"""An annotation used in the body of a [`NamedTuple`][typing.NamedTuple].
228
244
229
245
```python
@@ -235,7 +251,7 @@ class NT(NamedTuple):
235
251
**Allowed type qualifiers:** none.
236
252
"""
237
253
238
- FUNCTION = 5
254
+ FUNCTION = auto ()
239
255
"""An annotation used in a function, either for a parameter or the return value.
240
256
241
257
```python
@@ -246,13 +262,13 @@ def func(a: int) -> str:
246
262
**Allowed type qualifiers:** none.
247
263
"""
248
264
249
- ANY = 6
265
+ ANY = auto ()
250
266
"""An annotation that might come from any source.
251
267
252
268
**Allowed type qualifiers:** all.
253
269
"""
254
270
255
- BARE = 7
271
+ BARE = auto ()
256
272
"""An annotation that is inspected as is.
257
273
258
274
**Allowed type qualifiers:** none.
@@ -266,12 +282,14 @@ def allowed_qualifiers(self) -> set[Qualifier]:
266
282
return {'final' }
267
283
elif self is AnnotationSource .CLASS :
268
284
return {'final' , 'class_var' }
285
+ elif self is AnnotationSource .DATACLASS :
286
+ return {'final' , 'class_var' , 'init_var' }
269
287
elif self is AnnotationSource .TYPED_DICT :
270
288
return {'required' , 'not_required' , 'read_only' }
271
289
elif self in (AnnotationSource .NAMED_TUPLE , AnnotationSource .FUNCTION , AnnotationSource .BARE ):
272
290
return set ()
273
291
elif self is AnnotationSource .ANY :
274
- return { 'required' , 'not_required' , 'read_only' , 'class_var' , 'final' }
292
+ return _all_qualifiers
275
293
else : # pragma: no cover
276
294
assert_never (self )
277
295
@@ -327,7 +345,7 @@ class C:
327
345
"""The annotated metadata."""
328
346
329
347
330
- def inspect_annotation (
348
+ def inspect_annotation ( # noqa: PLR0915
331
349
annotation : Any ,
332
350
/ ,
333
351
* ,
@@ -423,11 +441,15 @@ def inspect_annotation(
423
441
else :
424
442
# origin is not None but not a type qualifier nor `Annotated` (e.g. `list[int]`):
425
443
break
444
+ elif isinstance (annotation , InitVar ):
445
+ if 'init_var' not in allowed_qualifiers :
446
+ raise ForbiddenQualifier ('init_var' )
447
+ qualifiers .add ('init_var' )
448
+ annotation = cast (Any , annotation .type )
426
449
else :
427
450
break
428
451
429
- # `Final` and `ClassVar` are type qualifiers allowed to be used as a bare annotation
430
- # (`ClassVar` is not explicitly specified, but will be: https://discuss.python.org/t/81705).
452
+ # `Final`, `ClassVar` and `InitVar` are type qualifiers allowed to be used as a bare annotation:
431
453
if typing_objects .is_final (annotation ):
432
454
if 'final' not in allowed_qualifiers :
433
455
raise ForbiddenQualifier ('final' )
@@ -438,6 +460,11 @@ def inspect_annotation(
438
460
raise ForbiddenQualifier ('class_var' )
439
461
qualifiers .add ('class_var' )
440
462
annotation = UNKNOWN
463
+ elif annotation is InitVar :
464
+ if 'init_var' not in allowed_qualifiers :
465
+ raise ForbiddenQualifier ('init_var' )
466
+ qualifiers .add ('init_var' )
467
+ annotation = UNKNOWN
441
468
442
469
return InspectedAnnotation (annotation , qualifiers , metadata )
443
470
0 commit comments