Chromium Code Reviews| OLD | NEW |
|---|---|
| 1 #!/usr/bin/env python | 1 #!/usr/bin/env python |
| 2 | 2 |
| 3 # Copyright (c) 2012 The Chromium Authors. All rights reserved. | 3 # Copyright (c) 2012 The Chromium Authors. All rights reserved. |
| 4 # Use of this source code is governed by a BSD-style license that can be | 4 # Use of this source code is governed by a BSD-style license that can be |
| 5 # found in the LICENSE file. | 5 # found in the LICENSE file. |
| 6 """Server for viewing the compiled C++ code from tools/json_schema_compiler. | 6 """Server for viewing the compiled C++ code from tools/json_schema_compiler. |
| 7 """ | 7 """ |
| 8 | 8 |
| 9 import cc_generator | 9 import cc_generator |
| 10 import code | 10 import code |
| 11 import cpp_type_generator | 11 import cpp_type_generator |
| 12 import cpp_util | 12 import cpp_util |
| 13 import h_generator | 13 import h_generator |
| 14 import idl_schema | 14 import idl_schema |
| 15 import json_schema | 15 import json_schema |
| 16 import model | 16 import model |
| 17 import optparse | 17 import optparse |
| 18 import os | 18 import os |
| 19 import schema_loader | 19 import shlex |
| 20 import urlparse | 20 import urlparse |
| 21 from highlighters import ( | 21 from highlighters import ( |
| 22 pygments_highlighter, none_highlighter, hilite_me_highlighter) | 22 pygments_highlighter, none_highlighter, hilite_me_highlighter) |
| 23 from BaseHTTPServer import BaseHTTPRequestHandler, HTTPServer | 23 from BaseHTTPServer import BaseHTTPRequestHandler, HTTPServer |
| 24 from cpp_namespace_environment import CppNamespaceEnvironment | |
| 25 from schema_loader import SchemaLoader | |
| 24 | 26 |
| 25 | 27 |
| 26 class CompilerHandler(BaseHTTPRequestHandler): | 28 class CompilerHandler(BaseHTTPRequestHandler): |
| 27 """A HTTPRequestHandler that outputs the result of tools/json_schema_compiler. | 29 """A HTTPRequestHandler that outputs the result of tools/json_schema_compiler. |
| 28 """ | 30 """ |
| 29 def do_GET(self): | 31 def do_GET(self): |
| 30 parsed_url = urlparse.urlparse(self.path) | 32 parsed_url = urlparse.urlparse(self.path) |
| 31 request_path = self._GetRequestPath(parsed_url) | 33 request_path = self._GetRequestPath(parsed_url) |
| 32 | 34 |
| 33 chromium_favicon = 'http://codereview.chromium.org/static/favicon.ico' | 35 chromium_favicon = 'http://codereview.chromium.org/static/favicon.ico' |
| (...skipping 135 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
| 169 var cachedValue = localStorage[highlighterStyle.id + 'Value']; | 171 var cachedValue = localStorage[highlighterStyle.id + 'Value']; |
| 170 if (cachedValue) | 172 if (cachedValue) |
| 171 highlighterStyle.value = cachedValue; | 173 highlighterStyle.value = cachedValue; |
| 172 }); | 174 }); |
| 173 })(); | 175 })(); |
| 174 | 176 |
| 175 window.addEventListener('hashchange', updateEverything, false); | 177 window.addEventListener('hashchange', updateEverything, false); |
| 176 updateEverything(); | 178 updateEverything(); |
| 177 </script>''') | 179 </script>''') |
| 178 | 180 |
| 179 def _LoadModel(self, basedir, name): | |
| 180 """Loads and returns the model for the |name| API from either its JSON or | |
| 181 IDL file, e.g. | |
| 182 name=contextMenus will be loaded from |basedir|/context_menus.json, | |
| 183 name=alarms will be loaded from |basedir|/alarms.idl. | |
| 184 """ | |
| 185 loaders = { | |
| 186 'json': json_schema.Load, | |
| 187 'idl': idl_schema.Load | |
| 188 } | |
| 189 # APIs are referred to like "webRequest" but that's in a file | |
| 190 # "web_request.json" so we need to unixify the name. | |
| 191 unix_name = model.UnixName(name) | |
| 192 for loader_ext, loader_fn in loaders.items(): | |
| 193 file_path = '%s/%s.%s' % (basedir, unix_name, loader_ext) | |
| 194 if os.path.exists(file_path): | |
| 195 # For historical reasons these files contain a singleton list with the | |
| 196 # model, so just return that single object. | |
| 197 return (loader_fn(file_path)[0], file_path) | |
| 198 raise ValueError('File for model "%s" not found' % name) | |
| 199 | |
| 200 def _ShowCompiledFile(self, parsed_url, head, body): | 181 def _ShowCompiledFile(self, parsed_url, head, body): |
| 201 """Show the compiled version of a json or idl file given the path to the | 182 """Show the compiled version of a json or idl file given the path to the |
| 202 compiled file. | 183 compiled file. |
| 203 """ | 184 """ |
| 204 api_model = model.Model() | 185 api_model = model.Model() |
| 205 | 186 |
| 206 request_path = self._GetRequestPath(parsed_url) | 187 request_path = self._GetRequestPath(parsed_url) |
| 207 (file_root, file_ext) = os.path.splitext(request_path) | 188 (file_root, file_ext) = os.path.splitext(request_path) |
| 208 (filedir, filename) = os.path.split(file_root) | 189 (filedir, filename) = os.path.split(file_root) |
| 209 | 190 |
| 191 schema_loader = SchemaLoader(filedir, | |
| 192 filedir, | |
| 193 self.server.include_rules, | |
| 194 '') | |
|
not at google - send to devlin
2014/08/28 19:41:35
Maybe make this like "fake_namespace" or something
lfg
2014/08/28 23:14:59
I've added the option to specify the namespace thr
| |
| 210 try: | 195 try: |
| 211 # Get main file. | 196 # Get main file. |
| 212 (api_def, file_path) = self._LoadModel(filedir, filename) | 197 namespace = schema_loader.ResolveNamespace(filename) |
| 213 namespace = api_model.AddNamespace(api_def, file_path) | |
| 214 type_generator = cpp_type_generator.CppTypeGenerator( | 198 type_generator = cpp_type_generator.CppTypeGenerator( |
| 215 api_model, | 199 api_model, |
| 216 schema_loader.SchemaLoader(filedir), | 200 schema_loader, |
| 217 namespace) | 201 namespace) |
| 218 | 202 |
| 219 # Get the model's dependencies. | |
| 220 for dependency in api_def.get('dependencies', []): | |
| 221 # Dependencies can contain : in which case they don't refer to APIs, | |
| 222 # rather, permissions or manifest keys. | |
| 223 if ':' in dependency: | |
| 224 continue | |
| 225 (api_def, file_path) = self._LoadModel(filedir, dependency) | |
| 226 referenced_namespace = api_model.AddNamespace(api_def, file_path) | |
| 227 if referenced_namespace: | |
| 228 type_generator.AddNamespace(referenced_namespace, | |
| 229 cpp_util.Classname(referenced_namespace.name).lower()) | |
| 230 | |
| 231 # Generate code | 203 # Generate code |
| 232 cpp_namespace = 'generated_api_schemas' | 204 cpp_namespace = 'generated_api_schemas' |
| 233 if file_ext == '.h': | 205 if file_ext == '.h': |
| 234 cpp_code = (h_generator.HGenerator(type_generator, cpp_namespace) | 206 cpp_code = (h_generator.HGenerator(type_generator) |
| 235 .Generate(namespace).Render()) | 207 .Generate(namespace).Render()) |
| 236 elif file_ext == '.cc': | 208 elif file_ext == '.cc': |
| 237 cpp_code = (cc_generator.CCGenerator(type_generator, cpp_namespace) | 209 cpp_code = (cc_generator.CCGenerator(type_generator) |
| 238 .Generate(namespace).Render()) | 210 .Generate(namespace).Render()) |
| 239 else: | 211 else: |
| 240 self.send_error(404, "File not found: %s" % request_path) | 212 self.send_error(404, "File not found: %s" % request_path) |
| 241 return | 213 return |
| 242 | 214 |
| 243 # Do highlighting on the generated code | 215 # Do highlighting on the generated code |
| 244 (highlighter_param, style_param) = self._GetHighlighterParams(parsed_url) | 216 (highlighter_param, style_param) = self._GetHighlighterParams(parsed_url) |
| 245 head.Append('<style>' + | 217 head.Append('<style>' + |
| 246 self.server.highlighters[highlighter_param].GetCSS(style_param) + | 218 self.server.highlighters[highlighter_param].GetCSS(style_param) + |
| 247 '</style>') | 219 '</style>') |
| (...skipping 73 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
| 321 # cc/h panes will automatically update via the hash change event. | 293 # cc/h panes will automatically update via the hash change event. |
| 322 html.Append('<li><a href="#%s">%s</a>' % | 294 html.Append('<li><a href="#%s">%s</a>' % |
| 323 (filename, filename)) | 295 (filename, filename)) |
| 324 | 296 |
| 325 html.Append('</ul>') | 297 html.Append('</ul>') |
| 326 | 298 |
| 327 return html.Render() | 299 return html.Render() |
| 328 | 300 |
| 329 | 301 |
| 330 class PreviewHTTPServer(HTTPServer, object): | 302 class PreviewHTTPServer(HTTPServer, object): |
| 331 def __init__(self, server_address, handler, highlighters): | 303 def __init__(self, server_address, handler, highlighters, include_rules): |
| 332 super(PreviewHTTPServer, self).__init__(server_address, handler) | 304 super(PreviewHTTPServer, self).__init__(server_address, handler) |
| 333 self.highlighters = highlighters | 305 self.highlighters = highlighters |
| 306 self.include_rules = include_rules | |
| 334 | 307 |
| 335 | 308 |
| 336 if __name__ == '__main__': | 309 if __name__ == '__main__': |
| 337 parser = optparse.OptionParser( | 310 parser = optparse.OptionParser( |
| 338 description='Runs a server to preview the json_schema_compiler output.', | 311 description='Runs a server to preview the json_schema_compiler output.', |
| 339 usage='usage: %prog [option]...') | 312 usage='usage: %prog [option]...') |
| 340 parser.add_option('-p', '--port', default='8000', | 313 parser.add_option('-p', '--port', default='8000', |
| 341 help='port to run the server on') | 314 help='port to run the server on') |
| 315 parser.add_option('-I', '--include-rules', | |
| 316 help='A list of paths to include when searching for referenced objects,' | |
| 317 ' with the namespace separated by a \':\'. Example: ' | |
| 318 '/foo/bar:Foo::Bar::%(namespace)s') | |
| 342 | 319 |
| 343 (opts, argv) = parser.parse_args() | 320 (opts, argv) = parser.parse_args() |
| 344 | 321 |
| 322 def split_path_and_namespace(path_and_namespace): | |
| 323 if ':' not in path_and_namespace: | |
| 324 raise ValueError('Invalid include rule "%s". Rules must be of ' | |
| 325 'the form path:namespace' % path_and_namespace) | |
| 326 return path_and_namespace.split(':', 1) | |
| 327 | |
| 328 include_rules = [] | |
| 329 if opts.include_rules: | |
| 330 include_rules = map(split_path_and_namespace, | |
| 331 shlex.split(opts.include_rules)) | |
| 332 | |
| 345 try: | 333 try: |
| 346 print('Starting previewserver on port %s' % opts.port) | 334 print('Starting previewserver on port %s' % opts.port) |
| 347 print('The extension documentation can be found at:') | 335 print('The extension documentation can be found at:') |
| 348 print('') | 336 print('') |
| 349 print(' http://localhost:%s/chrome/common/extensions/api' % opts.port) | 337 print(' http://localhost:%s/chrome/common/extensions/api' % opts.port) |
| 350 print('') | 338 print('') |
| 351 | 339 |
| 352 highlighters = { | 340 highlighters = { |
| 353 'hilite': hilite_me_highlighter.HiliteMeHighlighter(), | 341 'hilite': hilite_me_highlighter.HiliteMeHighlighter(), |
| 354 'none': none_highlighter.NoneHighlighter() | 342 'none': none_highlighter.NoneHighlighter() |
| 355 } | 343 } |
| 356 try: | 344 try: |
| 357 highlighters['pygments'] = pygments_highlighter.PygmentsHighlighter() | 345 highlighters['pygments'] = pygments_highlighter.PygmentsHighlighter() |
| 358 except ImportError as e: | 346 except ImportError as e: |
| 359 pass | 347 pass |
| 360 | 348 |
| 361 server = PreviewHTTPServer(('', int(opts.port)), | 349 server = PreviewHTTPServer(('', int(opts.port)), |
| 362 CompilerHandler, | 350 CompilerHandler, |
| 363 highlighters) | 351 highlighters, |
| 352 include_rules) | |
| 364 server.serve_forever() | 353 server.serve_forever() |
| 365 except KeyboardInterrupt: | 354 except KeyboardInterrupt: |
| 366 server.socket.close() | 355 server.socket.close() |
| OLD | NEW |