| // Copyright 2017 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. | 
 |  | 
 | #include "base/value_iterators.h" | 
 |  | 
 | #include <type_traits> | 
 |  | 
 | #include "base/memory/ptr_util.h" | 
 | #include "base/values.h" | 
 | #include "testing/gmock/include/gmock/gmock.h" | 
 | #include "testing/gtest/include/gtest/gtest.h" | 
 |  | 
 | namespace base { | 
 |  | 
 | namespace detail { | 
 |  | 
 | namespace { | 
 |  | 
 | // Implementation of std::equal variant that is missing in C++11. | 
 | template <class BinaryPredicate, class InputIterator1, class InputIterator2> | 
 | bool are_equal(InputIterator1 first1, | 
 |                InputIterator1 last1, | 
 |                InputIterator2 first2, | 
 |                InputIterator2 last2, | 
 |                BinaryPredicate pred) { | 
 |   for (; first1 != last1 && first2 != last2; ++first1, ++first2) { | 
 |     if (!pred(*first1, *first2)) | 
 |       return false; | 
 |   } | 
 |   return first1 == last1 && first2 == last2; | 
 | } | 
 |  | 
 | }  // namespace | 
 |  | 
 | TEST(ValueIteratorsTest, SameDictStorage) { | 
 |   static_assert(std::is_same<Value::DictStorage, DictStorage>::value, | 
 |                 "DictStorage differs between Value and Value Iterators."); | 
 | } | 
 |  | 
 | TEST(ValueIteratorsTest, IsAssignable) { | 
 |   static_assert( | 
 |       !std::is_assignable<dict_iterator::reference::first_type, std::string>(), | 
 |       "Can assign strings to dict_iterator"); | 
 |  | 
 |   static_assert( | 
 |       std::is_assignable<dict_iterator::reference::second_type, Value>(), | 
 |       "Can't assign Values to dict_iterator"); | 
 |  | 
 |   static_assert(!std::is_assignable<const_dict_iterator::reference::first_type, | 
 |                                     std::string>(), | 
 |                 "Can assign strings to const_dict_iterator"); | 
 |  | 
 |   static_assert( | 
 |       !std::is_assignable<const_dict_iterator::reference::second_type, Value>(), | 
 |       "Can assign Values to const_dict_iterator"); | 
 | } | 
 |  | 
 | TEST(ValueIteratorsTest, DictIteratorOperatorStar) { | 
 |   DictStorage storage; | 
 |   storage.emplace("0", std::make_unique<Value>(0)); | 
 |  | 
 |   using iterator = dict_iterator; | 
 |   iterator iter(storage.begin()); | 
 |   EXPECT_EQ("0", (*iter).first); | 
 |   EXPECT_EQ(Value(0), (*iter).second); | 
 |  | 
 |   (*iter).second = Value(1); | 
 |   EXPECT_EQ(Value(1), *storage["0"]); | 
 | } | 
 |  | 
 | TEST(ValueIteratorsTest, DictIteratorOperatorArrow) { | 
 |   DictStorage storage; | 
 |   storage.emplace("0", std::make_unique<Value>(0)); | 
 |  | 
 |   using iterator = dict_iterator; | 
 |   iterator iter(storage.begin()); | 
 |   EXPECT_EQ("0", iter->first); | 
 |   EXPECT_EQ(Value(0), iter->second); | 
 |  | 
 |   iter->second = Value(1); | 
 |   EXPECT_EQ(Value(1), *storage["0"]); | 
 | } | 
 |  | 
 | TEST(ValueIteratorsTest, DictIteratorPreIncrement) { | 
 |   DictStorage storage; | 
 |   storage.emplace("0", std::make_unique<Value>(0)); | 
 |   storage.emplace("1", std::make_unique<Value>(1)); | 
 |  | 
 |   using iterator = dict_iterator; | 
 |   iterator iter(storage.begin()); | 
 |   EXPECT_EQ("0", iter->first); | 
 |   EXPECT_EQ(Value(0), iter->second); | 
 |  | 
 |   iterator& iter_ref = ++iter; | 
 |   EXPECT_EQ(&iter, &iter_ref); | 
 |  | 
 |   EXPECT_EQ("1", iter_ref->first); | 
 |   EXPECT_EQ(Value(1), iter_ref->second); | 
 | } | 
 |  | 
 | TEST(ValueIteratorsTest, DictIteratorPostIncrement) { | 
 |   DictStorage storage; | 
 |   storage.emplace("0", std::make_unique<Value>(0)); | 
 |   storage.emplace("1", std::make_unique<Value>(1)); | 
 |  | 
 |   using iterator = dict_iterator; | 
 |   iterator iter(storage.begin()); | 
 |   iterator iter_old = iter++; | 
 |  | 
 |   EXPECT_EQ("0", iter_old->first); | 
 |   EXPECT_EQ(Value(0), iter_old->second); | 
 |  | 
 |   EXPECT_EQ("1", iter->first); | 
 |   EXPECT_EQ(Value(1), iter->second); | 
 | } | 
 |  | 
 | TEST(ValueIteratorsTest, DictIteratorPreDecrement) { | 
 |   DictStorage storage; | 
 |   storage.emplace("0", std::make_unique<Value>(0)); | 
 |   storage.emplace("1", std::make_unique<Value>(1)); | 
 |  | 
 |   using iterator = dict_iterator; | 
 |   iterator iter(++storage.begin()); | 
 |   EXPECT_EQ("1", iter->first); | 
 |   EXPECT_EQ(Value(1), iter->second); | 
 |  | 
 |   iterator& iter_ref = --iter; | 
 |   EXPECT_EQ(&iter, &iter_ref); | 
 |  | 
 |   EXPECT_EQ("0", iter_ref->first); | 
 |   EXPECT_EQ(Value(0), iter_ref->second); | 
 | } | 
 |  | 
 | TEST(ValueIteratorsTest, DictIteratorPostDecrement) { | 
 |   DictStorage storage; | 
 |   storage.emplace("0", std::make_unique<Value>(0)); | 
 |   storage.emplace("1", std::make_unique<Value>(1)); | 
 |  | 
 |   using iterator = dict_iterator; | 
 |   iterator iter(++storage.begin()); | 
 |   iterator iter_old = iter--; | 
 |  | 
 |   EXPECT_EQ("1", iter_old->first); | 
 |   EXPECT_EQ(Value(1), iter_old->second); | 
 |  | 
 |   EXPECT_EQ("0", iter->first); | 
 |   EXPECT_EQ(Value(0), iter->second); | 
 | } | 
 |  | 
 | TEST(ValueIteratorsTest, DictIteratorOperatorEQ) { | 
 |   DictStorage storage; | 
 |   using iterator = dict_iterator; | 
 |   EXPECT_EQ(iterator(storage.begin()), iterator(storage.begin())); | 
 |   EXPECT_EQ(iterator(storage.end()), iterator(storage.end())); | 
 | } | 
 |  | 
 | TEST(ValueIteratorsTest, DictIteratorOperatorNE) { | 
 |   DictStorage storage; | 
 |   storage.emplace("0", std::make_unique<Value>(0)); | 
 |  | 
 |   using iterator = dict_iterator; | 
 |   EXPECT_NE(iterator(storage.begin()), iterator(storage.end())); | 
 | } | 
 |  | 
 | TEST(ValueIteratorsTest, ConstDictIteratorOperatorStar) { | 
 |   DictStorage storage; | 
 |   storage.emplace("0", std::make_unique<Value>(0)); | 
 |  | 
 |   using iterator = const_dict_iterator; | 
 |   iterator iter(storage.begin()); | 
 |   EXPECT_EQ("0", (*iter).first); | 
 |   EXPECT_EQ(Value(0), (*iter).second); | 
 | } | 
 |  | 
 | TEST(ValueIteratorsTest, ConstDictIteratorOperatorArrow) { | 
 |   DictStorage storage; | 
 |   storage.emplace("0", std::make_unique<Value>(0)); | 
 |  | 
 |   using iterator = const_dict_iterator; | 
 |   iterator iter(storage.begin()); | 
 |   EXPECT_EQ("0", iter->first); | 
 |   EXPECT_EQ(Value(0), iter->second); | 
 | } | 
 |  | 
 | TEST(ValueIteratorsTest, ConstDictIteratorPreIncrement) { | 
 |   DictStorage storage; | 
 |   storage.emplace("0", std::make_unique<Value>(0)); | 
 |   storage.emplace("1", std::make_unique<Value>(1)); | 
 |  | 
 |   using iterator = const_dict_iterator; | 
 |   iterator iter(storage.begin()); | 
 |   EXPECT_EQ("0", iter->first); | 
 |   EXPECT_EQ(Value(0), iter->second); | 
 |  | 
 |   iterator& iter_ref = ++iter; | 
 |   EXPECT_EQ(&iter, &iter_ref); | 
 |  | 
 |   EXPECT_EQ("1", iter_ref->first); | 
 |   EXPECT_EQ(Value(1), iter_ref->second); | 
 | } | 
 |  | 
 | TEST(ValueIteratorsTest, ConstDictIteratorPostIncrement) { | 
 |   DictStorage storage; | 
 |   storage.emplace("0", std::make_unique<Value>(0)); | 
 |   storage.emplace("1", std::make_unique<Value>(1)); | 
 |  | 
 |   using iterator = const_dict_iterator; | 
 |   iterator iter(storage.begin()); | 
 |   iterator iter_old = iter++; | 
 |  | 
 |   EXPECT_EQ("0", iter_old->first); | 
 |   EXPECT_EQ(Value(0), iter_old->second); | 
 |  | 
 |   EXPECT_EQ("1", iter->first); | 
 |   EXPECT_EQ(Value(1), iter->second); | 
 | } | 
 |  | 
 | TEST(ValueIteratorsTest, ConstDictIteratorPreDecrement) { | 
 |   DictStorage storage; | 
 |   storage.emplace("0", std::make_unique<Value>(0)); | 
 |   storage.emplace("1", std::make_unique<Value>(1)); | 
 |  | 
 |   using iterator = const_dict_iterator; | 
 |   iterator iter(++storage.begin()); | 
 |   EXPECT_EQ("1", iter->first); | 
 |   EXPECT_EQ(Value(1), iter->second); | 
 |  | 
 |   iterator& iter_ref = --iter; | 
 |   EXPECT_EQ(&iter, &iter_ref); | 
 |  | 
 |   EXPECT_EQ("0", iter_ref->first); | 
 |   EXPECT_EQ(Value(0), iter_ref->second); | 
 | } | 
 |  | 
 | TEST(ValueIteratorsTest, ConstDictIteratorPostDecrement) { | 
 |   DictStorage storage; | 
 |   storage.emplace("0", std::make_unique<Value>(0)); | 
 |   storage.emplace("1", std::make_unique<Value>(1)); | 
 |  | 
 |   using iterator = const_dict_iterator; | 
 |   iterator iter(++storage.begin()); | 
 |   iterator iter_old = iter--; | 
 |  | 
 |   EXPECT_EQ("1", iter_old->first); | 
 |   EXPECT_EQ(Value(1), iter_old->second); | 
 |  | 
 |   EXPECT_EQ("0", iter->first); | 
 |   EXPECT_EQ(Value(0), iter->second); | 
 | } | 
 |  | 
 | TEST(ValueIteratorsTest, ConstDictIteratorOperatorEQ) { | 
 |   DictStorage storage; | 
 |   using iterator = const_dict_iterator; | 
 |   EXPECT_EQ(iterator(storage.begin()), iterator(storage.begin())); | 
 |   EXPECT_EQ(iterator(storage.end()), iterator(storage.end())); | 
 | } | 
 |  | 
 | TEST(ValueIteratorsTest, ConstDictIteratorOperatorNE) { | 
 |   DictStorage storage; | 
 |   storage.emplace("0", std::make_unique<Value>(0)); | 
 |  | 
 |   using iterator = const_dict_iterator; | 
 |   EXPECT_NE(iterator(storage.begin()), iterator(storage.end())); | 
 | } | 
 |  | 
 | TEST(ValueIteratorsTest, DictIteratorProxy) { | 
 |   DictStorage storage; | 
 |   storage.emplace("null", std::make_unique<Value>(Value::Type::NONE)); | 
 |   storage.emplace("bool", std::make_unique<Value>(Value::Type::BOOLEAN)); | 
 |   storage.emplace("int", std::make_unique<Value>(Value::Type::INTEGER)); | 
 |   storage.emplace("double", std::make_unique<Value>(Value::Type::DOUBLE)); | 
 |   storage.emplace("string", std::make_unique<Value>(Value::Type::STRING)); | 
 |   storage.emplace("blob", std::make_unique<Value>(Value::Type::BINARY)); | 
 |   storage.emplace("dict", std::make_unique<Value>(Value::Type::DICTIONARY)); | 
 |   storage.emplace("list", std::make_unique<Value>(Value::Type::LIST)); | 
 |  | 
 |   using iterator = const_dict_iterator; | 
 |   using iterator_proxy = dict_iterator_proxy; | 
 |   iterator_proxy proxy(&storage); | 
 |  | 
 |   auto equal_to = [](const DictStorage::value_type& lhs, | 
 |                      const iterator::reference& rhs) { | 
 |     return std::tie(lhs.first, *lhs.second) == std::tie(rhs.first, rhs.second); | 
 |   }; | 
 |  | 
 |   EXPECT_TRUE(are_equal(storage.begin(), storage.end(), proxy.begin(), | 
 |                         proxy.end(), equal_to)); | 
 |  | 
 |   EXPECT_TRUE(are_equal(storage.rbegin(), storage.rend(), proxy.rbegin(), | 
 |                         proxy.rend(), equal_to)); | 
 |  | 
 |   EXPECT_TRUE(are_equal(storage.cbegin(), storage.cend(), proxy.cbegin(), | 
 |                         proxy.cend(), equal_to)); | 
 |  | 
 |   EXPECT_TRUE(are_equal(storage.crbegin(), storage.crend(), proxy.crbegin(), | 
 |                         proxy.crend(), equal_to)); | 
 | } | 
 |  | 
 | TEST(ValueIteratorsTest, ConstDictIteratorProxy) { | 
 |   DictStorage storage; | 
 |   storage.emplace("null", std::make_unique<Value>(Value::Type::NONE)); | 
 |   storage.emplace("bool", std::make_unique<Value>(Value::Type::BOOLEAN)); | 
 |   storage.emplace("int", std::make_unique<Value>(Value::Type::INTEGER)); | 
 |   storage.emplace("double", std::make_unique<Value>(Value::Type::DOUBLE)); | 
 |   storage.emplace("string", std::make_unique<Value>(Value::Type::STRING)); | 
 |   storage.emplace("blob", std::make_unique<Value>(Value::Type::BINARY)); | 
 |   storage.emplace("dict", std::make_unique<Value>(Value::Type::DICTIONARY)); | 
 |   storage.emplace("list", std::make_unique<Value>(Value::Type::LIST)); | 
 |  | 
 |   using iterator = const_dict_iterator; | 
 |   using iterator_proxy = const_dict_iterator_proxy; | 
 |   iterator_proxy proxy(&storage); | 
 |  | 
 |   auto equal_to = [](const DictStorage::value_type& lhs, | 
 |                      const iterator::reference& rhs) { | 
 |     return std::tie(lhs.first, *lhs.second) == std::tie(rhs.first, rhs.second); | 
 |   }; | 
 |  | 
 |   EXPECT_TRUE(are_equal(storage.begin(), storage.end(), proxy.begin(), | 
 |                         proxy.end(), equal_to)); | 
 |  | 
 |   EXPECT_TRUE(are_equal(storage.rbegin(), storage.rend(), proxy.rbegin(), | 
 |                         proxy.rend(), equal_to)); | 
 |  | 
 |   EXPECT_TRUE(are_equal(storage.cbegin(), storage.cend(), proxy.cbegin(), | 
 |                         proxy.cend(), equal_to)); | 
 |  | 
 |   EXPECT_TRUE(are_equal(storage.crbegin(), storage.crend(), proxy.crbegin(), | 
 |                         proxy.crend(), equal_to)); | 
 | } | 
 |  | 
 | }  // namespace detail | 
 |  | 
 | }  // namespace base |