mirror of
https://github.com/LukeHagar/connexion.git
synced 2025-12-06 04:19:26 +00:00
This PR updates the examples for Connexion 3.0 and merges them for OpenAPI and Swagger. 2 examples required some changes to make them work: - The reverse proxy example required some fixes to the SwaggerUIMiddleware to leverage the `root_path` correctly. This is included in the PR. - The enforced defaults example requires the json validator to adapt the body and pass it on. We currently pass on the original body after validation, and I'm not sure if we should change this. I'll submit a separate PR to discuss this.
437 lines
15 KiB
Python
437 lines
15 KiB
Python
import json
|
|
from struct import unpack
|
|
|
|
import yaml
|
|
from connexion.apps.flask_app import FlaskJSONProvider
|
|
from werkzeug.test import Client, EnvironBuilder
|
|
|
|
|
|
def test_app(simple_app):
|
|
app_client = simple_app.app.test_client()
|
|
|
|
# by default the Swagger UI is enabled
|
|
swagger_ui = app_client.get("/v1.0/ui/") # type: flask.Response
|
|
assert swagger_ui.status_code == 200
|
|
assert b"Swagger UI" in swagger_ui.data
|
|
|
|
# test return Swagger UI static files
|
|
swagger_icon = app_client.get("/v1.0/ui/swagger-ui.js") # type: flask.Response
|
|
assert swagger_icon.status_code == 200
|
|
|
|
post_greeting_url = app_client.post(
|
|
"/v1.0/greeting/jsantos/the/third/of/his/name", data={}
|
|
) # type: flask.Response
|
|
assert post_greeting_url.status_code == 200
|
|
assert post_greeting_url.content_type == "application/json"
|
|
greeting_response_url = json.loads(post_greeting_url.data.decode("utf-8"))
|
|
assert (
|
|
greeting_response_url["greeting"]
|
|
== "Hello jsantos thanks for the/third/of/his/name"
|
|
)
|
|
|
|
post_greeting = app_client.post(
|
|
"/v1.0/greeting/jsantos", data={}
|
|
) # type: flask.Response
|
|
assert post_greeting.status_code == 200
|
|
assert post_greeting.content_type == "application/json"
|
|
greeting_response = json.loads(post_greeting.data.decode("utf-8"))
|
|
assert greeting_response["greeting"] == "Hello jsantos"
|
|
|
|
get_bye = app_client.get("/v1.0/bye/jsantos") # type: flask.Response
|
|
assert get_bye.status_code == 200
|
|
assert get_bye.data == b"Goodbye jsantos"
|
|
|
|
post_greeting = app_client.post(
|
|
"/v1.0/greeting/jsantos", data={}
|
|
) # type: flask.Response
|
|
assert post_greeting.status_code == 200
|
|
assert post_greeting.content_type == "application/json"
|
|
greeting_response = json.loads(post_greeting.data.decode("utf-8"))
|
|
assert greeting_response["greeting"] == "Hello jsantos"
|
|
|
|
|
|
def test_openapi_yaml_behind_proxy(reverse_proxied_app):
|
|
"""Verify the swagger.json file is returned with base_path updated
|
|
according to X-Original-URI header.
|
|
"""
|
|
app_client = reverse_proxied_app.app.test_client()
|
|
|
|
headers = {"X-Forwarded-Path": "/behind/proxy"}
|
|
|
|
swagger_ui = app_client.get("/v1.0/ui/", headers=headers)
|
|
assert swagger_ui.status_code == 200
|
|
|
|
openapi_yaml = app_client.get(
|
|
"/v1.0/" + reverse_proxied_app._spec_file, headers=headers
|
|
)
|
|
assert openapi_yaml.status_code == 200
|
|
assert openapi_yaml.headers.get("Content-Type").startswith("text/yaml")
|
|
spec = yaml.load(openapi_yaml.data.decode("utf-8"), Loader=yaml.BaseLoader)
|
|
|
|
if reverse_proxied_app._spec_file == "swagger.yaml":
|
|
assert b'url = "/behind/proxy/v1.0/swagger.json"' in swagger_ui.data
|
|
assert (
|
|
spec.get("basePath") == "/behind/proxy/v1.0"
|
|
), "basePath should contains original URI"
|
|
else:
|
|
assert b'url: "/behind/proxy/v1.0/openapi.json"' in swagger_ui.data
|
|
url = spec.get("servers", [{}])[0].get("url")
|
|
assert url == "/behind/proxy/v1.0", "basePath should contains original URI"
|
|
|
|
|
|
def test_produce_decorator(simple_app):
|
|
app_client = simple_app.app.test_client()
|
|
|
|
get_bye = app_client.get("/v1.0/bye/jsantos") # type: flask.Response
|
|
assert get_bye.content_type == "text/plain; charset=utf-8"
|
|
|
|
|
|
def test_returning_flask_response_tuple(simple_app):
|
|
app_client = simple_app.app.test_client()
|
|
|
|
result = app_client.get("/v1.0/flask_response_tuple") # type: flask.Response
|
|
assert result.status_code == 201
|
|
assert result.content_type == "application/json"
|
|
result_data = json.loads(result.data.decode("utf-8", "replace"))
|
|
assert result_data == {"foo": "bar"}
|
|
|
|
|
|
def test_jsonifier(simple_app):
|
|
app_client = simple_app.app.test_client()
|
|
|
|
post_greeting = app_client.post(
|
|
"/v1.0/greeting/jsantos", data={}
|
|
) # type: flask.Response
|
|
assert post_greeting.status_code == 200
|
|
assert post_greeting.content_type == "application/json"
|
|
greeting_reponse = json.loads(post_greeting.data.decode("utf-8", "replace"))
|
|
assert greeting_reponse["greeting"] == "Hello jsantos"
|
|
|
|
get_list_greeting = app_client.get(
|
|
"/v1.0/list/jsantos", data={}
|
|
) # type: flask.Response
|
|
assert get_list_greeting.status_code == 200
|
|
assert get_list_greeting.content_type == "application/json"
|
|
greeting_reponse = json.loads(get_list_greeting.data.decode("utf-8", "replace"))
|
|
assert len(greeting_reponse) == 2
|
|
assert greeting_reponse[0] == "hello"
|
|
assert greeting_reponse[1] == "jsantos"
|
|
|
|
get_greetings = app_client.get(
|
|
"/v1.0/greetings/jsantos", data={}
|
|
) # type: flask.Response
|
|
assert get_greetings.status_code == 200
|
|
assert get_greetings.content_type == "application/x.connexion+json"
|
|
greetings_reponse = json.loads(get_greetings.data.decode("utf-8", "replace"))
|
|
assert len(greetings_reponse) == 1
|
|
assert greetings_reponse["greetings"] == "Hello jsantos"
|
|
|
|
|
|
def test_not_content_response(simple_app):
|
|
app_client = simple_app.app.test_client()
|
|
|
|
get_no_content_response = app_client.get("/v1.0/test_no_content_response")
|
|
assert get_no_content_response.status_code == 204
|
|
assert get_no_content_response.content_length is None
|
|
|
|
|
|
def test_pass_through(simple_app):
|
|
app_client = simple_app.app.test_client()
|
|
|
|
response = app_client.get("/v1.0/multimime", data={}) # type: flask.Response
|
|
assert response.status_code == 200
|
|
|
|
|
|
def test_empty(simple_app):
|
|
app_client = simple_app.app.test_client()
|
|
|
|
response = app_client.get("/v1.0/empty") # type: flask.Response
|
|
assert response.status_code == 204
|
|
assert not response.data
|
|
|
|
|
|
def test_exploded_deep_object_param_endpoint_openapi_simple(simple_openapi_app):
|
|
app_client = simple_openapi_app.app.test_client()
|
|
|
|
response = app_client.get(
|
|
"/v1.0/exploded-deep-object-param?id[foo]=bar"
|
|
) # type: flask.Response
|
|
assert response.status_code == 200
|
|
response_data = json.loads(response.data.decode("utf-8", "replace"))
|
|
assert response_data == {"foo": "bar", "foo4": "blubb"}
|
|
|
|
|
|
def test_exploded_deep_object_param_endpoint_openapi_multiple_data_types(
|
|
simple_openapi_app,
|
|
):
|
|
app_client = simple_openapi_app.app.test_client()
|
|
|
|
response = app_client.get(
|
|
"/v1.0/exploded-deep-object-param?id[foo]=bar&id[fooint]=2&id[fooboo]=false"
|
|
) # type: flask.Response
|
|
assert response.status_code == 200, response.text
|
|
response_data = json.loads(response.data.decode("utf-8", "replace"))
|
|
assert response_data == {
|
|
"foo": "bar",
|
|
"fooint": 2,
|
|
"fooboo": False,
|
|
"foo4": "blubb",
|
|
}
|
|
|
|
|
|
def test_exploded_deep_object_param_endpoint_openapi_additional_properties(
|
|
simple_openapi_app,
|
|
):
|
|
app_client = simple_openapi_app.app.test_client()
|
|
|
|
response = app_client.get(
|
|
"/v1.0/exploded-deep-object-param-additional-properties?id[foo]=bar&id[fooint]=2"
|
|
) # type: flask.Response
|
|
assert response.status_code == 200
|
|
response_data = json.loads(response.data.decode("utf-8", "replace"))
|
|
assert response_data == {"foo": "bar", "fooint": "2"}
|
|
|
|
|
|
def test_exploded_deep_object_param_endpoint_openapi_additional_properties_false(
|
|
simple_openapi_app,
|
|
):
|
|
app_client = simple_openapi_app.app.test_client()
|
|
|
|
response = app_client.get(
|
|
"/v1.0/exploded-deep-object-param?id[foo]=bar&id[foofoo]=barbar"
|
|
) # type: flask.Response
|
|
assert response.status_code == 400
|
|
|
|
|
|
def test_exploded_deep_object_param_endpoint_openapi_with_dots(simple_openapi_app):
|
|
app_client = simple_openapi_app.app.test_client()
|
|
|
|
response = app_client.get(
|
|
"/v1.0/exploded-deep-object-param-additional-properties?id[foo]=bar&id[foo.foo]=barbar"
|
|
) # type: flask.Response
|
|
assert response.status_code == 200
|
|
response_data = json.loads(response.data.decode("utf-8", "replace"))
|
|
assert response_data == {"foo": "bar", "foo.foo": "barbar"}
|
|
|
|
|
|
def test_nested_exploded_deep_object_param_endpoint_openapi(simple_openapi_app):
|
|
app_client = simple_openapi_app.app.test_client()
|
|
|
|
response = app_client.get(
|
|
"/v1.0/nested-exploded-deep-object-param?id[foo][foo2]=bar&id[foofoo]=barbar"
|
|
) # type: flask.Response
|
|
assert response.status_code == 200
|
|
response_data = json.loads(response.data.decode("utf-8", "replace"))
|
|
assert response_data == {
|
|
"foo": {"foo2": "bar", "foo3": "blubb"},
|
|
"foofoo": "barbar",
|
|
}
|
|
|
|
|
|
def test_redirect_endpoint(simple_app):
|
|
app_client = simple_app.app.test_client()
|
|
resp = app_client.get("/v1.0/test-redirect-endpoint")
|
|
assert resp.status_code == 302
|
|
|
|
|
|
def test_redirect_response_endpoint(simple_app):
|
|
app_client = simple_app.app.test_client()
|
|
resp = app_client.get("/v1.0/test-redirect-response-endpoint")
|
|
assert resp.status_code == 302
|
|
|
|
|
|
def test_default_object_body(simple_app):
|
|
app_client = simple_app.app.test_client()
|
|
resp = app_client.post("/v1.0/test-default-object-body")
|
|
assert resp.status_code == 200
|
|
response = json.loads(resp.data.decode("utf-8", "replace"))
|
|
assert response["stack"] == {"image_version": "default_image"}
|
|
|
|
resp = app_client.post("/v1.0/test-default-integer-body")
|
|
assert resp.status_code == 200
|
|
response = json.loads(resp.data.decode("utf-8", "replace"))
|
|
assert response == 1
|
|
|
|
|
|
def test_empty_object_body(simple_app):
|
|
app_client = simple_app.app.test_client()
|
|
resp = app_client.post(
|
|
"/v1.0/test-empty-object-body",
|
|
data=json.dumps({}),
|
|
headers={"Content-Type": "application/json"},
|
|
)
|
|
assert resp.status_code == 200
|
|
response = json.loads(resp.data.decode("utf-8", "replace"))
|
|
assert response["stack"] == {}
|
|
|
|
|
|
def test_nested_additional_properties(simple_openapi_app):
|
|
app_client = simple_openapi_app.app.test_client()
|
|
resp = app_client.post(
|
|
"/v1.0/test-nested-additional-properties",
|
|
data=json.dumps({"nested": {"object": True}}),
|
|
headers={"Content-Type": "application/json"},
|
|
)
|
|
assert resp.status_code == 200
|
|
response = json.loads(resp.data.decode("utf-8", "replace"))
|
|
assert response == {"nested": {"object": True}}
|
|
|
|
|
|
def test_custom_provider(simple_app):
|
|
class CustomProvider(FlaskJSONProvider):
|
|
def default(self, o):
|
|
if o.__class__.__name__ == "DummyClass":
|
|
return "cool result"
|
|
return super().default(o)
|
|
|
|
flask_app = simple_app.app
|
|
flask_app.json = CustomProvider(flask_app)
|
|
app_client = flask_app.test_client()
|
|
|
|
resp = app_client.get("/v1.0/custom-json-response")
|
|
assert resp.status_code == 200
|
|
response = json.loads(resp.data.decode("utf-8", "replace"))
|
|
assert response["theResult"] == "cool result"
|
|
|
|
|
|
def test_content_type_not_json(simple_app):
|
|
app_client = simple_app.app.test_client()
|
|
|
|
resp = app_client.get("/v1.0/blob-response")
|
|
assert resp.status_code == 200
|
|
|
|
# validate binary content
|
|
text, number = unpack("!4sh", resp.data)
|
|
assert text == b"cool"
|
|
assert number == 8
|
|
|
|
|
|
def test_maybe_blob_or_json(simple_app):
|
|
app_client = simple_app.app.test_client()
|
|
|
|
resp = app_client.get("/v1.0/binary-response")
|
|
assert resp.status_code == 200
|
|
assert resp.content_type == "application/octet-stream"
|
|
# validate binary content
|
|
text, number = unpack("!4sh", resp.data)
|
|
assert text == b"cool"
|
|
assert number == 8
|
|
|
|
|
|
def test_bad_operations(bad_operations_app):
|
|
# Bad operationIds in bad_operations_app should result in 501
|
|
app_client = bad_operations_app.app.test_client()
|
|
|
|
resp = app_client.get("/v1.0/welcome")
|
|
assert resp.status_code == 501
|
|
|
|
resp = app_client.put("/v1.0/welcome")
|
|
assert resp.status_code == 501
|
|
|
|
resp = app_client.post("/v1.0/welcome")
|
|
assert resp.status_code == 501
|
|
|
|
|
|
def test_text_request(simple_app):
|
|
app_client = simple_app.app.test_client()
|
|
|
|
resp = app_client.post("/v1.0/text-request", data="text")
|
|
assert resp.status_code == 200
|
|
|
|
|
|
def test_operation_handler_returns_flask_object(invalid_resp_allowed_app):
|
|
app_client = invalid_resp_allowed_app.app.test_client()
|
|
resp = app_client.get("/v1.0/get_non_conforming_response")
|
|
assert resp.status_code == 200
|
|
|
|
|
|
def test_post_wrong_content_type(simple_app):
|
|
app_client = simple_app.app.test_client()
|
|
resp = app_client.post(
|
|
"/v1.0/post_wrong_content_type",
|
|
content_type="application/xml",
|
|
data=json.dumps({"some": "data"}),
|
|
)
|
|
assert resp.status_code == 415
|
|
|
|
resp = app_client.post(
|
|
"/v1.0/post_wrong_content_type",
|
|
content_type="application/x-www-form-urlencoded",
|
|
data="a=1&b=2",
|
|
)
|
|
assert resp.status_code == 415
|
|
|
|
resp = app_client.post(
|
|
"/v1.0/post_wrong_content_type",
|
|
content_type="application/json",
|
|
data="not a valid json",
|
|
)
|
|
assert (
|
|
resp.status_code == 400
|
|
), "Should return 400 when Content-Type is json but content not parsable"
|
|
|
|
|
|
def test_get_unicode_response(simple_app):
|
|
app_client = simple_app.app.test_client()
|
|
resp = app_client.get("/v1.0/get_unicode_response")
|
|
actualJson = {"currency": "\xa3", "key": "leena"}
|
|
assert json.loads(resp.data.decode("utf-8", "replace")) == actualJson
|
|
|
|
|
|
def test_get_enum_response(simple_app):
|
|
app_client = simple_app.app.test_client()
|
|
resp = app_client.get("/v1.0/get_enum_response")
|
|
assert resp.status_code == 200
|
|
|
|
|
|
def test_get_httpstatus_response(simple_app):
|
|
app_client = simple_app.app.test_client()
|
|
resp = app_client.get("/v1.0/get_httpstatus_response")
|
|
assert resp.status_code == 200
|
|
|
|
|
|
def test_get_bad_default_response(simple_app):
|
|
app_client = simple_app.app.test_client()
|
|
resp = app_client.get("/v1.0/get_bad_default_response/200")
|
|
assert resp.status_code == 200
|
|
|
|
resp = app_client.get("/v1.0/get_bad_default_response/202")
|
|
assert resp.status_code == 500
|
|
|
|
|
|
def test_streaming_response(simple_app):
|
|
app_client = simple_app.app.test_client()
|
|
resp = app_client.get("/v1.0/get_streaming_response")
|
|
assert resp.status_code == 200
|
|
|
|
|
|
def test_oneof(simple_openapi_app):
|
|
app_client = simple_openapi_app.app.test_client()
|
|
|
|
post_greeting = app_client.post( # type: flask.Response
|
|
"/v1.0/oneof_greeting",
|
|
data=json.dumps({"name": 3}),
|
|
content_type="application/json",
|
|
)
|
|
assert post_greeting.status_code == 200
|
|
assert post_greeting.content_type == "application/json"
|
|
greeting_reponse = json.loads(post_greeting.data.decode("utf-8", "replace"))
|
|
assert greeting_reponse["greeting"] == "Hello 3"
|
|
|
|
post_greeting = app_client.post( # type: flask.Response
|
|
"/v1.0/oneof_greeting",
|
|
data=json.dumps({"name": True}),
|
|
content_type="application/json",
|
|
)
|
|
assert post_greeting.status_code == 200
|
|
assert post_greeting.content_type == "application/json"
|
|
greeting_reponse = json.loads(post_greeting.data.decode("utf-8", "replace"))
|
|
assert greeting_reponse["greeting"] == "Hello True"
|
|
|
|
post_greeting = app_client.post( # type: flask.Response
|
|
"/v1.0/oneof_greeting",
|
|
data=json.dumps({"name": "jsantos"}),
|
|
content_type="application/json",
|
|
)
|
|
assert post_greeting.status_code == 400
|