| // 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 <chromeos/variant_dictionary.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, |
| chromeos::ErrorPtr* error) { |
| std::string value_as_string; |
| base::JSONWriter::Write(value_in, &value_as_string); |
| chromeos::Error::AddToPrintf(error, FROM_HERE, 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 base::Value* value_in, T*, |
| chromeos::ErrorPtr* error) { |
| ReportJsonTypeMismatch(value_in, |
| PropType::GetTypeStringFromType(GetValueType<T>()), |
| error); |
| return false; |
| } |
| |
| bool ErrorMissingProperty(chromeos::ErrorPtr* error, const char* param_name) { |
| chromeos::Error::AddToPrintf(error, FROM_HERE, 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::Value> TypedValueToJson(bool value, |
| chromeos::ErrorPtr* error) { |
| return std::unique_ptr<base::Value>(new base::FundamentalValue(value)); |
| } |
| |
| std::unique_ptr<base::Value> TypedValueToJson(int value, |
| chromeos::ErrorPtr* error) { |
| return std::unique_ptr<base::Value>(new base::FundamentalValue(value)); |
| } |
| |
| std::unique_ptr<base::Value> TypedValueToJson(double value, |
| chromeos::ErrorPtr* error) { |
| return std::unique_ptr<base::Value>(new base::FundamentalValue(value)); |
| } |
| |
| std::unique_ptr<base::Value> TypedValueToJson(const std::string& value, |
| chromeos::ErrorPtr* error) { |
| return std::unique_ptr<base::Value>(new base::StringValue(value)); |
| } |
| |
| std::unique_ptr<base::Value> TypedValueToJson(const native_types::Object& value, |
| chromeos::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, chromeos::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, chromeos::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, chromeos::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, chromeos::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, |
| chromeos::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)) { |
| chromeos::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))); |
| } 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()) { |
| chromeos::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)) { |
| chromeos::Error::AddToPrintf(error, FROM_HERE, errors::commands::kDomain, |
| errors::commands::kInvalidPropValue, |
| "Invalid 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; |
| } |
| |
| chromeos::Any PropValueToDBusVariant(const PropValue* value) { |
| if (value->GetType() != ValueType::Object) |
| return value->GetValueAsAny(); |
| // Special case for object types. |
| // Convert native_types::Object to chromeos::VariantDictionary |
| chromeos::VariantDictionary dict; |
| for (const auto& pair : value->GetObject()->GetValue()) { |
| // Since we are inserting the elements from native_types::Object which is |
| // a map, the keys are already sorted. So use the "end()" position as a hint |
| // for dict.insert() so the destination map can optimize its insertion |
| // time. |
| chromeos::Any prop = PropValueToDBusVariant(pair.second.get()); |
| dict.emplace_hint(dict.end(), pair.first, std::move(prop)); |
| } |
| return chromeos::Any(std::move(dict)); |
| } |
| |
| std::shared_ptr<const PropValue> PropValueFromDBusVariant( |
| const PropType* type, |
| const chromeos::Any& value, |
| chromeos::ErrorPtr* error) { |
| std::shared_ptr<const PropValue> result; |
| if (type->GetType() != ValueType::Object) { |
| result = type->CreateValue(value, error); |
| if (result && !type->ValidateConstraints(*result, error)) |
| result.reset(); |
| return result; |
| } |
| |
| // Special case for object types. |
| // We expect the |value| to contain chromeos::VariantDictionary, while |
| // PropValue must use native_types::Object instead. Do the conversion. |
| if (!value.IsTypeCompatible<chromeos::VariantDictionary>()) { |
| type->GenerateErrorValueTypeMismatch(error); |
| return result; |
| } |
| const auto& dict = value.Get<chromeos::VariantDictionary>(); |
| native_types::Object obj; |
| CHECK(nullptr != type->GetObjectSchemaPtr()) |
| << "An object type must have a schema defined for it"; |
| std::set<std::string> keys_processed; |
| // First go over all object parameters defined by type's object schema and |
| // extract the corresponding parameters from the source dictionary. |
| for (const auto& pair : type->GetObjectSchemaPtr()->GetProps()) { |
| const PropValue* def_value = pair.second->GetDefaultValue(); |
| auto it = dict.find(pair.first); |
| if (it != dict.end()) { |
| const PropType* prop_type = pair.second.get(); |
| CHECK(prop_type) << "Value property type must be available"; |
| auto prop_value = PropValueFromDBusVariant(prop_type, it->second, error); |
| if (!prop_value) { |
| chromeos::Error::AddToPrintf(error, FROM_HERE, |
| errors::commands::kDomain, |
| errors::commands::kInvalidPropValue, |
| "Invalid value for property '%s'", |
| pair.first.c_str()); |
| return result; |
| } |
| obj.emplace_hint(obj.end(), pair.first, std::move(prop_value)); |
| } else if (def_value) { |
| std::shared_ptr<const PropValue> prop_value = def_value->Clone(); |
| obj.emplace_hint(obj.end(), pair.first, std::move(prop_value)); |
| } else { |
| ErrorMissingProperty(error, pair.first.c_str()); |
| return result; |
| } |
| keys_processed.insert(pair.first); |
| } |
| |
| // Make sure that we processed all the necessary properties and there weren't |
| // any extra (unknown) ones specified, unless the schema allows them. |
| if (!type->GetObjectSchemaPtr()->GetExtraPropertiesAllowed()) { |
| for (const auto& pair : dict) { |
| if (keys_processed.find(pair.first) == keys_processed.end()) { |
| chromeos::Error::AddToPrintf(error, FROM_HERE, |
| errors::commands::kDomain, |
| errors::commands::kUnknownProperty, |
| "Unrecognized property '%s'", |
| pair.first.c_str()); |
| return result; |
| } |
| } |
| } |
| |
| result = type->CreateValue(std::move(obj), error); |
| if (result && !type->ValidateConstraints(*result, error)) |
| result.reset(); |
| return result; |
| } |
| |
| } // namespace buffet |