Movatterモバイル変換


[0]ホーム

URL:


Google Git
Sign in
chromium /chromium /src /refs/heads/main /. /base /values_unittest.cc
blob: b8ca5d8e5b3014cddca41ba81c2a9966ac10b9c9 [file] [log] [blame] [edit]
// Copyright 2012 The Chromium Authors
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#ifdef UNSAFE_BUFFERS_BUILD
// TODO(crbug.com/40284755): Remove this and spanify to fix the errors.
#pragma allow_unsafe_buffers
#endif
#include"base/values.h"
#include<stddef.h>
#include<algorithm>
#include<functional>
#include<iterator>
#include<limits>
#include<memory>
#include<optional>
#include<string>
#include<string_view>
#include<type_traits>
#include<utility>
#include<variant>
#include<vector>
#include"base/bits.h"
#include"base/containers/adapters.h"
#include"base/containers/contains.h"
#include"base/strings/utf_string_conversions.h"
#include"base/test/gtest_util.h"
#include"build/build_config.h"
#include"testing/gmock/include/gmock/gmock.h"
#include"testing/gtest/include/gtest/gtest.h"
#include"third_party/perfetto/include/perfetto/test/traced_value_test_support.h"
namespace base{
#ifdef NDEBUG
// `Value` should have a (relatively) small size to avoid creating excess
// overhead, e.g. for lists of values that are all ints.
//
// This test is limited to NDEBUG builds, since some containers may require
// extra storage for supporting debug checks for things like iterators.
TEST(ValuesTest,SizeOfValue){
#if defined(__GLIBCXX__)
// libstdc++ std::string takes already 4 machine words, so the std::variant
// takes 5
constexprsize_t kExpectedSize=5*sizeof(void*);
#else// !defined(__GLIBCXX__)
// libc++'s std::string and std::vector both take 3 machine words. An
// additional word is used by std::variant for the type index.
constexprsize_t kExpectedSize=4*sizeof(void*);
#endif// defined(__GLIBCXX__)
// Use std::integral_constant so the compiler error message includes the
// evaluated size. In future versions of clang, it should be possible to
// simplify this to an equality comparison (i.e. newer clangs print out
// "comparison reduces to '(1 == 2)'").
static_assert(std::is_same_v<std::integral_constant<size_t,sizeof(Value)>,
std::integral_constant<size_t, kExpectedSize>>,
"base::Value has an unexpected size!");
}
#endif
TEST(ValuesTest,TestNothrow){
static_assert(std::is_nothrow_move_constructible_v<Value>,
"IsNothrowMoveConstructible");
static_assert(std::is_nothrow_default_constructible_v<Value>,
"IsNothrowDefaultConstructible");
static_assert(std::is_nothrow_constructible_v<Value, std::string&&>,
"IsNothrowMoveConstructibleFromString");
static_assert(std::is_nothrow_constructible_v<Value,Value::BlobStorage&&>,
"IsNothrowMoveConstructibleFromBlob");
static_assert(std::is_nothrow_move_assignable_v<Value>,
"IsNothrowMoveAssignable");
}
TEST(ValuesTest,EmptyValue){
Value value;
EXPECT_EQ(Value::Type::NONE, value.type());
EXPECT_EQ(std::nullopt, value.GetIfBool());
EXPECT_EQ(std::nullopt, value.GetIfInt());
EXPECT_EQ(std::nullopt, value.GetIfDouble());
EXPECT_EQ(nullptr, value.GetIfString());
EXPECT_EQ(nullptr, value.GetIfBlob());
}
// Group of tests for the value constructors.
TEST(ValuesTest,ConstructBool){
Value true_value(true);
EXPECT_EQ(Value::Type::BOOLEAN, true_value.type());
EXPECT_THAT(true_value.GetIfBool(), testing::Optional(true));
EXPECT_TRUE(true_value.GetBool());
Value false_value(false);
EXPECT_EQ(Value::Type::BOOLEAN, false_value.type());
EXPECT_THAT(false_value.GetIfBool(), testing::Optional(false));
EXPECT_FALSE(false_value.GetBool());
}
TEST(ValuesTest,ConstructFromPtrs){
static_assert(!std::is_constructible_v<Value,int*>,"");
static_assert(!std::is_constructible_v<Value,constint*>,"");
static_assert(!std::is_constructible_v<Value,wchar_t*>,"");
static_assert(!std::is_constructible_v<Value,constwchar_t*>,"");
static_assert(std::is_constructible_v<Value,char*>,"");
static_assert(std::is_constructible_v<Value,constchar*>,"");
static_assert(std::is_constructible_v<Value,char16_t*>,"");
static_assert(std::is_constructible_v<Value,constchar16_t*>,"");
}
TEST(ValuesTest,ConstructInt){
Value value(-37);
EXPECT_EQ(Value::Type::INTEGER, value.type());
EXPECT_THAT(value.GetIfInt(), testing::Optional(-37));
EXPECT_EQ(-37, value.GetInt());
EXPECT_THAT(value.GetIfDouble(), testing::Optional(-37.0));
EXPECT_EQ(-37.0, value.GetDouble());
}
TEST(ValuesTest,ConstructDouble){
Value value(-4.655);
EXPECT_EQ(Value::Type::DOUBLE, value.type());
EXPECT_THAT(value.GetIfDouble(), testing::Optional(-4.655));
EXPECT_EQ(-4.655, value.GetDouble());
}
TEST(ValuesTest,ConstructStringFromConstCharPtr){
constchar* str="foobar";
Value value(str);
EXPECT_EQ(Value::Type::STRING, value.type());
EXPECT_THAT(value.GetIfString(), testing::Pointee(std::string("foobar")));
EXPECT_EQ("foobar", value.GetString());
}
TEST(ValuesTest,ConstructStringFromStringPiece){
std::string str="foobar";
Value value{std::string_view(str)};
EXPECT_EQ(Value::Type::STRING, value.type());
EXPECT_THAT(value.GetIfString(), testing::Pointee(std::string("foobar")));
EXPECT_EQ("foobar", value.GetString());
}
TEST(ValuesTest,ConstructStringFromStdStringRRef){
std::string str="foobar";
Value value(std::move(str));
EXPECT_EQ(Value::Type::STRING, value.type());
EXPECT_THAT(value.GetIfString(), testing::Pointee(std::string("foobar")));
EXPECT_EQ("foobar", value.GetString());
}
TEST(ValuesTest,ConstructStringFromConstChar16Ptr){
std::u16string str= u"foobar";
Value value(str.c_str());
EXPECT_EQ(Value::Type::STRING, value.type());
EXPECT_THAT(value.GetIfString(), testing::Pointee(std::string("foobar")));
EXPECT_EQ("foobar", value.GetString());
}
TEST(ValuesTest,ConstructStringFromStringPiece16){
std::u16string str= u"foobar";
Value value{std::u16string_view(str)};
EXPECT_EQ(Value::Type::STRING, value.type());
EXPECT_THAT(value.GetIfString(), testing::Pointee(std::string("foobar")));
EXPECT_EQ("foobar", value.GetString());
}
TEST(ValuesTest,ConstructBinary){
Value::BlobStorage blob={0xF,0x0,0x0,0xB,0xA,0x2};
Value value(blob);
EXPECT_EQ(Value::Type::BINARY, value.type());
EXPECT_THAT(value.GetIfBlob(), testing::Pointee(blob));
EXPECT_EQ(blob, value.GetBlob());
}
TEST(ValuesTest,ConstructDict){
Value::Dict value;
EXPECT_EQ(Value::Type::DICT,Value(std::move(value)).type());
}
TEST(ValuesTest,ConstructDictFromValueDict){
Value::Dict dict;
dict.Set("foo","bar");
{
Value value(dict.Clone());
EXPECT_EQ(Value::Type::DICT, value.type());
EXPECT_TRUE(value.GetIfDict());
EXPECT_TRUE(value.GetDict().FindString("foo"));
EXPECT_EQ("bar",*value.GetDict().FindString("foo"));
}
dict.Set("foo","baz");
{
Value value(std::move(dict));
EXPECT_EQ(Value::Type::DICT, value.type());
EXPECT_TRUE(value.GetIfDict());
EXPECT_TRUE(value.GetDict().FindString("foo"));
EXPECT_EQ("baz",*value.GetDict().FindString("foo"));
}
}
TEST(ValuesTest,ConstructList){
Value value(Value::List{});
EXPECT_EQ(Value::Type::LIST, value.type());
}
TEST(ValuesTest,UseTestingEachOnValueList){
Value::Listlist;
list.Append(true);
list.Append(true);
// This will only work if `Value::List::value_type` is defined.
EXPECT_THAT(list, testing::Each(testing::ResultOf(
[](constValue& value){return value.GetBool();},
testing::Eq(true))));
}
TEST(ValuesTest,ConstructListFromValueList){
Value::Listlist;
list.Append("foo");
{
Value value(list.Clone());
EXPECT_EQ(Value::Type::LIST, value.type());
EXPECT_EQ(1u, value.GetList().size());
EXPECT_EQ(Value::Type::STRING, value.GetList()[0].type());
EXPECT_EQ("foo", value.GetList()[0].GetString());
}
list.back()= base::Value("bar");
{
Value value(std::move(list));
EXPECT_EQ(Value::Type::LIST, value.type());
EXPECT_EQ(1u, value.GetList().size());
EXPECT_EQ(Value::Type::STRING, value.GetList()[0].type());
EXPECT_EQ("bar", value.GetList()[0].GetString());
}
}
TEST(ValuesTest,HardenTests){
Value value;
ASSERT_EQ(value.type(),Value::Type::NONE);
EXPECT_DEATH_IF_SUPPORTED(value.GetBool(),"");
EXPECT_DEATH_IF_SUPPORTED(value.GetInt(),"");
EXPECT_DEATH_IF_SUPPORTED(value.GetDouble(),"");
EXPECT_DEATH_IF_SUPPORTED(value.GetString(),"");
EXPECT_DEATH_IF_SUPPORTED(value.GetBlob(),"");
}
// Group of tests for the copy constructors and copy-assigmnent. For equality
// checks comparisons of the interesting fields are done instead of relying on
// Equals being correct.
TEST(ValuesTest,CopyBool){
Value true_value(true);
Value copied_true_value(true_value.Clone());
EXPECT_EQ(true_value.type(), copied_true_value.type());
EXPECT_EQ(true_value.GetBool(), copied_true_value.GetBool());
Value false_value(false);
Value copied_false_value(false_value.Clone());
EXPECT_EQ(false_value.type(), copied_false_value.type());
EXPECT_EQ(false_value.GetBool(), copied_false_value.GetBool());
Value blank;
blank= true_value.Clone();
EXPECT_EQ(true_value.type(), blank.type());
EXPECT_EQ(true_value.GetBool(), blank.GetBool());
blank= false_value.Clone();
EXPECT_EQ(false_value.type(), blank.type());
EXPECT_EQ(false_value.GetBool(), blank.GetBool());
}
TEST(ValuesTest,CopyInt){
Value value(74);
Value copied_value(value.Clone());
EXPECT_EQ(value.type(), copied_value.type());
EXPECT_EQ(value.GetInt(), copied_value.GetInt());
Value blank;
blank= value.Clone();
EXPECT_EQ(value.type(), blank.type());
EXPECT_EQ(value.GetInt(), blank.GetInt());
}
TEST(ValuesTest,CopyDouble){
Value value(74.896);
Value copied_value(value.Clone());
EXPECT_EQ(value.type(), copied_value.type());
EXPECT_EQ(value.GetDouble(), copied_value.GetDouble());
Value blank;
blank= value.Clone();
EXPECT_EQ(value.type(), blank.type());
EXPECT_EQ(value.GetDouble(), blank.GetDouble());
}
TEST(ValuesTest,CopyString){
Value value("foobar");
Value copied_value(value.Clone());
EXPECT_EQ(value.type(), copied_value.type());
EXPECT_EQ(value.GetString(), copied_value.GetString());
Value blank;
blank= value.Clone();
EXPECT_EQ(value.type(), blank.type());
EXPECT_EQ(value.GetString(), blank.GetString());
}
TEST(ValuesTest,CopyBinary){
Value value(Value::BlobStorage({0xF,0x0,0x0,0xB,0xA,0x2}));
Value copied_value(value.Clone());
EXPECT_EQ(value.type(), copied_value.type());
EXPECT_EQ(value.GetBlob(), copied_value.GetBlob());
Value blank;
blank= value.Clone();
EXPECT_EQ(value.type(), blank.type());
EXPECT_EQ(value.GetBlob(), blank.GetBlob());
}
TEST(ValuesTest,CopyDictionary){
Value::Dict dict;
dict.Set("Int",123);
Value value(std::move(dict));
Value copied_value(value.Clone());
EXPECT_EQ(value, copied_value);
Value blank;
blank= value.Clone();
EXPECT_EQ(value, blank);
}
TEST(ValuesTest,CopyList){
Value::Listlist;
list.Append(123);
Value value(std::move(list));
Value copied_value(value.Clone());
EXPECT_EQ(value, copied_value);
Value blank;
blank= value.Clone();
EXPECT_EQ(value, blank);
}
// Group of tests for the move constructors and move-assigmnent.
TEST(ValuesTest,MoveBool){
Value true_value(true);
Value moved_true_value(std::move(true_value));
EXPECT_EQ(Value::Type::BOOLEAN, moved_true_value.type());
EXPECT_TRUE(moved_true_value.GetBool());
Value false_value(false);
Value moved_false_value(std::move(false_value));
EXPECT_EQ(Value::Type::BOOLEAN, moved_false_value.type());
EXPECT_FALSE(moved_false_value.GetBool());
Value blank;
blank=Value(true);
EXPECT_EQ(Value::Type::BOOLEAN, blank.type());
EXPECT_TRUE(blank.GetBool());
blank=Value(false);
EXPECT_EQ(Value::Type::BOOLEAN, blank.type());
EXPECT_FALSE(blank.GetBool());
}
TEST(ValuesTest,MoveInt){
Value value(74);
Value moved_value(std::move(value));
EXPECT_EQ(Value::Type::INTEGER, moved_value.type());
EXPECT_EQ(74, moved_value.GetInt());
Value blank;
blank=Value(47);
EXPECT_EQ(Value::Type::INTEGER, blank.type());
EXPECT_EQ(47, blank.GetInt());
}
TEST(ValuesTest,MoveDouble){
Value value(74.896);
Value moved_value(std::move(value));
EXPECT_EQ(Value::Type::DOUBLE, moved_value.type());
EXPECT_EQ(74.896, moved_value.GetDouble());
Value blank;
blank=Value(654.38);
EXPECT_EQ(Value::Type::DOUBLE, blank.type());
EXPECT_EQ(654.38, blank.GetDouble());
}
TEST(ValuesTest,MoveString){
Value value("foobar");
Value moved_value(std::move(value));
EXPECT_EQ(Value::Type::STRING, moved_value.type());
EXPECT_EQ("foobar", moved_value.GetString());
Value blank;
blank=Value("foobar");
EXPECT_EQ(Value::Type::STRING, blank.type());
EXPECT_EQ("foobar", blank.GetString());
}
TEST(ValuesTest,MoveBinary){
constValue::BlobStorage buffer={0xF,0x0,0x0,0xB,0xA,0x2};
Value value(buffer);
Value moved_value(std::move(value));
EXPECT_EQ(Value::Type::BINARY, moved_value.type());
EXPECT_EQ(buffer, moved_value.GetBlob());
Value blank;
blank=Value(buffer);
EXPECT_EQ(Value::Type::BINARY, blank.type());
EXPECT_EQ(buffer, blank.GetBlob());
}
TEST(ValuesTest,MoveConstructDictionary){
Value::Dict dict;
dict.Set("Int",123);
Value value(std::move(dict));
Value moved_value(std::move(value));
EXPECT_EQ(Value::Type::DICT, moved_value.type());
EXPECT_EQ(123, moved_value.GetDict().Find("Int")->GetInt());
}
TEST(ValuesTest,MoveAssignDictionary){
Value::Dict dict;
dict.Set("Int",123);
Value blank;
blank=Value(std::move(dict));
EXPECT_EQ(Value::Type::DICT, blank.type());
EXPECT_EQ(123, blank.GetDict().Find("Int")->GetInt());
}
TEST(ValuesTest,ConstructDictWithIterators){
std::vector<std::pair<std::string,Value>> values;
values.emplace_back("Int",123);
Value blank;
blank=Value(Value::Dict(std::make_move_iterator(values.begin()),
std::make_move_iterator(values.end())));
EXPECT_EQ(Value::Type::DICT, blank.type());
EXPECT_EQ(123, blank.GetDict().Find("Int")->GetInt());
}
TEST(ValuesTest,MoveList){
Value::Listlist;
list.Append(123);
Value value(list.Clone());
Value moved_value(std::move(value));
EXPECT_EQ(Value::Type::LIST, moved_value.type());
EXPECT_EQ(123, moved_value.GetList().back().GetInt());
Value blank;
blank=Value(std::move(list));
EXPECT_EQ(Value::Type::LIST, blank.type());
EXPECT_EQ(123, blank.GetList().back().GetInt());
}
TEST(ValuesTest,Append){
Value::Listlist;
list.Append(true);
EXPECT_TRUE(list.back().is_bool());
list.Append(123);
EXPECT_TRUE(list.back().is_int());
list.Append(3.14);
EXPECT_TRUE(list.back().is_double());
std::string str="foo";
list.Append(str.c_str());
EXPECT_TRUE(list.back().is_string());
list.Append(std::string_view(str));
EXPECT_TRUE(list.back().is_string());
list.Append(std::move(str));
EXPECT_TRUE(list.back().is_string());
std::u16string str16= u"bar";
list.Append(str16.c_str());
EXPECT_TRUE(list.back().is_string());
list.Append(std::u16string_view(str16));
EXPECT_TRUE(list.back().is_string());
list.Append(Value());
EXPECT_TRUE(list.back().is_none());
list.Append(Value::Dict());
EXPECT_TRUE(list.back().is_dict());
list.Append(Value::List());
EXPECT_TRUE(list.back().is_list());
}
TEST(ValuesTest,ListInsert){
Value::Listlist;
constValue::List& const_list=list;
auto iter=list.Insert(list.end(),Value(true));
EXPECT_TRUE(list.begin()== iter);
EXPECT_EQ(*iter,true);
iter=list.Insert(const_list.begin(),Value(123));
EXPECT_TRUE(const_list.begin()== iter);
EXPECT_EQ(*iter,123);
iter=list.Insert(list.begin()+1,Value("Hello world!"));
EXPECT_TRUE(list.begin()+1== iter);
EXPECT_EQ(*iter,"Hello world!");
}
TEST(ValuesTest,ListResize){
autolist= base::Value::List().Append("Hello world!");
EXPECT_EQ(list.size(),1U);
list.resize(2);
// Adds an empty entry to the back to match the size.
EXPECT_EQ(list.size(),2U);
EXPECT_TRUE(list[0].is_string());
EXPECT_TRUE(list[1].is_none());
list.resize(1);
// Shrinks the list and kicks the new entry out.
EXPECT_EQ(list.size(),1U);
EXPECT_TRUE(list[0].is_string());
list.resize(0);
// Removes the remaining entry too.
EXPECT_EQ(list.size(),0U);
}
TEST(ValuesTest,ReverseIter){
Value::Listlist;
constValue::List& const_list=list;
list.Append(Value(true));
list.Append(Value(123));
list.Append(Value("Hello world!"));
auto iter=list.rbegin();
EXPECT_TRUE(const_list.rbegin()== iter);
EXPECT_EQ(*iter,"Hello world!");
++iter;
EXPECT_EQ(*iter,123);
++iter;
EXPECT_EQ(*iter,true);
++iter;
EXPECT_TRUE(list.rend()== iter);
EXPECT_TRUE(const_list.rend()== iter);
}
// Test all three behaviors of EnsureDict() (Create a new dict where no
// matchining values exist, return an existing dict, create a dict overwriting
// a value of another type).
TEST(ValuesTest,DictEnsureDict){
Value::Dict root;
// This call should create a new nested dictionary.
Value::Dict* foo_dict= root.EnsureDict("foo");
EXPECT_TRUE(foo_dict->empty());
foo_dict->Set("a","b");
// This call should retrieve the dictionary created above, rather than
// creating a new one.
std::string* a_string= root.EnsureDict("foo")->FindString("a");
ASSERT_NE(nullptr, a_string);
EXPECT_EQ(*a_string,"b");
// Use EnsureDict() to overwrite an existing non-dictionary value.
root.Set("bar",3);
Value::Dict* bar_dict= root.EnsureDict("bar");
EXPECT_TRUE(bar_dict->empty());
bar_dict->Set("b","c");
// Test that the above call created a "bar" entry.
bar_dict= root.FindDict("bar");
ASSERT_NE(nullptr, bar_dict);
std::string* b_string= bar_dict->FindString("b");
ASSERT_NE(nullptr, b_string);
EXPECT_EQ(*b_string,"c");
}
// Test all three behaviors of EnsureList() (Create a new list where no
// matchining value exists, return an existing list, create a list overwriting
// a value of another type).
TEST(ValuesTest,DictEnsureList){
Value::Dict root;
// This call should create a new list.
Value::List* foo_list= root.EnsureList("foo");
EXPECT_TRUE(foo_list->empty());
foo_list->Append("a");
// This call should retrieve the list created above, rather than creating a
// new one.
foo_list= root.EnsureList("foo");
ASSERT_EQ(1u, foo_list->size());
EXPECT_EQ((*foo_list)[0],Value("a"));
// Use EnsureList() to overwrite an existing non-list value.
root.Set("bar",3);
Value::List* bar_list= root.EnsureList("bar");
EXPECT_TRUE(bar_list->empty());
bar_list->Append("b");
// Test that the above call created a "bar" entry.
bar_list= root.FindList("bar");
ASSERT_NE(nullptr, bar_list);
ASSERT_EQ(1u, bar_list->size());
EXPECT_EQ((*bar_list)[0],Value("b"));
}
// TODO(dcheng): Add more tests directly exercising the updated dictionary and
// list APIs. For now, most of the updated APIs are tested indirectly via the
// legacy APIs that are largely backed by the updated APIs.
TEST(ValuesTest,DictFindByDottedPath){
Value::Dict dict;
EXPECT_EQ(nullptr, dict.FindByDottedPath("a.b.c"));
Value::Dict& a_dict= dict.Set("a",Value::Dict())->GetDict();
EXPECT_EQ(nullptr, dict.FindByDottedPath("a.b.c"));
Value::Dict& b_dict= a_dict.Set("b",Value::Dict())->GetDict();
EXPECT_EQ(nullptr, dict.FindByDottedPath("a.b.c"));
b_dict.Set("c",true);
constValue* value= dict.FindByDottedPath("a.b.c");
ASSERT_NE(nullptr, value);
EXPECT_TRUE(value->GetBool());
}
TEST(ValuesTest,DictSetByDottedPath){
Value::Dict dict;
Value* c= dict.SetByDottedPath("a.b.c",Value());
ASSERT_TRUE(c);
Value::Dict* a= dict.FindDict("a");
ASSERT_TRUE(a);
EXPECT_EQ(1U, a->size());
Value::Dict* b= a->FindDict("b");
ASSERT_TRUE(b);
EXPECT_EQ(1U, b->size());
EXPECT_EQ(c, b->Find("c"));
}
TEST(ValuesTest,RvalueDictSetByDottedPath){
Value::Dict dict=
Value::Dict()
.SetByDottedPath("nested.dictionary.null",Value())
.SetByDottedPath("nested.dictionary.bool",false)
.SetByDottedPath("nested.dictionary.int",42)
.SetByDottedPath("nested.dictionary.double",1.2)
.SetByDottedPath("nested.dictionary.string","value")
.SetByDottedPath("nested.dictionary.u16-string", u"u16-value")
.SetByDottedPath("nested.dictionary.std-string",
std::string("std-value"))
.SetByDottedPath("nested.dictionary.blob",Value::BlobStorage({1,2}))
.SetByDottedPath("nested.dictionary.list",
Value::List().Append("value in list"))
.SetByDottedPath("nested.dictionary.dict",
Value::Dict().Set("key","value"));
Value::Dict expected=
Value::Dict()//
.Set("nested",
base::Value::Dict()//
.Set("dictionary",
base::Value::Dict()
.Set("null",Value())
.Set("bool",false)
.Set("int",42)
.Set("double",1.2)
.Set("string","value")
.Set("u16-string", u"u16-value")
.Set("std-string", std::string("std-value"))
.Set("blob",Value::BlobStorage({1,2}))
.Set("list",Value::List().Append("value in list"))
.Set("dict",Value::Dict().Set("key","value"))));
EXPECT_EQ(dict, expected);
}
TEST(ValuesTest,DictSetWithDottedKey){
Value::Dict dict;
Value* abc= dict.Set("a.b.c",Value());
ASSERT_TRUE(abc);
EXPECT_FALSE(dict.FindByDottedPath("a"));
EXPECT_FALSE(dict.FindByDottedPath("a.b"));
EXPECT_FALSE(dict.FindByDottedPath("a.b.c"));
EXPECT_EQ(abc, dict.Find("a.b.c"));
}
TEST(ValuesTest,ListFront){
Value::Listlist;
constValue::List& const_list=list;
list.Append(1);
list.Append(2);
list.Append(3);
EXPECT_EQ(Value(1),list.front());
EXPECT_EQ(Value(1), const_list.front());
}
TEST(ValuesTest,ListFrontWhenEmpty){
Value::Listlist;
constValue::List& const_list=list;
EXPECT_CHECK_DEATH(list.front());
EXPECT_CHECK_DEATH(const_list.front());
}
TEST(ValuesTest,ListBack){
Value::Listlist;
constValue::List& const_list=list;
list.Append(1);
list.Append(2);
list.Append(3);
EXPECT_EQ(Value(3),list.back());
EXPECT_EQ(Value(3), const_list.back());
}
TEST(ValuesTest,ListBackWhenEmpty){
Value::Listlist;
constValue::List& const_list=list;
EXPECT_CHECK_DEATH(list.back());
EXPECT_CHECK_DEATH(const_list.back());
}
TEST(ValuesTest,ListContains){
Value::Listlist;
list.Append(false);
list.Append(1);
list.Append(2.3);
list.Append("banana");
Value::BlobStorage blob={0xF,0x0,0x0,0xB,0xA,0x2};
list.Append(Value(blob));
Value::Dict dict;
dict.Set("foo","bar");
list.Append(dict.Clone());
Value::List list2;
list2.Append(99);
list.Append(list2.Clone());
EXPECT_TRUE(list.contains(false));
EXPECT_TRUE(list.contains(1));
EXPECT_TRUE(list.contains(2.3));
EXPECT_TRUE(list.contains("banana"));
EXPECT_TRUE(list.contains(std::string_view("banana")));
EXPECT_TRUE(list.contains(std::string("banana")));
EXPECT_TRUE(list.contains(blob));
EXPECT_TRUE(list.contains(dict));
EXPECT_TRUE(list.contains(list2));
EXPECT_FALSE(list.contains(true));
EXPECT_FALSE(list.contains(0));
EXPECT_FALSE(list.contains(4.5));
EXPECT_FALSE(list.contains("orange"));
EXPECT_FALSE(list.contains(Value::BlobStorage({1,2,3})));
EXPECT_FALSE(list.contains(Value::Dict()));
EXPECT_FALSE(list.contains(list));
}
TEST(ValuesTest,ListErase){
Value::Listlist;
list.Append(1);
list.Append(2);
list.Append(3);
auto next_it=list.erase(list.begin()+1);
ASSERT_EQ(2u,list.size());
EXPECT_EQ(list[0],Value(1));
EXPECT_EQ(list[1],Value(3));
EXPECT_EQ(*next_it,Value(3));
EXPECT_EQ(next_it+1,list.end());
}
TEST(ValuesTest,ListEraseRange){
Value::Listlist;
list.Append(1);
list.Append(2);
list.Append(3);
list.Append(4);
auto next_it=list.erase(list.begin()+1,list.begin()+3);
ASSERT_EQ(2u,list.size());
EXPECT_EQ(list[0],Value(1));
EXPECT_EQ(list[1],Value(4));
EXPECT_EQ(*next_it,Value(4));
EXPECT_EQ(next_it+1,list.end());
next_it=list.erase(list.begin()+1,list.begin()+1);
ASSERT_EQ(2u,list.size());
EXPECT_EQ(list[0],Value(1));
EXPECT_EQ(list[1],Value(4));
EXPECT_EQ(*next_it,Value(4));
EXPECT_EQ(next_it+1,list.end());
next_it=list.erase(list.begin()+1,list.end());
ASSERT_EQ(1u,list.size());
EXPECT_EQ(list[0],Value(1));
EXPECT_EQ(next_it,list.end());
list.clear();
next_it=list.erase(list.begin(),list.begin());
ASSERT_EQ(0u,list.size());
EXPECT_EQ(next_it,list.begin());
EXPECT_EQ(next_it,list.end());
}
TEST(ValuesTest,ListEraseValue){
Value::Listlist;
list.Append(1);
list.Append(2);
list.Append(2);
list.Append(3);
EXPECT_EQ(2u,list.EraseValue(Value(2)));
EXPECT_EQ(2u,list.size());
EXPECT_EQ(1,list[0]);
EXPECT_EQ(3,list[1]);
EXPECT_EQ(1u,list.EraseValue(Value(1)));
EXPECT_EQ(1u,list.size());
EXPECT_EQ(3,list[0]);
EXPECT_EQ(1u,list.EraseValue(Value(3)));
EXPECT_TRUE(list.empty());
EXPECT_EQ(0u,list.EraseValue(Value(3)));
}
TEST(ValuesTest,ListEraseIf){
Value::Listlist;
list.Append(1);
list.Append(2);
list.Append(2);
list.Append(3);
EXPECT_EQ(3u,list.EraseIf([](constauto& val){return val>=Value(2);}));
EXPECT_EQ(1u,list.size());
EXPECT_EQ(1,list[0]);
EXPECT_EQ(1u,list.EraseIf([](constauto& val){returntrue;}));
EXPECT_TRUE(list.empty());
EXPECT_EQ(0u,list.EraseIf([](constauto& val){returntrue;}));
}
TEST(ValuesTest,ClearList){
Value::Listlist;
list.Append(1);
list.Append(2);
list.Append(3);
EXPECT_EQ(3u,list.size());
EXPECT_FALSE(list.empty());
list.clear();
EXPECT_EQ(0u,list.size());
EXPECT_TRUE(list.empty());
// list.clear() should be idempotent.
list.clear();
EXPECT_EQ(0u,list.size());
EXPECT_TRUE(list.empty());
}
TEST(ValuesTest,FindKey){
Value::Dict dict;
dict.Set("foo","bar");
Value value(std::move(dict));
EXPECT_NE(nullptr, value.GetDict().Find("foo"));
EXPECT_EQ(nullptr, value.GetDict().Find("baz"));
}
TEST(ValuesTest,FindKeyChangeValue){
Value::Dict dict;
dict.Set("foo","bar");
Value* found= dict.Find("foo");
ASSERT_NE(nullptr, found);
EXPECT_EQ("bar", found->GetString());
*found=Value(123);
EXPECT_EQ(123, dict.Find("foo")->GetInt());
}
TEST(ValuesTest,FindKeyConst){
Value::Dict dict;
dict.Set("foo","bar");
constValue value(std::move(dict));
EXPECT_NE(nullptr, value.GetDict().Find("foo"));
EXPECT_EQ(nullptr, value.GetDict().Find("baz"));
}
TEST(ValuesTest,FindBoolKey){
Value::Dict dict;
dict.Set("null",Value());
dict.Set("bool",false);
dict.Set("int",0);
dict.Set("double",0.0);
dict.Set("string", std::string());
dict.Set("blob",Value(Value::BlobStorage()));
dict.Set("list",Value::List());
dict.Set("dict",Value::Dict());
EXPECT_EQ(std::nullopt, dict.FindBool("null"));
EXPECT_NE(std::nullopt, dict.FindBool("bool"));
EXPECT_EQ(std::nullopt, dict.FindBool("int"));
EXPECT_EQ(std::nullopt, dict.FindBool("double"));
EXPECT_EQ(std::nullopt, dict.FindBool("string"));
EXPECT_EQ(std::nullopt, dict.FindBool("blob"));
EXPECT_EQ(std::nullopt, dict.FindBool("list"));
EXPECT_EQ(std::nullopt, dict.FindBool("dict"));
}
TEST(ValuesTest,FindIntKey){
Value::Dict dict;
dict.Set("null",Value());
dict.Set("bool",false);
dict.Set("int",0);
dict.Set("double",0.0);
dict.Set("string", std::string());
dict.Set("blob",Value(Value::BlobStorage()));
dict.Set("list",Value::List());
dict.Set("dict",Value::Dict());
EXPECT_EQ(std::nullopt, dict.FindInt("null"));
EXPECT_EQ(std::nullopt, dict.FindInt("bool"));
EXPECT_NE(std::nullopt, dict.FindInt("int"));
EXPECT_EQ(std::nullopt, dict.FindInt("double"));
EXPECT_EQ(std::nullopt, dict.FindInt("string"));
EXPECT_EQ(std::nullopt, dict.FindInt("blob"));
EXPECT_EQ(std::nullopt, dict.FindInt("list"));
EXPECT_EQ(std::nullopt, dict.FindInt("dict"));
}
TEST(ValuesTest,FindStringKey){
Value::Dict dict;
dict.Set("null",Value());
dict.Set("bool",false);
dict.Set("int",0);
dict.Set("double",0.0);
dict.Set("string", std::string());
dict.Set("blob",Value(Value::BlobStorage()));
dict.Set("list",Value::List());
dict.Set("dict",Value::Dict());
EXPECT_EQ(nullptr, dict.FindString("null"));
EXPECT_EQ(nullptr, dict.FindString("bool"));
EXPECT_EQ(nullptr, dict.FindString("int"));
EXPECT_EQ(nullptr, dict.FindString("double"));
EXPECT_NE(nullptr, dict.FindString("string"));
EXPECT_EQ(nullptr, dict.FindString("blob"));
EXPECT_EQ(nullptr, dict.FindString("list"));
EXPECT_EQ(nullptr, dict.FindString("dict"));
}
TEST(ValuesTest,MutableFindStringKey){
Value::Dict dict;
dict.Set("string","foo");
*(dict.FindString("string"))="bar";
Value::Dict expected_dict;
expected_dict.Set("string","bar");
EXPECT_EQ(expected_dict, dict);
Value value(std::move(dict));
Value expected_value(std::move(expected_dict));
EXPECT_EQ(expected_value, value);
}
TEST(ValuesTest,MutableFindBlobKey){
Value::BlobStorage original_blob={0xF,0x0,0x0,0xB,0xA,0x2};
Value::Dict dict;
dict.Set("blob", std::move(original_blob));
Value::BlobStorage new_blob={0x0,0x3,0x0};
*(dict.FindBlob("blob"))= new_blob;
Value::Dict expected_dict;
expected_dict.Set("blob", std::move(new_blob));
EXPECT_EQ(expected_dict, dict);
Value value(std::move(dict));
Value expected_value(std::move(expected_dict));
EXPECT_EQ(expected_value, value);
}
TEST(ValuesTest,FindDictKey){
Value::Dict dict;
dict.Set("null",Value());
dict.Set("bool",false);
dict.Set("int",0);
dict.Set("double",0.0);
dict.Set("string", std::string());
dict.Set("blob",Value(Value::BlobStorage()));
dict.Set("list",Value::List());
dict.Set("dict",Value::Dict());
EXPECT_EQ(nullptr, dict.FindDict("null"));
EXPECT_EQ(nullptr, dict.FindDict("bool"));
EXPECT_EQ(nullptr, dict.FindDict("int"));
EXPECT_EQ(nullptr, dict.FindDict("double"));
EXPECT_EQ(nullptr, dict.FindDict("string"));
EXPECT_EQ(nullptr, dict.FindDict("blob"));
EXPECT_EQ(nullptr, dict.FindDict("list"));
EXPECT_NE(nullptr, dict.FindDict("dict"));
}
TEST(ValuesTest,FindListKey){
Value::Dict dict;
dict.Set("null",Value());
dict.Set("bool",false);
dict.Set("int",0);
dict.Set("double",0.0);
dict.Set("string", std::string());
dict.Set("blob",Value(Value::BlobStorage()));
dict.Set("list",Value::List());
dict.Set("dict",Value::Dict());
EXPECT_EQ(nullptr, dict.FindList("null"));
EXPECT_EQ(nullptr, dict.FindList("bool"));
EXPECT_EQ(nullptr, dict.FindList("int"));
EXPECT_EQ(nullptr, dict.FindList("double"));
EXPECT_EQ(nullptr, dict.FindList("string"));
EXPECT_EQ(nullptr, dict.FindList("blob"));
EXPECT_NE(nullptr, dict.FindList("list"));
EXPECT_EQ(nullptr, dict.FindList("dict"));
}
TEST(ValuesTest,FindBlob){
Value::Dict dict;
dict.Set("null",Value());
dict.Set("bool",false);
dict.Set("int",0);
dict.Set("double",0.0);
dict.Set("string", std::string());
dict.Set("blob",Value(Value::BlobStorage()));
dict.Set("list",Value::List());
dict.Set("dict",Value::Dict());
EXPECT_EQ(nullptr, dict.FindBlob("null"));
EXPECT_EQ(nullptr, dict.FindBlob("bool"));
EXPECT_EQ(nullptr, dict.FindBlob("int"));
EXPECT_EQ(nullptr, dict.FindBlob("double"));
EXPECT_EQ(nullptr, dict.FindBlob("string"));
EXPECT_NE(nullptr, dict.FindBlob("blob"));
EXPECT_EQ(nullptr, dict.FindBlob("list"));
EXPECT_EQ(nullptr, dict.FindBlob("dict"));
}
TEST(ValuesTest,SetKey){
Value::Dict dict;
dict.Set("null",Value());
dict.Set("bool",false);
dict.Set("int",0);
dict.Set("double",0.0);
dict.Set("string", std::string());
dict.Set("blob",Value(Value::BlobStorage()));
dict.Set("list",Value::List());
dict.Set("dict",Value::Dict());
Value::Dict dict2;
dict2.Set(std::string_view("null"),Value(Value::Type::NONE));
dict2.Set(std::string_view("bool"),Value(Value::Type::BOOLEAN));
dict2.Set(std::string("int"),Value(Value::Type::INTEGER));
dict2.Set(std::string("double"),Value(Value::Type::DOUBLE));
dict2.Set(std::string("string"),Value(Value::Type::STRING));
dict2.Set("blob",Value(Value::Type::BINARY));
dict2.Set("list",Value(Value::Type::LIST));
dict2.Set("dict",Value(Value::Type::DICT));
EXPECT_EQ(dict, dict2);
EXPECT_EQ(Value(std::move(dict)),Value(std::move(dict2)));
}
TEST(ValuesTest,SetBoolKey){
std::optional<bool> value;
Value::Dict dict;
dict.Set("true_key",true);
dict.Set("false_key",false);
value= dict.FindBool("true_key");
ASSERT_TRUE(value);
ASSERT_TRUE(*value);
value= dict.FindBool("false_key");
ASSERT_TRUE(value);
ASSERT_FALSE(*value);
value= dict.FindBool("missing_key");
ASSERT_FALSE(value);
}
TEST(ValuesTest,SetIntKey){
std::optional<int> value;
Value::Dict dict;
dict.Set("one_key",1);
dict.Set("minus_one_key",-1);
value= dict.FindInt("one_key");
ASSERT_TRUE(value);
ASSERT_EQ(1,*value);
value= dict.FindInt("minus_one_key");
ASSERT_TRUE(value);
ASSERT_EQ(-1,*value);
value= dict.FindInt("missing_key");
ASSERT_FALSE(value);
}
TEST(ValuesTest,SetDoubleKey){
Value::Dict dict;
dict.Set("one_key",1.0);
dict.Set("minus_one_key",-1.0);
dict.Set("pi_key",3.1415);
constValue* value;
value= dict.Find("one_key");
ASSERT_TRUE(value);
EXPECT_TRUE(value->is_double());
EXPECT_EQ(1.0, value->GetDouble());
value= dict.Find("minus_one_key");
ASSERT_TRUE(value);
EXPECT_TRUE(value->is_double());
EXPECT_EQ(-1.0, value->GetDouble());
value= dict.Find("pi_key");
ASSERT_TRUE(value);
EXPECT_TRUE(value->is_double());
EXPECT_EQ(3.1415, value->GetDouble());
}
TEST(ValuesTest,SetStringKey){
Value::Dict dict;
dict.Set("one_key","one");
dict.Set("hello_key","hello world");
std::string movable_value("movable_value");
dict.Set("movable_key", std::move(movable_value));
ASSERT_TRUE(movable_value.empty());// NOLINT(bugprone-use-after-move)
const std::string* value;
value= dict.FindString("one_key");
ASSERT_TRUE(value);
ASSERT_EQ("one",*value);
value= dict.FindString("hello_key");
ASSERT_TRUE(value);
ASSERT_EQ("hello world",*value);
value= dict.FindString("movable_key");
ASSERT_TRUE(value);
ASSERT_EQ("movable_value",*value);
value= dict.FindString("missing_key");
ASSERT_FALSE(value);
}
TEST(ValuesTest,RvalueSet){
Value::Dict dict=Value::Dict()
.Set("null",Value())
.Set("bool",false)
.Set("int",42)
.Set("double",1.2)
.Set("string","value")
.Set("u16-string", u"u16-value")
.Set("std-string", std::string("std-value"))
.Set("blob",Value::BlobStorage({1,2}))
.Set("list",Value::List().Append("value in list"))
.Set("dict",Value::Dict().Set("key","value"));
Value::Dict expected;
expected.Set("null",Value());
expected.Set("bool",false);
expected.Set("int",42);
expected.Set("double",1.2);
expected.Set("string","value");
expected.Set("u16-string", u"u16-value");
expected.Set("std-string", std::string("std-value"));
expected.Set("blob",Value::BlobStorage({1,2}));
Value::List nested_list;
nested_list.Append("value in list");
expected.Set("list", std::move(nested_list));
Value::Dict nested_dict;
nested_dict.Set("key","value");
expected.Set("dict", std::move(nested_dict));
EXPECT_EQ(dict, expected);
}
TEST(ValuesTest,FindPath){
// Construct a dictionary path {root}.foo.bar = 123
Value::Dict foo;
foo.Set("bar",Value(123));
Value::Dict root;
root.Set("foo", std::move(foo));
// Double key, second not found.
Value* found= root.FindByDottedPath("foo.notfound");
EXPECT_FALSE(found);
// Double key, found.
found= root.FindByDottedPath("foo.bar");
EXPECT_TRUE(found);
EXPECT_TRUE(found->is_int());
EXPECT_EQ(123, found->GetInt());
}
TEST(ValuesTest,SetByDottedPath){
Value::Dict root;
Value* inserted= root.SetByDottedPath("one.two",Value(123));
Value* found= root.FindByDottedPath("one.two");
ASSERT_TRUE(found);
EXPECT_EQ(found->type(),Value::Type::INTEGER);
EXPECT_EQ(inserted, found);
EXPECT_EQ(123, found->GetInt());
inserted= root.SetByDottedPath("foo.bar",Value(123));
found= root.FindByDottedPath("foo.bar");
ASSERT_TRUE(found);
EXPECT_EQ(found->type(),Value::Type::INTEGER);
EXPECT_EQ(inserted, found);
EXPECT_EQ(123, found->GetInt());
// Overwrite with a different value.
root.SetByDottedPath("foo.bar",Value("hello"));
found= root.FindByDottedPath("foo.bar");
ASSERT_TRUE(found);
EXPECT_EQ(found->type(),Value::Type::STRING);
EXPECT_EQ("hello", found->GetString());
// Can't change existing non-dictionary keys to dictionaries.
found= root.SetByDottedPath("foo.bar.baz",Value(123));
EXPECT_FALSE(found);
}
TEST(ValuesTest,SetBoolPath){
Value::Dict root;
Value* inserted= root.SetByDottedPath("foo.bar",true);
Value* found= root.FindByDottedPath("foo.bar");
ASSERT_TRUE(found);
EXPECT_EQ(inserted, found);
ASSERT_TRUE(found->is_bool());
EXPECT_TRUE(found->GetBool());
// Overwrite with a different value.
root.SetByDottedPath("foo.bar",false);
found= root.FindByDottedPath("foo.bar");
ASSERT_TRUE(found);
ASSERT_TRUE(found->is_bool());
EXPECT_FALSE(found->GetBool());
// Can't change existing non-dictionary keys.
ASSERT_FALSE(root.SetByDottedPath("foo.bar.zoo",true));
}
TEST(ValuesTest,SetIntPath){
Value::Dict root;
Value* inserted= root.SetByDottedPath("foo.bar",123);
Value* found= root.FindByDottedPath("foo.bar");
ASSERT_TRUE(found);
EXPECT_EQ(inserted, found);
ASSERT_TRUE(found->is_int());
EXPECT_EQ(123, found->GetInt());
// Overwrite with a different value.
root.SetByDottedPath("foo.bar",234);
found= root.FindByDottedPath("foo.bar");
ASSERT_TRUE(found);
ASSERT_TRUE(found->is_int());
EXPECT_EQ(234, found->GetInt());
// Can't change existing non-dictionary keys.
ASSERT_FALSE(root.SetByDottedPath("foo.bar.zoo",567));
}
TEST(ValuesTest,SetDoublePath){
Value::Dict root;
Value* inserted= root.SetByDottedPath("foo.bar",1.23);
Value* found= root.FindByDottedPath("foo.bar");
ASSERT_TRUE(found);
EXPECT_EQ(inserted, found);
ASSERT_TRUE(found->is_double());
EXPECT_EQ(1.23, found->GetDouble());
// Overwrite with a different value.
root.SetByDottedPath("foo.bar",2.34);
found= root.FindByDottedPath("foo.bar");
ASSERT_TRUE(found);
ASSERT_TRUE(found->is_double());
EXPECT_EQ(2.34, found->GetDouble());
// Can't change existing non-dictionary keys.
ASSERT_FALSE(root.SetByDottedPath("foo.bar.zoo",5.67));
}
TEST(ValuesTest,SetStringPath){
Value::Dict root;
Value* inserted= root.SetByDottedPath("foo.bar","hello world");
Value* found= root.FindByDottedPath("foo.bar");
ASSERT_TRUE(found);
EXPECT_EQ(inserted, found);
ASSERT_TRUE(found->is_string());
EXPECT_EQ("hello world", found->GetString());
// Overwrite with a different value.
root.SetByDottedPath("foo.bar","bonjour monde");
found= root.FindByDottedPath("foo.bar");
ASSERT_TRUE(found);
ASSERT_TRUE(found->is_string());
EXPECT_EQ("bonjour monde", found->GetString());
ASSERT_TRUE(root.SetByDottedPath("foo.bar", std::string_view("rah rah")));
ASSERT_TRUE(root.SetByDottedPath("foo.bar", std::string("temp string")));
ASSERT_TRUE(root.SetByDottedPath("foo.bar", u"temp string"));
// Can't change existing non-dictionary keys.
ASSERT_FALSE(root.SetByDottedPath("foo.bar.zoo","ola mundo"));
}
TEST(ValuesTest,Remove){
Value::Dict root;
root.Set("one",Value(123));
// Removal of missing key should fail.
EXPECT_FALSE(root.Remove("two"));
// Removal of existing key should succeed.
EXPECT_TRUE(root.Remove("one"));
// Second removal of previously existing key should fail.
EXPECT_FALSE(root.Remove("one"));
}
TEST(ValuesTest,Extract){
Value::Dict root;
root.Set("one",Value(123));
// Extraction of missing key should fail.
EXPECT_EQ(std::nullopt, root.Extract("two"));
// Extraction of existing key should succeed.
EXPECT_EQ(Value(123), root.Extract("one"));
// Second extraction of previously existing key should fail.
EXPECT_EQ(std::nullopt, root.Extract("one"));
}
TEST(ValuesTest,RemoveByDottedPath){
Value::Dict root;
root.SetByDottedPath("one.two.three",Value(123));
// Removal of missing key should fail.
EXPECT_FALSE(root.RemoveByDottedPath("one.two.four"));
// Removal of existing key should succeed.
EXPECT_TRUE(root.RemoveByDottedPath("one.two.three"));
// Second removal of previously existing key should fail.
EXPECT_FALSE(root.RemoveByDottedPath("one.two.three"));
// Intermediate empty dictionaries should be cleared.
EXPECT_EQ(nullptr, root.Find("one"));
root.SetByDottedPath("one.two.three",Value(123));
root.SetByDottedPath("one.two.four",Value(124));
EXPECT_TRUE(root.RemoveByDottedPath("one.two.three"));
// Intermediate non-empty dictionaries should be kept.
EXPECT_NE(nullptr, root.Find("one"));
EXPECT_NE(nullptr, root.FindByDottedPath("one.two"));
EXPECT_NE(nullptr, root.FindByDottedPath("one.two.four"));
}
TEST(ValuesTest,ExtractByDottedPath){
Value::Dict root;
root.SetByDottedPath("one.two.three",Value(123));
// Extraction of missing key should fail.
EXPECT_EQ(std::nullopt, root.ExtractByDottedPath("one.two.four"));
// Extraction of existing key should succeed.
EXPECT_EQ(Value(123), root.ExtractByDottedPath("one.two.three"));
// Second extraction of previously existing key should fail.
EXPECT_EQ(std::nullopt, root.ExtractByDottedPath("one.two.three"));
// Intermediate empty dictionaries should be cleared.
EXPECT_EQ(nullptr, root.Find("one"));
root.SetByDottedPath("one.two.three",Value(123));
root.SetByDottedPath("one.two.four",Value(124));
EXPECT_EQ(Value(123), root.ExtractByDottedPath("one.two.three"));
// Intermediate non-empty dictionaries should be kept.
EXPECT_NE(nullptr, root.Find("one"));
EXPECT_NE(nullptr, root.FindByDottedPath("one.two"));
EXPECT_NE(nullptr, root.FindByDottedPath("one.two.four"));
}
TEST(ValuesTest,Basic){
// Test basic dictionary getting/setting
Value::Dict settings;
ASSERT_FALSE(settings.FindByDottedPath("global.homepage"));
ASSERT_FALSE(settings.Find("global"));
settings.Set("global",Value(true));
ASSERT_TRUE(settings.Find("global"));
settings.Remove("global");
settings.SetByDottedPath("global.homepage",Value("http://scurvy.com"));
ASSERT_TRUE(settings.Find("global"));
const std::string* homepage=
settings.FindStringByDottedPath("global.homepage");
ASSERT_TRUE(homepage);
ASSERT_EQ(std::string("http://scurvy.com"),*homepage);
// Test storing a dictionary in a list.
ASSERT_FALSE(settings.FindByDottedPath("global.toolbar.bookmarks"));
Value::List new_toolbar_bookmarks;
settings.SetByDottedPath("global.toolbar.bookmarks",
std::move(new_toolbar_bookmarks));
Value::List* toolbar_bookmarks=
settings.FindListByDottedPath("global.toolbar.bookmarks");
ASSERT_TRUE(toolbar_bookmarks);
Value::Dict new_bookmark;
new_bookmark.Set("name",Value("Froogle"));
new_bookmark.Set("url",Value("http://froogle.com"));
toolbar_bookmarks->Append(std::move(new_bookmark));
Value* bookmark_list= settings.FindByDottedPath("global.toolbar.bookmarks");
ASSERT_TRUE(bookmark_list);
ASSERT_EQ(1U, bookmark_list->GetList().size());
Value* bookmark=&bookmark_list->GetList()[0];
ASSERT_TRUE(bookmark);
ASSERT_TRUE(bookmark->is_dict());
const std::string* bookmark_name= bookmark->GetDict().FindString("name");
ASSERT_TRUE(bookmark_name);
ASSERT_EQ(std::string("Froogle"),*bookmark_name);
const std::string* bookmark_url= bookmark->GetDict().FindString("url");
ASSERT_TRUE(bookmark_url);
ASSERT_EQ(std::string("http://froogle.com"),*bookmark_url);
}
TEST(ValuesTest,List){
Value::List mixed_list;
mixed_list.Append(true);
mixed_list.Append(42);
mixed_list.Append(88.8);
mixed_list.Append("foo");
ASSERT_EQ(4u, mixed_list.size());
EXPECT_EQ(true, mixed_list[0]);
EXPECT_EQ(42, mixed_list[1]);
EXPECT_EQ(88.8, mixed_list[2]);
EXPECT_EQ("foo", mixed_list[3]);
// Try searching in the mixed list.
ASSERT_TRUE(Contains(mixed_list,42,&Value::GetIfInt));
ASSERT_FALSE(Contains(mixed_list,false,&Value::GetIfBool));
}
TEST(ValuesTest,RvalueAppend){
Value::Listlist=Value::List()
.Append(Value())
.Append(false)
.Append(42)
.Append(1.2)
.Append("value")
.Append(u"u16-value")
.Append(std::string("std-value"))
.Append(Value::BlobStorage({1,2}))
.Append(Value::List().Append("value in list"))
.Append(Value::Dict().Set("key","value"));
Value::List expected;
expected.Append(Value());
expected.Append(false);
expected.Append(42);
expected.Append(1.2);
expected.Append("value");
expected.Append(u"u16-value");
expected.Append(std::string("std-value"));
expected.Append(Value::BlobStorage({1,2}));
Value::List nested_list;
nested_list.Append("value in list");
expected.Append(std::move(nested_list));
Value::Dict nested_dict;
nested_dict.Set("key","value");
expected.Append(std::move(nested_dict));
EXPECT_EQ(list, expected);
}
TEST(ValuesTest,ListWithCapacity){
Value::List list_with_capacity=
Value::List::with_capacity(3).Append(true).Append(42).Append(88.8);
ASSERT_EQ(3u, list_with_capacity.size());
}
TEST(ValuesTest,BinaryValue){
// Default constructor creates a BinaryValue with a buffer of size 0.
Value binary(Value::Type::BINARY);
ASSERT_TRUE(binary.GetBlob().empty());
// Test the common case of a non-empty buffer
Value::BlobStorage buffer(15);
uint8_t* original_buffer= buffer.data();
binary=Value(std::move(buffer));
ASSERT_TRUE(binary.GetBlob().data());
ASSERT_EQ(original_buffer, binary.GetBlob().data());
ASSERT_EQ(15U, binary.GetBlob().size());
char stack_buffer[42];
memset(stack_buffer,'!',42);
binary=Value(Value::BlobStorage(stack_buffer, stack_buffer+42));
ASSERT_TRUE(binary.GetBlob().data());
ASSERT_NE(stack_buffer,
reinterpret_cast<constchar*>(binary.GetBlob().data()));
ASSERT_EQ(42U, binary.GetBlob().size());
ASSERT_EQ(0, memcmp(stack_buffer, binary.GetBlob().data(),
binary.GetBlob().size()));
}
TEST(ValuesTest,StringValue){
// Test overloaded StringValue constructor.
std::unique_ptr<Value> narrow_value(newValue("narrow"));
ASSERT_TRUE(narrow_value.get());
ASSERT_TRUE(narrow_value->is_string());
std::unique_ptr<Value> utf16_value(newValue(u"utf16"));
ASSERT_TRUE(utf16_value.get());
ASSERT_TRUE(utf16_value->is_string());
ASSERT_TRUE(narrow_value->is_string());
ASSERT_EQ(std::string("narrow"), narrow_value->GetString());
ASSERT_TRUE(utf16_value->is_string());
ASSERT_EQ(std::string("utf16"), utf16_value->GetString());
}
TEST(ValuesTest,DictionaryDeletion){
std::string key="test";
Value::Dict dict;
dict.Set(key,Value());
EXPECT_FALSE(dict.empty());
EXPECT_EQ(1U, dict.size());
dict.clear();
EXPECT_TRUE(dict.empty());
EXPECT_TRUE(dict.empty());
EXPECT_EQ(0U, dict.size());
}
TEST(ValuesTest,DictionarySetReturnsPointer){
{
Value::Dict dict;
Value* blank_ptr= dict.Set("foo.bar",Value());
EXPECT_EQ(Value::Type::NONE, blank_ptr->type());
}
{
Value::Dict dict;
Value* blank_ptr= dict.Set("foo.bar",Value());
EXPECT_EQ(Value::Type::NONE, blank_ptr->type());
}
{
Value::Dict dict;
Value* int_ptr= dict.Set("foo.bar",42);
EXPECT_EQ(Value::Type::INTEGER, int_ptr->type());
EXPECT_EQ(42, int_ptr->GetInt());
}
{
Value::Dict dict;
Value* string_ptr= dict.Set("foo.bar","foo");
EXPECT_EQ(Value::Type::STRING, string_ptr->type());
EXPECT_EQ("foo", string_ptr->GetString());
}
{
Value::Dict dict;
Value* string16_ptr= dict.Set("foo.bar", u"baz");
EXPECT_EQ(Value::Type::STRING, string16_ptr->type());
EXPECT_EQ("baz", string16_ptr->GetString());
}
{
Value::Dict dict;
Value* dict_ptr= dict.Set("foo.bar",Value::Dict());
EXPECT_EQ(Value::Type::DICT, dict_ptr->type());
}
{
Value::Dict dict;
Value* list_ptr= dict.Set("foo.bar",Value::List());
EXPECT_EQ(Value::Type::LIST, list_ptr->type());
}
}
TEST(ValuesTest,Clone){
Value original_null;
Value original_bool(true);
Value original_int(42);
Value original_double(3.14);
Value original_string("hello");
Value original_string16(u"hello16");
Value original_binary(Value::BlobStorage(42,'!'));
Value::Listlist;
list.Append(0);
list.Append(1);
Value original_list(std::move(list));
Value original_dict(Value::Dict()
.Set("null", original_null.Clone())
.Set("bool", original_bool.Clone())
.Set("int", original_int.Clone())
.Set("double", original_double.Clone())
.Set("string", original_string.Clone())
.Set("string16", original_string16.Clone())
.Set("binary", original_binary.Clone())
.Set("list", original_list.Clone()));
Value copy_value= original_dict.Clone();
constValue::Dict& copy_dict= copy_value.GetDict();
EXPECT_EQ(original_dict, copy_dict);
EXPECT_EQ(original_null,*copy_dict.Find("null"));
EXPECT_EQ(original_bool,*copy_dict.Find("bool"));
EXPECT_EQ(original_int,*copy_dict.Find("int"));
EXPECT_EQ(original_double,*copy_dict.Find("double"));
EXPECT_EQ(original_string,*copy_dict.Find("string"));
EXPECT_EQ(original_string16,*copy_dict.Find("string16"));
EXPECT_EQ(original_binary,*copy_dict.Find("binary"));
EXPECT_EQ(original_list,*copy_dict.Find("list"));
}
TEST(ValuesTest,TakeString){
Value value("foo");
std::string taken= std::move(value).TakeString();
EXPECT_EQ(taken,"foo");
}
// Check that the value can still be used after `TakeString()` was called, as
// long as a new value was assigned to it.
TEST(ValuesTest,PopulateAfterTakeString){
Value value("foo");
std::string taken= std::move(value).TakeString();
value=Value(false);
EXPECT_EQ(value,Value(false));
}
TEST(ValuesTest,TakeBlob){
Value::BlobStorage original_blob={0xF,0x0,0x0,0xB,0xA,0x2};
Value value(original_blob);
Value::BlobStorage taken= std::move(value).TakeBlob();
EXPECT_EQ(taken, original_blob);
}
TEST(ValuesTest,PopulateAfterTakeBlob){
Value::BlobStorage original_blob={0xF,0x0,0x0,0xB,0xA,0x2};
Value value(original_blob);
Value::BlobStorage taken= std::move(value).TakeBlob();
value=Value(false);
EXPECT_EQ(value,Value(false));
}
TEST(ValuesTest,TakeDict){
Value::Dict dict;
dict.Set("foo",123);
Value value(std::move(dict));
Value clone= value.Clone();
Value::Dict taken= std::move(value).TakeDict();
EXPECT_EQ(taken, clone);
}
// Check that the value can still be used after `TakeDict()` was called, as long
// as a new value was assigned to it.
TEST(ValuesTest,PopulateAfterTakeDict){
Value::Dict dict;
dict.Set("foo",123);
Value value(std::move(dict));
Value::Dict taken= std::move(value).TakeDict();
value=Value(false);
EXPECT_EQ(value,Value(false));
}
TEST(ValuesTest,TakeList){
Value::Listlist;
list.Append(true);
list.Append(123);
Value value(std::move(list));
Value clone= value.Clone();
Value::List taken= std::move(value).TakeList();
EXPECT_EQ(taken, clone);
}
// Check that the value can still be used after `TakeList()` was called, as long
// as a new value was assigned to it.
TEST(ValuesTest,PopulateAfterTakeList){
Value::Listlist;
list.Append("hello");
Value value(std::move(list));
Value::List taken= std::move(value).TakeList();
value=Value(false);
EXPECT_EQ(value,Value(false));
}
TEST(ValuesTest,SpecializedEquals){
std::vector<Value> values;
values.emplace_back(false);
values.emplace_back(true);
values.emplace_back(0);
values.emplace_back(1);
values.emplace_back(1.0);
values.emplace_back(2.0);
values.emplace_back("hello");
values.emplace_back("world");
base::Value::Dict dict;
dict.Set("hello","world");
values.emplace_back(std::move(dict));
base::Value::Dict dict2;
dict2.Set("world","hello");
values.emplace_back(std::move(dict2));
base::Value::Listlist;
list.Append("hello");
list.Append("world");
values.emplace_back(std::move(list));
base::Value::List list2;
list2.Append("world");
list2.Append("hello");
values.emplace_back(std::move(list2));
for(constValue& outer_value: values){
for(constValue& inner_value: values){
SCOPED_TRACE(::testing::Message()
<<"Outer: "<< outer_value<<"Inner: "<< inner_value);
constbool should_be_equal=&outer_value==&inner_value;
if(should_be_equal){
EXPECT_EQ(outer_value, inner_value);
EXPECT_EQ(inner_value, outer_value);
EXPECT_FALSE(outer_value!= inner_value);
EXPECT_FALSE(inner_value!= outer_value);
}else{
EXPECT_NE(outer_value, inner_value);
EXPECT_NE(inner_value, outer_value);
EXPECT_FALSE(outer_value== inner_value);
EXPECT_FALSE(inner_value== outer_value);
}
// Also test the various overloads for operator== against concrete
// subtypes.
outer_value.Visit([&](constauto& outer_member){
using T= std::decay_t<decltype(outer_member)>;
ifconstexpr(!std::is_same_v<T, std::monostate>&&
!std::is_same_v<T,Value::BlobStorage>){
if(should_be_equal){
EXPECT_EQ(outer_member, inner_value);
EXPECT_EQ(inner_value, outer_member);
EXPECT_FALSE(outer_member!= inner_value);
EXPECT_FALSE(inner_value!= outer_member);
}else{
EXPECT_NE(outer_member, inner_value);
EXPECT_NE(inner_value, outer_member);
EXPECT_FALSE(outer_member== inner_value);
EXPECT_FALSE(inner_value== outer_member);
}
}
});
}
// A copy of a Value should also compare equal to itself.
Value copied_value= outer_value.Clone();
EXPECT_EQ(outer_value, copied_value);
EXPECT_EQ(copied_value, outer_value);
EXPECT_FALSE(outer_value!= copied_value);
EXPECT_FALSE(copied_value!= outer_value);
}
}
// Test that a literal string comparison does not end up using the bool (!!)
// overload.
TEST(ValuesTest,LiteralStringEquals){
EXPECT_EQ("hello world", base::Value("hello world"));
EXPECT_EQ(base::Value("hello world"),"hello world");
EXPECT_NE("hello world", base::Value(true));
EXPECT_NE(base::Value(true),"hello world");
}
TEST(ValuesTest,Equals){
auto null1= std::make_unique<Value>();
auto null2= std::make_unique<Value>();
EXPECT_NE(null1.get(), null2.get());
EXPECT_EQ(*null1,*null2);
Value boolean(false);
EXPECT_NE(*null1, boolean);
Value::Dict dv;
dv.Set("a",false);
dv.Set("b",2);
dv.Set("c",2.5);
dv.Set("d1","string");
dv.Set("d2", u"http://google.com");
dv.Set("e",Value());
Value::Dict copy= dv.Clone();
EXPECT_EQ(dv, copy);
Value::Listlist;
list.Append(Value());
list.Append(Value(Value::Type::DICT));
Value::List list_copy(list.Clone());
Value* list_weak= dv.Set("f", std::move(list));
EXPECT_NE(dv, copy);
copy.Set("f", std::move(list_copy));
EXPECT_EQ(dv, copy);
list_weak->GetList().Append(true);
EXPECT_NE(dv, copy);
// Check if Equals detects differences in only the keys.
copy= dv.Clone();
EXPECT_EQ(dv, copy);
copy.Remove("a");
copy.Set("aa",false);
EXPECT_NE(dv, copy);
}
TEST(ValuesTest,Comparisons){
// Test None Values.
Value null1;
Value null2;
EXPECT_EQ(null1, null2);
EXPECT_FALSE(null1!= null2);
EXPECT_FALSE(null1< null2);
EXPECT_FALSE(null1> null2);
EXPECT_LE(null1, null2);
EXPECT_GE(null1, null2);
// Test Bool Values.
Value bool1(false);
Value bool2(true);
EXPECT_FALSE(bool1== bool2);
EXPECT_NE(bool1, bool2);
EXPECT_LT(bool1, bool2);
EXPECT_FALSE(bool1> bool2);
EXPECT_LE(bool1, bool2);
EXPECT_FALSE(bool1>= bool2);
// Test Int Values.
Valueint1(1);
Valueint2(2);
EXPECT_FALSE(int1==int2);
EXPECT_NE(int1,int2);
EXPECT_LT(int1,int2);
EXPECT_FALSE(int1>int2);
EXPECT_LE(int1,int2);
EXPECT_FALSE(int1>=int2);
// Test Double Values.
Value double1(1.0);
Value double2(2.0);
EXPECT_FALSE(double1== double2);
EXPECT_NE(double1, double2);
EXPECT_LT(double1, double2);
EXPECT_FALSE(double1> double2);
EXPECT_LE(double1, double2);
EXPECT_FALSE(double1>= double2);
// Test String Values.
Value string1("1");
Value string2("2");
EXPECT_FALSE(string1== string2);
EXPECT_NE(string1, string2);
EXPECT_LT(string1, string2);
EXPECT_FALSE(string1> string2);
EXPECT_LE(string1, string2);
EXPECT_FALSE(string1>= string2);
// Test Binary Values.
Value binary1(Value::BlobStorage{0x01});
Value binary2(Value::BlobStorage{0x02});
EXPECT_FALSE(binary1== binary2);
EXPECT_NE(binary1, binary2);
EXPECT_LT(binary1, binary2);
EXPECT_FALSE(binary1> binary2);
EXPECT_LE(binary1, binary2);
EXPECT_FALSE(binary1>= binary2);
// Test Empty List Values.
Value::List null_list1;
Value::List null_list2;
EXPECT_EQ(null_list1, null_list2);
EXPECT_FALSE(null_list1!= null_list2);
EXPECT_FALSE(null_list1< null_list2);
EXPECT_FALSE(null_list1> null_list2);
EXPECT_LE(null_list1, null_list2);
EXPECT_GE(null_list1, null_list2);
// Test Non Empty List Values.
Value::List int_list1;
Value::List int_list2;
int_list1.Append(1);
int_list2.Append(2);
EXPECT_FALSE(int_list1== int_list2);
EXPECT_NE(int_list1, int_list2);
EXPECT_LT(int_list1, int_list2);
EXPECT_FALSE(int_list1> int_list2);
EXPECT_LE(int_list1, int_list2);
EXPECT_FALSE(int_list1>= int_list2);
// Test Empty Dict Values.
Value::Dict null_dict1;
Value::Dict null_dict2;
EXPECT_EQ(null_dict1, null_dict2);
EXPECT_FALSE(null_dict1!= null_dict2);
EXPECT_FALSE(null_dict1< null_dict2);
EXPECT_FALSE(null_dict1> null_dict2);
EXPECT_LE(null_dict1, null_dict2);
EXPECT_GE(null_dict1, null_dict2);
// Test Non Empty Dict Values.
Value::Dict int_dict1;
Value::Dict int_dict2;
int_dict1.Set("key",1);
int_dict2.Set("key",2);
EXPECT_FALSE(int_dict1== int_dict2);
EXPECT_NE(int_dict1, int_dict2);
EXPECT_LT(int_dict1, int_dict2);
EXPECT_FALSE(int_dict1> int_dict2);
EXPECT_LE(int_dict1, int_dict2);
EXPECT_FALSE(int_dict1>= int_dict2);
// Test Values of different types.
std::vector<Value> values;
values.emplace_back(std::move(null1));
values.emplace_back(std::move(bool1));
values.emplace_back(std::move(int1));
values.emplace_back(std::move(double1));
values.emplace_back(std::move(string1));
values.emplace_back(std::move(binary1));
values.emplace_back(std::move(int_dict1));
values.emplace_back(std::move(int_list1));
for(size_t i=0; i< values.size();++i){
for(size_t j= i+1; j< values.size();++j){
EXPECT_FALSE(values[i]== values[j]);
EXPECT_NE(values[i], values[j]);
EXPECT_LT(values[i], values[j]);
EXPECT_FALSE(values[i]> values[j]);
EXPECT_LE(values[i], values[j]);
EXPECT_FALSE(values[i]>= values[j]);
}
}
}
TEST(ValuesTest,Merge){
Value::Dict base;
base.Set("base_key","base_key_value_base");
base.Set("collide_key","collide_key_value_base");
Value::Dict base_sub_dict;
base_sub_dict.Set("sub_base_key","sub_base_key_value_base");
base_sub_dict.Set("sub_collide_key","sub_collide_key_value_base");
base.Set("sub_dict_key", std::move(base_sub_dict));
Value::Dict merge;
merge.Set("merge_key","merge_key_value_merge");
merge.Set("collide_key","collide_key_value_merge");
Value::Dict merge_sub_dict;
merge_sub_dict.Set("sub_merge_key","sub_merge_key_value_merge");
merge_sub_dict.Set("sub_collide_key","sub_collide_key_value_merge");
merge.Set("sub_dict_key", std::move(merge_sub_dict));
base.Merge(std::move(merge));
EXPECT_EQ(4U, base.size());
const std::string* base_key_value= base.FindString("base_key");
ASSERT_TRUE(base_key_value);
EXPECT_EQ("base_key_value_base",*base_key_value);// Base value preserved.
const std::string* collide_key_value= base.FindString("collide_key");
ASSERT_TRUE(collide_key_value);
EXPECT_EQ("collide_key_value_merge",*collide_key_value);// Replaced.
const std::string* merge_key_value= base.FindString("merge_key");
ASSERT_TRUE(merge_key_value);
EXPECT_EQ("merge_key_value_merge",*merge_key_value);// Merged in.
Value::Dict* res_sub_dict= base.FindDict("sub_dict_key");
ASSERT_TRUE(res_sub_dict);
EXPECT_EQ(3U, res_sub_dict->size());
const std::string* sub_base_key_value=
res_sub_dict->FindString("sub_base_key");
ASSERT_TRUE(sub_base_key_value);
EXPECT_EQ("sub_base_key_value_base",*sub_base_key_value);// Preserved.
const std::string* sub_collide_key_value=
res_sub_dict->FindString("sub_collide_key");
ASSERT_TRUE(sub_collide_key_value);
EXPECT_EQ("sub_collide_key_value_merge",
*sub_collide_key_value);// Replaced.
const std::string* sub_merge_key_value=
res_sub_dict->FindString("sub_merge_key");
ASSERT_TRUE(sub_merge_key_value);
EXPECT_EQ("sub_merge_key_value_merge",*sub_merge_key_value);// Merged in.
}
TEST(ValuesTest,DictionaryIterator){
Value::Dict dict;
for(Value::Dict::iterator it= dict.begin(); it!= dict.end();++it){
ADD_FAILURE();
}
Value value1("value1");
dict.Set("key1", value1.Clone());
bool seen1=false;
for(Value::Dict::iterator it= dict.begin(); it!= dict.end();++it){
EXPECT_FALSE(seen1);
EXPECT_EQ("key1", it->first);
EXPECT_EQ(value1, it->second);
seen1=true;
}
EXPECT_TRUE(seen1);
Value value2("value2");
dict.Set("key2", value2.Clone());
bool seen2= seen1=false;
for(Value::Dict::iterator it= dict.begin(); it!= dict.end();++it){
if(it->first=="key1"){
EXPECT_FALSE(seen1);
EXPECT_EQ(value1, it->second);
seen1=true;
}elseif(it->first=="key2"){
EXPECT_FALSE(seen2);
EXPECT_EQ(value2, it->second);
seen2=true;
}else{
ADD_FAILURE();
}
}
EXPECT_TRUE(seen1);
EXPECT_TRUE(seen2);
}
TEST(ValuesTest,MutatingCopiedPairsInDictMutatesUnderlyingValues){
Value::Dict dict;
dict.Set("key",Value("initial value"));
// Because the non-const dict iterates over <const std::string&, Value&>
// pairs, it's possible to alter iterated-over values in place even when
// "copying" the key-value pair:
for(auto kv: dict){
kv.second.GetString()="replacement";
}
std::string* found= dict.FindString("key");
ASSERT_TRUE(found);
EXPECT_EQ(*found,"replacement");
}
TEST(ValuesTest,StdDictionaryIterator){
Value::Dict dict;
for(auto it= dict.begin(); it!= dict.end();++it){
ADD_FAILURE();
}
Value value1("value1");
dict.Set("key1", value1.Clone());
bool seen1=false;
for(auto it: dict){
EXPECT_FALSE(seen1);
EXPECT_EQ("key1", it.first);
EXPECT_EQ(value1, it.second);
seen1=true;
}
EXPECT_TRUE(seen1);
Value value2("value2");
dict.Set("key2", value2.Clone());
bool seen2= seen1=false;
for(auto it: dict){
if(it.first=="key1"){
EXPECT_FALSE(seen1);
EXPECT_EQ(value1, it.second);
seen1=true;
}elseif(it.first=="key2"){
EXPECT_FALSE(seen2);
EXPECT_EQ(value2, it.second);
seen2=true;
}else{
ADD_FAILURE();
}
}
EXPECT_TRUE(seen1);
EXPECT_TRUE(seen2);
}
TEST(ValuesTest,SelfSwap){
base::Value test(1);
std::swap(test, test);
EXPECT_EQ(1, test.GetInt());
}
TEST(ValuesTest,FromToUniquePtrValue){
std::unique_ptr<Value> dict= std::make_unique<Value>(Value::Type::DICT);
dict->GetDict().Set("name","Froogle");
dict->GetDict().Set("url","http://froogle.com");
Value dict_copy= dict->Clone();
Value dict_converted=Value::FromUniquePtrValue(std::move(dict));
EXPECT_EQ(dict_copy, dict_converted);
std::unique_ptr<Value> val=
Value::ToUniquePtrValue(std::move(dict_converted));
EXPECT_EQ(dict_copy,*val);
}
TEST(ValuesTest,MutableFindStringPath){
Value::Dict dict;
dict.SetByDottedPath("foo.bar","value");
*(dict.FindStringByDottedPath("foo.bar"))="new_value";
Value::Dict expected_dict;
expected_dict.SetByDottedPath("foo.bar","new_value");
EXPECT_EQ(expected_dict, dict);
}
TEST(ValuesTest,MutableGetString){
Value value("value");
value.GetString()="new_value";
EXPECT_EQ("new_value", value.GetString());
}
TEST(ValuesTest,MutableFindBlobPath){
Value::BlobStorage original_blob={0xF,0x0,0x0,0xB,0xA,0x2};
Value::Dict dict;
dict.SetByDottedPath("foo.bar", std::move(original_blob));
Value::BlobStorage new_blob={0x0,0x3,0x0};
*(dict.FindBlobByDottedPath("foo.bar"))= new_blob;
Value::Dict expected_dict;
expected_dict.SetByDottedPath("foo.bar", std::move(new_blob));
EXPECT_EQ(expected_dict, dict);
}
TEST(ValuesTest,MutableGetBlob){
Value::BlobStorage original_blob={0xF,0x0,0x0,0xB,0xA,0x2};
Value value(std::move(original_blob));
Value::BlobStorage new_blob={0x0,0x3,0x0};
value.GetBlob()= new_blob;
EXPECT_EQ(new_blob, value.GetBlob());
}
TEST(ValuesTest,TracingSupport){
EXPECT_EQ(perfetto::TracedValueToString(Value(false)),"false");
EXPECT_EQ(perfetto::TracedValueToString(Value(1)),"1");
EXPECT_EQ(perfetto::TracedValueToString(Value(1.5)),"1.5");
EXPECT_EQ(perfetto::TracedValueToString(Value("value")),"value");
EXPECT_EQ(perfetto::TracedValueToString(Value(Value::Type::NONE)),"<none>");
{
Value::Listlist;
EXPECT_EQ(perfetto::TracedValueToString(list),"{}");
list.Append(2);
list.Append(3);
EXPECT_EQ(perfetto::TracedValueToString(list),"[2,3]");
EXPECT_EQ(perfetto::TracedValueToString(Value(std::move(list))),"[2,3]");
}
{
Value::Dict dict;
EXPECT_EQ(perfetto::TracedValueToString(dict),"{}");
dict.Set("key","value");
EXPECT_EQ(perfetto::TracedValueToString(dict),"{key:value}");
EXPECT_EQ(perfetto::TracedValueToString(Value(std::move(dict))),
"{key:value}");
}
}
TEST(ValueViewTest,BasicConstruction){
{
ValueView v=true;
EXPECT_EQ(true, std::get<bool>(v.data_view_for_test()));
}
{
ValueView v=25;
EXPECT_EQ(25, std::get<int>(v.data_view_for_test()));
}
{
ValueView v=3.14;
EXPECT_DOUBLE_EQ(3.14, std::get<ValueView::DoubleStorageForTest>(
v.data_view_for_test()));
}
{
ValueView v= std::string_view("hello world");
EXPECT_EQ("hello world",
std::get<std::string_view>(v.data_view_for_test()));
}
{
ValueView v="hello world";
EXPECT_EQ("hello world",
std::get<std::string_view>(v.data_view_for_test()));
}
{
std::string str="hello world";
ValueView v= str;
EXPECT_EQ("hello world",
std::get<std::string_view>(v.data_view_for_test()));
}
{
Value::Dict dict;
dict.Set("hello","world");
ValueView v= dict;
EXPECT_EQ(dict, std::get<std::reference_wrapper<constValue::Dict>>(
v.data_view_for_test()));
}
{
Value::Listlist;
list.Append("hello");
list.Append("world");
ValueView v=list;
EXPECT_EQ(list, std::get<std::reference_wrapper<constValue::List>>(
v.data_view_for_test()));
}
}
TEST(ValueViewTest,ValueConstruction){
{
Value val(true);
ValueView v= val;
EXPECT_EQ(true, std::get<bool>(v.data_view_for_test()));
}
{
Value val(25);
ValueView v= val;
EXPECT_EQ(25, std::get<int>(v.data_view_for_test()));
}
{
Value val(3.14);
ValueView v= val;
EXPECT_DOUBLE_EQ(3.14, std::get<ValueView::DoubleStorageForTest>(
v.data_view_for_test()));
}
{
Value val("hello world");
ValueView v= val;
EXPECT_EQ("hello world",
std::get<std::string_view>(v.data_view_for_test()));
}
{
Value::Dict dict;
dict.Set("hello","world");
Value val(dict.Clone());
ValueView v= val;
EXPECT_EQ(dict, std::get<std::reference_wrapper<constValue::Dict>>(
v.data_view_for_test()));
}
{
Value::Listlist;
list.Append("hello");
list.Append("world");
Value val(list.Clone());
ValueView v= val;
EXPECT_EQ(list, std::get<std::reference_wrapper<constValue::List>>(
v.data_view_for_test()));
}
}
TEST(ValueViewTest,ToValue){
{
Value val(true);
Value to_val=ValueView(val).ToValue();
EXPECT_EQ(val, to_val);
}
{
Value val(25);
Value to_val=ValueView(val).ToValue();
EXPECT_EQ(val, to_val);
}
{
Value val(3.14);
Value to_val=ValueView(val).ToValue();
EXPECT_EQ(val, to_val);
}
{
Value val("hello world");
Value to_val=ValueView(val).ToValue();
EXPECT_EQ(val, to_val);
}
{
Value::Dict dict;
dict.Set("hello","world");
Value val(dict.Clone());
Value to_val=ValueView(val).ToValue();
EXPECT_EQ(val, to_val);
}
{
Value::Listlist;
list.Append("hello");
list.Append("world");
Value val(list.Clone());
Value to_val=ValueView(val).ToValue();
EXPECT_EQ(val, to_val);
}
}
}// namespace base

[8]ページ先頭

©2009-2025 Movatter.jp