mirror of
https://github.com/correl/openapi-core.git
synced 2025-01-01 11:03:19 +00:00
Merge pull request #131 from MCapitani/links
add support for "links" in Response
This commit is contained in:
commit
f0cfa2d0e0
7 changed files with 207 additions and 1 deletions
0
openapi_core/schema/links/__init__.py
Normal file
0
openapi_core/schema/links/__init__.py
Normal file
44
openapi_core/schema/links/generators.py
Normal file
44
openapi_core/schema/links/generators.py
Normal file
|
@ -0,0 +1,44 @@
|
|||
"""OpenAPI core links generators module"""
|
||||
from six import iteritems
|
||||
|
||||
from openapi_core.compat import lru_cache
|
||||
from openapi_core.schema.links.models import Link
|
||||
from openapi_core.schema.parameters.generators import ParametersGenerator
|
||||
from openapi_core.schema.servers.generators import ServersGenerator
|
||||
|
||||
|
||||
class LinksGenerator(object):
|
||||
|
||||
def __init__(self, dereferencer, schemas_registry):
|
||||
self.dereferencer = dereferencer
|
||||
self.schemas_registry = schemas_registry
|
||||
|
||||
def generate(self, links):
|
||||
for link_name, link in iteritems(links):
|
||||
link_deref = self.dereferencer.dereference(link)
|
||||
operation_id = link_deref.get('operationId')
|
||||
parameters = link_deref.get('parameters', {})
|
||||
request_body = link_deref.get('requestBody') # string or dict
|
||||
description = link_deref.get('description')
|
||||
server_spec = link_deref.get('server')
|
||||
server = self.servers_generator.generate(server_spec) \
|
||||
if server_spec is not None \
|
||||
else None
|
||||
|
||||
yield link_name, Link(
|
||||
operation_id,
|
||||
parameters,
|
||||
request_body,
|
||||
description,
|
||||
server
|
||||
)
|
||||
|
||||
@property
|
||||
@lru_cache()
|
||||
def parameters_generator(self):
|
||||
return ParametersGenerator(self.dereferencer, self.schemas_registry)
|
||||
|
||||
@property
|
||||
@lru_cache()
|
||||
def servers_generator(self):
|
||||
return ServersGenerator(self.dereferencer)
|
26
openapi_core/schema/links/models.py
Normal file
26
openapi_core/schema/links/models.py
Normal file
|
@ -0,0 +1,26 @@
|
|||
"""OpenAPI core links models module"""
|
||||
|
||||
|
||||
class Link(object):
|
||||
"""Represents an OpenAPI Link."""
|
||||
|
||||
def __init__(
|
||||
self,
|
||||
operation_id,
|
||||
parameters,
|
||||
request_body,
|
||||
description,
|
||||
server
|
||||
):
|
||||
"""
|
||||
request_body is assumed to be either a string (JSON, YAML or
|
||||
runtime expression) or an object (deserialized JSON or YAML)
|
||||
"""
|
||||
self.operationId = operation_id
|
||||
self.description = description
|
||||
self.server = server
|
||||
self.parameters = dict(parameters) if parameters else {}
|
||||
self.request_body = request_body
|
||||
|
||||
def __getitem__(self, item):
|
||||
return self.parameters[item]
|
|
@ -2,6 +2,7 @@
|
|||
from six import iteritems
|
||||
|
||||
from openapi_core.compat import lru_cache
|
||||
from openapi_core.schema.links.generators import LinksGenerator
|
||||
from openapi_core.schema.media_types.generators import MediaTypeGenerator
|
||||
from openapi_core.schema.parameters.generators import ParametersGenerator
|
||||
from openapi_core.schema.responses.models import Response
|
||||
|
@ -19,6 +20,8 @@ class ResponsesGenerator(object):
|
|||
description = response_deref['description']
|
||||
headers = response_deref.get('headers')
|
||||
content = response_deref.get('content')
|
||||
links_dict = response_deref.get('links', {})
|
||||
links = self.links_generator.generate(links_dict)
|
||||
|
||||
media_types = None
|
||||
if content:
|
||||
|
@ -30,7 +33,7 @@ class ResponsesGenerator(object):
|
|||
|
||||
yield http_status, Response(
|
||||
http_status, description,
|
||||
content=media_types, headers=parameters)
|
||||
content=media_types, headers=parameters, links=links)
|
||||
|
||||
@property
|
||||
@lru_cache()
|
||||
|
@ -41,3 +44,8 @@ class ResponsesGenerator(object):
|
|||
@lru_cache()
|
||||
def parameters_generator(self):
|
||||
return ParametersGenerator(self.dereferencer, self.schemas_registry)
|
||||
|
||||
@property
|
||||
@lru_cache()
|
||||
def links_generator(self):
|
||||
return LinksGenerator(self.dereferencer, self.schemas_registry)
|
||||
|
|
48
tests/integration/data/v3.0/links.yaml
Normal file
48
tests/integration/data/v3.0/links.yaml
Normal file
|
@ -0,0 +1,48 @@
|
|||
openapi: "3.0.0"
|
||||
info:
|
||||
title: Minimal valid OpenAPI specification
|
||||
version: "0.1"
|
||||
paths:
|
||||
/linked/noParam:
|
||||
get:
|
||||
operationId: noParOp
|
||||
responses:
|
||||
default:
|
||||
description: the linked result
|
||||
/linked/withParam:
|
||||
get:
|
||||
operationId: paramOp
|
||||
parameters:
|
||||
- name: opParam
|
||||
in: query
|
||||
description: test
|
||||
schema:
|
||||
type: string
|
||||
responses:
|
||||
default:
|
||||
description: the linked result
|
||||
/status:
|
||||
get:
|
||||
responses:
|
||||
default:
|
||||
description: Return something
|
||||
links:
|
||||
noParamLink:
|
||||
operationId: noParOp
|
||||
/status/{resourceId}:
|
||||
get:
|
||||
parameters:
|
||||
- name: resourceId
|
||||
in: path
|
||||
required: true
|
||||
schema:
|
||||
type: string
|
||||
responses:
|
||||
default:
|
||||
description: Return something else
|
||||
links:
|
||||
paramLink:
|
||||
operationId: paramOp
|
||||
parameters:
|
||||
opParam: $request.path.resourceId
|
||||
requestBody: test
|
36
tests/integration/test_link_spec.py
Normal file
36
tests/integration/test_link_spec.py
Normal file
|
@ -0,0 +1,36 @@
|
|||
from openapi_core.shortcuts import create_spec
|
||||
|
||||
|
||||
class TestLinkSpec(object):
|
||||
|
||||
def test_no_param(self, factory):
|
||||
spec_dict = factory.spec_from_file("data/v3.0/links.yaml")
|
||||
spec = create_spec(spec_dict)
|
||||
resp = spec['/status']['get'].get_response()
|
||||
|
||||
assert len(resp.links) == 1
|
||||
|
||||
link = resp.links['noParamLink']
|
||||
|
||||
assert link.operationId == 'noParOp'
|
||||
assert link.server is None
|
||||
assert link.request_body is None
|
||||
assert len(link.parameters) == 0
|
||||
|
||||
def test_param(self, factory):
|
||||
spec_dict = factory.spec_from_file("data/v3.0/links.yaml")
|
||||
spec = create_spec(spec_dict)
|
||||
resp = spec['/status/{resourceId}']['get'].get_response()
|
||||
|
||||
assert len(resp.links) == 1
|
||||
|
||||
link = resp.links['paramLink']
|
||||
|
||||
assert link.operationId == 'paramOp'
|
||||
assert link.server is None
|
||||
assert link.request_body == 'test'
|
||||
assert len(link.parameters) == 1
|
||||
|
||||
param = link.parameters['opParam']
|
||||
|
||||
assert param == '$request.path.resourceId'
|
44
tests/unit/schema/test_links.py
Normal file
44
tests/unit/schema/test_links.py
Normal file
|
@ -0,0 +1,44 @@
|
|||
import mock
|
||||
import pytest
|
||||
|
||||
from openapi_core.schema.links.models import Link
|
||||
from openapi_core.schema.servers.models import Server
|
||||
|
||||
|
||||
class TestLinks(object):
|
||||
|
||||
@pytest.fixture
|
||||
def link_factory(self):
|
||||
def link_factory(request_body, server):
|
||||
parameters = {
|
||||
'par1': mock.sentinel.par1,
|
||||
'par2': mock.sentinel.par2,
|
||||
}
|
||||
return Link(
|
||||
'op_id',
|
||||
parameters,
|
||||
request_body,
|
||||
'Test link',
|
||||
server
|
||||
)
|
||||
return link_factory
|
||||
|
||||
servers = [
|
||||
None,
|
||||
Server("https://bad.remote.domain.net/"),
|
||||
Server("http://localhost")
|
||||
]
|
||||
|
||||
request_body_list = [
|
||||
None,
|
||||
"request",
|
||||
'{"request": "value", "opt": 2}',
|
||||
{"request": "value", "opt": 2}
|
||||
]
|
||||
|
||||
@pytest.mark.parametrize("server", servers)
|
||||
@pytest.mark.parametrize("request_body", request_body_list)
|
||||
def test_iteritems(self, link_factory, request_body, server):
|
||||
link = link_factory(request_body, server)
|
||||
for par_name in link.parameters.keys():
|
||||
assert link[par_name] == link.parameters[par_name]
|
Loading…
Reference in a new issue