@@ -40,7 +40,21 @@ def create_endpoint_model(endpoint_func: EndpointFunc, api_spec: dict[str, Any]
40
40
if parameter_objects := operation_obj .get ("parameters" ):
41
41
_parse_parameter_objects (method , parameter_objects , path_param_fields , body_or_query_param_fields )
42
42
if request_body := operation_obj .get ("requestBody" ):
43
- content_type = _parse_request_body_object (request_body , body_or_query_param_fields )
43
+ try :
44
+ content_type = _parse_request_body_object (request_body , body_or_query_param_fields )
45
+ except Exception :
46
+ logger .warning (f"Unable to parse the following requestBody obj:\n { request_body } " )
47
+ raise
48
+
49
+ expected_path_params = re .findall (r"{([^}]+)}" , path )
50
+ documented_path_params = [x [0 ] for x in path_param_fields ]
51
+ if undocumented_path_params := [x for x in expected_path_params if x not in documented_path_params ]:
52
+ logger .warning (f"{ method .upper ()} { path } : Found undocumented path parameters: { undocumented_path_params } " )
53
+ # Some OpenAPI specs don't properly document path parameters at all, or path parameters could be documented
54
+ # as incorrect "in" like "query". We fix this by adding the missing path parameters, and remove them from
55
+ # body/query params if any
56
+ path_param_fields = [DataclassModelField (x , str ) for x in expected_path_params ]
57
+ body_or_query_param_fields = [x for x in body_or_query_param_fields if x [0 ] not in expected_path_params ]
44
58
else :
45
59
# Generate model fields from the function signature
46
60
sig = inspect .signature (endpoint_func ._original_func )
@@ -54,19 +68,6 @@ def create_endpoint_model(endpoint_func: EndpointFunc, api_spec: dict[str, Any]
54
68
# keyword arguments (body/query parameters)
55
69
_add_body_or_query_param_field (body_or_query_param_fields , name , param_obj .annotation )
56
70
57
- if hasattr (endpoint_func , "endpoint" ):
58
- method = endpoint_func .endpoint .method
59
- path = endpoint_func .endpoint .path
60
- expected_path_params = re .findall (r"{([^}]+)}" , path )
61
- documented_path_params = [x [0 ] for x in path_param_fields ]
62
- if undocumented_path_params := [x for x in expected_path_params if x not in documented_path_params ]:
63
- logger .warning (f"{ method .upper ()} { path } : Found undocumented path parameters: { undocumented_path_params } " )
64
- # Some OpenAPI specs don't properly document path parameters at all, or path parameters could be documented
65
- # as incorrect "in" like "query". We fix this by adding the missing path parameters, and remove them from
66
- # body/query params if any
67
- path_param_fields = [DataclassModelField (x , str ) for x in expected_path_params ]
68
- body_or_query_param_fields = [x for x in body_or_query_param_fields if x [0 ] not in expected_path_params ]
69
-
70
71
# Address the case where a path param name conflicts with body/query param name
71
72
for i , (field_name , field_type , _ ) in enumerate (path_param_fields ):
72
73
if field_name in [x [0 ] for x in body_or_query_param_fields ]:
@@ -75,8 +76,9 @@ def create_endpoint_model(endpoint_func: EndpointFunc, api_spec: dict[str, Any]
75
76
# Some OpenAPI specs define a parameter name using characters we can't use as a python variable name.
76
77
# We will use the cleaned name as the model field and annotate it as `Annotated[field_type, Alias(<original_val>)]`
77
78
# When calling an endpoint function, the actual name will be automatically resolved in the payload/query parameters
78
- param_model_util .alias_illegal_model_field_names (model_name , path_param_fields )
79
- param_model_util .alias_illegal_model_field_names (model_name , body_or_query_param_fields )
79
+ endpoint = f"{ endpoint_func .method .upper ()} { endpoint_func .path } "
80
+ param_model_util .alias_illegal_model_field_names (endpoint , path_param_fields )
81
+ param_model_util .alias_illegal_model_field_names (endpoint , body_or_query_param_fields )
80
82
81
83
fields = path_param_fields + body_or_query_param_fields
82
84
return cast (
@@ -211,7 +213,7 @@ def _parse_request_body_object(
211
213
# TODO: Support multiple content types
212
214
content_type = next (iter (contents ))
213
215
214
- def parse_schema_obj (obj : dict [str , Any ]) -> list [dict [str , Any ]] | None :
216
+ def parse_schema_obj (obj : dict [str , Any ]) -> list [dict [str , Any ] | list [ dict [ str , Any ]] ] | None :
215
217
# This part has some variations, and sometimes not consistent
216
218
if not (properties := obj .get ("properties" , {})):
217
219
schema_type = obj .get ("type" )
@@ -235,8 +237,12 @@ def parse_schema_obj(obj: dict[str, Any]) -> list[dict[str, Any]] | None:
235
237
):
236
238
# The API directly takes data that is not form-encoded (eg. send tar binary data)
237
239
properties = {"data" : schema_obj }
240
+ elif obj == {}:
241
+ # Empty schema
242
+ properties = {}
238
243
else :
239
- raise NotImplementedError (f"Unsupported request body:\n { json .dumps (obj , indent = 4 , default = str )} " )
244
+ # An example from actual OpenAPI spec: {"content": {"application/json": {"schema": {"type": 'string"}}
245
+ raise NotImplementedError (f"Unsupported schema obj:\n { json .dumps (obj , indent = 4 , default = str )} " )
240
246
241
247
for param_name in properties :
242
248
param_obj = properties [param_name ]
0 commit comments