|  | // 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. | 
|  |  | 
|  | #ifndef BASE_JSON_JSON_VALUE_CONVERTER_H_ | 
|  | #define BASE_JSON_JSON_VALUE_CONVERTER_H_ | 
|  |  | 
|  | #include <stddef.h> | 
|  |  | 
|  | #include <memory> | 
|  | #include <string> | 
|  | #include <string_view> | 
|  | #include <vector> | 
|  |  | 
|  | #include "base/logging.h" | 
|  | #include "base/macros.h" | 
|  | #include "base/memory/ptr_util.h" | 
|  | #include "base/values.h" | 
|  |  | 
|  | // JSONValueConverter converts a JSON value into a C++ struct in a | 
|  | // lightweight way. | 
|  | // | 
|  | // Usage: | 
|  | // For real examples, you may want to refer to _unittest.cc file. | 
|  | // | 
|  | // Assume that you have a struct like this: | 
|  | //   struct Message { | 
|  | //     int foo; | 
|  | //     std::string bar; | 
|  | //     static void RegisterJSONConverter( | 
|  | //         JSONValueConverter<Message>* converter); | 
|  | //   }; | 
|  | // | 
|  | // And you want to parse a json data into this struct.  First, you | 
|  | // need to declare RegisterJSONConverter() method in your struct. | 
|  | //   // static | 
|  | //   void Message::RegisterJSONConverter( | 
|  | //       JSONValueConverter<Message>* converter) { | 
|  | //     converter->RegisterIntField("foo", &Message::foo); | 
|  | //     converter->RegisterStringField("bar", &Message::bar); | 
|  | //   } | 
|  | // | 
|  | // Then, you just instantiate your JSONValueConverter of your type and call | 
|  | // Convert() method. | 
|  | //   Message message; | 
|  | //   JSONValueConverter<Message> converter; | 
|  | //   converter.Convert(json, &message); | 
|  | // | 
|  | // Convert() returns false when it fails.  Here "fail" means that the value is | 
|  | // structurally different from expected, such like a string value appears | 
|  | // for an int field.  Do not report failures for missing fields. | 
|  | // Also note that Convert() will modify the passed |message| even when it | 
|  | // fails for performance reason. | 
|  | // | 
|  | // For nested field, the internal message also has to implement the registration | 
|  | // method.  Then, just use RegisterNestedField() from the containing struct's | 
|  | // RegisterJSONConverter method. | 
|  | //   struct Nested { | 
|  | //     Message foo; | 
|  | //     static void RegisterJSONConverter(...) { | 
|  | //       ... | 
|  | //       converter->RegisterNestedField("foo", &Nested::foo); | 
|  | //     } | 
|  | //   }; | 
|  | // | 
|  | // For repeated field, we just assume std::vector<std::unique_ptr<ElementType>> | 
|  | // for its container and you can put RegisterRepeatedInt or some other types. | 
|  | // Use RegisterRepeatedMessage for nested repeated fields. | 
|  | // | 
|  | // Sometimes JSON format uses string representations for other types such | 
|  | // like enum, timestamp, or URL.  You can use RegisterCustomField method | 
|  | // and specify a function to convert a std::string_view to your type. | 
|  | //   bool ConvertFunc(std::string_view s, YourEnum* result) { | 
|  | //     // do something and return true if succeed... | 
|  | //   } | 
|  | //   struct Message { | 
|  | //     YourEnum ye; | 
|  | //     ... | 
|  | //     static void RegisterJSONConverter(...) { | 
|  | //       ... | 
|  | //       converter->RegsiterCustomField<YourEnum>( | 
|  | //           "your_enum", &Message::ye, &ConvertFunc); | 
|  | //     } | 
|  | //   }; | 
|  |  | 
|  | namespace base { | 
|  |  | 
|  | template <typename StructType> | 
|  | class JSONValueConverter; | 
|  |  | 
|  | namespace internal { | 
|  |  | 
|  | template <typename StructType> | 
|  | class FieldConverterBase { | 
|  | public: | 
|  | explicit FieldConverterBase(const std::string& path) : field_path_(path) {} | 
|  | virtual ~FieldConverterBase() = default; | 
|  | virtual bool ConvertField(const base::Value& value, | 
|  | StructType* obj) const = 0; | 
|  | const std::string& field_path() const { return field_path_; } | 
|  |  | 
|  | private: | 
|  | std::string field_path_; | 
|  | DISALLOW_COPY_AND_ASSIGN(FieldConverterBase); | 
|  | }; | 
|  |  | 
|  | template <typename FieldType> | 
|  | class ValueConverter { | 
|  | public: | 
|  | virtual ~ValueConverter() = default; | 
|  | virtual bool Convert(const base::Value& value, FieldType* field) const = 0; | 
|  | }; | 
|  |  | 
|  | template <typename StructType, typename FieldType> | 
|  | class FieldConverter : public FieldConverterBase<StructType> { | 
|  | public: | 
|  | explicit FieldConverter(const std::string& path, | 
|  | FieldType StructType::*field, | 
|  | ValueConverter<FieldType>* converter) | 
|  | : FieldConverterBase<StructType>(path), | 
|  | field_pointer_(field), | 
|  | value_converter_(converter) {} | 
|  |  | 
|  | bool ConvertField(const base::Value& value, StructType* dst) const override { | 
|  | return value_converter_->Convert(value, &(dst->*field_pointer_)); | 
|  | } | 
|  |  | 
|  | private: | 
|  | FieldType StructType::*field_pointer_; | 
|  | std::unique_ptr<ValueConverter<FieldType>> value_converter_; | 
|  | DISALLOW_COPY_AND_ASSIGN(FieldConverter); | 
|  | }; | 
|  |  | 
|  | template <typename FieldType> | 
|  | class BasicValueConverter; | 
|  |  | 
|  | template <> | 
|  | class BasicValueConverter<int> : public ValueConverter<int> { | 
|  | public: | 
|  | BasicValueConverter() = default; | 
|  |  | 
|  | bool Convert(const base::Value& value, int* field) const override; | 
|  |  | 
|  | private: | 
|  | DISALLOW_COPY_AND_ASSIGN(BasicValueConverter); | 
|  | }; | 
|  |  | 
|  | template <> | 
|  | class BasicValueConverter<std::string> : public ValueConverter<std::string> { | 
|  | public: | 
|  | BasicValueConverter() = default; | 
|  |  | 
|  | bool Convert(const base::Value& value, std::string* field) const override; | 
|  |  | 
|  | private: | 
|  | DISALLOW_COPY_AND_ASSIGN(BasicValueConverter); | 
|  | }; | 
|  |  | 
|  | template <> | 
|  | class BasicValueConverter<std::u16string> | 
|  | : public ValueConverter<std::u16string> { | 
|  | public: | 
|  | BasicValueConverter() = default; | 
|  |  | 
|  | bool Convert(const base::Value& value, std::u16string* field) const override; | 
|  |  | 
|  | private: | 
|  | DISALLOW_COPY_AND_ASSIGN(BasicValueConverter); | 
|  | }; | 
|  |  | 
|  | template <> | 
|  | class BasicValueConverter<double> : public ValueConverter<double> { | 
|  | public: | 
|  | BasicValueConverter() = default; | 
|  |  | 
|  | bool Convert(const base::Value& value, double* field) const override; | 
|  |  | 
|  | private: | 
|  | DISALLOW_COPY_AND_ASSIGN(BasicValueConverter); | 
|  | }; | 
|  |  | 
|  | template <> | 
|  | class BasicValueConverter<bool> : public ValueConverter<bool> { | 
|  | public: | 
|  | BasicValueConverter() = default; | 
|  |  | 
|  | bool Convert(const base::Value& value, bool* field) const override; | 
|  |  | 
|  | private: | 
|  | DISALLOW_COPY_AND_ASSIGN(BasicValueConverter); | 
|  | }; | 
|  |  | 
|  | template <typename FieldType> | 
|  | class ValueFieldConverter : public ValueConverter<FieldType> { | 
|  | public: | 
|  | typedef bool (*ConvertFunc)(const base::Value* value, FieldType* field); | 
|  |  | 
|  | explicit ValueFieldConverter(ConvertFunc convert_func) | 
|  | : convert_func_(convert_func) {} | 
|  |  | 
|  | bool Convert(const base::Value& value, FieldType* field) const override { | 
|  | return convert_func_(&value, field); | 
|  | } | 
|  |  | 
|  | private: | 
|  | ConvertFunc convert_func_; | 
|  |  | 
|  | DISALLOW_COPY_AND_ASSIGN(ValueFieldConverter); | 
|  | }; | 
|  |  | 
|  | template <typename FieldType> | 
|  | class CustomFieldConverter : public ValueConverter<FieldType> { | 
|  | public: | 
|  | typedef bool (*ConvertFunc)(std::string_view value, FieldType* field); | 
|  |  | 
|  | explicit CustomFieldConverter(ConvertFunc convert_func) | 
|  | : convert_func_(convert_func) {} | 
|  |  | 
|  | bool Convert(const base::Value& value, FieldType* field) const override { | 
|  | std::string string_value; | 
|  | return value.GetAsString(&string_value) && | 
|  | convert_func_(string_value, field); | 
|  | } | 
|  |  | 
|  | private: | 
|  | ConvertFunc convert_func_; | 
|  |  | 
|  | DISALLOW_COPY_AND_ASSIGN(CustomFieldConverter); | 
|  | }; | 
|  |  | 
|  | template <typename NestedType> | 
|  | class NestedValueConverter : public ValueConverter<NestedType> { | 
|  | public: | 
|  | NestedValueConverter() = default; | 
|  |  | 
|  | bool Convert(const base::Value& value, NestedType* field) const override { | 
|  | return converter_.Convert(value, field); | 
|  | } | 
|  |  | 
|  | private: | 
|  | JSONValueConverter<NestedType> converter_; | 
|  | DISALLOW_COPY_AND_ASSIGN(NestedValueConverter); | 
|  | }; | 
|  |  | 
|  | template <typename Element> | 
|  | class RepeatedValueConverter | 
|  | : public ValueConverter<std::vector<std::unique_ptr<Element>>> { | 
|  | public: | 
|  | RepeatedValueConverter() = default; | 
|  |  | 
|  | bool Convert(const base::Value& value, | 
|  | std::vector<std::unique_ptr<Element>>* field) const override { | 
|  | const base::ListValue* list = NULL; | 
|  | if (!value.GetAsList(&list)) { | 
|  | // The field is not a list. | 
|  | return false; | 
|  | } | 
|  |  | 
|  | field->reserve(list->GetSize()); | 
|  | for (size_t i = 0; i < list->GetSize(); ++i) { | 
|  | const base::Value* element = NULL; | 
|  | if (!list->Get(i, &element)) | 
|  | continue; | 
|  |  | 
|  | std::unique_ptr<Element> e(new Element); | 
|  | if (basic_converter_.Convert(*element, e.get())) { | 
|  | field->push_back(std::move(e)); | 
|  | } else { | 
|  | return false; | 
|  | } | 
|  | } | 
|  | return true; | 
|  | } | 
|  |  | 
|  | private: | 
|  | BasicValueConverter<Element> basic_converter_; | 
|  | DISALLOW_COPY_AND_ASSIGN(RepeatedValueConverter); | 
|  | }; | 
|  |  | 
|  | template <typename NestedType> | 
|  | class RepeatedMessageConverter | 
|  | : public ValueConverter<std::vector<std::unique_ptr<NestedType>>> { | 
|  | public: | 
|  | RepeatedMessageConverter() = default; | 
|  |  | 
|  | bool Convert(const base::Value& value, | 
|  | std::vector<std::unique_ptr<NestedType>>* field) const override { | 
|  | const base::ListValue* list = NULL; | 
|  | if (!value.GetAsList(&list)) | 
|  | return false; | 
|  |  | 
|  | field->reserve(list->GetSize()); | 
|  | for (size_t i = 0; i < list->GetSize(); ++i) { | 
|  | const base::Value* element = NULL; | 
|  | if (!list->Get(i, &element)) | 
|  | continue; | 
|  |  | 
|  | std::unique_ptr<NestedType> nested(new NestedType); | 
|  | if (converter_.Convert(*element, nested.get())) { | 
|  | field->push_back(std::move(nested)); | 
|  | } else { | 
|  | return false; | 
|  | } | 
|  | } | 
|  | return true; | 
|  | } | 
|  |  | 
|  | private: | 
|  | JSONValueConverter<NestedType> converter_; | 
|  | DISALLOW_COPY_AND_ASSIGN(RepeatedMessageConverter); | 
|  | }; | 
|  |  | 
|  | template <typename NestedType> | 
|  | class RepeatedCustomValueConverter | 
|  | : public ValueConverter<std::vector<std::unique_ptr<NestedType>>> { | 
|  | public: | 
|  | typedef bool (*ConvertFunc)(const base::Value* value, NestedType* field); | 
|  |  | 
|  | explicit RepeatedCustomValueConverter(ConvertFunc convert_func) | 
|  | : convert_func_(convert_func) {} | 
|  |  | 
|  | bool Convert(const base::Value& value, | 
|  | std::vector<std::unique_ptr<NestedType>>* field) const override { | 
|  | const base::ListValue* list = NULL; | 
|  | if (!value.GetAsList(&list)) | 
|  | return false; | 
|  |  | 
|  | field->reserve(list->GetSize()); | 
|  | for (size_t i = 0; i < list->GetSize(); ++i) { | 
|  | const base::Value* element = NULL; | 
|  | if (!list->Get(i, &element)) | 
|  | continue; | 
|  |  | 
|  | std::unique_ptr<NestedType> nested(new NestedType); | 
|  | if ((*convert_func_)(element, nested.get())) { | 
|  | field->push_back(std::move(nested)); | 
|  | } else { | 
|  | return false; | 
|  | } | 
|  | } | 
|  | return true; | 
|  | } | 
|  |  | 
|  | private: | 
|  | ConvertFunc convert_func_; | 
|  | DISALLOW_COPY_AND_ASSIGN(RepeatedCustomValueConverter); | 
|  | }; | 
|  |  | 
|  | }  // namespace internal | 
|  |  | 
|  | template <class StructType> | 
|  | class JSONValueConverter { | 
|  | public: | 
|  | JSONValueConverter() { StructType::RegisterJSONConverter(this); } | 
|  |  | 
|  | void RegisterIntField(const std::string& field_name, int StructType::*field) { | 
|  | fields_.push_back( | 
|  | std::make_unique<internal::FieldConverter<StructType, int>>( | 
|  | field_name, field, new internal::BasicValueConverter<int>)); | 
|  | } | 
|  |  | 
|  | void RegisterStringField(const std::string& field_name, | 
|  | std::string StructType::*field) { | 
|  | fields_.push_back( | 
|  | std::make_unique<internal::FieldConverter<StructType, std::string>>( | 
|  | field_name, field, new internal::BasicValueConverter<std::string>)); | 
|  | } | 
|  |  | 
|  | void RegisterStringField(const std::string& field_name, | 
|  | std::u16string StructType::*field) { | 
|  | fields_.push_back( | 
|  | std::make_unique<internal::FieldConverter<StructType, std::u16string>>( | 
|  | field_name, field, | 
|  | new internal::BasicValueConverter<std::u16string>)); | 
|  | } | 
|  |  | 
|  | void RegisterBoolField(const std::string& field_name, | 
|  | bool StructType::*field) { | 
|  | fields_.push_back( | 
|  | std::make_unique<internal::FieldConverter<StructType, bool>>( | 
|  | field_name, field, new internal::BasicValueConverter<bool>)); | 
|  | } | 
|  |  | 
|  | void RegisterDoubleField(const std::string& field_name, | 
|  | double StructType::*field) { | 
|  | fields_.push_back( | 
|  | std::make_unique<internal::FieldConverter<StructType, double>>( | 
|  | field_name, field, new internal::BasicValueConverter<double>)); | 
|  | } | 
|  |  | 
|  | template <class NestedType> | 
|  | void RegisterNestedField(const std::string& field_name, | 
|  | NestedType StructType::*field) { | 
|  | fields_.push_back( | 
|  | std::make_unique<internal::FieldConverter<StructType, NestedType>>( | 
|  | field_name, field, new internal::NestedValueConverter<NestedType>)); | 
|  | } | 
|  |  | 
|  | template <typename FieldType> | 
|  | void RegisterCustomField(const std::string& field_name, | 
|  | FieldType StructType::*field, | 
|  | bool (*convert_func)(std::string_view, FieldType*)) { | 
|  | fields_.push_back( | 
|  | std::make_unique<internal::FieldConverter<StructType, FieldType>>( | 
|  | field_name, field, | 
|  | new internal::CustomFieldConverter<FieldType>(convert_func))); | 
|  | } | 
|  |  | 
|  | template <typename FieldType> | 
|  | void RegisterCustomValueField(const std::string& field_name, | 
|  | FieldType StructType::*field, | 
|  | bool (*convert_func)(const base::Value*, | 
|  | FieldType*)) { | 
|  | fields_.push_back( | 
|  | std::make_unique<internal::FieldConverter<StructType, FieldType>>( | 
|  | field_name, field, | 
|  | new internal::ValueFieldConverter<FieldType>(convert_func))); | 
|  | } | 
|  |  | 
|  | void RegisterRepeatedInt( | 
|  | const std::string& field_name, | 
|  | std::vector<std::unique_ptr<int>> StructType::*field) { | 
|  | fields_.push_back(std::make_unique<internal::FieldConverter< | 
|  | StructType, std::vector<std::unique_ptr<int>>>>( | 
|  | field_name, field, new internal::RepeatedValueConverter<int>)); | 
|  | } | 
|  |  | 
|  | void RegisterRepeatedString( | 
|  | const std::string& field_name, | 
|  | std::vector<std::unique_ptr<std::string>> StructType::*field) { | 
|  | fields_.push_back( | 
|  | std::make_unique<internal::FieldConverter< | 
|  | StructType, std::vector<std::unique_ptr<std::string>>>>( | 
|  | field_name, field, | 
|  | new internal::RepeatedValueConverter<std::string>)); | 
|  | } | 
|  |  | 
|  | void RegisterRepeatedString( | 
|  | const std::string& field_name, | 
|  | std::vector<std::unique_ptr<std::u16string>> StructType::*field) { | 
|  | fields_.push_back( | 
|  | std::make_unique<internal::FieldConverter< | 
|  | StructType, std::vector<std::unique_ptr<std::u16string>>>>( | 
|  | field_name, field, | 
|  | new internal::RepeatedValueConverter<std::u16string>)); | 
|  | } | 
|  |  | 
|  | void RegisterRepeatedDouble( | 
|  | const std::string& field_name, | 
|  | std::vector<std::unique_ptr<double>> StructType::*field) { | 
|  | fields_.push_back(std::make_unique<internal::FieldConverter< | 
|  | StructType, std::vector<std::unique_ptr<double>>>>( | 
|  | field_name, field, new internal::RepeatedValueConverter<double>)); | 
|  | } | 
|  |  | 
|  | void RegisterRepeatedBool( | 
|  | const std::string& field_name, | 
|  | std::vector<std::unique_ptr<bool>> StructType::*field) { | 
|  | fields_.push_back(std::make_unique<internal::FieldConverter< | 
|  | StructType, std::vector<std::unique_ptr<bool>>>>( | 
|  | field_name, field, new internal::RepeatedValueConverter<bool>)); | 
|  | } | 
|  |  | 
|  | template <class NestedType> | 
|  | void RegisterRepeatedCustomValue( | 
|  | const std::string& field_name, | 
|  | std::vector<std::unique_ptr<NestedType>> StructType::*field, | 
|  | bool (*convert_func)(const base::Value*, NestedType*)) { | 
|  | fields_.push_back( | 
|  | std::make_unique<internal::FieldConverter< | 
|  | StructType, std::vector<std::unique_ptr<NestedType>>>>( | 
|  | field_name, field, | 
|  | new internal::RepeatedCustomValueConverter<NestedType>( | 
|  | convert_func))); | 
|  | } | 
|  |  | 
|  | template <class NestedType> | 
|  | void RegisterRepeatedMessage( | 
|  | const std::string& field_name, | 
|  | std::vector<std::unique_ptr<NestedType>> StructType::*field) { | 
|  | fields_.push_back( | 
|  | std::make_unique<internal::FieldConverter< | 
|  | StructType, std::vector<std::unique_ptr<NestedType>>>>( | 
|  | field_name, field, | 
|  | new internal::RepeatedMessageConverter<NestedType>)); | 
|  | } | 
|  |  | 
|  | bool Convert(const base::Value& value, StructType* output) const { | 
|  | const DictionaryValue* dictionary_value = NULL; | 
|  | if (!value.GetAsDictionary(&dictionary_value)) | 
|  | return false; | 
|  |  | 
|  | for (size_t i = 0; i < fields_.size(); ++i) { | 
|  | const internal::FieldConverterBase<StructType>* field_converter = | 
|  | fields_[i].get(); | 
|  | const base::Value* field = NULL; | 
|  | if (dictionary_value->Get(field_converter->field_path(), &field)) { | 
|  | if (!field_converter->ConvertField(*field, output)) { | 
|  | return false; | 
|  | } | 
|  | } | 
|  | } | 
|  | return true; | 
|  | } | 
|  |  | 
|  | private: | 
|  | std::vector<std::unique_ptr<internal::FieldConverterBase<StructType>>> | 
|  | fields_; | 
|  |  | 
|  | DISALLOW_COPY_AND_ASSIGN(JSONValueConverter); | 
|  | }; | 
|  |  | 
|  | }  // namespace base | 
|  |  | 
|  | #endif  // BASE_JSON_JSON_VALUE_CONVERTER_H_ |