Skip to content

Commit e9ac17b

Browse files
authored
Handle bare ClassVar type qualifiers, rename INFERRED sentinel (#26)
While currently not explicitly allowed by the typing specification, `ClassVar` is allowed as a bare type qualifier. Unlike `Final`, the actual type doesn't have to be inferred from the assignment (e.g. one can use `Any`). For this reason, the `INFERRED` sentinel was renamed to `UNKNOWN`.
1 parent ee1f9ad commit e9ac17b

File tree

4 files changed

+73
-34
lines changed

4 files changed

+73
-34
lines changed

docs/api/introspection.md

+1-2
Original file line numberDiff line numberDiff line change
@@ -9,6 +9,5 @@
99
- AnnotationSource
1010
- Qualifier
1111
- InspectedAnnotation
12-
- INFERRED
13-
- InferredType
12+
- UNKNOWN
1413
- ForbiddenQualifier

docs/usage.md

+24-12
Original file line numberDiff line numberDiff line change
@@ -98,28 +98,40 @@ If you want to allow all of them, use the [`AnnotationSource.ANY`][typing_inspec
9898
source.
9999

100100
The result of the [`inspect_annotation()`][typing_inspection.introspection.inspect_annotation] function contains the underlying
101-
[type expression][], the qualifiers and the annotated metadata. Note that some qualifiers are allowed to be used without any
102-
type expression. In this case, the type should be inferred from the assigned value:
101+
[type expression][], the qualifiers and the annotated metadata.
102+
103+
#### Handling bare type qualifiers
104+
105+
Note that some qualifiers are allowed to be used without any
106+
type expression. In this case, the [`InspectedAnnotation.type`][typing_inspection.introspection.InspectedAnnotation.type] attribute
107+
will take the value of the [`UNKNOWN`][typing_inspection.introspection.UNKNOWN] sentinel.
108+
109+
Depending on the type qualifier that was used, you can infer the actual type in different ways:
103110

104111
```python
112+
from typing import get_type_hints
113+
114+
from typing_inspection.introspection import UNKNOWN, AnnotationSource, inspect_annotation
115+
116+
105117
class A:
106-
x: Annotated[Final, 'meta'] = 1 # type of x should be inferred to `int`.
107-
```
118+
# For `Final` annotations, the type should be inferred from the assignment
119+
# (and you may error if no assignment is available).
120+
# In this case, you can infer to either `int` or `Literal[1]`:
121+
x: Annotated[Final, 'meta'] = 1
108122

109-
In this case, the [`InspectedAnnotation.type`][typing_inspection.introspection.InspectedAnnotation.type] attribute is set
110-
to the [`INFERRED`][typing_inspection.introspection.INFERRED] sentinel value, so you should check for this sentinel value
111-
before doing any processing on the type:
123+
# For `ClassVar` annotations, the type can be inferred as `Any`,
124+
# or from the assignment if available (both options are valid in all cases):
125+
y: ClassVar
112126

113-
```python
114-
from typing_inspection.introspection import INFERRED, AnnotationSource, inspect_annotation
115127

116128
inspected_annotation = inspect_annotation(
117-
Annotated[Final, 'meta'],
129+
get_type_hints(A)['x'],
118130
annotation_source=AnnotationSource.CLASS,
119131
)
120132

121-
if inspected_annotation.type is INFERRED:
122-
ann_type = type(assigned_value) # assigned_value would come from the class
133+
if inspected_annotation.type is UNKNOWN:
134+
ann_type = type(A.x)
123135
else:
124136
ann_type = inspected_annotation.type
125137
```

src/typing_inspection/introspection.py

+23-17
Original file line numberDiff line numberDiff line change
@@ -277,33 +277,37 @@ def __init__(self, qualifier: Qualifier, /) -> None:
277277
self.qualifier = qualifier
278278

279279

280-
class _InferredTypeEnum(Enum):
281-
INFERRED = auto()
280+
class _UnknownTypeEnum(Enum):
281+
UNKNOWN = auto()
282+
283+
def __str__(self) -> str:
284+
return 'UNKNOWN'
282285

283286
def __repr__(self) -> str:
284-
return 'INFERRED'
287+
return '<UNKNOWN>'
285288

286289

287-
INFERRED = _InferredTypeEnum.INFERRED
288-
"""A sentinel value used when no [type expression][] is used, indicating
289-
that the type should be inferred from the assigned value.
290-
"""
290+
UNKNOWN = _UnknownTypeEnum.UNKNOWN
291+
"""A sentinel value used when no [type expression][] is present."""
291292

292-
InferredType: TypeAlias = Literal[_InferredTypeEnum.INFERRED]
293-
"""The type of the [`INFERRED`][typing_inspection.introspection.INFERRED] sentinel value."""
293+
_UnkownType: TypeAlias = Literal[_UnknownTypeEnum.UNKNOWN]
294+
"""The type of the [`UNKNOWN`][typing_inspection.introspection.UNKNOWN] sentinel value."""
294295

295296

296297
class InspectedAnnotation(NamedTuple):
297298
"""The result of the inspected annotation."""
298299

299-
type: Any | InferredType
300+
type: Any | _UnkownType
300301
"""The final [type expression][], with [type qualifiers][type qualifier] and annotated metadata stripped.
301302
302-
If no type expression is available, the [`INFERRED`][typing_inspection.introspection.INFERRED] sentinel
303+
If no type expression is available, the [`UNKNOWN`][typing_inspection.introspection.UNKNOWN] sentinel
303304
value is used instead. This is the case when a [type qualifier][] is used with no type annotation:
304305
305306
```python
306307
ID: Final = 1
308+
309+
class C:
310+
x: ClassVar = 'test'
307311
```
308312
"""
309313

@@ -413,16 +417,18 @@ def inspect_annotation(
413417
else:
414418
break
415419

416-
# `Final` is the only type qualifier allowed to be used as a bare annotation
417-
# (`ClassVar` is under discussion: https://discuss.python.org/t/81705):
420+
# `Final` and `ClassVar` are type qualifiers allowed to be used as a bare annotation
421+
# (`ClassVar` is not explicitly specified, but will be: https://discuss.python.org/t/81705).
418422
if typing_objects.is_final(annotation):
419423
if 'final' not in allowed_qualifiers:
420424
raise ForbiddenQualifier('final')
421425
qualifiers.add('final')
422-
# No type expression is available, the type should be inferred from the
423-
# assigned value.
424-
# See https://typing.readthedocs.io/en/latest/spec/qualifiers.html#syntax
425-
annotation = INFERRED
426+
annotation = UNKNOWN
427+
elif typing_objects.is_classvar(annotation):
428+
if 'class_var' not in allowed_qualifiers:
429+
raise ForbiddenQualifier('class_var')
430+
qualifiers.add('class_var')
431+
annotation = UNKNOWN
426432

427433
return InspectedAnnotation(annotation, qualifiers, metadata)
428434

tests/introspection/test_inspect_annotation.py

+25-3
Original file line numberDiff line numberDiff line change
@@ -6,7 +6,13 @@
66
import pytest
77
import typing_extensions as t_e
88

9-
from typing_inspection.introspection import INFERRED, AnnotationSource, ForbiddenQualifier, inspect_annotation
9+
from typing_inspection.introspection import UNKNOWN, AnnotationSource, ForbiddenQualifier, inspect_annotation
10+
11+
12+
def test_unknown_repr() -> None:
13+
assert str(UNKNOWN) == 'UNKNOWN'
14+
assert repr(UNKNOWN) == '<UNKNOWN>'
15+
1016

1117
_all_qualifiers: list[Any] = [
1218
t_e.ClassVar[int],
@@ -51,14 +57,30 @@ def test_annotation_source_invalid_qualifiers(source: AnnotationSource, annotati
5157
inspect_annotation(annotation, annotation_source=source)
5258

5359

54-
def test_bare_final_qualifier() -> None:
60+
def test_final_bare_final_qualifier() -> None:
5561
result = inspect_annotation(
5662
t.Final,
5763
annotation_source=AnnotationSource.ANY,
5864
)
5965

6066
assert result.qualifiers == {'final'}
61-
assert result.type is INFERRED
67+
assert result.type is UNKNOWN
68+
69+
with pytest.raises(ForbiddenQualifier):
70+
inspect_annotation(t.Final, annotation_source=AnnotationSource.BARE)
71+
72+
73+
def test_class_var_bare_final_qualifier() -> None:
74+
result = inspect_annotation(
75+
t.ClassVar,
76+
annotation_source=AnnotationSource.ANY,
77+
)
78+
79+
assert result.qualifiers == {'class_var'}
80+
assert result.type is UNKNOWN
81+
82+
with pytest.raises(ForbiddenQualifier):
83+
inspect_annotation(t.ClassVar, annotation_source=AnnotationSource.BARE)
6284

6385

6486
def test_nested_metadata_and_qualifiers() -> None:

0 commit comments

Comments
 (0)