19
19
import openapi_test_client .libraries .api .api_functions .utils .param_model as param_model_util
20
20
from openapi_test_client .libraries .api .types import Alias , Constraint , Format , ParamAnnotationType , ParamDef
21
21
from openapi_test_client .libraries .common .constants import BACKSLASH
22
+ from openapi_test_client .libraries .common .misc import dedup
22
23
23
24
logger = get_logger (__name__ )
24
25
@@ -37,6 +38,8 @@ def get_type_annotation_as_str(tp: Any) -> str:
37
38
"""
38
39
if isinstance (tp , str ):
39
40
return repr (tp )
41
+ elif tp is Ellipsis :
42
+ return "..."
40
43
elif get_origin (tp ) is Annotated :
41
44
orig_type = get_type_annotation_as_str (tp .__origin__ )
42
45
metadata_types = ", " .join (get_type_annotation_as_str (m ) for m in tp .__metadata__ )
@@ -68,7 +71,7 @@ def get_type_annotation_as_str(tp: Any) -> str:
68
71
if v is not None
69
72
)
70
73
return f"{ type (tp ).__name__ } ({ const } )"
71
- elif tp is NoneType :
74
+ elif tp in [ NoneType , None ] :
72
75
return "None"
73
76
else :
74
77
if inspect .isclass (tp ):
@@ -192,8 +195,8 @@ def resolve(param_type: str, param_format: str | None = None):
192
195
return type_annotation
193
196
194
197
195
- def get_inner_type (tp : Any , return_if_container_type : bool = False ) -> Any | list [Any ]:
196
- """Get the inner type (=actual type) from the type annotation
198
+ def get_base_type (tp : Any , return_if_container_type : bool = False ) -> Any | list [Any ]:
199
+ """Get the base type from the type annotation
197
200
198
201
eg:
199
202
Optional[str] -> str
@@ -214,26 +217,26 @@ def get_inner_type(tp: Any, return_if_container_type: bool = False) -> Any | lis
214
217
215
218
if is_union_type (tp ):
216
219
args_without_nonetype = [x for x in get_args (tp ) if x is not NoneType ]
217
- return generate_union_type ([get_inner_type (x ) for x in args_without_nonetype ])
220
+ return generate_union_type ([get_base_type (x ) for x in args_without_nonetype ])
218
221
elif origin_type is Annotated :
219
- return get_inner_type (tp .__origin__ )
222
+ return get_base_type (tp .__origin__ )
220
223
elif origin_type is list :
221
224
if return_if_container_type :
222
225
return tp
223
226
else :
224
- return get_inner_type (get_args (tp )[0 ])
227
+ return get_base_type (get_args (tp )[0 ])
225
228
return tp
226
229
227
230
228
- def replace_inner_type (tp : Any , new_type : Any , replace_container_type : bool = False ) -> Any :
229
- """Replace an inner type of in the type annotation to something else
231
+ def replace_base_type (tp : Any , new_type : Any , replace_container_type : bool = False ) -> Any :
232
+ """Replace the base type of the type annotation to something else
230
233
231
234
:param tp: The original type annotation
232
- :param new_type: A new type to replace the inner type with
233
- :param replace_container_type: Treat container types like list and tuple as an inner type
235
+ :param new_type: A new type to replace the base type with
236
+ :param replace_container_type: Treat container types like list and tuple as an base type
234
237
235
238
>>> tp = Optional[Annotated[int, "metadata"]]
236
- >>> new_tp = replace_inner_type (tp, str)
239
+ >>> new_tp = replace_base_type (tp, str)
237
240
>>> print(new_tp)
238
241
typing.Optional[typing.Annotated[str, 'metadata']]
239
242
"""
@@ -242,16 +245,16 @@ def replace_inner_type(tp: Any, new_type: Any, replace_container_type: bool = Fa
242
245
args = get_args (tp )
243
246
if is_union_type (tp ):
244
247
if is_optional_type (tp ):
245
- return Optional [replace_inner_type (args [0 ], new_type )] # noqa: UP007
248
+ return Optional [replace_base_type (args [0 ], new_type )] # noqa: UP007
246
249
else :
247
- return replace_inner_type (args , new_type )
250
+ return replace_base_type (args , new_type )
248
251
elif origin_type is Annotated :
249
- return Annotated [replace_inner_type (tp .__origin__ , new_type ), * tp .__metadata__ ]
252
+ return Annotated [replace_base_type (tp .__origin__ , new_type ), * tp .__metadata__ ]
250
253
elif origin_type in [list , tuple ]:
251
254
if replace_container_type :
252
255
return new_type
253
256
else :
254
- return origin_type [replace_inner_type (args , new_type )]
257
+ return origin_type [replace_base_type (args , new_type )]
255
258
else :
256
259
return new_type
257
260
else :
@@ -358,16 +361,22 @@ def generate_optional_type(tp: Any) -> Any:
358
361
return Union [tp , None ] # noqa: UP007
359
362
360
363
361
- def generate_annotated_type (tp : Any , metadata : Any ):
362
- """Add `Annotated` type to the type annotation with the metadata
364
+ def generate_annotated_type (tp : Any , * metadata : Any ):
365
+ """Add `Annotated` type to the type annotation with the metadata.
366
+
367
+ If the given type is already annotated, the specified metadata will be merged into the existing annotated type
363
368
364
369
:param tp: Type annotation
370
+ :param metadata: Metadata to add to Annotated[]
365
371
"""
366
- if is_optional_type (tp ):
372
+ if get_origin (tp ) is Annotated :
373
+ # merge metadata
374
+ return generate_annotated_type (get_args (tp )[0 ], * (dedup (* tp .__metadata__ , * metadata )))
375
+ elif is_optional_type (tp ):
367
376
inner_type = get_args (tp )[0 ]
368
- return Optional [Annotated [ inner_type , metadata ] ] # noqa: UP007
377
+ return Optional [generate_annotated_type ( inner_type , * metadata ) ] # noqa: UP007
369
378
else :
370
- return Annotated [tp , metadata ]
379
+ return Annotated [tp , * metadata ]
371
380
372
381
373
382
def get_annotated_type (tp : Any ) -> _AnnotatedAlias | None :
@@ -392,16 +401,6 @@ def merge_annotation_types(tp1: Any, tp2: Any) -> Any:
392
401
Note: This is still experimental
393
402
"""
394
403
395
- def dedup (* args : Any ) -> tuple [Any , ...]:
396
- """Deduplicate items by retaining the order"""
397
- seen = set ()
398
- deduped_args = []
399
- for arg in args :
400
- if arg not in seen :
401
- deduped_args .append (arg )
402
- seen .add (arg )
403
- return tuple (deduped_args )
404
-
405
404
def merge_args_per_origin (args : Sequence [Any ]) -> tuple [Any , ...]:
406
405
"""Merge type annotations per its origiin type"""
407
406
origin_type_order = {Literal : 1 , Annotated : 2 , Union : 3 , UnionType : 4 , list : 5 , dict : 6 , None : 10 }
0 commit comments