# Copyright (c) 2012 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. from code import Code from model import PropertyType import any_helper import cpp_util import model import schema_util import sys import util_cc_helper class CCGenerator(object): """A .cc generator for a namespace. """ def __init__(self, namespace, cpp_type_generator): self._cpp_type_generator = cpp_type_generator self._namespace = namespace self._target_namespace = ( self._cpp_type_generator.GetCppNamespaceName(self._namespace)) self._util_cc_helper = ( util_cc_helper.UtilCCHelper(self._cpp_type_generator)) self._any_helper = any_helper.AnyHelper() def Generate(self): """Generates a Code object with the .cc for a single namespace. """ c = Code() (c.Append(cpp_util.CHROMIUM_LICENSE) .Append() .Append(cpp_util.GENERATED_FILE_MESSAGE % self._namespace.source_file) .Append() .Append(self._util_cc_helper.GetIncludePath()) .Append('#include "%s/%s.h"' % (self._namespace.source_file_dir, self._namespace.unix_name)) ) includes = self._cpp_type_generator.GenerateIncludes() if not includes.IsEmpty(): (c.Concat(includes) .Append() ) (c.Append() .Concat(self._cpp_type_generator.GetRootNamespaceStart()) .Concat(self._cpp_type_generator.GetNamespaceStart()) .Append() ) if self._namespace.properties: (c.Append('//') .Append('// Properties') .Append('//') .Append() ) for property in self._namespace.properties.values(): property_code = self._cpp_type_generator.GeneratePropertyValues( property, 'const %(type)s %(name)s = %(value)s;', nodoc=True) if property_code: c.Concat(property_code).Append() if self._namespace.types: (c.Append('//') .Append('// Types') .Append('//') .Append() ) for type_ in self._namespace.types.values(): (c.Concat(self._GenerateType( schema_util.StripSchemaNamespace(type_.name), type_)).Append() ) if self._namespace.functions: (c.Append('//') .Append('// Functions') .Append('//') .Append() ) for function in self._namespace.functions.values(): (c.Concat(self._GenerateFunction( cpp_util.Classname(function.name), function)) .Append() ) if self._namespace.events: (c.Append('//') .Append('// Events') .Append('//') .Append() ) for event in self._namespace.events.values(): (c.Concat(self._GenerateCreateCallbackArguments( cpp_util.Classname(event.name), event, generate_to_json=True)) .Append() ) (c.Concat(self._cpp_type_generator.GetNamespaceEnd()) .Concat(self._cpp_type_generator.GetRootNamespaceEnd()) .Append() ) return c def _GenerateType(self, cpp_namespace, type_): """Generates the function definitions for a type. """ classname = cpp_util.Classname(schema_util.StripSchemaNamespace(type_.name)) c = Code() if type_.functions: for function in type_.functions.values(): (c.Concat( self._GenerateFunction( cpp_namespace + '::' + cpp_util.Classname(function.name), function)) .Append()) elif type_.type_ == PropertyType.OBJECT: (c.Concat(self._GeneratePropertyFunctions( cpp_namespace, type_.properties.values())) .Sblock('%(namespace)s::%(classname)s()') .Concat(self._GenerateInitializersAndBody(type_)) .Eblock('%(namespace)s::~%(classname)s() {}') .Append()) if type_.from_json: (c.Concat(self._GenerateTypePopulate(cpp_namespace, type_)) .Append()) if type_.from_client: (c.Concat(self._GenerateTypeToValue(cpp_namespace, type_)) .Append()) elif self._cpp_type_generator.IsEnumOrEnumRef(type_): (c.Concat(self._GenerateCreateEnumTypeValue(cpp_namespace, type_)) .Append() .Concat(self._GenerateEnumFromString(cpp_namespace, type_)) .Append() .Concat(self._GenerateEnumToString(cpp_namespace, type_)) .Append()) c.Substitute({'classname': classname, 'namespace': cpp_namespace}) return c def _GenerateInitializersAndBody(self, type_): items = [] for prop in type_.properties.values(): if prop.optional: continue t = prop.type_ if t == PropertyType.INTEGER: items.append('%s(0)' % prop.unix_name) elif t == PropertyType.DOUBLE: items.append('%s(0.0)' % prop.unix_name) elif t == PropertyType.BOOLEAN: items.append('%s(false)' % prop.unix_name) elif t == PropertyType.BINARY: items.append('%s(NULL)' % prop.unix_name) elif (t == PropertyType.ADDITIONAL_PROPERTIES or t == PropertyType.ANY or t == PropertyType.ARRAY or t == PropertyType.CHOICES or t == PropertyType.ENUM or t == PropertyType.OBJECT or t == PropertyType.FUNCTION or t == PropertyType.REF or t == PropertyType.STRING): # TODO(miket): It would be nice to initialize CHOICES and ENUM, but we # don't presently have the semantics to indicate which one of a set # should be the default. continue else: sys.exit("Unhandled PropertyType: %s" % t) if items: s = ': %s' % (', '.join(items)) else: s = '' s = s + ' {}' return Code().Append(s) def _GenerateTypePopulate(self, cpp_namespace, type_): """Generates the function for populating a type given a pointer to it. E.g for type "Foo", generates Foo::Populate() """ classname = cpp_util.Classname(schema_util.StripSchemaNamespace(type_.name)) c = Code() (c.Append('// static') .Sblock('bool %(namespace)s::Populate' '(const base::Value& value, %(name)s* out) {') .Append('if (!value.IsType(base::Value::TYPE_DICTIONARY))') .Append(' return false;') ) if type_.properties: (c.Append('const base::DictionaryValue* dict = ' 'static_cast(&value);') .Append() ) for prop in type_.properties.values(): c.Concat(self._InitializePropertyToDefault(prop, 'out')) for prop in type_.properties.values(): if prop.type_ == PropertyType.ADDITIONAL_PROPERTIES: c.Append('out->additional_properties.MergeDictionary(dict);') # remove all keys that are actual properties for cur_prop in type_.properties.values(): if prop != cur_prop: c.Append('out->additional_properties' '.RemoveWithoutPathExpansion("%s", NULL);' % cur_prop.name) c.Append() else: c.Concat(self._GenerateTypePopulateProperty(prop, 'dict', 'out')) (c.Append('return true;') .Eblock('}') ) c.Substitute({'namespace': cpp_namespace, 'name': classname}) return c def _GenerateTypePopulateProperty(self, prop, src, dst): """Generate the code to populate a single property in a type. src: base::DictionaryValue* dst: Type* """ c = Code() value_var = prop.unix_name + '_value' c.Append('const base::Value* %(value_var)s = NULL;') if prop.optional: (c.Sblock( 'if (%(src)s->GetWithoutPathExpansion("%(key)s", &%(value_var)s)) {') .Concat(self._GeneratePopulatePropertyFromValue( prop, value_var, dst, 'false'))) if self._cpp_type_generator.IsEnumOrEnumRef(prop): (c.Append('} else {') .Append('%%(dst)s->%%(name)s = %s;' % self._cpp_type_generator.GetEnumNoneValue(prop))) c.Eblock('}') else: (c.Append( 'if (!%(src)s->GetWithoutPathExpansion("%(key)s", &%(value_var)s))') .Append(' return false;') .Concat(self._GeneratePopulatePropertyFromValue( prop, value_var, dst, 'false')) ) c.Append() c.Substitute({ 'value_var': value_var, 'key': prop.name, 'src': src, 'dst': dst, 'name': prop.unix_name }) return c def _GenerateTypeToValue(self, cpp_namespace, type_): """Generates a function that serializes the type into a |base::DictionaryValue|. E.g. for type "Foo" generates Foo::ToValue() """ c = Code() (c.Sblock('scoped_ptr %s::ToValue() const {' % cpp_namespace) .Append('scoped_ptr value(' 'new base::DictionaryValue());') .Append() ) for prop in type_.properties.values(): if prop.type_ == PropertyType.ADDITIONAL_PROPERTIES: c.Append('value->MergeDictionary(&%s);' % prop.unix_name) else: if prop.optional: if self._cpp_type_generator.IsEnumOrEnumRef(prop): c.Sblock('if (%s != %s) {' % (prop.unix_name, self._cpp_type_generator.GetEnumNoneValue(prop))) elif prop.type_ == PropertyType.CHOICES: c.Sblock('if (%s_type != %s) {' % (prop.unix_name, self._cpp_type_generator.GetEnumNoneValue(prop))) else: c.Sblock('if (%s.get()) {' % prop.unix_name) if prop.type_ == prop.compiled_type: c.Append('value->SetWithoutPathExpansion("%s", %s);' % ( prop.name, self._CreateValueFromProperty(prop, 'this->' + prop.unix_name))) else: conversion_src = 'this->' + prop.unix_name if prop.optional: conversion_src = '*' + conversion_src (c.Append('%s %s;' % (self._cpp_type_generator.GetType(prop), prop.unix_name)) .Append(cpp_util.GenerateCompiledTypeToTypeConversion( self._cpp_type_generator.GetReferencedProperty(prop), conversion_src, prop.unix_name) + ';') .Append('value->SetWithoutPathExpansion("%s", %s);' % ( prop.unix_name, self._CreateValueFromProperty(prop, prop.unix_name))) ) if prop.optional: c.Eblock('}'); (c.Append() .Append('return value.Pass();') .Eblock('}') ) return c def _GenerateFunction(self, cpp_namespace, function): """Generates the definitions for function structs. """ c = Code() # Params::Populate function if function.params: c.Concat(self._GeneratePropertyFunctions(cpp_namespace + '::Params', function.params)) (c.Append('%(cpp_namespace)s::Params::Params() {}') .Append('%(cpp_namespace)s::Params::~Params() {}') .Append() .Concat(self._GenerateFunctionParamsCreate(cpp_namespace, function)) .Append() ) # Results::Create function if function.callback: c.Concat(self._GenerateCreateCallbackArguments( "%s::Results" % cpp_namespace, function.callback)) c.Substitute({'cpp_namespace': cpp_namespace}) return c def _CreateValueFromProperty(self, prop, var): """Creates a base::Value given a property. Generated code passes ownership to caller. var: variable or variable* E.g for std::string, generate base::Value::CreateStringValue(var) """ if prop.type_ == PropertyType.CHOICES: return 'Get%sChoiceValue().release()' % cpp_util.Classname(prop.name) elif self._IsObjectOrObjectRef(prop): if prop.optional: return '%s->ToValue().release()' % var else: return '%s.ToValue().release()' % var elif prop.type_ == PropertyType.ANY: return '%s.DeepCopy()' % self._any_helper.GetValue(prop, var) elif prop.type_ == PropertyType.ADDITIONAL_PROPERTIES: return '%s.DeepCopy()' % var elif prop.type_ == PropertyType.FUNCTION: if prop.optional: vardot = var + '->' else: vardot = var + '.' return '%sDeepCopy()' % vardot elif self._cpp_type_generator.IsEnumOrEnumRef(prop): return 'base::Value::CreateStringValue(ToString(%s))' % var elif prop.type_ == PropertyType.BINARY: if prop.optional: vardot = var + '->' else: vardot = var + '.' return ('base::BinaryValue::CreateWithCopiedBuffer(%sdata(), %ssize())' % (vardot, vardot)) elif self._IsArrayOrArrayRef(prop): return '%s.release()' % self._util_cc_helper.CreateValueFromArray( self._cpp_type_generator.GetReferencedProperty(prop), var, prop.optional) elif self._IsFundamentalOrFundamentalRef(prop): # If prop.type != prop.compiled_type, then no asterisk is necessary # because the target is a local variable and not a dereferenced scoped # pointer. The asterisk is instead prepended to conversion_src around line # 273. if prop.optional and prop.type_ == prop.compiled_type: var = '*' + var prop = self._cpp_type_generator.GetReferencedProperty(prop); return { PropertyType.STRING: 'base::Value::CreateStringValue(%s)', PropertyType.BOOLEAN: 'base::Value::CreateBooleanValue(%s)', PropertyType.INTEGER: 'base::Value::CreateIntegerValue(%s)', PropertyType.DOUBLE: 'base::Value::CreateDoubleValue(%s)', }[prop.type_] % var else: raise NotImplementedError('Conversion of %s to base::Value not ' 'implemented' % repr(prop.type_)) def _GenerateParamsCheck(self, function, var): """Generates a check for the correct number of arguments when creating Params. """ c = Code() num_required = 0 for param in function.params: if not param.optional: num_required += 1 if num_required == len(function.params): c.Append('if (%(var)s.GetSize() != %(total)d)') elif not num_required: c.Append('if (%(var)s.GetSize() > %(total)d)') else: c.Append('if (%(var)s.GetSize() < %(required)d' ' || %(var)s.GetSize() > %(total)d)') c.Append(' return scoped_ptr();') c.Substitute({ 'var': var, 'required': num_required, 'total': len(function.params), }) return c def _GenerateFunctionParamsCreate(self, cpp_namespace, function): """Generate function to create an instance of Params. The generated function takes a base::ListValue of arguments. E.g for function "Bar", generate Bar::Params::Create() """ c = Code() (c.Append('// static') .Sblock('scoped_ptr<%(cpp_namespace)s::Params> ' '%(cpp_namespace)s::Params::Create(const base::ListValue& args) {') .Concat(self._GenerateParamsCheck(function, 'args')) .Append('scoped_ptr params(new Params());') ) c.Substitute({'cpp_namespace': cpp_namespace}) for param in function.params: c.Concat(self._InitializePropertyToDefault(param, 'params')) for i, param in enumerate(function.params): # Any failure will cause this function to return. If any argument is # incorrect or missing, those following it are not processed. Note that # for optional arguments, we allow missing arguments and proceed because # there may be other arguments following it. failure_value = 'scoped_ptr()' c.Append() value_var = param.unix_name + '_value' (c.Append('const base::Value* %(value_var)s = NULL;') .Append('if (args.Get(%(i)s, &%(value_var)s) &&\n' ' !%(value_var)s->IsType(base::Value::TYPE_NULL))') .Sblock('{') .Concat(self._GeneratePopulatePropertyFromValue( param, value_var, 'params', failure_value)) .Eblock('}') ) if not param.optional: (c.Sblock('else {') .Append('return %s;' % failure_value) .Eblock('}') ) c.Substitute({'value_var': value_var, 'i': i}) (c.Append() .Append('return params.Pass();') .Eblock('}') .Append() ) return c def _GeneratePopulatePropertyFromValue( self, prop, value_var, dst, failure_value, check_type=True): """Generates code to populate a model.Property given a base::Value*. The existence of data inside the base::Value* is assumed so checks for existence should be performed before the code this generates. prop: the property the code is populating. value_var: a base::Value* that should represent |prop|. dst: the object with |prop| as a member. failure_value: the value to return if |prop| cannot be extracted from |value_var| check_type: if true, will check if |value_var| is the correct base::Value::Type """ c = Code() c.Sblock('{') if self._IsFundamentalOrFundamentalRef(prop): self._GenerateFundamentalOrFundamentalRefPopulate(c, prop, value_var, dst) elif self._IsObjectOrObjectRef(prop): self._GenerateObjectOrObjectRefPopulate(c, prop) elif prop.type_ == PropertyType.FUNCTION: self._GenerateFunctionPopulate(c, prop) elif prop.type_ == PropertyType.ANY: self._GenerateAnyPopulate(c, prop, value_var, dst) elif self._IsArrayOrArrayRef(prop): self._GenerateArrayOrArrayRefPopulate(c, prop, dst) elif prop.type_ == PropertyType.CHOICES: self._GenerateChoicePopulate(c, prop, value_var, dst, failure_value) elif self._cpp_type_generator.IsEnumOrEnumRef(prop): self._GenerateEnumPopulate(c, prop, value_var) elif prop.type_ == PropertyType.BINARY: self._GenerateBinaryPopulate(c, prop) else: raise NotImplementedError(prop.type_) c.Eblock('}') sub = { 'value_var': value_var, 'name': prop.unix_name, 'dst': dst, 'failure_value': failure_value, } if prop.type_ not in (PropertyType.CHOICES, PropertyType.ANY): sub['ctype'] = self._cpp_type_generator.GetType(prop) sub['compiled_ctype'] = self._cpp_type_generator.GetCompiledType(prop) sub['value_type'] = cpp_util.GetValueType(self._cpp_type_generator .GetReferencedProperty(prop).type_) c.Substitute(sub) return c def _GenerateFundamentalOrFundamentalRefPopulate(self, c, prop, value_var, dst): if prop.optional: (c.Append('%(ctype)s temp;') .Append('if (!%s)' % cpp_util.GetAsFundamentalValue( self._cpp_type_generator.GetReferencedProperty(prop), value_var, '&temp')) .Append(' return %(failure_value)s;') ) if prop.type_ != prop.compiled_type: (c.Append('%(compiled_ctype)s temp2;') .Append('if (!%s)' % cpp_util.GenerateTypeToCompiledTypeConversion( self._cpp_type_generator.GetReferencedProperty(prop), 'temp', 'temp2')) .Append(' return %(failure_value)s;') .Append('%(dst)s->%(name)s.reset(new %(compiled_ctype)s(temp2));') ) else: c.Append('%(dst)s->%(name)s.reset(new %(ctype)s(temp));') else: if prop.type_ == prop.compiled_type: assignment_target = '&%s->%s' % (dst, prop.unix_name) else: c.Append('%(ctype)s temp;') assignment_target = '&temp' (c.Append('if (!%s)' % cpp_util.GetAsFundamentalValue( self._cpp_type_generator.GetReferencedProperty(prop), value_var, assignment_target)) .Append(' return %(failure_value)s;') ) if prop.type_ != prop.compiled_type: (c.Append('if (!%s)' % cpp_util.GenerateTypeToCompiledTypeConversion( self._cpp_type_generator.GetReferencedProperty(prop), 'temp', '%s->%s' % (dst, prop.unix_name))) .Append(' return %(failure_value)s;') ) def _GenerateObjectOrObjectRefPopulate(self, c, prop): if prop.optional: (c.Append('const base::DictionaryValue* dictionary = NULL;') .Append('if (!%(value_var)s->GetAsDictionary(&dictionary))') .Append(' return %(failure_value)s;') .Append('scoped_ptr<%(ctype)s> temp(new %(ctype)s());') .Append('if (!%(ctype)s::Populate(*dictionary, temp.get()))') .Append(' return %(failure_value)s;') .Append('%(dst)s->%(name)s = temp.Pass();') ) else: (c.Append('const base::DictionaryValue* dictionary = NULL;') .Append('if (!%(value_var)s->GetAsDictionary(&dictionary))') .Append(' return %(failure_value)s;') .Append( 'if (!%(ctype)s::Populate(*dictionary, &%(dst)s->%(name)s))') .Append(' return %(failure_value)s;') ) def _GenerateFunctionPopulate(self, c, prop): if prop.optional: c.Append('%(dst)s->%(name)s.reset(new base::DictionaryValue());') def _GenerateAnyPopulate(self, c, prop, value_var, dst): if prop.optional: c.Append('%(dst)s->%(name)s.reset(new ' + any_helper.ANY_CLASS + '());') c.Append(self._any_helper.Init(prop, value_var, dst) + ';') def _GenerateArrayOrArrayRefPopulate(self, c, prop, dst): # util_cc_helper deals with optional and required arrays (c.Append('const base::ListValue* list = NULL;') .Append('if (!%(value_var)s->GetAsList(&list))') .Append(' return %(failure_value)s;')) if prop.item_type.type_ == PropertyType.ENUM: self._GenerateListValueToEnumArrayConversion(c, prop) else: (c.Append('if (!%s)' % self._util_cc_helper.PopulateArrayFromList( self._cpp_type_generator.GetReferencedProperty(prop), 'list', dst + '->' + prop.unix_name, prop.optional)) .Append(' return %(failure_value)s;') ) def _GenerateChoicePopulate(self, c, prop, value_var, dst, failure_value): type_var = '%(dst)s->%(name)s_type' c.Sblock('switch (%(value_var)s->GetType()) {') for choice in self._cpp_type_generator.ExpandParams([prop]): (c.Sblock('case %s: {' % cpp_util.GetValueType( self._cpp_type_generator.GetReferencedProperty(choice).type_)) .Concat(self._GeneratePopulatePropertyFromValue( choice, value_var, dst, failure_value, check_type=False)) .Append('%s = %s;' % (type_var, self._cpp_type_generator.GetEnumValue( prop, choice.type_.name))) .Append('break;') .Eblock('}') ) (c.Append('default:') .Append(' return %(failure_value)s;') ) c.Eblock('}') def _GenerateEnumPopulate(self, c, prop, value_var): c.Sblock('{') self._GenerateStringToEnumConversion(c, prop, value_var, 'enum_temp') c.Append('%(dst)s->%(name)s = enum_temp;') c.Eblock('}') def _GenerateBinaryPopulate(self, c, prop): (c.Append('if (!%(value_var)s->IsType(%(value_type)s))') .Append(' return %(failure_value)s;') .Append('const base::BinaryValue* binary_value =') .Append(' static_cast(%(value_var)s);') ) if prop.optional: (c.Append('%(dst)s->%(name)s.reset(') .Append(' new std::string(binary_value->GetBuffer(),') .Append(' binary_value->GetSize()));') ) else: (c.Append('%(dst)s->%(name)s.assign(binary_value->GetBuffer(),') .Append(' binary_value->GetSize());') ) def _GenerateListValueToEnumArrayConversion(self, c, prop): """Appends code that converts a ListValue of string contstants to an array of enums in dst. Leaves dst, name, and failure_value unsubstituted. c: the Code object that is being appended to. prop: the property that the code is populating. """ accessor = '.' if prop.optional: c.Append('%(dst)s->%(name)s.reset(new std::vector<' + ( self._cpp_type_generator.GetType(prop.item_type) + '>);')) accessor = '->' c.Sblock('for (ListValue::const_iterator it = list->begin(); ' 'it != list->end(); ++it) {') self._GenerateStringToEnumConversion( c, prop.item_type, '(*it)', 'enum_temp') c.Append('%(dst)s->%(name)s' + accessor + 'push_back(enum_temp);') c.Eblock('}') def _GenerateStringToEnumConversion(self, c, prop, value_var, enum_temp): """Appends code that converts a string to an enum. Leaves failure_value unsubstituted. c: the code that is appended to. prop: the property that the code is populating. value_var: the string value that is being converted. enum_temp: the name used to store the temporary enum value. """ (c.Append('std::string enum_as_string;') .Append('if (!%s->GetAsString(&enum_as_string))' % value_var) .Append(' return %(failure_value)s;') .Append('%(type)s %(enum)s = From%(type)sString(enum_as_string);' % { 'type': self._cpp_type_generator.GetCompiledType(prop), 'enum': enum_temp }) .Append('if (%s == %s)' % (enum_temp, self._cpp_type_generator.GetEnumNoneValue(prop))) .Append(' return %(failure_value)s;')) def _GeneratePropertyFunctions(self, param_namespace, params): """Generate the functions for structures generated by a property such as CreateEnumValue for ENUMs and Populate/ToValue for Params/Results objects. """ c = Code() for param in params: if param.type_ == PropertyType.OBJECT: c.Concat(self._GenerateType( param_namespace + '::' + cpp_util.Classname(param.name), param)) c.Append() elif param.type_ == PropertyType.ARRAY: c.Concat(self._GeneratePropertyFunctions( param_namespace, [param.item_type])) elif param.type_ == PropertyType.CHOICES: c.Concat(self._GeneratePropertyFunctions( param_namespace, param.choices.values())) if param.from_client: c.Concat(self._GenerateGetChoiceValue(param_namespace, param)) elif param.type_ == PropertyType.ENUM: (c.Concat(self._GenerateCreateEnumValue(param_namespace, param)) .Append() .Concat(self._GenerateEnumFromString(param_namespace, param, use_namespace=True)) .Append() .Concat(self._GenerateEnumToString(param_namespace, param, use_namespace=True)) .Append()) return c def _GenerateGetChoiceValue(self, cpp_namespace, prop): """Generates GetChoiceValue() that returns a scoped_ptr representing the choice value. """ c = Code() (c.Sblock('scoped_ptr ' '%(cpp_namespace)s::Get%(choice)sChoiceValue() const {') .Sblock('switch (%s_type) {' % prop.unix_name) .Concat(self._GenerateReturnCase( self._cpp_type_generator.GetEnumNoneValue(prop), 'scoped_ptr()'))) for choice in self._cpp_type_generator.ExpandParams([prop]): c.Concat(self._GenerateReturnCase( self._cpp_type_generator.GetEnumValue(prop, choice.type_.name), 'make_scoped_ptr(%s)' % self._CreateValueFromProperty(choice, choice.unix_name))) (c.Eblock('}') .Append('return scoped_ptr();') .Eblock('}') .Append() .Substitute({ 'cpp_namespace': cpp_namespace, 'choice': cpp_util.Classname(prop.name) }) ) return c def _GenerateCreateEnumTypeValue(self, cpp_namespace, prop): """Generates CreateEnumValue() that returns the base::StringValue representation of an enum type. """ c = Code() classname = cpp_util.Classname(schema_util.StripSchemaNamespace(prop.name)) (c.Sblock('scoped_ptr CreateEnumValue(%s %s) {' % (classname, classname.lower())) .Append('std::string enum_temp = ToString(%s);' % classname.lower()) .Append('if (enum_temp.empty())') .Append(' return scoped_ptr();') .Append('return scoped_ptr(' 'base::Value::CreateStringValue(enum_temp));') .Eblock('}')) return c def _GenerateEnumToString(self, cpp_namespace, prop, use_namespace=False): """Generates ToString() which gets the string representation of an enum. """ c = Code() classname = cpp_util.Classname(schema_util.StripSchemaNamespace(prop.name)) if use_namespace: namespace = '%s::' % cpp_namespace else: namespace = '' (c.Append('// static') .Sblock('std::string %(namespace)sToString(%(class)s enum_param) {')) enum_prop = self._cpp_type_generator.GetReferencedProperty(prop) c.Sblock('switch (enum_param) {') for enum_value in enum_prop.enum_values: c.Concat(self._GenerateReturnCase( self._cpp_type_generator.GetEnumValue(prop, enum_value), '"%s"' % enum_value)) (c.Append('case %s:' % self._cpp_type_generator.GetEnumNoneValue(prop)) .Append(' return "";') .Eblock('}') .Append('return "";') .Eblock('}') .Substitute({ 'namespace': namespace, 'class': classname })) return c def _GenerateEnumFromString(self, cpp_namespace, prop, use_namespace=False): """Generates FromClassNameString() which gets an enum from its string representation. """ c = Code() classname = cpp_util.Classname(schema_util.StripSchemaNamespace(prop.name)) if use_namespace: namespace = '%s::' % cpp_namespace else: namespace = '' (c.Append('// static') .Sblock('%(namespace)s%(class)s' ' %(namespace)sFrom%(class)sString(' 'const std::string& enum_string) {')) enum_prop = self._cpp_type_generator.GetReferencedProperty(prop) for i, enum_value in enumerate( self._cpp_type_generator.GetReferencedProperty(prop).enum_values): # This is broken up into all ifs with no else ifs because we get # "fatal error C1061: compiler limit : blocks nested too deeply" # on Windows. (c.Append('if (enum_string == "%s")' % enum_value) .Append(' return %s;' % self._cpp_type_generator.GetEnumValue(prop, enum_value))) (c.Append('return %s;' % self._cpp_type_generator.GetEnumNoneValue(prop)) .Eblock('}') .Substitute({ 'namespace': namespace, 'class': classname })) return c # TODO(chebert): This is basically the same as GenerateCreateEnumTypeValue(). # The plan is to phase out the old-style enums, and make all enums into REF # types. def _GenerateCreateEnumValue(self, cpp_namespace, prop): """Generates CreateEnumValue() that returns the base::StringValue representation of an enum. """ c = Code() (c.Append('// static') .Sblock('scoped_ptr %(cpp_namespace)s::CreateEnumValue(' '%(arg)s) {') .Append('std::string enum_temp = ToString(%s);' % prop.unix_name) .Append('if (enum_temp.empty())') .Append(' return scoped_ptr();') .Append('return scoped_ptr(' 'base::Value::CreateStringValue(enum_temp));') .Eblock('}') .Substitute({ 'cpp_namespace': cpp_namespace, 'arg': cpp_util.GetParameterDeclaration( prop, self._cpp_type_generator.GetType(prop)) })) return c def _GenerateReturnCase(self, case_value, return_value): """Generates a single return case for a switch block. """ c = Code() (c.Append('case %s:' % case_value) .Append(' return %s;' % return_value) ) return c def _GenerateCreateCallbackArguments(self, function_scope, callback, generate_to_json=False): """Generate all functions to create Value parameters for a callback. E.g for function "Bar", generate Bar::Results::Create E.g for event "Baz", generate Baz::Create function_scope: the function scope path, e.g. Foo::Bar for the function Foo::Bar::Baz(). callback: the Function object we are creating callback arguments for. generate_to_json: Generate a ToJson method. """ c = Code() params = callback.params expanded_params = self._cpp_type_generator.ExpandParams(params) c.Concat(self._GeneratePropertyFunctions(function_scope, expanded_params)) param_lists = self._cpp_type_generator.GetAllPossibleParameterLists(params) for param_list in param_lists: (c.Sblock('scoped_ptr %(function_scope)s::' 'Create(%(declaration_list)s) {') .Append('scoped_ptr create_results(' 'new base::ListValue());') ) declaration_list = [] for param in param_list: # We treat this argument as 'required' to avoid wrapping it in a # scoped_ptr if it's optional. param_copy = param.Copy() param_copy.optional = False declaration_list.append("const %s" % cpp_util.GetParameterDeclaration( param_copy, self._cpp_type_generator.GetCompiledType(param_copy))) param_name = param_copy.unix_name if param_copy.type_ != param_copy.compiled_type: param_name = 'temp_' + param_name (c.Append('%s %s;' % (self._cpp_type_generator.GetType(param_copy), param_name)) .Append(cpp_util.GenerateCompiledTypeToTypeConversion( param_copy, param_copy.unix_name, param_name) + ';') ) c.Append('create_results->Append(%s);' % self._CreateValueFromProperty(param_copy, param_name)) c.Append('return create_results.Pass();') c.Eblock('}') if generate_to_json: c.Append() (c.Sblock('std::string %(function_scope)s::' 'ToJson(%(declaration_list)s) {') .Append('scoped_ptr create_results = ' '%(function_scope)s::Create(%(param_list)s);') .Append('std::string json;') .Append('base::JSONWriter::Write(create_results.get(), &json);') .Append('return json;') ) c.Eblock('}') c.Substitute({ 'function_scope': function_scope, 'declaration_list': ', '.join(declaration_list), 'param_list': ', '.join(param.unix_name for param in param_list) }) return c def _InitializePropertyToDefault(self, prop, dst): """Initialize a model.Property to its default value inside an object. E.g for optional enum "state", generate dst->state = STATE_NONE; dst: Type* """ c = Code() if (self._cpp_type_generator.IsEnumOrEnumRef(prop) or prop.type_ == PropertyType.CHOICES): if prop.optional: prop_name = prop.unix_name if prop.type_ == PropertyType.CHOICES: prop_name = prop.unix_name + '_type' c.Append('%s->%s = %s;' % ( dst, prop_name, self._cpp_type_generator.GetEnumNoneValue(prop))) return c def _IsObjectOrObjectRef(self, prop): """Determines if this property is an Object or is a ref to an Object. """ return (self._cpp_type_generator.GetReferencedProperty(prop).type_ == PropertyType.OBJECT) def _IsArrayOrArrayRef(self, prop): """Determines if this property is an Array or is a ref to an Array. """ return (self._cpp_type_generator.GetReferencedProperty(prop).type_ == PropertyType.ARRAY) def _IsFundamentalOrFundamentalRef(self, prop): """Determines if this property is a Fundamental type or is a ref to a Fundamental type. """ return (self._cpp_type_generator.GetReferencedProperty(prop).type_. is_fundamental)