Skip to content

Commit 98094b7

Browse files
committed
More Inspector features. Introduced inspect_type function to replace can_validate in its TypeInspector-returning behaviour.
1 parent 9b2e8d4 commit 98094b7

File tree

4 files changed

+86
-16
lines changed

4 files changed

+86
-16
lines changed

docs/api/typing_validation.inspector.rst

+1
Original file line numberDiff line numberDiff line change
@@ -9,6 +9,7 @@ TypeInspector
99
.. autoclass:: typing_validation.inspector.TypeInspector
1010
:show-inheritance:
1111
:members:
12+
:special-members: __repr__
1213

1314
UnsupportedType
1415
---------------

docs/api/typing_validation.validation.rst

+5
Original file line numberDiff line numberDiff line change
@@ -20,6 +20,11 @@ can_validate
2020

2121
.. autofunction:: typing_validation.validation.can_validate
2222

23+
inspect_type
24+
------------
25+
26+
.. autofunction:: typing_validation.validation.inspect_type
27+
2328
is_valid
2429
--------
2530

typing_validation/inspector.py

+49-6
Original file line numberDiff line numberDiff line change
@@ -44,6 +44,11 @@
4444
else:
4545
TypeConstructorArgs = typing.Tuple[str, Any]
4646

47+
if sys.version_info[1] >= 11:
48+
from typing import Self
49+
else:
50+
from typing_extensions import Self
51+
4752
_typing_equiv = {
4853
list: typing.List,
4954
tuple: typing.Tuple,
@@ -113,7 +118,7 @@ class TypeInspector:
113118
"_pending_generic_type_constr",
114119
)
115120

116-
def __new__(cls) -> "TypeInspector":
121+
def __new__(cls) -> Self:
117122
instance = super().__new__(cls)
118123
instance._recorded_constructors = []
119124
instance._unsupported_types = []
@@ -134,6 +139,34 @@ def unsupported_types(self) -> typing.Tuple[Any, ...]:
134139
r"""The sequence of unsupported types encountered during validation."""
135140
return tuple(self._unsupported_types)
136141

142+
@property
143+
def type_structure(self) -> str:
144+
"""
145+
The structure of the recorded type:
146+
147+
1. The string spans multiple lines, with indentation levels matching
148+
the nesting level of inner types.
149+
2. Any unsupported types encountered are wrapped using the generic type
150+
:obj:`UnsupportedType`.
151+
152+
"""
153+
return "\n".join(self._repr()[0])
154+
155+
@property
156+
def type_annotation(self) -> str:
157+
"""
158+
The type annotation for the recorded type.
159+
Differs from the output of :attr:`type_structure` in the following ways:
160+
161+
1. The annotation is on a single line.
162+
2. Unsupported types are not wrapped.
163+
164+
"""
165+
return "".join(
166+
line.strip()
167+
for line in self._repr(mark_unsupported=False)[0]
168+
)
169+
137170
def _recorded_type(self, idx: int) -> typing.Tuple[Any, int]:
138171
# pylint: disable = too-many-return-statements, too-many-branches
139172
param: Any
@@ -297,15 +330,23 @@ def __bool__(self) -> bool:
297330
return not self._unsupported_types
298331

299332
def __repr__(self) -> str:
300-
# addr = "0x"+f"{id(self):x}"
301-
header = f"The following type can{'' if self else 'not'} be validated against:"
302-
return header + "\n" + "\n".join(self._repr()[0])
333+
"""
334+
Representation of the inspector, including the :attr:`type_structure`.
335+
336+
:meta public:
337+
"""
338+
return (
339+
"TypeInspector instance for the following type:\n"
340+
+self.type_structure
341+
)
303342

