forked from SpaceApi/website
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathgenerate_api_docs.py
127 lines (108 loc) · 3.91 KB
/
generate_api_docs.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
import collections
import json
import sys
from typing import List
def visit_generic(name: str, data, nullable: bool, required: bool):
print('<header>')
full_type = data['type']
if full_type == 'array':
full_type += ' of %s' % data['items']['type']
print('<h3>%s</h3><span class="type">(%s)</span>' % (name, full_type))
if required:
print('<span class="tag required">required</span>')
if nullable:
print('<span class="tag nullable">nullable</span>')
print('</header>')
print('<details class="togglable">')
print('<summary>Details</summary>')
print('<div>')
if 'description' in data:
print('<p>%s</p>' % data['description'])
if 'enum' in data:
print('<h4>Valid values</h4>')
print('<p><code>%s</code></p>' % '</code> | <code>'.join(data['enum']))
if 'minItems' in data:
print('<h4>Minimum number of items</h4>')
print('<p>%s</p>' % data['minItems'])
def visit_object(name, data):
assert 'properties' in data
print('<h4>Nested object properties</h4>')
print('<ul class="group">')
visit(data['properties'], data.get('required', []))
print('</ul>')
def visit_array(name, data):
assert 'items' in data
items = data['items']
print('<h4>Nested array items</h4>')
if items['type'] == 'string':
print('<span>string</span>')
if 'enum' in items:
print('<h5>Valid values</h5>')
print('<p><code>%s</code></p>' % '</code> | <code>'.join(items['enum']))
elif items['type'] == 'object':
print('<ul class="group">')
visit(items['properties'], items.get('required', []))
print('</ul>')
def visit_string(name, data):
pass
def visit_number(name, data):
pass
def visit_boolean(name, data):
pass
def visit(properties, required_fields: List[str]):
for k, v in properties.items():
print('<li><section class="item">')
nullable = False
a = isinstance(v['type'], list)
b = len(v['type']) == 2
c = 'null' in v['type']
if a and b and c:
nullable = True
v['type'].remove('null')
v['type'] = v['type'][0]
visit_generic(k, v, nullable, k in required_fields)
if v['type'] == 'object':
visit_object(k, v)
elif v['type'] == 'string':
visit_string(k, v)
elif v['type'] == 'array':
visit_array(k, v)
elif v['type'] == 'number':
visit_number(k, v)
elif v['type'] == 'boolean':
visit_boolean(k, v)
else:
print('Non object: %s' % v['type'], file=sys.stderr)
sys.exit(1)
print('</div></details>')
print('</section></li>')
def process_version(path):
with open(path, 'r') as f:
schema = json.loads(f.read(), object_pairs_hook=collections.OrderedDict)
assert schema['type'] == 'object'
assert 'properties' in schema
print('_model: page')
print('---')
print('title: API Documentation')
print('---')
print('body:')
print()
print("It's highly recommended to use the explicit specified fields from the")
print("reference. If you need other fields additionally, please make a change request.")
print("Or prefix custom fields with `ext_` to make it clear the field is not part of")
print("the documented API. Consumers are not obligated to interpret any custom fields.")
print()
print('Most types are not nullable. That means that they may not contain the value "null",')
print("but they may be left away if they're not required.")
print()
print('<ul class="group apidocs">')
visit(schema['properties'], schema.get('required', []))
print('</ul>')
print('---')
print('_discoverable: yes')
print('---')
print('_slug: docs')
if len(sys.argv) != 2:
print('Usage: {} <path/to/schema.json>'.format(sys.argv[0]))
sys.exit(1)
process_version(sys.argv[1])