2020-03-27 05:03:26 +01:00
|
|
|
import json
|
2020-06-11 00:54:34 +02:00
|
|
|
import os
|
|
|
|
import re
|
2020-06-11 21:35:13 +02:00
|
|
|
from typing import Any, Dict, List, Mapping, Sequence
|
2017-12-27 08:57:47 +01:00
|
|
|
|
2020-06-11 00:54:34 +02:00
|
|
|
import markdown
|
2018-08-14 02:50:05 +02:00
|
|
|
from django.utils.html import escape as escape_html
|
2017-12-27 08:57:47 +01:00
|
|
|
from markdown.extensions import Extension
|
|
|
|
from markdown.preprocessors import Preprocessor
|
2020-06-11 00:54:34 +02:00
|
|
|
|
2021-09-17 19:01:36 +02:00
|
|
|
from zerver.lib.markdown.preprocessor_priorities import PREPROCESSOR_PRIORITES
|
2021-06-23 17:27:22 +02:00
|
|
|
from zerver.openapi.openapi import (
|
|
|
|
check_deprecated_consistency,
|
|
|
|
get_openapi_parameters,
|
|
|
|
get_parameters_description,
|
|
|
|
)
|
2017-12-27 08:57:47 +01:00
|
|
|
|
2021-02-12 08:20:45 +01:00
|
|
|
REGEXP = re.compile(r"\{generate_api_arguments_table\|\s*(.+?)\s*\|\s*(.+)\s*\}")
|
2017-12-27 08:57:47 +01:00
|
|
|
|
2021-11-16 16:19:24 +01:00
|
|
|
API_PARAMETER_TEMPLATE = """
|
|
|
|
<div class="api-argument" id="parameter-{argument}">
|
|
|
|
<p class="api-argument-name"><strong>{argument}</strong> <span class="api-field-type">{type}</span> {required} {deprecated} <a href="#parameter-{argument}" class="api-argument-hover-link"><i class="fa fa-chain"></i></a></p>
|
|
|
|
<div class="api-example">
|
|
|
|
<span class="api-argument-example-label">Example</span>: <code>{example}</code>
|
|
|
|
</div>
|
|
|
|
<div class="api-description">{description}{object_details}</div>
|
|
|
|
<hr>
|
|
|
|
</div>
|
|
|
|
""".strip()
|
|
|
|
|
|
|
|
OBJECT_DETAILS_TEMPLATE = """
|
|
|
|
<p><strong>{argument}</strong> object details:</p>
|
|
|
|
<ul>
|
|
|
|
{values}
|
|
|
|
</ul>
|
|
|
|
""".strip()
|
|
|
|
|
|
|
|
OBJECT_LIST_ITEM_TEMPLATE = """
|
|
|
|
<li>
|
2022-01-25 14:14:56 +01:00
|
|
|
<code>{value}</code>: <span class=api-field-type>{data_type}</span> {required} {description}{object_details}
|
2021-11-16 16:19:24 +01:00
|
|
|
</li>
|
|
|
|
""".strip()
|
|
|
|
|
|
|
|
OBJECT_DESCRIPTION_TEMPLATE = """
|
|
|
|
{description}
|
|
|
|
<p>{additional_information}</p>
|
|
|
|
""".strip()
|
|
|
|
|
|
|
|
OBJECT_CODE_TEMPLATE = "<code>{value}</code>".strip()
|
|
|
|
|
2017-12-27 08:57:47 +01:00
|
|
|
|
|
|
|
class MarkdownArgumentsTableGenerator(Extension):
|
2020-06-13 03:34:01 +02:00
|
|
|
def __init__(self, configs: Mapping[str, Any] = {}) -> None:
|
2017-12-27 08:57:47 +01:00
|
|
|
self.config = {
|
2021-02-12 08:20:45 +01:00
|
|
|
"base_path": [
|
|
|
|
".",
|
|
|
|
"Default location from which to evaluate relative paths for the JSON files.",
|
2021-02-12 08:19:30 +01:00
|
|
|
],
|
2017-12-27 08:57:47 +01:00
|
|
|
}
|
|
|
|
for key, value in configs.items():
|
|
|
|
self.setConfig(key, value)
|
|
|
|
|
2020-10-20 01:28:13 +02:00
|
|
|
def extendMarkdown(self, md: markdown.Markdown) -> None:
|
|
|
|
md.preprocessors.register(
|
2021-09-17 19:01:36 +02:00
|
|
|
APIArgumentsTablePreprocessor(md, self.getConfigs()),
|
|
|
|
"generate_api_arguments",
|
|
|
|
PREPROCESSOR_PRIORITES["generate_api_arguments"],
|
2017-12-27 08:57:47 +01:00
|
|
|
)
|
|
|
|
|
|
|
|
|
|
|
|
class APIArgumentsTablePreprocessor(Preprocessor):
|
2020-10-20 02:49:02 +02:00
|
|
|
def __init__(self, md: markdown.Markdown, config: Mapping[str, Any]) -> None:
|
2020-04-09 21:51:58 +02:00
|
|
|
super().__init__(md)
|
2021-02-12 08:20:45 +01:00
|
|
|
self.base_path = config["base_path"]
|
2017-12-27 08:57:47 +01:00
|
|
|
|
|
|
|
def run(self, lines: List[str]) -> List[str]:
|
|
|
|
done = False
|
|
|
|
while not done:
|
|
|
|
for line in lines:
|
|
|
|
loc = lines.index(line)
|
|
|
|
match = REGEXP.search(line)
|
|
|
|
|
2018-07-13 14:09:03 +02:00
|
|
|
if not match:
|
|
|
|
continue
|
|
|
|
|
|
|
|
filename = match.group(1)
|
|
|
|
doc_name = match.group(2)
|
|
|
|
filename = os.path.expanduser(filename)
|
|
|
|
|
2021-02-12 08:20:45 +01:00
|
|
|
is_openapi_format = filename.endswith(".yaml")
|
2018-07-13 14:09:03 +02:00
|
|
|
|
|
|
|
if not os.path.isabs(filename):
|
|
|
|
parent_dir = self.base_path
|
|
|
|
filename = os.path.normpath(os.path.join(parent_dir, filename))
|
|
|
|
|
|
|
|
if is_openapi_format:
|
2021-02-12 08:20:45 +01:00
|
|
|
endpoint, method = doc_name.rsplit(":", 1)
|
python: Convert assignment type annotations to Python 3.6 style.
This commit was split by tabbott; this piece covers the vast majority
of files in Zulip, but excludes scripts/, tools/, and puppet/ to help
ensure we at least show the right error messages for Xenial systems.
We can likely further refine the remaining pieces with some testing.
Generated by com2ann, with whitespace fixes and various manual fixes
for runtime issues:
- invoiced_through: Optional[LicenseLedger] = models.ForeignKey(
+ invoiced_through: Optional["LicenseLedger"] = models.ForeignKey(
-_apns_client: Optional[APNsClient] = None
+_apns_client: Optional["APNsClient"] = None
- notifications_stream: Optional[Stream] = models.ForeignKey('Stream', related_name='+', null=True, blank=True, on_delete=CASCADE)
- signup_notifications_stream: Optional[Stream] = models.ForeignKey('Stream', related_name='+', null=True, blank=True, on_delete=CASCADE)
+ notifications_stream: Optional["Stream"] = models.ForeignKey('Stream', related_name='+', null=True, blank=True, on_delete=CASCADE)
+ signup_notifications_stream: Optional["Stream"] = models.ForeignKey('Stream', related_name='+', null=True, blank=True, on_delete=CASCADE)
- author: Optional[UserProfile] = models.ForeignKey('UserProfile', blank=True, null=True, on_delete=CASCADE)
+ author: Optional["UserProfile"] = models.ForeignKey('UserProfile', blank=True, null=True, on_delete=CASCADE)
- bot_owner: Optional[UserProfile] = models.ForeignKey('self', null=True, on_delete=models.SET_NULL)
+ bot_owner: Optional["UserProfile"] = models.ForeignKey('self', null=True, on_delete=models.SET_NULL)
- default_sending_stream: Optional[Stream] = models.ForeignKey('zerver.Stream', null=True, related_name='+', on_delete=CASCADE)
- default_events_register_stream: Optional[Stream] = models.ForeignKey('zerver.Stream', null=True, related_name='+', on_delete=CASCADE)
+ default_sending_stream: Optional["Stream"] = models.ForeignKey('zerver.Stream', null=True, related_name='+', on_delete=CASCADE)
+ default_events_register_stream: Optional["Stream"] = models.ForeignKey('zerver.Stream', null=True, related_name='+', on_delete=CASCADE)
-descriptors_by_handler_id: Dict[int, ClientDescriptor] = {}
+descriptors_by_handler_id: Dict[int, "ClientDescriptor"] = {}
-worker_classes: Dict[str, Type[QueueProcessingWorker]] = {}
-queues: Dict[str, Dict[str, Type[QueueProcessingWorker]]] = {}
+worker_classes: Dict[str, Type["QueueProcessingWorker"]] = {}
+queues: Dict[str, Dict[str, Type["QueueProcessingWorker"]]] = {}
-AUTH_LDAP_REVERSE_EMAIL_SEARCH: Optional[LDAPSearch] = None
+AUTH_LDAP_REVERSE_EMAIL_SEARCH: Optional["LDAPSearch"] = None
Signed-off-by: Anders Kaseorg <anders@zulipchat.com>
2020-04-22 01:09:50 +02:00
|
|
|
arguments: List[Dict[str, Any]] = []
|
2018-07-13 14:09:03 +02:00
|
|
|
|
|
|
|
try:
|
|
|
|
arguments = get_openapi_parameters(endpoint, method)
|
|
|
|
except KeyError as e:
|
|
|
|
# Don't raise an exception if the "parameters"
|
|
|
|
# field is missing; we assume that's because the
|
|
|
|
# endpoint doesn't accept any parameters
|
2021-02-12 08:20:45 +01:00
|
|
|
if e.args != ("parameters",):
|
2018-07-13 14:09:03 +02:00
|
|
|
raise e
|
|
|
|
else:
|
2020-04-09 21:51:58 +02:00
|
|
|
with open(filename) as fp:
|
2020-03-27 05:03:26 +01:00
|
|
|
json_obj = json.load(fp)
|
2018-07-13 14:09:03 +02:00
|
|
|
arguments = json_obj[doc_name]
|
|
|
|
|
|
|
|
if arguments:
|
2021-11-16 16:19:24 +01:00
|
|
|
text = self.render_parameters(arguments)
|
2021-06-23 17:27:22 +02:00
|
|
|
# We want to show this message only if the parameters
|
|
|
|
# description doesn't say anything else.
|
|
|
|
elif is_openapi_format and get_parameters_description(endpoint, method) == "":
|
2021-02-12 08:20:45 +01:00
|
|
|
text = ["This endpoint does not accept any parameters."]
|
2021-06-23 17:27:22 +02:00
|
|
|
else:
|
|
|
|
text = []
|
2018-07-13 14:09:03 +02:00
|
|
|
# The line that contains the directive to include the macro
|
|
|
|
# may be preceded or followed by text or tags, in that case
|
|
|
|
# we need to make sure that any preceding or following text
|
|
|
|
# stays the same.
|
|
|
|
line_split = REGEXP.split(line, maxsplit=0)
|
|
|
|
preceding = line_split[0]
|
|
|
|
following = line_split[-1]
|
2020-09-02 06:59:07 +02:00
|
|
|
text = [preceding, *text, following]
|
2021-02-12 08:19:30 +01:00
|
|
|
lines = lines[:loc] + text + lines[loc + 1 :]
|
2018-07-13 14:09:03 +02:00
|
|
|
break
|
2017-12-27 08:57:47 +01:00
|
|
|
else:
|
|
|
|
done = True
|
|
|
|
return lines
|
|
|
|
|
2021-11-16 16:19:24 +01:00
|
|
|
def render_parameters(self, arguments: Sequence[Mapping[str, Any]]) -> List[str]:
|
|
|
|
parameters = []
|
2017-12-27 08:57:47 +01:00
|
|
|
|
|
|
|
md_engine = markdown.Markdown(extensions=[])
|
2021-02-12 08:20:45 +01:00
|
|
|
arguments = sorted(arguments, key=lambda argument: "deprecated" in argument)
|
2017-12-27 08:57:47 +01:00
|
|
|
for argument in arguments:
|
2021-11-16 16:19:24 +01:00
|
|
|
name = argument.get("argument") or argument.get("name")
|
2021-02-12 08:20:45 +01:00
|
|
|
description = argument["description"]
|
2022-01-27 20:18:59 +01:00
|
|
|
enums = argument.get("schema", {}).get("enum")
|
|
|
|
if enums is not None:
|
|
|
|
formatted_enums = [
|
|
|
|
OBJECT_CODE_TEMPLATE.format(value=json.dumps(enum)) for enum in enums
|
|
|
|
]
|
|
|
|
description += "\nMust be one of: {}. ".format(", ".join(formatted_enums))
|
2018-07-26 20:48:34 +02:00
|
|
|
|
2021-02-12 08:20:45 +01:00
|
|
|
default = argument.get("schema", {}).get("default")
|
2018-07-26 20:48:34 +02:00
|
|
|
if default is not None:
|
2021-02-12 08:20:45 +01:00
|
|
|
description += f"\nDefaults to `{json.dumps(default)}`."
|
2020-12-10 15:58:12 +01:00
|
|
|
data_type = ""
|
2021-02-12 08:20:45 +01:00
|
|
|
if "schema" in argument:
|
|
|
|
data_type = generate_data_type(argument["schema"])
|
2020-12-10 15:58:12 +01:00
|
|
|
else:
|
2021-02-12 08:20:45 +01:00
|
|
|
data_type = generate_data_type(argument["content"]["application/json"]["schema"])
|
2018-07-26 20:48:34 +02:00
|
|
|
|
2020-03-27 05:03:26 +01:00
|
|
|
# TODO: OpenAPI allows indicating where the argument goes
|
|
|
|
# (path, querystring, form data...). We should document this detail.
|
2020-05-11 16:26:33 +02:00
|
|
|
example = ""
|
2021-02-12 08:20:45 +01:00
|
|
|
if "example" in argument:
|
2021-04-07 18:47:48 +02:00
|
|
|
# We use this style without explicit JSON encoding for
|
|
|
|
# integers, strings, and booleans.
|
|
|
|
# * For booleans, JSON encoding correctly corrects for Python's
|
|
|
|
# str(True)="True" not matching the encoding of "true".
|
|
|
|
# * For strings, doing so nicely results in strings being quoted
|
|
|
|
# in the documentation, improving readability.
|
|
|
|
# * For integers, it is a noop, since json.dumps(3) == str(3) == "3".
|
|
|
|
example = json.dumps(argument["example"])
|
2020-05-11 16:26:33 +02:00
|
|
|
else:
|
2021-02-12 08:20:45 +01:00
|
|
|
example = json.dumps(argument["content"]["application/json"]["example"])
|
2020-05-11 16:26:33 +02:00
|
|
|
|
2020-06-11 21:35:13 +02:00
|
|
|
required_string: str = "required"
|
2021-02-12 08:20:45 +01:00
|
|
|
if argument.get("in", "") == "path":
|
2020-06-11 21:35:13 +02:00
|
|
|
# Any path variable is required
|
2021-02-12 08:20:45 +01:00
|
|
|
assert argument["required"]
|
|
|
|
required_string = "required in path"
|
2020-06-11 21:35:13 +02:00
|
|
|
|
2021-02-12 08:20:45 +01:00
|
|
|
if argument.get("required", False):
|
2020-06-11 21:35:13 +02:00
|
|
|
required_block = f'<span class="api-argument-required">{required_string}</span>'
|
|
|
|
else:
|
|
|
|
required_block = '<span class="api-argument-optional">optional</span>'
|
|
|
|
|
2021-02-16 13:01:36 +01:00
|
|
|
check_deprecated_consistency(argument, description)
|
2021-02-12 08:20:45 +01:00
|
|
|
if argument.get("deprecated", False):
|
2020-06-26 16:18:27 +02:00
|
|
|
deprecated_block = '<span class="api-argument-deprecated">Deprecated</span>'
|
|
|
|
else:
|
2021-02-12 08:20:45 +01:00
|
|
|
deprecated_block = ""
|
2020-06-26 16:18:27 +02:00
|
|
|
|
2021-11-16 16:19:24 +01:00
|
|
|
object_block = ""
|
|
|
|
# TODO: There are some endpoint parameters with object properties
|
|
|
|
# that are not defined in `zerver/openapi/zulip.yaml`
|
|
|
|
if "object" in data_type:
|
|
|
|
if "schema" in argument:
|
|
|
|
object_schema = argument["schema"]
|
|
|
|
else:
|
|
|
|
object_schema = argument["content"]["application/json"]["schema"]
|
|
|
|
|
|
|
|
if "items" in object_schema and "properties" in object_schema["items"]:
|
|
|
|
object_block = self.render_object_details(object_schema["items"], str(name))
|
|
|
|
elif "properties" in object_schema:
|
|
|
|
object_block = self.render_object_details(object_schema, str(name))
|
|
|
|
|
|
|
|
parameters.append(
|
|
|
|
API_PARAMETER_TEMPLATE.format(
|
|
|
|
argument=name,
|
2021-02-12 08:19:30 +01:00
|
|
|
example=escape_html(example),
|
|
|
|
required=required_block,
|
|
|
|
deprecated=deprecated_block,
|
|
|
|
description=md_engine.convert(description),
|
2021-11-16 16:19:24 +01:00
|
|
|
type=(data_type),
|
|
|
|
object_details=object_block,
|
|
|
|
)
|
|
|
|
)
|
|
|
|
|
|
|
|
return parameters
|
|
|
|
|
|
|
|
def render_object_details(self, schema: Mapping[str, Any], name: str) -> str:
|
|
|
|
md_engine = markdown.Markdown(extensions=[])
|
|
|
|
li_elements = []
|
|
|
|
|
|
|
|
object_values = schema.get("properties", {})
|
|
|
|
for value in object_values:
|
|
|
|
|
|
|
|
description = ""
|
|
|
|
if "description" in object_values[value]:
|
|
|
|
description = object_values[value]["description"]
|
|
|
|
|
|
|
|
# check for default, enum, required or example in documentation
|
|
|
|
additions: List[str] = []
|
|
|
|
|
|
|
|
default = object_values.get(value, {}).get("default")
|
|
|
|
if default is not None:
|
|
|
|
formatted_default = OBJECT_CODE_TEMPLATE.format(value=json.dumps(default))
|
|
|
|
additions += f"Defaults to {formatted_default}. "
|
|
|
|
|
|
|
|
enums = object_values.get(value, {}).get("enum")
|
|
|
|
if enums is not None:
|
|
|
|
formatted_enums = [
|
|
|
|
OBJECT_CODE_TEMPLATE.format(value=json.dumps(enum)) for enum in enums
|
|
|
|
]
|
|
|
|
additions += "Must be one of: {}. ".format(", ".join(formatted_enums))
|
|
|
|
|
|
|
|
if "example" in object_values[value]:
|
|
|
|
example = json.dumps(object_values[value]["example"])
|
|
|
|
formatted_example = OBJECT_CODE_TEMPLATE.format(value=escape_html(example))
|
2022-01-25 14:14:56 +01:00
|
|
|
additions += (
|
|
|
|
f'<span class="api-argument-example-label">Example</span>: {formatted_example}'
|
|
|
|
)
|
2021-11-16 16:19:24 +01:00
|
|
|
|
|
|
|
if len(additions) > 0:
|
|
|
|
additional_information = "".join(additions).strip()
|
|
|
|
description_final = OBJECT_DESCRIPTION_TEMPLATE.format(
|
|
|
|
description=md_engine.convert(description),
|
|
|
|
additional_information=additional_information,
|
2021-02-12 08:19:30 +01:00
|
|
|
)
|
2021-11-16 16:19:24 +01:00
|
|
|
else:
|
|
|
|
description_final = md_engine.convert(description)
|
|
|
|
|
2022-01-25 14:14:56 +01:00
|
|
|
required_block = ""
|
|
|
|
if "required" in schema:
|
|
|
|
if value in schema["required"]:
|
|
|
|
required_block = '<span class="api-argument-required">required</span>'
|
|
|
|
else:
|
|
|
|
required_block = '<span class="api-argument-optional">optional</span>'
|
|
|
|
|
2021-11-16 16:19:24 +01:00
|
|
|
data_type = generate_data_type(object_values[value])
|
|
|
|
|
|
|
|
details = ""
|
|
|
|
if "object" in data_type and "properties" in object_values[value]:
|
|
|
|
details += self.render_object_details(object_values[value], str(value))
|
|
|
|
|
|
|
|
li = OBJECT_LIST_ITEM_TEMPLATE.format(
|
|
|
|
value=value,
|
|
|
|
data_type=data_type,
|
2022-01-25 14:14:56 +01:00
|
|
|
required=required_block,
|
2021-11-16 16:19:24 +01:00
|
|
|
description=description_final,
|
|
|
|
object_details=details,
|
2021-02-12 08:19:30 +01:00
|
|
|
)
|
2017-12-27 08:57:47 +01:00
|
|
|
|
2021-11-16 16:19:24 +01:00
|
|
|
li_elements.append(li)
|
|
|
|
|
|
|
|
object_details = OBJECT_DETAILS_TEMPLATE.format(
|
|
|
|
argument=name,
|
|
|
|
values="\n".join(li_elements),
|
|
|
|
)
|
|
|
|
return object_details
|
2017-12-27 08:57:47 +01:00
|
|
|
|
2021-02-12 08:19:30 +01:00
|
|
|
|
2017-12-27 08:57:47 +01:00
|
|
|
def makeExtension(*args: Any, **kwargs: str) -> MarkdownArgumentsTableGenerator:
|
|
|
|
return MarkdownArgumentsTableGenerator(kwargs)
|
2020-12-10 15:58:12 +01:00
|
|
|
|
2021-02-12 08:19:30 +01:00
|
|
|
|
2020-12-10 15:58:12 +01:00
|
|
|
def generate_data_type(schema: Mapping[str, Any]) -> str:
|
|
|
|
data_type = ""
|
2021-02-12 08:20:45 +01:00
|
|
|
if "oneOf" in schema:
|
|
|
|
for item in schema["oneOf"]:
|
2020-12-10 15:58:12 +01:00
|
|
|
data_type = data_type + generate_data_type(item) + " | "
|
|
|
|
data_type = data_type[:-3]
|
2021-02-12 08:20:45 +01:00
|
|
|
elif "items" in schema:
|
|
|
|
data_type = "(" + generate_data_type(schema["items"]) + ")[]"
|
2020-12-10 15:58:12 +01:00
|
|
|
else:
|
2021-02-12 08:20:45 +01:00
|
|
|
data_type = schema["type"]
|
2021-11-16 16:37:24 +01:00
|
|
|
if "nullable" in schema and schema["nullable"]:
|
|
|
|
data_type = data_type + " | null"
|
2020-12-10 15:58:12 +01:00
|
|
|
return data_type
|