blob: ed8618289b7bc233b8bf4c651db07ef54eec294a [file] [log] [blame]
// 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