mirror of
https://github.com/correl/openapi-core.git
synced 2024-11-28 11:09:52 +00:00
202 lines
5.8 KiB
Python
202 lines
5.8 KiB
Python
|
import json
|
||
|
import pytest
|
||
|
from six import iteritems
|
||
|
|
||
|
from openapi_core.exceptions import (
|
||
|
MissingParameterError, InvalidContentTypeError,
|
||
|
)
|
||
|
from openapi_core.media_types import MediaType
|
||
|
from openapi_core.operations import Operation
|
||
|
from openapi_core.paths import Path
|
||
|
from openapi_core.request_bodies import RequestBody
|
||
|
from openapi_core.schemas import Schema
|
||
|
from openapi_core.shortcuts import create_spec
|
||
|
from openapi_core.wrappers import BaseOpenAPIRequest
|
||
|
|
||
|
|
||
|
class RequestMock(BaseOpenAPIRequest):
|
||
|
|
||
|
def __init__(
|
||
|
self, method, path, path_pattern=None, args=None, view_args=None,
|
||
|
headers=None, cookies=None, data=None,
|
||
|
content_type='application/json'):
|
||
|
self.path = path
|
||
|
self.path_pattern = path_pattern or path
|
||
|
self.method = method
|
||
|
|
||
|
self.args = args or {}
|
||
|
self.view_args = view_args or {}
|
||
|
self.headers = headers or {}
|
||
|
self.cookies = cookies or {}
|
||
|
self.data = data or ''
|
||
|
|
||
|
self.content_type = content_type
|
||
|
|
||
|
|
||
|
class TestPetstore(object):
|
||
|
|
||
|
@pytest.fixture
|
||
|
def spec_dict(self, factory):
|
||
|
return factory.spec_from_file("data/v3.0/petstore.yaml")
|
||
|
|
||
|
@pytest.fixture
|
||
|
def spec(self, spec_dict):
|
||
|
return create_spec(spec_dict)
|
||
|
|
||
|
def test_spec(self, spec, spec_dict):
|
||
|
assert spec.servers == spec_dict['servers']
|
||
|
assert spec.get_server_url() == spec_dict['servers'][0]['url']
|
||
|
|
||
|
for path_name, path in iteritems(spec.paths):
|
||
|
assert type(path) == Path
|
||
|
assert path.name == path_name
|
||
|
|
||
|
for http_method, operation in iteritems(path.operations):
|
||
|
assert type(operation) == Operation
|
||
|
assert operation.path_name == path_name
|
||
|
assert operation.http_method == http_method
|
||
|
|
||
|
operation_spec = spec_dict['paths'][path_name][http_method]
|
||
|
request_body_spec = operation_spec.get('requestBody')
|
||
|
|
||
|
assert bool(request_body_spec) == bool(operation.request_body)
|
||
|
|
||
|
if not request_body_spec:
|
||
|
continue
|
||
|
|
||
|
assert type(operation.request_body) == RequestBody
|
||
|
assert bool(operation.request_body.required) ==\
|
||
|
request_body_spec.get('required', False)
|
||
|
|
||
|
for content_type, media_type in iteritems(
|
||
|
operation.request_body.content):
|
||
|
assert type(media_type) == MediaType
|
||
|
assert media_type.content_type == content_type
|
||
|
|
||
|
content_spec = request_body_spec['content'][content_type]
|
||
|
schema_spec = content_spec.get('schema')
|
||
|
assert bool(schema_spec) == bool(media_type.schema)
|
||
|
|
||
|
if not schema_spec:
|
||
|
continue
|
||
|
|
||
|
# @todo: test with defererence
|
||
|
if '$ref' in schema_spec:
|
||
|
continue
|
||
|
|
||
|
assert type(media_type.schema) == Schema
|
||
|
assert media_type.schema.type == schema_spec['type']
|
||
|
assert media_type.schema.required == schema_spec.get(
|
||
|
'required', False)
|
||
|
|
||
|
def test_get_pets(self, spec):
|
||
|
query_params = {
|
||
|
'limit': '20',
|
||
|
'ids': ['12', '13'],
|
||
|
}
|
||
|
|
||
|
request = RequestMock('get', '/pets', args=query_params)
|
||
|
|
||
|
parameters = request.get_parameters(spec)
|
||
|
|
||
|
assert parameters == {
|
||
|
'query': {
|
||
|
'limit': 20,
|
||
|
'ids': [12, 13],
|
||
|
}
|
||
|
}
|
||
|
|
||
|
def test_get_pets_raises_missing_required_param(self, spec):
|
||
|
request = RequestMock('get', '/pets')
|
||
|
|
||
|
with pytest.raises(MissingParameterError):
|
||
|
request.get_parameters(spec)
|
||
|
|
||
|
def test_get_pets_failed_to_cast(self, spec):
|
||
|
query_params = {
|
||
|
'limit': 'non_integer_value',
|
||
|
}
|
||
|
|
||
|
request = RequestMock('get', '/pets', args=query_params)
|
||
|
|
||
|
parameters = request.get_parameters(spec)
|
||
|
|
||
|
assert parameters == {
|
||
|
'query': {
|
||
|
'limit': 'non_integer_value',
|
||
|
}
|
||
|
}
|
||
|
|
||
|
def test_get_pets_empty_value(self, spec):
|
||
|
query_params = {
|
||
|
'limit': '',
|
||
|
}
|
||
|
|
||
|
request = RequestMock('get', '/pets', args=query_params)
|
||
|
|
||
|
parameters = request.get_parameters(spec)
|
||
|
|
||
|
assert parameters == {
|
||
|
'query': {
|
||
|
'limit': None,
|
||
|
}
|
||
|
}
|
||
|
|
||
|
def test_get_pets_none_value(self, spec):
|
||
|
query_params = {
|
||
|
'limit': None,
|
||
|
}
|
||
|
|
||
|
request = RequestMock('get', '/pets', args=query_params)
|
||
|
|
||
|
parameters = request.get_parameters(spec)
|
||
|
|
||
|
assert parameters == {
|
||
|
'query': {
|
||
|
'limit': None,
|
||
|
}
|
||
|
}
|
||
|
|
||
|
def test_post_pets(self, spec):
|
||
|
data_json = {
|
||
|
'name': 'Cat',
|
||
|
'tag': 'cats',
|
||
|
}
|
||
|
data = json.dumps(data_json)
|
||
|
|
||
|
request = RequestMock('post', '/pets', data=data)
|
||
|
|
||
|
body = request.get_body(spec)
|
||
|
|
||
|
assert body == data_json
|
||
|
|
||
|
def test_post_pets_raises_invalid_content_type(self, spec):
|
||
|
data_json = {
|
||
|
'name': 'Cat',
|
||
|
'tag': 'cats',
|
||
|
}
|
||
|
data = json.dumps(data_json)
|
||
|
|
||
|
request = RequestMock(
|
||
|
'post', '/pets', data=data, content_type='text/html')
|
||
|
|
||
|
with pytest.raises(InvalidContentTypeError):
|
||
|
request.get_body(spec)
|
||
|
|
||
|
def test_get_pet(self, spec):
|
||
|
view_args = {
|
||
|
'petId': '1',
|
||
|
}
|
||
|
request = RequestMock(
|
||
|
'get', '/pets/1', path_pattern='/pets/{petId}',
|
||
|
view_args=view_args,
|
||
|
)
|
||
|
|
||
|
parameters = request.get_parameters(spec)
|
||
|
|
||
|
assert parameters == {
|
||
|
'path': {
|
||
|
'petId': 1,
|
||
|
}
|
||
|
}
|