Skip to content

Commit

Permalink
Improve JSON req error on disallowed empty body (#1761)
Browse files Browse the repository at this point in the history
Fixes #1152.

Currently, when a request body is empty, the JSON request validator
would parse it into None, which is later passed down to the JSON Schema
validator. However, jsonschema's validation error message for this case
(when nullable is false) "None is not of type 'object'" is not
particularly friendly to either the API user, nor the website developer.

This change adds a specific check before the None value is passed to
jsonschema to emit a better error message directly.

I also added some drive-by improvements on function argument typing
since _validate in validators don't seem to expect receiving None, but
_parse (the result of which is passed to _validate) is totally allowed
to return None (or anything really). This does not seem to reflect the
logic well.

I’m not exactly sure if this is the best way to do this in Connexion
3.x. [We do have a patch in Connexion 2.x to achieve a similar
effect](apache/airflow@e89a7ee)
so if anyone understands how the two implementations correspond please
tell me whether the two do the same thing 🙂

---------

Co-authored-by: Robbe Sneyders <robbe.sneyders@ml6.eu>
  • Loading branch information
uranusjr and RobbeSneyders committed Nov 1, 2023
1 parent e7dc56c commit 9c02fdf
Show file tree
Hide file tree
Showing 3 changed files with 15 additions and 5 deletions.
9 changes: 5 additions & 4 deletions connexion/validators/form_data.py
Original file line number Diff line number Diff line change
Expand Up @@ -70,12 +70,13 @@ async def _parse(self, stream: t.AsyncGenerator[bytes, None], scope: Scope) -> d

return data

def _validate(self, data: dict) -> None:
def _validate(self, body: t.Any) -> t.Optional[dict]: # type: ignore[return]
if not isinstance(body, dict):
raise BadRequestProblem("Parsed body must be a mapping")
if self._strict_validation:
self._validate_params_strictly(data)

self._validate_params_strictly(body)
try:
self._validator.validate(data)
self._validator.validate(body)
except ValidationError as exception:
error_path_msg = format_error_with_path(exception=exception)
logger.error(
Expand Down
4 changes: 3 additions & 1 deletion connexion/validators/json.py
Original file line number Diff line number Diff line change
Expand Up @@ -61,7 +61,9 @@ async def _parse(
except json.decoder.JSONDecodeError as e:
raise BadRequestProblem(detail=str(e))

def _validate(self, body: dict) -> None:
def _validate(self, body: t.Any) -> t.Optional[dict]:
if not self._nullable and body is None:
raise BadRequestProblem("Request body must not be empty")
try:
return self._validator.validate(body)
except ValidationError as exception:
Expand Down
7 changes: 7 additions & 0 deletions tests/api/test_errors.py
Original file line number Diff line number Diff line change
Expand Up @@ -88,3 +88,10 @@ def test_errors(problem_app):
"Invalid Content-type (text/html)"
)
assert unsupported_media_type_body["status"] == 415


def test_should_raise_400_for_no_json(simple_app):
app_client = simple_app.test_client()
response = app_client.post("/v1.0/test-empty-object-body")
assert response.status_code == 400
assert response.json()["detail"] == "Request body must not be empty"

0 comments on commit 9c02fdf

Please sign in to comment.