| OLD | NEW |
| (Empty) |
| 1 # Copyright (c) 2012 The Chromium Authors. All rights reserved. | |
| 2 # Use of this source code is governed by a BSD-style license that can be | |
| 3 # found in the LICENSE file. | |
| 4 | |
| 5 import code | |
| 6 import cpp_util | |
| 7 from model import Platforms | |
| 8 from schema_util import CapitalizeFirstLetter | |
| 9 from schema_util import JsFunctionNameToClassName | |
| 10 | |
| 11 import json | |
| 12 import os | |
| 13 import re | |
| 14 | |
| 15 # TODO(miket/asargent) - parameterize this. | |
| 16 SOURCE_BASE_PATH = 'chrome/common/extensions/api' | |
| 17 | |
| 18 def _RemoveDescriptions(node): | |
| 19 """Returns a copy of |schema| with "description" fields removed. | |
| 20 """ | |
| 21 if isinstance(node, dict): | |
| 22 result = {} | |
| 23 for key, value in node.items(): | |
| 24 # Some schemas actually have properties called "description", so only | |
| 25 # remove descriptions that have string values. | |
| 26 if key == 'description' and isinstance(value, basestring): | |
| 27 continue | |
| 28 result[key] = _RemoveDescriptions(value) | |
| 29 return result | |
| 30 if isinstance(node, list): | |
| 31 return [_RemoveDescriptions(v) for v in node] | |
| 32 return node | |
| 33 | |
| 34 class SchemaBundleGenerator(object): | |
| 35 """This class contains methods to generate code based on multiple schemas. | |
| 36 """ | |
| 37 | |
| 38 def __init__(self, root, model, api_defs, cpp_type_generator): | |
| 39 self._root = root; | |
| 40 self._model = model | |
| 41 self._api_defs = api_defs | |
| 42 self._cpp_type_generator = cpp_type_generator | |
| 43 | |
| 44 def GenerateHeader(self, file_base, body_code): | |
| 45 """Generates a code.Code object for a header file | |
| 46 | |
| 47 Parameters: | |
| 48 - |file_base| - the base of the filename, e.g. 'foo' (for 'foo.h') | |
| 49 - |body_code| - the code to put in between the multiple inclusion guards""" | |
| 50 c = code.Code() | |
| 51 c.Append(cpp_util.CHROMIUM_LICENSE) | |
| 52 c.Append() | |
| 53 c.Append(cpp_util.GENERATED_BUNDLE_FILE_MESSAGE % SOURCE_BASE_PATH) | |
| 54 ifndef_name = cpp_util.GenerateIfndefName(SOURCE_BASE_PATH, file_base) | |
| 55 c.Append() | |
| 56 c.Append('#ifndef %s' % ifndef_name) | |
| 57 c.Append('#define %s' % ifndef_name) | |
| 58 c.Append() | |
| 59 c.Concat(body_code) | |
| 60 c.Append() | |
| 61 c.Append('#endif // %s' % ifndef_name) | |
| 62 c.Append() | |
| 63 return c | |
| 64 | |
| 65 def _GetPlatformIfdefs(self, model_object): | |
| 66 """Generates the "defined" conditional for an #if check if |model_object| | |
| 67 has platform restrictions. Returns None if there are no restrictions. | |
| 68 """ | |
| 69 if model_object.platforms is None: | |
| 70 return None | |
| 71 ifdefs = [] | |
| 72 for platform in model_object.platforms: | |
| 73 if platform == Platforms.CHROMEOS: | |
| 74 ifdefs.append('defined(OS_CHROMEOS)') | |
| 75 else: | |
| 76 raise ValueError("Unsupported platform ifdef: %s" % platform.name) | |
| 77 return ' and '.join(ifdefs) | |
| 78 | |
| 79 def GenerateAPIHeader(self): | |
| 80 """Generates the header for API registration / declaration""" | |
| 81 c = code.Code() | |
| 82 | |
| 83 c.Append('#include <string>') | |
| 84 c.Append() | |
| 85 c.Append('#include "base/basictypes.h"') | |
| 86 | |
| 87 for namespace in self._model.namespaces.values(): | |
| 88 namespace_name = namespace.unix_name.replace("experimental_", "") | |
| 89 implementation_header = namespace.compiler_options.get( | |
| 90 "implemented_in", | |
| 91 "chrome/browser/extensions/api/%s/%s_api.h" % (namespace_name, | |
| 92 namespace_name)) | |
| 93 if not os.path.exists( | |
| 94 os.path.join(self._root, os.path.normpath(implementation_header))): | |
| 95 if "implemented_in" in namespace.compiler_options: | |
| 96 raise ValueError('Header file for namespace "%s" specified in ' | |
| 97 'compiler_options not found: %s' % | |
| 98 (namespace.unix_name, implementation_header)) | |
| 99 continue | |
| 100 ifdefs = self._GetPlatformIfdefs(namespace) | |
| 101 if ifdefs is not None: | |
| 102 c.Append("#if %s" % ifdefs, indent_level=0) | |
| 103 | |
| 104 c.Append('#include "%s"' % implementation_header) | |
| 105 | |
| 106 if ifdefs is not None: | |
| 107 c.Append("#endif // %s" % ifdefs, indent_level=0) | |
| 108 | |
| 109 c.Append() | |
| 110 c.Append("class ExtensionFunctionRegistry;") | |
| 111 c.Append() | |
| 112 | |
| 113 c.Concat(self._cpp_type_generator.GetRootNamespaceStart()) | |
| 114 for namespace in self._model.namespaces.values(): | |
| 115 c.Append("// TODO(miket): emit code for %s" % (namespace.unix_name)) | |
| 116 c.Append() | |
| 117 c.Concat(self.GenerateFunctionRegistry()) | |
| 118 c.Concat(self._cpp_type_generator.GetRootNamespaceEnd()) | |
| 119 c.Append() | |
| 120 return self.GenerateHeader('generated_api', c) | |
| 121 | |
| 122 def _GetNamespaceFunctions(self, namespace): | |
| 123 functions = list(namespace.functions.values()) | |
| 124 if namespace.compiler_options.get("generate_type_functions", False): | |
| 125 for type_ in namespace.types.values(): | |
| 126 functions += list(type_.functions.values()) | |
| 127 return functions | |
| 128 | |
| 129 def GenerateFunctionRegistry(self): | |
| 130 c = code.Code() | |
| 131 c.Sblock("class GeneratedFunctionRegistry {") | |
| 132 c.Append(" public:") | |
| 133 c.Sblock("static void RegisterAll(ExtensionFunctionRegistry* registry) {") | |
| 134 for namespace in self._model.namespaces.values(): | |
| 135 namespace_ifdefs = self._GetPlatformIfdefs(namespace) | |
| 136 if namespace_ifdefs is not None: | |
| 137 c.Append("#if %s" % namespace_ifdefs, indent_level=0) | |
| 138 | |
| 139 namespace_name = CapitalizeFirstLetter(namespace.name.replace( | |
| 140 "experimental.", "")) | |
| 141 for function in self._GetNamespaceFunctions(namespace): | |
| 142 if function.nocompile: | |
| 143 continue | |
| 144 function_ifdefs = self._GetPlatformIfdefs(function) | |
| 145 if function_ifdefs is not None: | |
| 146 c.Append("#if %s" % function_ifdefs, indent_level=0) | |
| 147 | |
| 148 function_name = JsFunctionNameToClassName(namespace.name, function.name) | |
| 149 c.Append("registry->RegisterFunction<%sFunction>();" % ( | |
| 150 function_name)) | |
| 151 | |
| 152 if function_ifdefs is not None: | |
| 153 c.Append("#endif // %s" % function_ifdefs, indent_level=0) | |
| 154 | |
| 155 if namespace_ifdefs is not None: | |
| 156 c.Append("#endif // %s" % namespace_ifdefs, indent_level=0) | |
| 157 c.Eblock("}") | |
| 158 c.Eblock("};") | |
| 159 c.Append() | |
| 160 return c | |
| 161 | |
| 162 def GenerateSchemasHeader(self): | |
| 163 """Generates a code.Code object for the generated schemas .h file""" | |
| 164 c = code.Code() | |
| 165 c.Append('#include <map>') | |
| 166 c.Append('#include <string>') | |
| 167 c.Append(); | |
| 168 c.Append('#include "base/string_piece.h"') | |
| 169 c.Append() | |
| 170 c.Concat(self._cpp_type_generator.GetRootNamespaceStart()) | |
| 171 c.Append() | |
| 172 c.Sblock('class GeneratedSchemas {') | |
| 173 c.Append(' public:') | |
| 174 c.Append('// Puts all API schemas in |schemas|.') | |
| 175 c.Append('static void Get(' | |
| 176 'std::map<std::string, base::StringPiece>* schemas);') | |
| 177 c.Eblock('};'); | |
| 178 c.Append() | |
| 179 c.Concat(self._cpp_type_generator.GetRootNamespaceEnd()) | |
| 180 c.Append() | |
| 181 return self.GenerateHeader('generated_schemas', c) | |
| 182 | |
| 183 def GenerateSchemasCC(self): | |
| 184 """Generates a code.Code object for the generated schemas .cc file""" | |
| 185 c = code.Code() | |
| 186 c.Append(cpp_util.CHROMIUM_LICENSE) | |
| 187 c.Append() | |
| 188 c.Append('#include "%s"' % (os.path.join(SOURCE_BASE_PATH, | |
| 189 'generated_schemas.h'))) | |
| 190 c.Append() | |
| 191 c.Concat(self._cpp_type_generator.GetRootNamespaceStart()) | |
| 192 c.Append() | |
| 193 c.Append('// static') | |
| 194 c.Sblock('void GeneratedSchemas::Get(' | |
| 195 'std::map<std::string, base::StringPiece>* schemas) {') | |
| 196 for api in self._api_defs: | |
| 197 namespace = self._model.namespaces[api.get('namespace')] | |
| 198 # JSON parsing code expects lists of schemas, so dump a singleton list. | |
| 199 json_content = json.dumps([_RemoveDescriptions(api)], | |
| 200 separators=(',', ':')) | |
| 201 # Escape all double-quotes and backslashes. For this to output a valid | |
| 202 # JSON C string, we need to escape \ and ". | |
| 203 json_content = json_content.replace('\\', '\\\\').replace('"', '\\"') | |
| 204 c.Append('(*schemas)["%s"] = "%s";' % (namespace.name, json_content)) | |
| 205 c.Eblock('}') | |
| 206 c.Append() | |
| 207 c.Concat(self._cpp_type_generator.GetRootNamespaceEnd()) | |
| 208 c.Append() | |
| 209 return c | |
| OLD | NEW |