openapi-core/openapi_core/templating/paths/finders.py

104 lines
4.1 KiB
Python
Raw Normal View History

"""OpenAPI core templating paths finders module"""
2020-02-21 16:33:45 +00:00
from more_itertools import peekable
from six import iteritems
2020-03-17 18:37:44 +00:00
from six.moves.urllib.parse import urljoin, urlparse
2020-02-21 16:33:45 +00:00
2021-04-27 21:16:30 +00:00
from openapi_core.spec.servers import is_absolute
2020-02-21 16:33:45 +00:00
from openapi_core.templating.datatypes import TemplateResult
from openapi_core.templating.util import parse, search
from openapi_core.templating.paths.exceptions import (
PathNotFound, OperationNotFound, ServerNotFound,
)
class PathFinder(object):
2020-02-21 16:33:45 +00:00
def __init__(self, spec, base_url=None):
self.spec = spec
2020-02-21 16:33:45 +00:00
self.base_url = base_url
def find(self, request):
2020-02-21 16:33:45 +00:00
paths_iter = self._get_paths_iter(request.full_url_pattern)
paths_iter_peek = peekable(paths_iter)
if not paths_iter_peek:
raise PathNotFound(request.full_url_pattern)
operations_iter = self._get_operations_iter(
request.method, paths_iter_peek)
operations_iter_peek = peekable(operations_iter)
if not operations_iter_peek:
raise OperationNotFound(request.full_url_pattern, request.method)
servers_iter = self._get_servers_iter(
request.full_url_pattern, operations_iter_peek)
2020-02-21 16:33:45 +00:00
try:
return next(servers_iter)
except StopIteration:
raise ServerNotFound(request.full_url_pattern)
2020-02-21 16:33:45 +00:00
def _get_paths_iter(self, full_url_pattern):
template_paths = []
2021-04-23 11:36:27 +00:00
paths = self.spec / 'paths'
for path_pattern, path in paths.items():
2021-01-31 12:37:14 +00:00
# simple path.
# Return right away since it is always the most concrete
2020-02-21 16:33:45 +00:00
if full_url_pattern.endswith(path_pattern):
path_result = TemplateResult(path_pattern, {})
yield (path, path_result)
# template path
else:
result = search(path_pattern, full_url_pattern)
if result:
path_result = TemplateResult(path_pattern, result.named)
template_paths.append((path, path_result))
# Fewer variables -> more concrete path
for path in sorted(template_paths, key=lambda p: len(p[1].variables)):
yield path
2020-02-21 16:33:45 +00:00
def _get_operations_iter(self, request_method, paths_iter):
for path, path_result in paths_iter:
2021-04-23 11:36:27 +00:00
if request_method not in path:
2020-02-21 16:33:45 +00:00
continue
2021-04-23 11:36:27 +00:00
operation = path / request_method
2020-02-21 16:33:45 +00:00
yield (path, operation, path_result)
2020-02-21 16:33:45 +00:00
def _get_servers_iter(self, full_url_pattern, ooperations_iter):
for path, operation, path_result in ooperations_iter:
2021-04-23 11:36:27 +00:00
servers = path.get('servers', None) or \
operation.get('servers', None) or \
self.spec.get('servers', [{'url': '/'}])
2020-02-21 16:33:45 +00:00
for server in servers:
server_url_pattern = full_url_pattern.rsplit(
path_result.resolved, 1)[0]
2021-04-23 11:36:27 +00:00
server_url = server['url']
if not is_absolute(server_url):
2020-03-17 18:37:44 +00:00
# relative to absolute url
if self.base_url is not None:
2021-04-23 11:36:27 +00:00
server_url = urljoin(self.base_url, server['url'])
2020-03-17 18:37:44 +00:00
# if no base url check only path part
else:
server_url_pattern = urlparse(server_url_pattern).path
2020-02-21 16:33:45 +00:00
if server_url.endswith('/'):
server_url = server_url[:-1]
# simple path
2020-03-17 18:37:44 +00:00
if server_url_pattern == server_url:
2021-04-23 11:36:27 +00:00
server_result = TemplateResult(server['url'], {})
2020-02-21 16:33:45 +00:00
yield (
path, operation, server,
path_result, server_result,
)
# template path
else:
2021-04-23 11:36:27 +00:00
result = parse(server['url'], server_url_pattern)
2020-02-21 16:33:45 +00:00
if result:
server_result = TemplateResult(
2021-04-23 11:36:27 +00:00
server['url'], result.named)
2020-02-21 16:33:45 +00:00
yield (
path, operation, server,
path_result, server_result,
)