| // Copyright 2014 The Chromium OS Authors. All rights reserved. |
| // Use of this source code is governed by a BSD-style license that can be |
| // found in the LICENSE file. |
| |
| #include "buffet/commands/schema_utils.h" |
| |
| #include <algorithm> |
| #include <set> |
| #include <string> |
| |
| #include <base/json/json_writer.h> |
| |
| #include "buffet/commands/object_schema.h" |
| #include "buffet/commands/prop_types.h" |
| #include "buffet/commands/prop_values.h" |
| |
| namespace buffet { |
| namespace { |
| // Helper function to report "type mismatch" errors when parsing JSON. |
| void ReportJsonTypeMismatch(const base::Value* value_in, |
| const std::string& expected_type, |
| ErrorPtr* error) { |
| std::string value_as_string; |
| base::JSONWriter::Write(value_in, &value_as_string); |
| Error::AddToPrintf(error, commands::errors::kDomain, |
| commands::errors::kTypeMismatch, |
| "Unable to convert value %s into %s", |
| value_as_string.c_str(), expected_type.c_str()); |
| } |
| |
| // Template version of ReportJsonTypeMismatch that deduces the type of expected |
| // data from the value_out parameter passed to particular overload of |
| // TypedValueFromJson() function. Always returns false. |
| template<typename T> |
| bool ReportUnexpectedJson(const base::Value* value_in, T*, ErrorPtr* error) { |
| ReportJsonTypeMismatch(value_in, |
| PropType::GetTypeStringFromType(GetValueType<T>()), |
| error); |
| return false; |
| } |
| |
| bool ErrorMissingProperty(ErrorPtr* error, const char* param_name) { |
| Error::AddToPrintf(error, commands::errors::kDomain, |
| commands::errors::kPropertyMissing, |
| "Required parameter missing: %s", param_name); |
| return false; |
| } |
| } // namespace |
| |
| // Specializations of TypedValueToJson<T>() for supported C++ types. |
| std::unique_ptr<base::Value> TypedValueToJson(bool value, ErrorPtr* error) { |
| return std::unique_ptr<base::Value>(base::Value::CreateBooleanValue(value)); |
| } |
| |
| std::unique_ptr<base::Value> TypedValueToJson(int value, ErrorPtr* error) { |
| return std::unique_ptr<base::Value>(base::Value::CreateIntegerValue(value)); |
| } |
| |
| std::unique_ptr<base::Value> TypedValueToJson(double value, ErrorPtr* error) { |
| return std::unique_ptr<base::Value>(base::Value::CreateDoubleValue(value)); |
| } |
| |
| std::unique_ptr<base::Value> TypedValueToJson(const std::string& value, |
| ErrorPtr* error) { |
| return std::unique_ptr<base::Value>(base::Value::CreateStringValue(value)); |
| } |
| |
| std::unique_ptr<base::Value> TypedValueToJson(const native_types::Object& value, |
| ErrorPtr* error) { |
| std::unique_ptr<base::DictionaryValue> dict(new base::DictionaryValue); |
| for (const auto& pair : value) { |
| auto prop_value = pair.second->ToJson(error); |
| if (!prop_value) |
| return prop_value; |
| dict->SetWithoutPathExpansion(pair.first, prop_value.release()); |
| } |
| return std::move(dict); |
| } |
| |
| bool TypedValueFromJson(const base::Value* value_in, |
| const ObjectSchema* object_schema, |
| bool* value_out, ErrorPtr* error) { |
| return value_in->GetAsBoolean(value_out) || |
| ReportUnexpectedJson(value_in, value_out, error); |
| } |
| |
| bool TypedValueFromJson(const base::Value* value_in, |
| const ObjectSchema* object_schema, |
| int* value_out, ErrorPtr* error) { |
| return value_in->GetAsInteger(value_out) || |
| ReportUnexpectedJson(value_in, value_out, error); |
| } |
| |
| bool TypedValueFromJson(const base::Value* value_in, |
| const ObjectSchema* object_schema, |
| double* value_out, ErrorPtr* error) { |
| return value_in->GetAsDouble(value_out) || |
| ReportUnexpectedJson(value_in, value_out, error); |
| } |
| |
| bool TypedValueFromJson(const base::Value* value_in, |
| const ObjectSchema* object_schema, |
| std::string* value_out, ErrorPtr* error) { |
| return value_in->GetAsString(value_out) || |
| ReportUnexpectedJson(value_in, value_out, error); |
| } |
| |
| bool TypedValueFromJson(const base::Value* value_in, |
| const ObjectSchema* object_schema, |
| native_types::Object* value_out, ErrorPtr* error) { |
| const base::DictionaryValue* dict = nullptr; |
| if (!value_in->GetAsDictionary(&dict)) |
| return ReportUnexpectedJson(value_in, value_out, error); |
| |
| CHECK(object_schema) << "Object schema must be provided"; |
| |
| std::set<std::string> keys_processed; |
| for (const auto& pair : object_schema->GetProps()) { |
| const PropValue* def_value = pair.second->GetDefaultValue(); |
| if (dict->HasKey(pair.first)) { |
| std::shared_ptr<PropValue> value = pair.second->CreateValue(); |
| const base::Value* param_value = nullptr; |
| CHECK(dict->GetWithoutPathExpansion(pair.first, ¶m_value)) |
| << "Unable to get parameter"; |
| if (!value->FromJson(param_value, error)) |
| return false; |
| value_out->insert(std::make_pair(pair.first, std::move(value))); |
| } else if (def_value) { |
| std::shared_ptr<PropValue> value = def_value->Clone(); |
| value_out->insert(std::make_pair(pair.first, std::move(value))); |
| } else { |
| return ErrorMissingProperty(error, pair.first.c_str()); |
| } |
| keys_processed.insert(pair.first); |
| } |
| |
| // Just for sanity, make sure that we processed all the necessary properties |
| // and there weren't any extra (unknown) ones specified. If so, ignore |
| // them, but log as warnings... |
| base::DictionaryValue::Iterator iter(*dict); |
| while (!iter.IsAtEnd()) { |
| std::string key = iter.key(); |
| if (keys_processed.find(key) == keys_processed.end() && |
| !object_schema->GetExtraPropertiesAllowed()) { |
| Error::AddToPrintf(error, commands::errors::kDomain, |
| commands::errors::kUnknownProperty, |
| "Unrecognized parameter '%s'", key.c_str()); |
| return false; |
| } |
| iter.Advance(); |
| } |
| |
| // Now go over all property values and validate them. |
| for (const auto& pair : *value_out) { |
| const PropType* prop_type = pair.second->GetPropType(); |
| CHECK(prop_type) << "Value property type must be available"; |
| if (!prop_type->ValidateConstraints(*pair.second, error)) { |
| Error::AddToPrintf(error, commands::errors::kDomain, |
| commands::errors::kInvalidPropValue, |
| "Invalid parameter value for property '%s'", |
| pair.first.c_str()); |
| return false; |
| } |
| } |
| return true; |
| } |
| |
| // Compares two sets of key-value pairs from two Objects. |
| static bool obj_cmp(const native_types::Object::value_type& v1, |
| const native_types::Object::value_type& v2) { |
| return (v1.first == v2.first) && v1.second->IsEqual(v2.second.get()); |
| } |
| |
| bool operator==(const native_types::Object& obj1, |
| const native_types::Object& obj2) { |
| if (obj1.size() != obj2.size()) |
| return false; |
| |
| auto pair = std::mismatch(obj1.begin(), obj1.end(), obj2.begin(), obj_cmp); |
| return pair == std::make_pair(obj1.end(), obj2.end()); |
| } |
| |
| std::string ToString(const native_types::Object& obj) { |
| auto val = TypedValueToJson(obj, nullptr); |
| std::string str; |
| base::JSONWriter::Write(val.get(), &str); |
| return str; |
| } |
| |
| |
| } // namespace buffet |