304343
def _repr(
305-
self, idx: int = 0, level: int = 0
344+
self, idx: int = 0, level: int = 0,
345+
*,
346+
mark_unsupported: bool = True
306347
) -> typing.Tuple[typing.List[str], int]:
307348
# pylint: disable = too-many-return-statements, too-many-branches, too-many-statements, too-many-locals
308-
basic_indent = " "
349+
basic_indent = " "
309350
assert len(basic_indent) >= 2
310351
indent = basic_indent * level
311352
next_indent = basic_indent * (level + 1)
@@ -314,6 +355,8 @@ def _repr(
314355
lines: typing.List[str]
315356
tag, param = self._recorded_constructors[idx]
316357
if tag == "unsupported":
358+
if not mark_unsupported:
359+
return [indent+str(param)], idx
317360
return [
318361
indent + "UnsupportedType[",
319362
indent + " " + str(param),

typing_validation/validation.py

+31-10
Original file line numberDiff line numberDiff line change
@@ -904,29 +904,50 @@ def validate(val: Any, t: Any) -> Literal[True]:
904904

905905

906906
def can_validate(t: Any) -> TypeInspector:
907-
r"""
908-
Checks whether validation is supported for the given type ``t``: if not, :func:`validate` will raise :obj:`UnsupportedTypeError`.
907+
"""
908+
Checks whether validation is supported for the given type ``t``: if not,
909+
:func:`validate` will raise :obj:`UnsupportedTypeError`.
909910
910-
The returned :class:`TypeInspector` instance can be used wherever a boolean is expected, and will indicate whether the type is supported or not:
911+
.. warning::
912+
913+
The return type will be changed to :obj:`bool` in v1.3.0.
914+
To obtain a :class:`TypeInspector` object, please use the newly
915+
introduced :func:`inspect_type` instead.
916+
917+
:param t: the type to be checked for validation support
918+
:type t: :obj:`~typing.Any`
919+
920+
"""
921+
inspector = TypeInspector()
922+
validate(inspector, t)
923+
return inspector
924+
925+
926+
def inspect_type(t: Any) -> TypeInspector:
927+
r"""
928+
Returns a :class:`TypeInspector` instance can be used wherever a boolean is
929+
expected, and will indicate whether the type is supported or not:
911930
912931
>>> from typing import *
913-
>>> from typing_validation import can_validate
914-
>>> res = can_validate(tuple[list[str], Union[int, float, Callable[[int], int]]])
932+
>>> from typing_validation import inspect_type
933+
>>> res = inspect_type(tuple[list[str], Union[int, float, Callable[[int], int]]])
915934
>>> bool(res)
916935
False
917936
918-
However, it also records (with minimal added cost) the full structure of the type as the latter was validated, which it then exposes via its
937+
The instance also records (with minimal added cost) the full structure of
938+
the type as the latter was validated, which it then exposes via its
919939
:attr:`TypeInspector.recorded_type` property:
920940
921-
>>> res = can_validate(tuple[list[Union[str, int]],...])
941+
>>> res = inspect_type(tuple[list[Union[str, int]],...])
922942
>>> bool(res)
923943
True
924944
>>> res.recorded_type
925945
tuple[list[typing.Union[str, int]], ...]
926946
927-
Any unsupported subtype encountered during the validation is left in place, wrapped into an :class:`UnsupportedType`:
947+
Any unsupported subtype encountered during the validation is left in place,
948+
wrapped into an :class:`UnsupportedType`:
928949
929-
>>> can_validate(tuple[list[str], Union[int, float, Callable[[int], int]]])
950+
>>> inspect_type(tuple[list[str], Union[int, float, Callable[[int], int]]])
930951
The following type cannot be validated against:
931952
tuple[
932953
list[
@@ -945,7 +966,7 @@ def can_validate(t: Any) -> TypeInspector:
945966
946967
:param t: the type to be checked for validation support
947968
:type t: :obj:`~typing.Any`
948-
:raises AssertionError: if things go unexpectedly wrong with ``__args__`` for parametric types
969+
949970
"""
950971
inspector = TypeInspector()
951972
validate(inspector, t)

0 commit comments

Comments
 (0)