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

Unified Diff: content/browser/devtools/protocol/devtools_protocol_handler_generator.py

Issue 642263004: [DevTools] Make generated protocol structs wrappers around DictionaryValue (Closed) Base URL: https://chromium.googlesource.com/chromium/src.git@master
Patch Set: Fixed indentation in generated code Created 6 years, 1 month 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 side-by-side diff with in-line comments
Download patch
Index: content/browser/devtools/protocol/devtools_protocol_handler_generator.py
diff --git a/content/browser/devtools/protocol/devtools_protocol_handler_generator.py b/content/browser/devtools/protocol/devtools_protocol_handler_generator.py
index dd165faec5f34eb139e1db9ef62166ed3f250f8d..72219c2d2c0f611de1bc222281662ce61b82313f 100755
--- a/content/browser/devtools/protocol/devtools_protocol_handler_generator.py
+++ b/content/browser/devtools/protocol/devtools_protocol_handler_generator.py
@@ -34,6 +34,43 @@ namespace content {
class DevToolsProtocolHandlerImpl;
+class ListBuilderBase {
dgozman 2014/11/06 14:20:42 Let's put ListBuilder in content::devtools.
vkuzkokov 2014/11/06 16:30:12 Done.
+ public:
+ static scoped_ptr<base::ListValue> ToValue(ListBuilderBase src);
+
+ protected:
+ ListBuilderBase();
+ ListBuilderBase(scoped_ptr<base::ListValue> list_);
+
+ scoped_ptr<base::ListValue> list_;
+};
+
+template<class T>
+class ListBuilder : public ListBuilderBase {
+ MOVE_ONLY_TYPE_WITH_MOVE_CONSTRUCTOR_FOR_CPP_03(ListBuilder)
+ public:
+ ListBuilder() {}
+ ListBuilder(ListBuilder&& other) : ListBuilderBase(other.list_.Pass()) {
+ }
+
+ void push_back(T item) {
+ list_->Append(T::ToValue(item.Pass()).release());
+ }
+};
+
+template<>
+class ListBuilder<std::string> : public ListBuilderBase {
+ MOVE_ONLY_TYPE_WITH_MOVE_CONSTRUCTOR_FOR_CPP_03(ListBuilder)
+ public:
+ ListBuilder() {}
+ ListBuilder(ListBuilder&& other) : ListBuilderBase(other.list_.Pass()) {
+ }
+
+ void push_back(const std::string& item) {
+ list_->AppendString(item);
+ }
+};
+
namespace devtools {
${types}\
@@ -51,8 +88,6 @@ class DevToolsProtocolHandlerImpl : public DevToolsProtocol::Handler {
${setters}\
private:
-${friends}\
-
${methods}\
${fields}\
@@ -71,27 +106,72 @@ typedef ${param_type} ${declared_name};
tmpl_struct = string.Template("""\
namespace ${domain} {
-struct ${declared_name} {
+template<int MASK>
+struct ${declared_name}Builder {
+ MOVE_ONLY_TYPE_WITH_MOVE_CONSTRUCTOR_FOR_CPP_03(${declared_name}Builder)
public:
- ${declared_name}();
+ enum {
+ kAllSet = 0,
+${fields_enum}\
+ };
${methods}\
+ static ${declared_name}Builder<kNoneSet> Create() {
+ return ${declared_name}Builder<kNoneSet>();
+ }
+
+ static scoped_ptr<base::DictionaryValue> ToValue(
+ ${declared_name}Builder src) {
+ return src.dict_.Pass();
+ }
+
+ ${declared_name}Builder(${declared_name}Builder&& other)
+ : dict_(other.dict_.Pass()) {
+ }
+
private:
- friend class ::content::DevToolsProtocolHandlerImpl;
+ friend struct ${declared_name}Builder<0>;
-${fields}\
+ ${declared_name}Builder() : dict_(new base::DictionaryValue()) {
+ }
+
+ template<class T> T& ThisAs() {
+ COMPILE_ASSERT(sizeof(*this) == sizeof(T), cannot_cast);
+ return *reinterpret_cast<T*>(this);
+ }
+
+ scoped_ptr<base::DictionaryValue> dict_;
};
+
+typedef ${declared_name}Builder<0> ${declared_name};
+
} // namespace ${domain}
""")
-tmpl_struct_setter = string.Template("""\
- void set_${param}(${pass_type} ${param});
+tmpl_builder_setter_req = string.Template("""\
+ ${declared_name}Builder<MASK & ~k${Param}>&
+ set_${param}(${pass_type} ${param}) {
+ COMPILE_ASSERT(MASK & k${Param}, already_set);
+${builder_wrap}\
+ return ThisAs<${declared_name}Builder<MASK & ~k${Param}>>();
+ }
""")
-tmpl_struct_field = string.Template("""\
- ${param_type} ${param}_;
- bool has_${param}_;
+tmpl_builder_setter_opt = string.Template("""\
+ ${declared_name}Builder<MASK>&
+ set_${param}(${pass_type} ${param}) {
+${builder_wrap}\
+ return *this;
+ }
+""")
+
+tmpl_builder_enum = string.Template("""\
+ k${Param} = 1 << ${ordinal},
+""")
+
+tmpl_builder_none_set = string.Template("""\
+ kNoneSet = ${all_fields}
""")
tmpl_enum = string.Template("""\
@@ -130,13 +210,13 @@ ${methods}\
tmpl_event = string.Template("""\
void ${Command}(
- const ${Command}Params& params);
+ ${Command}Params params);
""")
tmpl_response = string.Template("""\
void Send${Command}Response(
scoped_refptr<DevToolsProtocol::Command> command,
- const ${Command}Response& params);
+ ${Command}Response params);
""")
tmpl_setter = string.Template("""\
@@ -144,21 +224,12 @@ tmpl_setter = string.Template("""\
devtools::${domain}::${Domain}Handler* ${domain}_handler);
""")
-tmpl_friend = string.Template("""\
- friend class devtools::${domain}::Client;
-""")
-
tmpl_callback = string.Template("""\
scoped_refptr<DevToolsProtocol::Response>
On${Domain}${Command}(
scoped_refptr<DevToolsProtocol::Command> command);
""")
-tmpl_to_value = string.Template("""\
- static base::DictionaryValue* ToValue(
- const devtools::${domain}::${declared_name}& src);
-""")
-
tmpl_field = string.Template("""\
devtools::${domain}::${Domain}Handler* ${domain}_handler_;
""")
@@ -172,6 +243,20 @@ ${includes}\
namespace content {
+ListBuilderBase::ListBuilderBase()
+ : list_(new base::ListValue) {
+}
+
+ListBuilderBase::ListBuilderBase(scoped_ptr<base::ListValue> list)
+ : list_(list.Pass()) {
+}
+
+// static
+scoped_ptr<base::ListValue> ListBuilderBase::ToValue(
+ ListBuilderBase src) {
+ return src.list_.Pass();
+}
+
DevToolsProtocolHandlerImpl::DevToolsProtocolHandlerImpl()
: ${fields_init} {
}
@@ -189,7 +274,7 @@ bool CreateCommonResponse(
scoped_refptr<DevToolsProtocol::Response>* protocol_response) {
switch (response.status()) {
case ResponseStatus::RESPONSE_STATUS_FALLTHROUGH:
- *protocol_response = NULL;
+ *protocol_response = nullptr;
break;
case ResponseStatus::RESPONSE_STATUS_OK:
return false;
@@ -223,7 +308,7 @@ tmpl_include = string.Template("""\
#include "content/browser/devtools/protocol/${domain}_handler.h"
""")
-tmpl_field_init = string.Template("${domain}_handler_(NULL)")
+tmpl_field_init = string.Template("${domain}_handler_(nullptr)")
tmpl_setter_impl = string.Template("""\
void DevToolsProtocolHandlerImpl::Set${Domain}Handler(
@@ -258,9 +343,9 @@ ${prep}\
scoped_refptr<DevToolsProtocol::Response> protocol_response;
if (CreateCommonResponse(command, response, &protocol_response))
return protocol_response;
- base::DictionaryValue* dict = new base::DictionaryValue();
+ base::DictionaryValue* result = new base::DictionaryValue();
${wrap}\
- return command->SuccessResponse(dict);
+ return command->SuccessResponse(result);
}
""")
@@ -279,16 +364,15 @@ params_prep = """\
tmpl_prep_req = string.Template("""\
${param_type} in_${param}${init};
- if (!params ||
- !params->Get${Type}("${proto_param}", &in_${param}))
+ if (!params || !params->Get${Type}("${proto_param}", &in_${param}))
return command->InvalidParamResponse("${proto_param}");
""")
tmpl_prep_req_list = string.Template("""\
- base::ListValue* list_${param} = NULL;
+ base::ListValue* list_${param} = nullptr;
if (!params || !params->GetList("${proto_param}", &list_${param}))
return command->InvalidParamResponse("${proto_param}");
- ${param_type} in_${param};
+ std::vector<${item_type}> in_${param};
for (base::ListValue::const_iterator it =
list_${param}->begin(); it != list_${param}->end(); ++it) {
${item_type} item${item_init};
@@ -312,45 +396,7 @@ tmpl_prep_output = string.Template("""\
tmpl_arg_req = string.Template("in_${param}")
tmpl_arg_opt = string.Template(
- "${param}_found ?\n &in_${param} : NULL")
-
-tmpl_arg_output = string.Template("&out_${param}")
-
-tmpl_to_value_impl = string.Template("""\
-// static
-base::DictionaryValue* DevToolsProtocolHandlerImpl::ToValue(
- const devtools::${domain}::${declared_name}& src) {
- base::DictionaryValue* dict = new base::DictionaryValue();
-${dchecks}\
-${wrap}\
- return dict;
-}
-""")
-
-tmpl_dcheck = string.Template("""\
- DCHECK(${cond_name});
-""")
-
-tmpl_struct_impl = string.Template("""\
-namespace ${domain} {
-
-${declared_name}::${declared_name}()${fields} {
-}
-
-${methods}\
-
-} // namespace ${domain}
-""")
-
-tmpl_struct_field_init = string.Template("has_${param}_(false)")
-
-tmpl_struct_setter_impl = string.Template("""\
-void ${declared_name}::set_${param}(
- ${pass_type} ${param}) {
- ${param}_ = ${param};
- has_${param}_ = true;
-}
-""")
+ "${param}_found ? &in_${param} : nullptr")
tmpl_client_impl = string.Template("""\
namespace ${domain} {
@@ -369,68 +415,45 @@ ${methods}\
tmpl_event_impl = string.Template("""\
void Client::${Command}(
- const ${Command}Params& params) {
+ ${Command}Params params) {
SendNotification("${Domain}.${command}",
- DevToolsProtocolHandlerImpl::ToValue(params));
+ ${Command}Params::ToValue(params.Pass()).Pass());
}
""")
tmpl_response_impl = string.Template("""\
void Client::Send${Command}Response(
scoped_refptr<DevToolsProtocol::Command> command,
- const ${Command}Response& params) {
+ ${Command}Response params) {
SendAsyncResponse(
- command->SuccessResponse(DevToolsProtocolHandlerImpl::ToValue(params)));
+ command->SuccessResponse(
+ ${Command}Response::ToValue(params.Pass()).release()));
}
""")
tmpl_wrap = string.Template("""\
- dict->Set${Type}("${proto_param}", ${var_name});
+ ${dict_var}->Set${Type}("${proto_param}", ${var_name});
""")
tmpl_wrap_dict = string.Template("""\
- dict->Set("${proto_param}",
- DevToolsProtocolHandlerImpl::ToValue(${var_name}));
+ ${dict_var}->Set("${proto_param}",
+ ${param_type}::ToValue(${var_name}.Pass()).release());
""")
-tmpl_wrap_obj = string.Template("""\
- dict->Set("${proto_param}", ${var_name});
-""")
-
-tmpl_wrap_list = string.Template("""\
- base::ListValue* list_${param} = new base::ListValue();
- for (${param_type}::const_iterator it =
- ${var_name}.begin(); it != ${var_name}.end(); ++it) {
-${append}\
- }
- dict->Set("${proto_param}", list_${param});
-""")
-
-tmpl_append = string.Template("""\
- list_${param}->Append${Type}(*it);
-""")
+tmpl_wrap_list = tmpl_wrap_dict;
-tmpl_append_dict = string.Template("""\
- list_${param}->Append(DevToolsProtocolHandlerImpl::ToValue(*it));
-""")
-
-tmpl_append_obj = string.Template("""\
- list_${param}->Append(*it);
-""")
-
-tmpl_wrap_opt = string.Template("""\
- if (${cond_name})
- dict->Set${Type}("${proto_param}", ${var_name});
+tmpl_wrap_obj = string.Template("""\
+ ${dict_var}->Set("${proto_param}", ${var_name}.release());
""")
tmpl_typename = string.Template("devtools::${domain}::${declared_name}")
+def Shift(prefix, s):
+ return "\n".join(map(lambda line: prefix + line, s.splitlines())) + "\n"
+
def Capitalize(s):
return s[:1].upper() + s[1:]
-def Decapitalize(s):
- return s.lower()
-
def Uncamelcase(s):
result = ""
for i, c in enumerate(s):
@@ -456,48 +479,35 @@ for json_domain in json_api["domains"]:
def DeclareStruct(json_properties, mapping):
methods = []
- fields = []
- fields_init = []
- method_impls = []
- dchecks = []
- wrap = []
+ fields_enum = []
+ enum_items = []
+ req_fields_num = 0
for json_prop in json_properties:
prop_map = mapping.copy()
prop_map["proto_param"] = json_prop["name"]
prop_map["param"] = Uncamelcase(json_prop["name"])
- prop_map["var_name"] = "src.%s_" % prop_map["param"]
- prop_map["cond_name"] = "src.has_%s_" % prop_map["param"]
+ prop_map["Param"] = Capitalize(json_prop["name"])
+ prop_map["var_name"] = "%s" % prop_map["param"]
+ prop_map["dict_var"] = "dict_";
ResolveType(json_prop, prop_map)
prop_map["declared_name"] = mapping["declared_name"]
- methods.append(tmpl_struct_setter.substitute(prop_map))
- fields.append(tmpl_struct_field.substitute(prop_map))
- fields_init.append(tmpl_struct_field_init.substitute(prop_map))
- method_impls.append(tmpl_struct_setter_impl.substitute(prop_map))
if json_prop.get("optional"):
- if param_map["Type"] in ["List", "Dictionary"]:
- # TODO(vkuzkokov) Implement.
- raise Exception(
- "Optional array and object properties are not implemented")
- wrap.append(tmpl_wrap_opt.substitute(prop_map))
+ methods.append(tmpl_builder_setter_opt.substitute(prop_map))
else:
- dchecks.append(tmpl_dcheck.substitute(prop_map));
- if not "wrap" in prop_map:
- raise Exception("Arrays of arrays are not implemented")
- wrap.append(prop_map["wrap"])
-
+ methods.append(tmpl_builder_setter_req.substitute(prop_map))
+ enum_items.append("k%s" % prop_map["Param"]);
+ fields_enum.append(tmpl_builder_enum.substitute(prop_map,
+ ordinal = req_fields_num))
+ req_fields_num += 1
+
+ all_fields = "kAllSet"
+ if len(enum_items) > 0:
+ all_fields = " | ".join(enum_items)
+ fields_enum.append(tmpl_builder_none_set.substitute(mapping,
+ all_fields = all_fields))
type_decls.append(tmpl_struct.substitute(mapping,
- methods = "".join(methods),
- fields = "".join(fields)))
- fields_init_str = ""
- if len(fields_init) > 0:
- fields_init_str = "\n : " + (",\n ".join(fields_init))
- type_impls.append(tmpl_struct_impl.substitute(mapping,
- fields = fields_init_str,
- methods = "\n".join(method_impls)))
- handler_methods.append(tmpl_to_value.substitute(mapping))
- handler_method_impls.append(tmpl_to_value_impl.substitute(mapping,
- dchecks = "".join(dchecks),
- wrap = "".join(wrap)))
+ methods = "\n".join(methods),
+ fields_enum = "".join(fields_enum)))
def ResolveRef(json, mapping):
dot_pos = json["$ref"].find(".")
@@ -510,7 +520,7 @@ def ResolveRef(json, mapping):
json_type = types["%s.%s" % (domain_name, type_name)]
mapping["declared_name"] = Capitalize(type_name)
mapping["Domain"] = domain_name
- mapping["domain"] = Decapitalize(domain_name)
+ mapping["domain"] = Uncamelcase(domain_name)
mapping["param_type"] = tmpl_typename.substitute(mapping)
if json_type.get("enum"):
# TODO(vkuzkokov) Implement. Approximate template:
@@ -528,17 +538,15 @@ def ResolveRef(json, mapping):
def ResolveArray(json, mapping):
items_map = mapping.copy()
ResolveType(json["items"], items_map)
- mapping["param_type"] = "std::vector<%s>" % items_map["param_type"]
+ mapping["param_type"] = "ListBuilder<%s>" % items_map["param_type"]
mapping["Type"] = "List"
- if "append" in items_map:
- mapping["wrap"] = tmpl_wrap_list.substitute(mapping,
- append = items_map["append"])
- mapping["pass_type"] = "const %s&" % mapping["param_type"]
+ mapping["wrap"] = tmpl_wrap_list.substitute(mapping)
+ mapping["pass_type"] = mapping["param_type"]
mapping["prep_req"] = tmpl_prep_req_list.substitute(mapping,
item_type = items_map["param_type"],
item_init = items_map["init"],
ItemType = items_map["Type"])
- # TODO(vkuzkokov) mapping["append"]: template for array of arrays.
+ mapping["arg_out"] = "&%s" % mapping["var_name"]
def ResolveObject(json, mapping):
mapping["Type"] = "Dictionary"
@@ -548,14 +556,14 @@ def ResolveObject(json, mapping):
(mapping["Command"], Capitalize(mapping["proto_param"])))
mapping["param_type"] = tmpl_typename.substitute(mapping)
DeclareStruct(json["properties"], mapping)
- mapping["append"] = tmpl_append_dict.substitute(mapping)
mapping["wrap"] = tmpl_wrap_dict.substitute(mapping)
- mapping["pass_type"] = "const %s&" % mapping["param_type"]
+ mapping["pass_type"] = mapping["param_type"]
+ mapping["arg_out"] = "&%s" % mapping["var_name"]
else:
- mapping["param_type"] = "base::DictionaryValue*"
- mapping["append"] = tmpl_append_obj.substitute(mapping)
+ mapping["param_type"] = "base::DictionaryValue"
mapping["wrap"] = tmpl_wrap_obj.substitute(mapping)
- mapping["pass_type"] = mapping["param_type"]
+ mapping["pass_type"] = "scoped_ptr<base::DictionaryValue>"
+ mapping["arg_out"] = "%s.get()" % mapping["var_name"]
def ResolvePrimitive(json, mapping):
jsonrpc_type = json["type"]
@@ -596,10 +604,10 @@ def ResolvePrimitive(json, mapping):
else:
raise Exception("Unknown type: %s" % json_type)
mapping["wrap"] = tmpl_wrap.substitute(mapping)
- mapping["append"] = tmpl_append.substitute(mapping)
mapping["prep_req"] = tmpl_prep_req.substitute(mapping)
if jsonrpc_type != "string":
mapping["pass_type"] = mapping["param_type"]
+ mapping["arg_out"] = "&%s" % mapping["var_name"]
def ResolveType(json, mapping):
mapping["init"] = ""
@@ -616,9 +624,9 @@ def ResolveType(json, mapping):
else:
raise Exception("Unknown type at %s.%s %s" %
(mapping["Domain"], mapping["command"], mapping["proto_param"]))
+ mapping["builder_wrap"] = Shift(" ", mapping["wrap"])
setters = []
-friends = []
fields = []
includes = []
@@ -627,7 +635,7 @@ fields_init = []
for json_domain in json_api["domains"]:
domain_map = {}
domain_map["Domain"] = json_domain["domain"]
- domain_map["domain"] = Decapitalize(json_domain["domain"])
+ domain_map["domain"] = Uncamelcase(json_domain["domain"])
initializations = []
client_methods = []
@@ -655,6 +663,7 @@ for json_domain in json_api["domains"]:
param_map["proto_param"] = json_param["name"]
param_map["param"] = Uncamelcase(json_param["name"])
param_map["var_name"] = "in_%s" % param_map["param"]
+ param_map["dict_var"] = "result"
ResolveType(json_param, param_map)
if len(prep) == 0:
@@ -696,13 +705,14 @@ for json_domain in json_api["domains"]:
param_map["proto_param"] = json_param["name"]
param_map["param"] = Uncamelcase(json_param["name"])
param_map["var_name"] = "out_%s" % param_map["param"]
+ param_map["dict_var"] = "result"
if json_param.get("optional"):
# TODO(vkuzkokov) Implement Optional<T> for value types.
raise Exception("Optional return values are not implemented")
ResolveType(json_param, param_map)
prep.append(tmpl_prep_output.substitute(param_map))
- args.append(tmpl_arg_output.substitute(param_map))
+ args.append(param_map["arg_out"])
if not "wrap" in param_map:
raise Exception("Arrays of arrays are not implemented")
wrap.append(param_map["wrap"])
@@ -749,7 +759,6 @@ for json_domain in json_api["domains"]:
if domain_needs_client:
type_decls.append(tmpl_client.substitute(domain_map,
methods = "".join(client_methods)))
- friends.append(tmpl_friend.substitute(domain_map))
initializations.append(tmpl_init_client.substitute(domain_map))
type_impls.append(tmpl_client_impl.substitute(domain_map,
methods = "\n".join(client_method_impls)))
@@ -763,7 +772,6 @@ output_cc_file = open(output_cc_path, "w")
output_h_file.write(template_h.substitute({},
types = "\n".join(type_decls),
setters = "".join(setters),
- friends = "".join(friends),
methods = "".join(handler_methods),
fields = "".join(fields)))
output_h_file.close()

Powered by Google App Engine
This is Rietveld 408576698