| // Copyright 2015 The Weave 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 "src/commands/schema_utils.h" |
| |
| #include <algorithm> |
| #include <set> |
| #include <string> |
| |
| #include <base/json/json_writer.h> |
| #include <base/logging.h> |
| |
| #include "src/commands/object_schema.h" |
| #include "src/commands/prop_types.h" |
| #include "src/commands/prop_values.h" |
| |
| namespace weave { |
| namespace { |
| // Helper function to report "type mismatch" errors when parsing JSON. |
| void ReportJsonTypeMismatch(const tracked_objects::Location& location, |
| 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, location, errors::commands::kDomain, |
| errors::commands::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 tracked_objects::Location& location, |
| const base::Value* value_in, |
| T*, |
| ErrorPtr* error) { |
| ReportJsonTypeMismatch(location, value_in, |
| PropType::GetTypeStringFromType(GetValueType<T>()), |
| error); |
| return false; |
| } |
| |
| bool ErrorMissingProperty(ErrorPtr* error, |
| const tracked_objects::Location& location, |
| const char* param_name) { |
| Error::AddToPrintf(error, location, errors::commands::kDomain, |
| errors::commands::kPropertyMissing, |
| "Required parameter missing: %s", param_name); |
| return false; |
| } |
| |
| } // namespace |
| |
| // Specializations of TypedValueToJson<T>() for supported C++ types. |
| std::unique_ptr<base::FundamentalValue> TypedValueToJson(bool value) { |
| return std::unique_ptr<base::FundamentalValue>( |
| new base::FundamentalValue(value)); |
| } |
| |
| std::unique_ptr<base::FundamentalValue> TypedValueToJson(int value) { |
| return std::unique_ptr<base::FundamentalValue>( |
| new base::FundamentalValue(value)); |
| } |
| |
| std::unique_ptr<base::FundamentalValue> TypedValueToJson(double value) { |
| return std::unique_ptr<base::FundamentalValue>( |
| new base::FundamentalValue(value)); |
| } |
| |
| std::unique_ptr<base::StringValue> TypedValueToJson(const std::string& value) { |
| return std::unique_ptr<base::StringValue>(new base::StringValue(value)); |
| } |
| |
| std::unique_ptr<base::DictionaryValue> TypedValueToJson(const ValueMap& value) { |
| std::unique_ptr<base::DictionaryValue> dict(new base::DictionaryValue); |
| for (const auto& pair : value) { |
| auto prop_value = pair.second->ToJson(); |
| CHECK(prop_value); |
| dict->SetWithoutPathExpansion(pair.first, prop_value.release()); |
| } |
| return dict; |
| } |
| |
| std::unique_ptr<base::ListValue> TypedValueToJson(const ValueVector& value) { |
| std::unique_ptr<base::ListValue> list(new base::ListValue); |
| for (const auto& item : value) { |
| auto json = item->ToJson(); |
| CHECK(json); |
| list->Append(json.release()); |
| } |
| return list; |
| } |
| |
| bool TypedValueFromJson(const base::Value* value_in, |
| const PropType* type, |
| bool* value_out, |
| ErrorPtr* error) { |
| return value_in->GetAsBoolean(value_out) || |
| ReportUnexpectedJson(FROM_HERE, value_in, value_out, error); |
| } |
| |
| bool TypedValueFromJson(const base::Value* value_in, |
| const PropType* type, |
| int* value_out, |
| ErrorPtr* error) { |
| return value_in->GetAsInteger(value_out) || |
| ReportUnexpectedJson(FROM_HERE, value_in, value_out, error); |
| } |
| |
| bool TypedValueFromJson(const base::Value* value_in, |
| const PropType* type, |
| double* value_out, |
| ErrorPtr* error) { |
| return value_in->GetAsDouble(value_out) || |
| ReportUnexpectedJson(FROM_HERE, value_in, value_out, error); |
| } |
| |
| bool TypedValueFromJson(const base::Value* value_in, |
| const PropType* type, |
| std::string* value_out, |
| ErrorPtr* error) { |
| return value_in->GetAsString(value_out) || |
| ReportUnexpectedJson(FROM_HERE, value_in, value_out, error); |
| } |
| |
| bool TypedValueFromJson(const base::Value* value_in, |
| const PropType* type, |
| ValueMap* value_out, |
| ErrorPtr* error) { |
| const base::DictionaryValue* dict = nullptr; |
| if (!value_in->GetAsDictionary(&dict)) |
| return ReportUnexpectedJson(FROM_HERE, value_in, value_out, error); |
| |
| CHECK(type) << "Object definition must be provided"; |
| CHECK(ValueType::Object == type->GetType()) << "Type must be Object"; |
| |
| const ObjectSchema* object_schema = type->GetObject()->GetObjectSchemaPtr(); |
| std::set<std::string> keys_processed; |
| value_out->clear(); // Clear possible default values already in |value_out|. |
| for (const auto& pair : object_schema->GetProps()) { |
| const PropValue* def_value = pair.second->GetDefaultValue(); |
| if (dict->HasKey(pair.first)) { |
| const base::Value* param_value = nullptr; |
| CHECK(dict->GetWithoutPathExpansion(pair.first, ¶m_value)) |
| << "Unable to get parameter"; |
| auto value = pair.second->CreatePropValue(*param_value, error); |
| if (!value) { |
| Error::AddToPrintf(error, FROM_HERE, errors::commands::kDomain, |
| errors::commands::kInvalidPropValue, |
| "Invalid value for property '%s'", |
| pair.first.c_str()); |
| return false; |
| } |
| value_out->insert(std::make_pair(pair.first, std::move(value))); |
| keys_processed.insert(pair.first); |
| } else if (def_value) { |
| value_out->insert(std::make_pair(pair.first, def_value->Clone())); |
| keys_processed.insert(pair.first); |
| } else if (pair.second->IsRequired()) { |
| return ErrorMissingProperty(error, FROM_HERE, pair.first.c_str()); |
| } |
| } |
| |
| // 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, FROM_HERE, errors::commands::kDomain, |
| errors::commands::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, FROM_HERE, errors::commands::kDomain, |
| errors::commands::kInvalidPropValue, |
| "Invalid value for property '%s'", pair.first.c_str()); |
| return false; |
| } |
| } |
| return true; |
| } |
| |
| bool TypedValueFromJson(const base::Value* value_in, |
| const PropType* type, |
| ValueVector* value_out, |
| ErrorPtr* error) { |
| const base::ListValue* list = nullptr; |
| if (!value_in->GetAsList(&list)) |
| return ReportUnexpectedJson(FROM_HERE, value_in, value_out, error); |
| |
| CHECK(type) << "Array type definition must be provided"; |
| CHECK(ValueType::Array == type->GetType()) << "Type must be Array"; |
| const PropType* item_type = type->GetArray()->GetItemTypePtr(); |
| CHECK(item_type) << "Incomplete array type definition"; |
| |
| // This value might already contain values from the type defaults. |
| // Clear them first before proceeding. |
| value_out->clear(); |
| value_out->reserve(list->GetSize()); |
| for (const base::Value* item : *list) { |
| std::unique_ptr<PropValue> prop_value = |
| item_type->CreatePropValue(*item, error); |
| if (!prop_value) |
| return false; |
| value_out->push_back(std::move(prop_value)); |
| } |
| return true; |
| } |
| |
| // Compares two sets of key-value pairs from two Objects. |
| static bool obj_cmp(const ValueMap::value_type& v1, |
| const ValueMap::value_type& v2) { |
| return (v1.first == v2.first) && v1.second->IsEqual(v2.second.get()); |
| } |
| |
| bool operator==(const ValueMap& obj1, const ValueMap& 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()); |
| } |
| |
| bool operator==(const ValueVector& arr1, const ValueVector& arr2) { |
| if (arr1.size() != arr2.size()) |
| return false; |
| |
| using Type = const ValueVector::value_type; |
| // Compare two array items. |
| auto arr_cmp = [](const Type& v1, const Type& v2) { |
| return v1->IsEqual(v2.get()); |
| }; |
| auto pair = std::mismatch(arr1.begin(), arr1.end(), arr2.begin(), arr_cmp); |
| return pair == std::make_pair(arr1.end(), arr2.end()); |
| } |
| |
| std::string ToString(const ValueMap& obj) { |
| auto val = TypedValueToJson(obj); |
| std::string str; |
| base::JSONWriter::Write(*val, &str); |
| return str; |
| } |
| |
| std::string ToString(const ValueVector& arr) { |
| auto val = TypedValueToJson(arr); |
| std::string str; |
| base::JSONWriter::Write(*val, &str); |
| return str; |
| } |
| |
| } // namespace weave |