mirror of
https://github.com/PiBrewing/craftbeerpi4.git
synced 2025-01-01 18:31:45 +01:00
185 lines
6 KiB
Python
185 lines
6 KiB
Python
|
from collections import defaultdict
|
||
|
from os.path import abspath, dirname, join
|
||
|
from inspect import isclass
|
||
|
|
||
|
import yaml
|
||
|
from aiohttp import web
|
||
|
from aiohttp.hdrs import METH_ANY, METH_ALL
|
||
|
from jinja2 import Environment, BaseLoader
|
||
|
|
||
|
try:
|
||
|
import ujson as json
|
||
|
except ImportError: # pragma: no cover
|
||
|
import json
|
||
|
|
||
|
|
||
|
SWAGGER_TEMPLATE = abspath(join(dirname(__file__), "..", "templates"))
|
||
|
|
||
|
|
||
|
def _extract_swagger_docs(end_point_doc, method="get"):
|
||
|
# Find Swagger start point in doc
|
||
|
end_point_swagger_start = 0
|
||
|
for i, doc_line in enumerate(end_point_doc):
|
||
|
if "---" in doc_line:
|
||
|
end_point_swagger_start = i + 1
|
||
|
break
|
||
|
|
||
|
# Build JSON YAML Obj
|
||
|
try:
|
||
|
end_point_swagger_doc = (
|
||
|
yaml.full_load("\n".join(end_point_doc[end_point_swagger_start:]))
|
||
|
)
|
||
|
except yaml.YAMLError:
|
||
|
end_point_swagger_doc = {
|
||
|
"description": "⚠ Swagger document could not be loaded "
|
||
|
"from docstring ⚠",
|
||
|
"tags": ["Invalid Swagger"]
|
||
|
}
|
||
|
return {method: end_point_swagger_doc}
|
||
|
|
||
|
|
||
|
def _build_doc_from_func_doc(route):
|
||
|
|
||
|
out = {}
|
||
|
if isclass(route.handler) and issubclass(route.handler, web.View):
|
||
|
for method_name in _get_method_names_for_handler(route):
|
||
|
method = getattr(route.handler, method_name)
|
||
|
if method.__doc__ is not None and "---" in method.__doc__:
|
||
|
end_point_doc = method.__doc__.splitlines()
|
||
|
out.update(_extract_swagger_docs(end_point_doc, method=method_name))
|
||
|
|
||
|
else:
|
||
|
try:
|
||
|
end_point_doc = route.handler.__doc__.splitlines()
|
||
|
except AttributeError:
|
||
|
return {}
|
||
|
out.update(_extract_swagger_docs(end_point_doc, method=str(route.method).lower()))
|
||
|
return out
|
||
|
|
||
|
def _get_method_names_for_handler(route):
|
||
|
# Return all valid method names in handler if the method is *,
|
||
|
# otherwise return the specific method.
|
||
|
if route.method == METH_ANY:
|
||
|
return {
|
||
|
attr for attr in dir(route.handler)
|
||
|
if attr.upper() in METH_ALL
|
||
|
}
|
||
|
else:
|
||
|
return {
|
||
|
attr for attr in dir(route.handler)
|
||
|
if attr.upper() in METH_ALL and attr.upper() == route.method
|
||
|
}
|
||
|
|
||
|
|
||
|
def generate_doc_from_each_end_point(
|
||
|
app: web.Application,
|
||
|
*,
|
||
|
ui_version: int = None,
|
||
|
api_base_url: str = "/",
|
||
|
description: str = "Swagger API definition",
|
||
|
api_version: str = "1.0.0",
|
||
|
title: str = "Swagger API",
|
||
|
contact: str = "",
|
||
|
template_path: str = None,
|
||
|
definitions: dict = None,
|
||
|
security_definitions: dict = None):
|
||
|
# Clean description
|
||
|
_start_desc = 0
|
||
|
for i, word in enumerate(description):
|
||
|
if word != '\n':
|
||
|
_start_desc = i
|
||
|
break
|
||
|
cleaned_description = " ".join(description[_start_desc:].splitlines())
|
||
|
|
||
|
def nesteddict2yaml(d, indent=10, result=""):
|
||
|
for key, value in d.items():
|
||
|
result += " " * indent + str(key) + ':'
|
||
|
if isinstance(value, dict):
|
||
|
result = nesteddict2yaml(value, indent + 2, result + "\n")
|
||
|
elif isinstance(value, str):
|
||
|
result += " \"" + str(value) + "\"\n"
|
||
|
else:
|
||
|
result += " " + str(value) + "\n"
|
||
|
return result
|
||
|
|
||
|
# Load base Swagger template
|
||
|
jinja2_env = Environment(loader=BaseLoader())
|
||
|
jinja2_env.filters['nesteddict2yaml'] = nesteddict2yaml
|
||
|
|
||
|
if template_path is None:
|
||
|
if ui_version == 3:
|
||
|
template_path = join(SWAGGER_TEMPLATE, "openapi.yaml")
|
||
|
else:
|
||
|
template_path = join(SWAGGER_TEMPLATE, "swagger.yaml")
|
||
|
|
||
|
with open(template_path, "r") as f:
|
||
|
swagger_base = (
|
||
|
jinja2_env.from_string(f.read()).render(
|
||
|
description=cleaned_description,
|
||
|
version=api_version,
|
||
|
title=title,
|
||
|
contact=contact,
|
||
|
base_path=api_base_url,
|
||
|
definitions=definitions,
|
||
|
security_definitions=security_definitions)
|
||
|
)
|
||
|
|
||
|
# The Swagger OBJ
|
||
|
swagger = yaml.full_load(swagger_base)
|
||
|
swagger["paths"] = defaultdict(dict)
|
||
|
|
||
|
for route in app.router.routes():
|
||
|
|
||
|
end_point_doc = None
|
||
|
|
||
|
# If route has a external link to doc, we use it, not function doc
|
||
|
if getattr(route.handler, "swagger_file", False):
|
||
|
try:
|
||
|
with open(route.handler.swagger_file, "r") as f:
|
||
|
end_point_doc = {
|
||
|
route.method.lower():
|
||
|
yaml.full_load(f.read())
|
||
|
}
|
||
|
except yaml.YAMLError:
|
||
|
end_point_doc = {
|
||
|
route.method.lower(): {
|
||
|
"description": "⚠ Swagger document could not be "
|
||
|
"loaded from file ⚠",
|
||
|
"tags": ["Invalid Swagger"]
|
||
|
}
|
||
|
}
|
||
|
except FileNotFoundError:
|
||
|
end_point_doc = {
|
||
|
route.method.lower(): {
|
||
|
"description":
|
||
|
"⚠ Swagger file not "
|
||
|
"found ({}) ⚠".format(route.handler.swagger_file),
|
||
|
"tags": ["Invalid Swagger"]
|
||
|
}
|
||
|
}
|
||
|
|
||
|
# Check if end-point has Swagger doc
|
||
|
else:
|
||
|
end_point_doc = _build_doc_from_func_doc(route)
|
||
|
|
||
|
# there is doc available?
|
||
|
if end_point_doc:
|
||
|
url_info = route._resource.get_info()
|
||
|
if url_info.get("path", None):
|
||
|
url = url_info.get("path")
|
||
|
else:
|
||
|
url = url_info.get("formatter")
|
||
|
|
||
|
swagger["paths"][url].update(end_point_doc)
|
||
|
|
||
|
return json.dumps(swagger)
|
||
|
|
||
|
|
||
|
def load_doc_from_yaml_file(doc_path: str):
|
||
|
with open(doc_path, "r") as f:
|
||
|
loaded_yaml = yaml.full_load(f.read())
|
||
|
return json.dumps(loaded_yaml)
|
||
|
|
||
|
|
||
|
__all__ = ("generate_doc_from_each_end_point", "load_doc_from_yaml_file")
|