1d5d6a0442
Includes: - Better windows support in json generator. - Add PDL2JSON, don't paste descriptions as comments into generated code. - [inspector_protocol] generated only used types R=pfeldman@chromium.org Bug: none Cq-Include-Trybots: master.tryserver.blink:linux_trusty_blink_rel Change-Id: I3c634359b3ac4b00293ac7c5ee224dab53aae7ca Reviewed-on: https://chromium-review.googlesource.com/810204 Commit-Queue: Aleksey Kozyatinskiy <kozyatinskiy@chromium.org> Reviewed-by: Pavel Feldman <pfeldman@chromium.org> Cr-Commit-Position: refs/heads/master@{#49949}
184 lines
5.6 KiB
Python
184 lines
5.6 KiB
Python
# Copyright 2017 The Chromium Authors. All rights reserved.
|
|
# Use of this source code is governed by a BSD-style license that can be
|
|
# found in the LICENSE file.
|
|
|
|
import collections
|
|
import json
|
|
import os.path
|
|
import re
|
|
import sys
|
|
|
|
file_name = None
|
|
description = ''
|
|
|
|
primitiveTypes = ['integer', 'number', 'boolean', 'string', 'object', 'any', 'array']
|
|
|
|
|
|
def assignType(item, type, isArray=False):
|
|
if isArray:
|
|
item['type'] = 'array'
|
|
item['items'] = collections.OrderedDict()
|
|
assignType(item['items'], type)
|
|
return
|
|
|
|
if type == 'enum':
|
|
type = 'string'
|
|
if type in primitiveTypes:
|
|
item['type'] = type
|
|
else:
|
|
item['$ref'] = type
|
|
|
|
|
|
def createItem(d, experimental, deprecated, name=None):
|
|
result = collections.OrderedDict(d)
|
|
if name:
|
|
result['name'] = name
|
|
global description
|
|
if description:
|
|
result['description'] = description.strip()
|
|
if experimental:
|
|
result['experimental'] = True
|
|
if deprecated:
|
|
result['deprecated'] = True
|
|
return result
|
|
|
|
|
|
def parse(data):
|
|
protocol = collections.OrderedDict()
|
|
protocol['version'] = collections.OrderedDict()
|
|
protocol['domains'] = []
|
|
domain = None
|
|
item = None
|
|
subitems = None
|
|
nukeDescription = False
|
|
global description
|
|
lines = data.split('\n')
|
|
for i in range(0, len(lines)):
|
|
if nukeDescription:
|
|
description = ''
|
|
nukeDescription = False
|
|
line = lines[i]
|
|
trimLine = line.strip()
|
|
|
|
if trimLine.startswith('#'):
|
|
if len(description):
|
|
description += '\n'
|
|
description += trimLine[2:]
|
|
continue
|
|
else:
|
|
nukeDescription = True
|
|
|
|
if len(trimLine) == 0:
|
|
continue
|
|
|
|
match = re.compile('^(experimental )?(deprecated )?domain (.*)').match(line)
|
|
if match:
|
|
domain = createItem({'domain' : match.group(3)}, match.group(1), match.group(2))
|
|
protocol['domains'].append(domain)
|
|
continue
|
|
|
|
match = re.compile('^ depends on ([^\s]+)').match(line)
|
|
if match:
|
|
if 'dependencies' not in domain:
|
|
domain['dependencies'] = []
|
|
domain['dependencies'].append(match.group(1))
|
|
continue
|
|
|
|
match = re.compile('^ (experimental )?(deprecated )?type (.*) extends (array of )?([^\s]+)').match(line)
|
|
if match:
|
|
if 'types' not in domain:
|
|
domain['types'] = []
|
|
item = createItem({'id': match.group(3)}, match.group(1), match.group(2))
|
|
assignType(item, match.group(5), match.group(4))
|
|
domain['types'].append(item)
|
|
continue
|
|
|
|
match = re.compile('^ (experimental )?(deprecated )?(command|event) (.*)').match(line)
|
|
if match:
|
|
list = []
|
|
if match.group(3) == 'command':
|
|
if 'commands' in domain:
|
|
list = domain['commands']
|
|
else:
|
|
list = domain['commands'] = []
|
|
else:
|
|
if 'events' in domain:
|
|
list = domain['events']
|
|
else:
|
|
list = domain['events'] = []
|
|
|
|
item = createItem({}, match.group(1), match.group(2), match.group(4))
|
|
list.append(item)
|
|
continue
|
|
|
|
match = re.compile('^ (experimental )?(deprecated )?(optional )?(array of )?([^\s]+) ([^\s]+)').match(line)
|
|
if match:
|
|
param = createItem({}, match.group(1), match.group(2), match.group(6))
|
|
if match.group(3):
|
|
param['optional'] = True
|
|
assignType(param, match.group(5), match.group(4))
|
|
if match.group(5) == 'enum':
|
|
enumliterals = param['enum'] = []
|
|
subitems.append(param)
|
|
continue
|
|
|
|
match = re.compile('^ (parameters|returns|properties)').match(line)
|
|
if match:
|
|
subitems = item[match.group(1)] = []
|
|
continue
|
|
|
|
match = re.compile('^ enum').match(line)
|
|
if match:
|
|
enumliterals = item['enum'] = []
|
|
continue
|
|
|
|
match = re.compile('^version').match(line)
|
|
if match:
|
|
continue
|
|
|
|
match = re.compile('^ major (\d+)').match(line)
|
|
if match:
|
|
protocol['version']['major'] = match.group(1)
|
|
continue
|
|
|
|
match = re.compile('^ minor (\d+)').match(line)
|
|
if match:
|
|
protocol['version']['minor'] = match.group(1)
|
|
continue
|
|
|
|
match = re.compile('^ redirect ([^\s]+)').match(line)
|
|
if match:
|
|
item['redirect'] = match.group(1)
|
|
continue
|
|
|
|
match = re.compile('^ ( )?[^\s]+$').match(line)
|
|
if match:
|
|
# enum literal
|
|
enumliterals.append(trimLine)
|
|
continue
|
|
|
|
print 'Error in %s:%s, illegal token: \t%s' % (file_name, i, line)
|
|
sys.exit(1)
|
|
return protocol
|
|
|
|
def main(argv):
|
|
if len(argv) < 2:
|
|
sys.stderr.write("Usage: %s <protocol.pdl> <protocol.json>\n" % sys.argv[0])
|
|
return 1
|
|
global file_name
|
|
file_name = os.path.normpath(argv[0])
|
|
input_file = open(file_name, "r")
|
|
pdl_string = input_file.read()
|
|
protocol = parse(pdl_string)
|
|
output_file = open(argv[0].replace('.pdl', '.json'), 'wb')
|
|
json.dump(protocol, output_file, indent=4, separators=(',', ': '))
|
|
output_file.close()
|
|
|
|
output_file = open(os.path.normpath(argv[1]), 'wb')
|
|
json.dump(protocol, output_file, indent=4, separators=(',', ': '))
|
|
output_file.close()
|
|
|
|
|
|
if __name__ == '__main__':
|
|
sys.exit(main(sys.argv[1:]))
|