2019-05-18 10:42:04 +02:00
|
|
|
import re
|
2019-06-03 14:22:03 +02:00
|
|
|
from typing import Dict, Tuple, NewType
|
2019-03-18 10:18:57 -06:00
|
|
|
|
|
|
|
|
2019-05-18 10:42:04 +02:00
|
|
|
FUNCTION_REGEX = r'^def [\w_]*'
|
|
|
|
|
2019-06-03 14:22:03 +02:00
|
|
|
SpecObject = NewType('SpecObjects', Tuple[Dict[str, str], Dict[str, str], Dict[str, str], Dict[str, str]])
|
2019-05-18 10:42:04 +02:00
|
|
|
|
2019-06-03 14:22:03 +02:00
|
|
|
|
|
|
|
def get_spec(file_name: str) -> SpecObject:
|
|
|
|
"""
|
|
|
|
Takes in the file name of a spec.md file, opens it and returns the following objects:
|
|
|
|
functions = {function_name: function_code}
|
|
|
|
constants= {constant_name: constant_code}
|
|
|
|
ssz_objects= {object_name: object}
|
|
|
|
|
|
|
|
Note: This function makes heavy use of the inherent ordering of dicts,
|
|
|
|
if this is not supported by your python version, it will not work.
|
|
|
|
"""
|
2019-05-24 16:51:21 +02:00
|
|
|
pulling_from = None # line number of start of latest object
|
|
|
|
current_name = None # most recent section title
|
2019-11-19 20:16:40 +01:00
|
|
|
functions: Dict[str, str] = {}
|
|
|
|
constants: Dict[str, str] = {}
|
|
|
|
ssz_objects: Dict[str, str] = {}
|
2019-05-18 10:42:04 +02:00
|
|
|
function_matcher = re.compile(FUNCTION_REGEX)
|
2019-06-17 10:48:33 -04:00
|
|
|
is_ssz = False
|
2019-11-19 20:16:40 +01:00
|
|
|
custom_types: Dict[str, str] = {}
|
2019-05-07 12:13:22 +01:00
|
|
|
for linenum, line in enumerate(open(file_name).readlines()):
|
2019-03-18 10:18:57 -06:00
|
|
|
line = line.rstrip()
|
|
|
|
if pulling_from is None and len(line) > 0 and line[0] == '#' and line[-1] == '`':
|
2019-03-19 11:39:19 +08:00
|
|
|
current_name = line[line[:-1].rfind('`') + 1: -1]
|
2019-03-18 10:18:57 -06:00
|
|
|
if line[:9] == '```python':
|
|
|
|
assert pulling_from is None
|
|
|
|
pulling_from = linenum + 1
|
|
|
|
elif line[:3] == '```':
|
2019-05-18 10:42:04 +02:00
|
|
|
pulling_from = None
|
2019-03-18 10:18:57 -06:00
|
|
|
else:
|
2019-06-03 14:22:03 +02:00
|
|
|
# Handle function definitions & ssz_objects
|
2019-05-18 10:42:04 +02:00
|
|
|
if pulling_from is not None:
|
2019-06-05 15:29:26 +02:00
|
|
|
# SSZ Object
|
|
|
|
if len(line) > 18 and line[:6] == 'class ' and line[-12:] == '(Container):':
|
|
|
|
name = line[6:-12]
|
|
|
|
# Check consistency with markdown header
|
|
|
|
assert name == current_name
|
|
|
|
is_ssz = True
|
|
|
|
# function definition
|
|
|
|
elif function_matcher.match(line) is not None:
|
|
|
|
current_name = function_matcher.match(line).group(0)
|
|
|
|
is_ssz = False
|
|
|
|
if is_ssz:
|
2019-05-20 14:00:54 +02:00
|
|
|
ssz_objects[current_name] = ssz_objects.get(current_name, '') + line + '\n'
|
2019-06-05 15:29:26 +02:00
|
|
|
else:
|
2019-05-20 14:00:54 +02:00
|
|
|
functions[current_name] = functions.get(current_name, '') + line + '\n'
|
2019-06-15 16:57:50 -04:00
|
|
|
# Handle constant and custom types table entries
|
2019-03-18 10:18:57 -06:00
|
|
|
elif pulling_from is None and len(line) > 0 and line[0] == '|':
|
|
|
|
row = line[1:].split('|')
|
|
|
|
if len(row) >= 2:
|
|
|
|
for i in range(2):
|
|
|
|
row[i] = row[i].strip().strip('`')
|
|
|
|
if '`' in row[i]:
|
|
|
|
row[i] = row[i][:row[i].find('`')]
|
2019-07-01 00:20:31 +02:00
|
|
|
is_constant_def = True
|
|
|
|
if row[0][0] not in 'ABCDEFGHIJKLMNOPQRSTUVWXYZ_':
|
|
|
|
is_constant_def = False
|
|
|
|
for c in row[0]:
|
|
|
|
if c not in 'ABCDEFGHIJKLMNOPQRSTUVWXYZ_0123456789':
|
|
|
|
is_constant_def = False
|
|
|
|
if is_constant_def:
|
2019-09-27 13:02:16 +09:00
|
|
|
constants[row[0]] = row[1].replace('**TBD**', '2**32')
|
2019-07-01 00:20:31 +02:00
|
|
|
elif row[1].startswith('uint') or row[1].startswith('Bytes'):
|
2019-06-15 16:57:50 -04:00
|
|
|
custom_types[row[0]] = row[1]
|
2019-11-19 20:16:40 +01:00
|
|
|
return SpecObject((functions, custom_types, constants, ssz_objects))
|