Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(91)

Side by Side Diff: tools/json_schema_compiler/compiler.py

Issue 12041098: Initial commit of the Dart Chrome Extension APIs generators (Closed) Base URL: http://git.chromium.org/chromium/src.git@file_path_bugfix
Patch Set: Small style & structure fixes to dart_generator, and similar. Created 7 years, 10 months ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View unified diff | Download patch
OLDNEW
1 #!/usr/bin/env python 1 #!/usr/bin/env python
2 # Copyright (c) 2012 The Chromium Authors. All rights reserved. 2 # Copyright (c) 2012 The Chromium Authors. All rights reserved.
3 # Use of this source code is governed by a BSD-style license that can be 3 # Use of this source code is governed by a BSD-style license that can be
4 # found in the LICENSE file. 4 # found in the LICENSE file.
5 """Generator for C++ structs from api json files. 5 """Generator for C++ structs from api json files.
6 6
7 The purpose of this tool is to remove the need for hand-written code that 7 The purpose of this tool is to remove the need for hand-written code that
8 converts to and from base::Value types when receiving javascript api calls. 8 converts to and from base::Value types when receiving javascript api calls.
9 Originally written for generating code for extension apis. Reference schemas 9 Originally written for generating code for extension apis. Reference schemas
10 are in chrome/common/extensions/api. 10 are in chrome/common/extensions/api.
11 11
12 Usage example: 12 Usage example:
13 compiler.py --root /home/Work/src --namespace extensions windows.json 13 compiler.py --root /home/Work/src --namespace extensions windows.json
14 tabs.json 14 tabs.json
15 compiler.py --destdir gen --root /home/Work/src 15 compiler.py --destdir gen --root /home/Work/src
16 --namespace extensions windows.json tabs.json 16 --namespace extensions windows.json tabs.json
17 """ 17 """
18 18
19 import cc_generator 19 from cpp_generator import CppGenerator
not at google - send to devlin 2013/02/02 00:45:48 I like this idiom. Let's make all the imports whic
sashab 2013/02/04 05:09:27 Done.
20 import cpp_type_generator 20 import cpp_type_generator
21 import h_generator 21 import h_generator
22 from dart_generator import DartGenerator
22 import idl_schema 23 import idl_schema
23 import json_schema 24 import json_schema
24 import model 25 import model
25 import schema_bundle_generator 26 from cpp_bundle_generator import CppBundleGenerator
26 27
27 import optparse 28 import optparse
28 import os.path 29 import os.path
29 import sys 30 import sys
30 31
31 def load_schema(schema): 32 def load_schema(schema):
32 schema_filename, schema_extension = os.path.splitext(schema) 33 schema_filename, schema_extension = os.path.splitext(schema)
33 34
34 if schema_extension == '.json': 35 if schema_extension == '.json':
35 api_defs = json_schema.Load(schema) 36 api_defs = json_schema.Load(schema)
36 elif schema_extension == '.idl': 37 elif schema_extension == '.idl':
37 api_defs = idl_schema.Load(schema) 38 api_defs = idl_schema.Load(schema)
38 else: 39 else:
39 sys.exit("Did not recognize file extension %s for schema %s" % 40 sys.exit('Did not recognize file extension %s for schema %s' %
40 (schema_extension, schema)) 41 (schema_extension, schema))
41 if len(api_defs) != 1: 42 if len(api_defs) != 1:
42 sys.exit("File %s has multiple schemas. Files are only allowed to contain a" 43 sys.exit('File %s has multiple schemas. Files are only allowed to contain a'
43 " single schema." % schema) 44 ' single schema.' % schema)
44 45
45 return api_defs 46 return api_defs
46 47
47 def handle_single_schema(filename, dest_dir, root, root_namespace): 48 if __name__ == '__main__':
48 schema = os.path.normpath(filename) 49 parser = optparse.OptionParser(
49 schema_filename, schema_extension = os.path.splitext(schema) 50 description='Generates a C++ model of an API from JSON schema',
50 path, short_filename = os.path.split(schema_filename) 51 usage='usage: %prog [option]... schema')
51 api_defs = json_schema.DeleteNocompileNodes(load_schema(schema)) 52 parser.add_option('-r', '--root', default='.',
53 help='logical include root directory. Path to schema files from specified'
54 'dir will be the include path.')
55 parser.add_option('-d', '--destdir',
56 help='root directory to output generated files.')
57 parser.add_option('-n', '--namespace', default='generated_api_schemas',
58 help='C++ namespace for generated files. e.g extensions::api.')
59 parser.add_option('-b', '--bundle', action='store_true', help=
60 '''if supplied, causes compiler to generate bundle files for the given set of
61 source files.''')
62 parser.add_option('-l', '--lang', default='c++',
63 help='''The language to generate the output in.''')
not at google - send to devlin 2013/02/02 00:45:48 Ok, one last complaint: --bundle and --lang are se
sashab 2013/02/04 05:09:27 Done. Not sure what nbd means, but I updated the .
not at google - send to devlin 2013/02/04 17:21:06 nbd = no big deal
64 parser.add_option('-D', '--dart-overrides-dir', dest='dart_overrides_dir',
65 help='Adds custom dart from files in the given directory (Dart only).')
66
67 (opts, filenames) = parser.parse_args()
68
69 if not filenames:
70 sys.exit(0) # This is OK as a no-op
71
72 # Unless in bundle mode, only one file should be specified.
73 if not opts.bundle and len(filenames) > 1:
74 # TODO(sashab): Could also just use filenames[0] here and not complain.
75 raise Exception(
76 "Unless in bundle mode, only one file can be specified at a time.")
77
78 # Merge the source files into a single list of schemas.
79 api_defs = []
80 for filename in filenames:
81 schema = os.path.normpath(filename)
82 schema_filename, schema_extension = os.path.splitext(schema)
83 path, short_filename = os.path.split(schema_filename)
84 api_def = load_schema(schema)
85
86 # If in a c++-related mode, delete 'nocompile' nodes.
87 if opts.bundle or opts.lang == 'c++':
88 api_def = json_schema.DeleteNodes(api_def, 'nocompile')
89 api_defs.extend(api_def)
52 90
53 api_model = model.Model() 91 api_model = model.Model()
54 92
93 # Load type dependencies into the model.
55 for target_namespace in api_defs: 94 for target_namespace in api_defs:
56 referenced_schemas = target_namespace.get('dependencies', [])
57 # Load type dependencies into the model.
58 # TODO(miket): do we need this in IDL? 95 # TODO(miket): do we need this in IDL?
not at google - send to devlin 2013/02/02 00:45:48 I think this comment might be out of date now.
sashab 2013/02/04 05:09:27 What does it mean? *Is* it needed in IDL? I'm pre
not at google - send to devlin 2013/02/04 17:21:06 I more mean that this code executes independently
sashab 2013/02/04 23:12:45 Done.
sashab 2013/02/04 23:12:45 Done.
59 for referenced_schema in referenced_schemas: 96 for referenced_schema in target_namespace.get('dependencies', []):
60 split_schema = referenced_schema.split(':', 1) 97 split_schema = referenced_schema.split(':', 1)
61 if len(split_schema) > 1: 98 if len(split_schema) > 1:
62 if split_schema[0] != 'api': 99 if split_schema[0] != 'api':
63 continue 100 continue
64 else: 101 else:
65 referenced_schema = split_schema[1] 102 referenced_schema = split_schema[1]
66 103
67 referenced_schema_path = os.path.join( 104 referenced_schema_path = os.path.join(
68 os.path.dirname(schema), referenced_schema + '.json') 105 os.path.dirname(schema), referenced_schema + '.json')
69 referenced_api_defs = json_schema.Load(referenced_schema_path) 106 referenced_api_defs = json_schema.Load(referenced_schema_path)
70 107
71 for namespace in referenced_api_defs: 108 for namespace in referenced_api_defs:
72 api_model.AddNamespace( 109 api_model.AddNamespace(
73 namespace, 110 namespace,
74 os.path.relpath(referenced_schema_path, opts.root)) 111 os.path.relpath(referenced_schema_path, opts.root))
75 112
76 # Gets the relative path from opts.root to the schema to correctly determine 113 # Load the actual namespaces into the model.
77 # the include path. 114 for target_namespace, schema_filename in zip(api_defs, filenames):
78 relpath = os.path.relpath(schema, opts.root) 115 relpath = os.path.relpath(os.path.normpath(schema_filename), opts.root)
79 namespace = api_model.AddNamespace(target_namespace, 116 namespace = api_model.AddNamespace(target_namespace,
80 relpath, 117 relpath,
81 include_compiler_options=True) 118 include_compiler_options=True)
82 if not namespace:
83 continue
84
85 if short_filename != namespace.unix_name:
86 sys.exit("Filename %s is illegal. Name files using unix_hacker style." %
87 filename)
88
89 # The output filename must match the input filename for gyp to deal with it
90 # properly.
91 out_file = namespace.unix_name
92 type_generator = cpp_type_generator.CppTypeGenerator(
93 root_namespace, namespace, namespace.unix_name)
94 for referenced_namespace in api_model.namespaces.values():
95 if referenced_namespace == namespace:
96 continue
97 type_generator.AddNamespace(
98 referenced_namespace,
99 referenced_namespace.unix_name)
100
101 h_code = (h_generator.HGenerator(namespace, type_generator)
102 .Generate().Render())
103 cc_code = (cc_generator.CCGenerator(namespace, type_generator)
104 .Generate().Render())
105
106 if dest_dir:
107 with open(
108 os.path.join(dest_dir, namespace.source_file_dir, out_file + '.cc'),
109 'w') as cc_file:
110 cc_file.write(cc_code)
111 with open(
112 os.path.join(dest_dir, namespace.source_file_dir, out_file + '.h'),
113 'w') as h_file:
114 h_file.write(h_code)
115 else:
116 print '%s.h' % out_file
117 print
118 print h_code
119 print
120 print '%s.cc' % out_file
121 print
122 print cc_code
123
124 def handle_bundle_schema(filenames, dest_dir, root, root_namespace):
125 # Merge the source files into a single list of schemas.
126 api_defs = []
127 for filename in filenames:
128 schema = os.path.normpath(filename)
129 schema_filename, schema_extension = os.path.splitext(schema)
130 api_defs.extend(load_schema(schema))
131
132 api_model = model.Model()
133 relpath = os.path.relpath(os.path.normpath(filenames[0]), root)
134
135 for target_namespace, schema_filename in zip(api_defs, filenames):
136 namespace = api_model.AddNamespace(target_namespace,
137 relpath,
138 include_compiler_options=True)
139 path, filename = os.path.split(schema_filename) 119 path, filename = os.path.split(schema_filename)
140 short_filename, extension = os.path.splitext(filename) 120 short_filename, extension = os.path.splitext(filename)
141 121
142 # Filenames are checked against the unix_names of the namespaces they 122 # Filenames are checked against the unix_names of the namespaces they
143 # generate because the gyp uses the names of the JSON files to generate 123 # generate because the gyp uses the names of the JSON files to generate
144 # the names of the .cc and .h files. We want these to be using unix_names. 124 # the names of the .cc and .h files. We want these to be using unix_names.
145 if namespace.unix_name != short_filename: 125 if namespace.unix_name != short_filename:
146 sys.exit("Filename %s is illegal. Name files using unix_hacker style." % 126 sys.exit("Filename %s is illegal. Name files using unix_hacker style." %
147 schema_filename) 127 schema_filename)
148 128
149 type_generator = cpp_type_generator.CppTypeGenerator(root_namespace) 129 # The output filename must match the input filename for gyp to deal with it
130 # properly.
131 out_file = namespace.unix_name
132
133 # Construct the type generator with all the namespaces in this model.
134 type_generator = cpp_type_generator.CppTypeGenerator(opts.namespace)
150 for referenced_namespace in api_model.namespaces.values(): 135 for referenced_namespace in api_model.namespaces.values():
151 type_generator.AddNamespace( 136 type_generator.AddNamespace(
152 referenced_namespace, 137 referenced_namespace,
153 referenced_namespace.unix_name) 138 referenced_namespace.unix_name)
154 139
155 generator = schema_bundle_generator.SchemaBundleGenerator( 140 # Generate the code for each namespace.
156 root, api_model, api_defs, type_generator) 141 for namespace in api_model.namespaces.values():
157 api_h_code = generator.GenerateAPIHeader().Render() 142 if opts.bundle:
158 schemas_h_code = generator.GenerateSchemasHeader().Render() 143 cpp_bundle_generator = CppBundleGenerator(opts.root, api_model, api_defs,
159 schemas_cc_code = generator.GenerateSchemasCC().Render() 144 type_generator)
145 generators = [
146 ('generated_api.h', cpp_bundle_generator.api_h_generator),
147 ('generated_schemas.cc', cpp_bundle_generator.schemas_cc_generator),
148 ('generated_schemas.h', cpp_bundle_generator.schemas_h_generator)
149 ]
150 elif opts.lang == 'c++':
151 cpp_generator = CppGenerator(type_generator)
152 generators = [
153 ('%s.h' % namespace.unix_name, cpp_generator.h_generator),
154 ('%s.cc' % namespace.unix_name, cpp_generator.cc_generator)
155 ]
156 elif opts.lang == 'dart':
157 generators = [
158 ('%s.dart' % namespace.unix_name, DartGenerator(
159 opts.dart_overrides_dir))
160 ]
161 else:
not at google - send to devlin 2013/02/02 00:45:48 I am so happy right now! Factored perfection!
sashab 2013/02/04 05:09:27 =D
162 raise Exception('Unrecognised language %s' % opts.lang)
160 163
161 if dest_dir: 164 for filename, generator in generators:
162 basedir = os.path.join(dest_dir, 'chrome/common/extensions/api') 165 code = generator.Generate(namespace).Render()
163 with open(os.path.join(basedir, 'generated_api.h'), 'w') as h_file: 166 if opts.destdir:
164 h_file.write(api_h_code) 167 with open(os.path.join(opts.destdir, namespace.source_file_dir,
165 with open(os.path.join(basedir, 'generated_schemas.h'), 'w') as h_file: 168 filename)) as f:
166 h_file.write(schemas_h_code) 169 f.write(code)
167 with open(os.path.join(basedir, 'generated_schemas.cc'), 'w') as cc_file: 170 else:
168 cc_file.write(schemas_cc_code) 171 print filename
169 else: 172 print
170 print 'generated_api.h' 173 print code
171 print 174 print
172 print api_h_code
173 print
174 print 'generated_schemas.h'
175 print
176 print schemas_h_code
177 print
178 print 'generated_schemas.cc'
179 print
180 print schemas_cc_code
181
182 if __name__ == '__main__':
183 parser = optparse.OptionParser(
184 description='Generates a C++ model of an API from JSON schema',
185 usage='usage: %prog [option]... schema')
186 parser.add_option('-r', '--root', default='.',
187 help='logical include root directory. Path to schema files from specified'
188 'dir will be the include path.')
189 parser.add_option('-d', '--destdir',
190 help='root directory to output generated files.')
191 parser.add_option('-n', '--namespace', default='generated_api_schemas',
192 help='C++ namespace for generated files. e.g extensions::api.')
193 parser.add_option('-b', '--bundle', action="store_true", help=
194 '''if supplied, causes compiler to generate bundle files for the given set of
195 source files.''')
196
197 (opts, args) = parser.parse_args()
198
199 if not args:
200 sys.exit(0) # This is OK as a no-op
201 dest_dir = opts.destdir
202 root_namespace = opts.namespace
203
204 if opts.bundle:
205 handle_bundle_schema(args, dest_dir, opts.root, root_namespace)
206 else:
207 handle_single_schema(args[0], dest_dir, opts.root, root_namespace)
OLDNEW

Powered by Google App Engine
This is Rietveld 408576698