forked from caproto/caproto
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathgenerate_headers.py
160 lines (135 loc) · 5.78 KB
/
generate_headers.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
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
# Parse excerpts from
# http://www.aps.anl.gov/epics/docs/CAproto.html#secVirtualCircuitLifeCycle
# and generate Python functions that return bytestrings.
import re
import os
from collections import namedtuple
from jinja2 import Environment, FileSystemLoader
from bs4 import BeautifulSoup
def getpath(*args):
"""Get absolute path of joined directories relative to this file"""
return os.path.abspath(os.path.join(os.path.dirname(__file__), *args))
Param = namedtuple('Param', ['standard_name', 'field', 'value', 'description'])
Command = namedtuple('Command', ['name', 'description',
'input_params', 'struct_args'])
COMMAND_NAME_PATTERN = re.compile("Command identifier for ([A-Z_]+).?")
def validate(params):
if not params[0].field == 'command':
raise ValueError("first cell of table should be 'Command', "
"not %s" % params[0].field)
if not len(params) == 6:
raise ValueError("table should have six rows")
def is_reserved(param):
return (param.field == 'reserved' or
'Must be 0' in param.description or
'Must be 1' in param.description or
'Payload size is constant' in param.description)
def handle_special_cases(command):
"""
Called by parse_commands.
This is here to separate the special-case manging from the rest of the
code.
"""
# These require the 'cid' param to be given twice for some reason.
if command.name in ('SearchRequest', 'NotFoundResponse'):
command = command._replace(input_params=command.input_params[:-1])
if command.name == 'RsrvIsUpResponse':
command = command._replace(name='Beacon')
return command
def parse_commands(h2):
"""
Parse HTML describing CA request/response spec into namedtuples.
"""
# Grab the description under the h2 heading for use in the docstring.
raw_desc = h2.find_next(string='Description: ').parent.parent.text
description = '\n'.join(raw_desc.replace('\t\t\t', ' ').split('\n')[1:])
# Select the <table> tags between this <h2> and the next </h2>.
next_h2 = h2.find_next('h2')
tables = h2.find_all_next('table')[:-len(next_h2.find_all_next('table'))]
# Figure out which tables are associated with 'Request' vs 'Response'.
# Ignore tables about 'Compatibility' with various versions of EPICS.
request_spec = []
response_spec = []
for table in tables:
table_title = table.previous_sibling.find('span').text
if table_title == 'Compatibility':
continue
h3_text = table.find_previous('h3').text
if 'Request' in h3_text:
request_spec.append(table)
elif 'Response' in h3_text:
response_spec.append(table)
else:
raise ValueError("expected 'Request' or 'Response' in h3text")
# Prase the tables under the headings 'Request' and 'Response' into
# namedtuples. Some have only one of these headings, or neither, so this
# may result in a list of 0, 1, or 2 commands.
commands = []
for suffix, tables in (('Request', request_spec), ('Response', response_spec)):
if not tables:
# There is no spec for this.
continue
if len(tables) == 1:
header_table, = tables
elif len(tables) == 2:
header_table, payload_table = tables
else:
raise ValueError("expected at most two tables per <h3>")
rows = header_table.find_all('tr')
params = []
STANDARD_NAMES = ('command payload_size data_type data_count '
'parameter1 parameter 2').split()
for row, sn in zip(rows[1:], STANDARD_NAMES): # exclude header row
field_td, val_td, desc_td = row.find_all('td')
field = field_td.find('tt').text.replace(' ', '_').lower()
if field == 'status_code':
field = 'status' # for consistency -- docs are loose with this
val = val_td.find('tt').text
desc = desc_td.text
params.append(Param(sn, field, val, desc))
validate(params)
name = COMMAND_NAME_PATTERN.match(params[0].description).group(1)
if name.startswith('CA_PROTO'):
name = name[len('CA_PROTO'):]
if name.startswith('CA_'):
name = name[len('CA_'):]
name = name.title().replace('_', '')
input_params = [p for p in params[1:] if not is_reserved(p)]
struct_args = [p.field
if p in input_params else int(p.value)
for p in params]
command = Command('{}{}'.format(name, suffix),
description, input_params, struct_args)
command = handle_special_cases(command)
commands.append(command)
return commands
JINJA_ENV = Environment(loader=FileSystemLoader(getpath('.')))
template = JINJA_ENV.get_template('headers.tpl')
def parse_section(h1):
"""
Apply ``parse_command`` to each <h2> tag betweeen this <h1> and the next.
"""
next_h1 = h1.find_next('h1')
h2s = h1.find_all_next('h2')[:-len(next_h1.find_all_next('h2'))]
commands = []
for h2 in h2s:
commands.extend(parse_commands(h2))
return commands
def write_commands(path=None):
"""
Generate _headers.py from headers.tpl and CAproto.html
"""
with open(getpath('CAproto.html')) as f:
soup = BeautifulSoup(f.read(), 'html.parser')
commands = []
for _id in ('secCommandsShared',
'secCommandsUDP',
'secCommandsTCP'):
h1 = soup.find(id=_id)
commands.extend(parse_section(h1))
if path is None:
path = getpath('.')
with open(os.path.join(path, 'caproto', '_headers.py'), 'w') as f:
f.write(template.render(commands=commands))
if __name__ == '__main__':
write_commands()