44
44
else :
45
45
TypeConstructorArgs = typing .Tuple [str , Any ]
46
46
47
+ if sys .version_info [1 ] >= 11 :
48
+ from typing import Self
49
+ else :
50
+ from typing_extensions import Self
51
+
47
52
_typing_equiv = {
48
53
list : typing .List ,
49
54
tuple : typing .Tuple ,
@@ -113,7 +118,7 @@ class TypeInspector:
113
118
"_pending_generic_type_constr" ,
114
119
)
115
120
116
- def __new__ (cls ) -> "TypeInspector" :
121
+ def __new__ (cls ) -> Self :
117
122
instance = super ().__new__ (cls )
118
123
instance ._recorded_constructors = []
119
124
instance ._unsupported_types = []
@@ -134,6 +139,34 @@ def unsupported_types(self) -> typing.Tuple[Any, ...]:
134
139
r"""The sequence of unsupported types encountered during validation."""
135
140
return tuple (self ._unsupported_types )
136
141
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
+
137
170
def _recorded_type (self , idx : int ) -> typing .Tuple [Any , int ]:
138
171
# pylint: disable = too-many-return-statements, too-many-branches
139
172
param : Any
@@ -297,15 +330,23 @@ def __bool__(self) -> bool:
297
330
return not self ._unsupported_types
298
331
299
332
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
+ )
303
342
304
343
def _repr (
305
- self , idx : int = 0 , level : int = 0
344
+ self , idx : int = 0 , level : int = 0 ,
345
+ * ,
346
+ mark_unsupported : bool = True
306
347
) -> typing .Tuple [typing .List [str ], int ]:
307
348
# pylint: disable = too-many-return-statements, too-many-branches, too-many-statements, too-many-locals
308
- basic_indent = " "
349
+ basic_indent = " "
309
350
assert len (basic_indent ) >= 2
310
351
indent = basic_indent * level
311
352
next_indent = basic_indent * (level + 1 )
@@ -314,6 +355,8 @@ def _repr(
314
355
lines : typing .List [str ]
315
356
tag , param = self ._recorded_constructors [idx ]
316
357
if tag == "unsupported" :
358
+ if not mark_unsupported :
359
+ return [indent + str (param )], idx
317
360
return [
318
361
indent + "UnsupportedType[" ,
319
362
indent + " " + str (param ),
0 commit comments