2017-12-27 08:57:47 +01:00
|
|
|
import re
|
|
|
|
import os
|
2020-03-27 05:03:26 +01:00
|
|
|
import json
|
2017-12-27 08:57:47 +01:00
|
|
|
|
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-02-23 18:10:42 +01:00
|
|
|
from zerver.openapi.openapi import get_openapi_parameters
|
2017-12-27 08:57:47 +01:00
|
|
|
from typing import Any, Dict, Optional, List
|
|
|
|
import markdown
|
|
|
|
|
2018-05-15 19:28:42 +02:00
|
|
|
REGEXP = re.compile(r'\{generate_api_arguments_table\|\s*(.+?)\s*\|\s*(.+)\s*\}')
|
2017-12-27 08:57:47 +01:00
|
|
|
|
|
|
|
|
|
|
|
class MarkdownArgumentsTableGenerator(Extension):
|
2018-05-17 17:32:40 +02:00
|
|
|
def __init__(self, configs: Optional[Dict[str, Any]]=None) -> None:
|
|
|
|
if configs is None:
|
|
|
|
configs = {}
|
2017-12-27 08:57:47 +01:00
|
|
|
self.config = {
|
|
|
|
'base_path': ['.', 'Default location from which to evaluate relative paths for the JSON files.'],
|
|
|
|
}
|
|
|
|
for key, value in configs.items():
|
|
|
|
self.setConfig(key, value)
|
|
|
|
|
|
|
|
def extendMarkdown(self, md: markdown.Markdown, md_globals: Dict[str, Any]) -> None:
|
|
|
|
md.preprocessors.add(
|
|
|
|
'generate_api_arguments', APIArgumentsTablePreprocessor(md, self.getConfigs()), '_begin'
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
|
|
class APIArgumentsTablePreprocessor(Preprocessor):
|
|
|
|
def __init__(self, md: markdown.Markdown, config: Dict[str, Any]) -> None:
|
2020-04-09 21:51:58 +02:00
|
|
|
super().__init__(md)
|
2017-12-27 08:57:47 +01:00
|
|
|
self.base_path = config['base_path']
|
|
|
|
|
|
|
|
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)
|
|
|
|
|
|
|
|
is_openapi_format = filename.endswith('.yaml')
|
|
|
|
|
|
|
|
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:
|
|
|
|
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
|
|
|
|
if e.args != ('parameters',):
|
|
|
|
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:
|
|
|
|
text = self.render_table(arguments)
|
|
|
|
else:
|
|
|
|
text = ['This endpoint does not consume any arguments.']
|
|
|
|
# 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]
|
|
|
|
text = [preceding] + text + [following]
|
|
|
|
lines = lines[:loc] + text + lines[loc+1:]
|
|
|
|
break
|
2017-12-27 08:57:47 +01:00
|
|
|
else:
|
|
|
|
done = True
|
|
|
|
return lines
|
|
|
|
|
|
|
|
def render_table(self, arguments: List[Dict[str, Any]]) -> List[str]:
|
2020-03-27 05:03:26 +01:00
|
|
|
# TODO: Fix naming now that this no longer renders a table.
|
2017-12-27 08:57:47 +01:00
|
|
|
table = []
|
2020-03-27 05:03:26 +01:00
|
|
|
argument_template = """
|
|
|
|
<div class="api-argument">
|
|
|
|
<p class="api-argument-name"><strong>{argument}</strong> {required}</p>
|
|
|
|
<div class="api-example">
|
|
|
|
<span class="api-argument-example-label">Example</span>: <code>{example}</code>
|
|
|
|
</div>
|
|
|
|
<div class="api-description">{description}</div>
|
|
|
|
<hr />
|
|
|
|
</div>"""
|
2017-12-27 08:57:47 +01:00
|
|
|
|
|
|
|
md_engine = markdown.Markdown(extensions=[])
|
|
|
|
|
|
|
|
for argument in arguments:
|
2018-07-26 20:48:34 +02:00
|
|
|
description = argument['description']
|
|
|
|
|
2019-10-07 10:18:18 +02:00
|
|
|
oneof = ['`' + str(item) + '`'
|
2018-05-15 19:28:42 +02:00
|
|
|
for item in argument.get('schema', {}).get('enum', [])]
|
|
|
|
if oneof:
|
|
|
|
description += '\nMust be one of: {}.'.format(', '.join(oneof))
|
2018-07-26 20:48:34 +02:00
|
|
|
|
|
|
|
default = argument.get('schema', {}).get('default')
|
|
|
|
if default is not None:
|
2020-03-27 05:03:26 +01:00
|
|
|
description += '\nDefaults to `{}`.'.format(json.dumps(default))
|
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.
|
|
|
|
table.append(argument_template.format(
|
2018-05-15 19:28:42 +02:00
|
|
|
argument=argument.get('argument') or argument.get('name'),
|
2018-07-13 06:59:48 +02:00
|
|
|
# Show this as JSON to avoid changing the quoting style, which
|
|
|
|
# may cause problems with JSON encoding.
|
2020-03-27 05:03:26 +01:00
|
|
|
example=escape_html(json.dumps(argument['example'])),
|
|
|
|
required='<span class="api-argument-required">required</span>' if argument.get('required')
|
|
|
|
else '<span class="api-argument-optional">optional</span>',
|
2018-05-15 19:28:42 +02:00
|
|
|
description=md_engine.convert(description),
|
2017-12-27 08:57:47 +01:00
|
|
|
))
|
|
|
|
|
|
|
|
return table
|
|
|
|
|
|
|
|
def makeExtension(*args: Any, **kwargs: str) -> MarkdownArgumentsTableGenerator:
|
|
|
|
return MarkdownArgumentsTableGenerator(kwargs)
|