blob: 806347c9387a317df0e895de2bed6fdef56b6cff [file] [log] [blame]
[email protected]3b63f8f42011-03-28 01:54:151// Copyright (c) 2011 The Chromium Authors. All rights reserved.
license.botbf09a502008-08-24 00:55:552// Use of this source code is governed by a BSD-style license that can be
3// found in the LICENSE file.
initial.commitd7cae122008-07-26 21:49:384
[email protected]836061b2008-08-13 14:57:515#include <limits>
6
[email protected]3b63f8f42011-03-28 01:54:157#include "base/memory/scoped_ptr.h"
[email protected]9101ef1e2010-01-15 20:09:038#include "base/string16.h"
[email protected]be1ce6a72010-08-03 14:35:229#include "base/utf_string_conversions.h"
[email protected]9101ef1e2010-01-15 20:09:0310#include "base/values.h"
initial.commitd7cae122008-07-26 21:49:3811#include "testing/gtest/include/gtest/gtest.h"
12
[email protected]58b916e2011-06-28 22:56:3313namespace base {
initial.commitd7cae122008-07-26 21:49:3814
[email protected]58b916e2011-06-28 22:56:3315TEST(ValuesTest, Basic) {
initial.commitd7cae122008-07-26 21:49:3816 // Test basic dictionary getting/setting
17 DictionaryValue settings;
[email protected]9e4cda7332010-07-31 04:56:1418 std::string homepage = "http://google.com";
19 ASSERT_FALSE(settings.GetString("global.homepage", &homepage));
20 ASSERT_EQ(std::string("http://google.com"), homepage);
21
22 ASSERT_FALSE(settings.Get("global", NULL));
23 settings.Set("global", Value::CreateBooleanValue(true));
24 ASSERT_TRUE(settings.Get("global", NULL));
25 settings.SetString("global.homepage", "http://scurvy.com");
26 ASSERT_TRUE(settings.Get("global", NULL));
27 homepage = "http://google.com";
28 ASSERT_TRUE(settings.GetString("global.homepage", &homepage));
29 ASSERT_EQ(std::string("http://scurvy.com"), homepage);
30
31 // Test storing a dictionary in a list.
32 ListValue* toolbar_bookmarks;
33 ASSERT_FALSE(
34 settings.GetList("global.toolbar.bookmarks", &toolbar_bookmarks));
35
36 toolbar_bookmarks = new ListValue;
37 settings.Set("global.toolbar.bookmarks", toolbar_bookmarks);
38 ASSERT_TRUE(settings.GetList("global.toolbar.bookmarks", &toolbar_bookmarks));
39
40 DictionaryValue* new_bookmark = new DictionaryValue;
41 new_bookmark->SetString("name", "Froogle");
42 new_bookmark->SetString("url", "http://froogle.com");
43 toolbar_bookmarks->Append(new_bookmark);
44
45 ListValue* bookmark_list;
46 ASSERT_TRUE(settings.GetList("global.toolbar.bookmarks", &bookmark_list));
47 DictionaryValue* bookmark;
48 ASSERT_EQ(1U, bookmark_list->GetSize());
49 ASSERT_TRUE(bookmark_list->GetDictionary(0, &bookmark));
50 std::string bookmark_name = "Unnamed";
51 ASSERT_TRUE(bookmark->GetString("name", &bookmark_name));
52 ASSERT_EQ(std::string("Froogle"), bookmark_name);
53 std::string bookmark_url;
54 ASSERT_TRUE(bookmark->GetString("url", &bookmark_url));
55 ASSERT_EQ(std::string("http://froogle.com"), bookmark_url);
56}
57
[email protected]58b916e2011-06-28 22:56:3358TEST(ValuesTest, List) {
[email protected]f82fb4952009-01-20 21:05:3259 scoped_ptr<ListValue> mixed_list(new ListValue());
60 mixed_list->Set(0, Value::CreateBooleanValue(true));
61 mixed_list->Set(1, Value::CreateIntegerValue(42));
[email protected]fb534c92011-02-01 01:02:0762 mixed_list->Set(2, Value::CreateDoubleValue(88.8));
[email protected]f82fb4952009-01-20 21:05:3263 mixed_list->Set(3, Value::CreateStringValue("foo"));
64 ASSERT_EQ(4u, mixed_list->GetSize());
[email protected]52a261f2009-03-03 15:01:1265
[email protected]f82fb4952009-01-20 21:05:3266 Value *value = NULL;
67 bool bool_value = false;
68 int int_value = 0;
69 double double_value = 0.0;
70 std::string string_value;
71
72 ASSERT_FALSE(mixed_list->Get(4, &value));
73
74 ASSERT_FALSE(mixed_list->GetInteger(0, &int_value));
75 ASSERT_EQ(0, int_value);
[email protected]d3accda2011-05-02 01:59:2176 ASSERT_FALSE(mixed_list->GetBoolean(1, &bool_value));
77 ASSERT_FALSE(bool_value);
[email protected]f82fb4952009-01-20 21:05:3278 ASSERT_FALSE(mixed_list->GetString(2, &string_value));
79 ASSERT_EQ("", string_value);
[email protected]d3accda2011-05-02 01:59:2180 ASSERT_FALSE(mixed_list->GetInteger(2, &int_value));
81 ASSERT_EQ(0, int_value);
[email protected]f82fb4952009-01-20 21:05:3282 ASSERT_FALSE(mixed_list->GetBoolean(3, &bool_value));
[email protected]74e3af72010-10-03 21:44:3983 ASSERT_FALSE(bool_value);
[email protected]f82fb4952009-01-20 21:05:3284
85 ASSERT_TRUE(mixed_list->GetBoolean(0, &bool_value));
[email protected]74e3af72010-10-03 21:44:3986 ASSERT_TRUE(bool_value);
[email protected]f82fb4952009-01-20 21:05:3287 ASSERT_TRUE(mixed_list->GetInteger(1, &int_value));
88 ASSERT_EQ(42, int_value);
[email protected]d3accda2011-05-02 01:59:2189 // implicit conversion from Integer to Double should be possible.
90 ASSERT_TRUE(mixed_list->GetDouble(1, &double_value));
91 ASSERT_EQ(42, double_value);
[email protected]fb534c92011-02-01 01:02:0792 ASSERT_TRUE(mixed_list->GetDouble(2, &double_value));
[email protected]f82fb4952009-01-20 21:05:3293 ASSERT_EQ(88.8, double_value);
94 ASSERT_TRUE(mixed_list->GetString(3, &string_value));
95 ASSERT_EQ("foo", string_value);
[email protected]5fb35372011-09-19 15:23:1096
97 // Try searching in the mixed list.
98 scoped_ptr<Value> sought_value(Value::CreateIntegerValue(42));
99 scoped_ptr<Value> not_found_value(Value::CreateBooleanValue(false));
100
101 ASSERT_NE(mixed_list->end(), mixed_list->Find(*sought_value));
102 ASSERT_TRUE((*mixed_list->Find(*sought_value))->GetAsInteger(&int_value));
103 ASSERT_EQ(42, int_value);
104 ASSERT_EQ(mixed_list->end(), mixed_list->Find(*not_found_value));
[email protected]f82fb4952009-01-20 21:05:32105}
106
[email protected]58b916e2011-06-28 22:56:33107TEST(ValuesTest, BinaryValue) {
initial.commitd7cae122008-07-26 21:49:38108 char* buffer = NULL;
109 // Passing a null buffer pointer doesn't yield a BinaryValue
[email protected]0a9a0fc2009-03-24 23:37:14110 scoped_ptr<BinaryValue> binary(BinaryValue::Create(buffer, 0));
111 ASSERT_FALSE(binary.get());
initial.commitd7cae122008-07-26 21:49:38112
113 // If you want to represent an empty binary value, use a zero-length buffer.
[email protected]cd679232008-08-13 02:39:51114 buffer = new char[1];
initial.commitd7cae122008-07-26 21:49:38115 ASSERT_TRUE(buffer);
[email protected]0a9a0fc2009-03-24 23:37:14116 binary.reset(BinaryValue::Create(buffer, 0));
117 ASSERT_TRUE(binary.get());
initial.commitd7cae122008-07-26 21:49:38118 ASSERT_TRUE(binary->GetBuffer());
119 ASSERT_EQ(buffer, binary->GetBuffer());
[email protected]f297d182008-08-21 16:24:51120 ASSERT_EQ(0U, binary->GetSize());
initial.commitd7cae122008-07-26 21:49:38121
122 // Test the common case of a non-empty buffer
123 buffer = new char[15];
[email protected]0a9a0fc2009-03-24 23:37:14124 binary.reset(BinaryValue::Create(buffer, 15));
125 ASSERT_TRUE(binary.get());
initial.commitd7cae122008-07-26 21:49:38126 ASSERT_TRUE(binary->GetBuffer());
127 ASSERT_EQ(buffer, binary->GetBuffer());
[email protected]f297d182008-08-21 16:24:51128 ASSERT_EQ(15U, binary->GetSize());
initial.commitd7cae122008-07-26 21:49:38129
130 char stack_buffer[42];
131 memset(stack_buffer, '!', 42);
[email protected]0a9a0fc2009-03-24 23:37:14132 binary.reset(BinaryValue::CreateWithCopiedBuffer(stack_buffer, 42));
133 ASSERT_TRUE(binary.get());
initial.commitd7cae122008-07-26 21:49:38134 ASSERT_TRUE(binary->GetBuffer());
135 ASSERT_NE(stack_buffer, binary->GetBuffer());
[email protected]f297d182008-08-21 16:24:51136 ASSERT_EQ(42U, binary->GetSize());
initial.commitd7cae122008-07-26 21:49:38137 ASSERT_EQ(0, memcmp(stack_buffer, binary->GetBuffer(), binary->GetSize()));
initial.commitd7cae122008-07-26 21:49:38138}
139
[email protected]58b916e2011-06-28 22:56:33140TEST(ValuesTest, StringValue) {
[email protected]4cd5f6a2008-12-11 01:23:17141 // Test overloaded CreateStringValue.
[email protected]0a9a0fc2009-03-24 23:37:14142 scoped_ptr<Value> narrow_value(Value::CreateStringValue("narrow"));
143 ASSERT_TRUE(narrow_value.get());
[email protected]4cd5f6a2008-12-11 01:23:17144 ASSERT_TRUE(narrow_value->IsType(Value::TYPE_STRING));
[email protected]e2e593d2010-08-03 15:42:58145 scoped_ptr<Value> utf16_value(
146 Value::CreateStringValue(ASCIIToUTF16("utf16")));
147 ASSERT_TRUE(utf16_value.get());
148 ASSERT_TRUE(utf16_value->IsType(Value::TYPE_STRING));
149
150 // Test overloaded GetString.
151 std::string narrow = "http://google.com";
152 string16 utf16 = ASCIIToUTF16("http://google.com");
153 ASSERT_TRUE(narrow_value->GetAsString(&narrow));
154 ASSERT_TRUE(narrow_value->GetAsString(&utf16));
155 ASSERT_EQ(std::string("narrow"), narrow);
156 ASSERT_EQ(ASCIIToUTF16("narrow"), utf16);
157
158 ASSERT_TRUE(utf16_value->GetAsString(&narrow));
159 ASSERT_TRUE(utf16_value->GetAsString(&utf16));
160 ASSERT_EQ(std::string("utf16"), narrow);
161 ASSERT_EQ(ASCIIToUTF16("utf16"), utf16);
162}
163
initial.commitd7cae122008-07-26 21:49:38164// This is a Value object that allows us to tell if it's been
165// properly deleted by modifying the value of external flag on destruction.
166class DeletionTestValue : public Value {
[email protected]8ab4809b2009-07-03 02:28:55167 public:
[email protected]2fdc86a2010-01-26 23:08:02168 explicit DeletionTestValue(bool* deletion_flag) : Value(TYPE_NULL) {
initial.commitd7cae122008-07-26 21:49:38169 Init(deletion_flag); // Separate function so that we can use ASSERT_*
170 }
171
172 void Init(bool* deletion_flag) {
173 ASSERT_TRUE(deletion_flag);
174 deletion_flag_ = deletion_flag;
175 *deletion_flag_ = false;
176 }
177
178 ~DeletionTestValue() {
179 *deletion_flag_ = true;
180 }
181
[email protected]8ab4809b2009-07-03 02:28:55182 private:
initial.commitd7cae122008-07-26 21:49:38183 bool* deletion_flag_;
184};
185
[email protected]58b916e2011-06-28 22:56:33186TEST(ValuesTest, ListDeletion) {
initial.commitd7cae122008-07-26 21:49:38187 bool deletion_flag = true;
188
189 {
190 ListValue list;
191 list.Append(new DeletionTestValue(&deletion_flag));
192 EXPECT_FALSE(deletion_flag);
193 }
194 EXPECT_TRUE(deletion_flag);
195
196 {
197 ListValue list;
198 list.Append(new DeletionTestValue(&deletion_flag));
199 EXPECT_FALSE(deletion_flag);
200 list.Clear();
201 EXPECT_TRUE(deletion_flag);
202 }
203
204 {
205 ListValue list;
206 list.Append(new DeletionTestValue(&deletion_flag));
207 EXPECT_FALSE(deletion_flag);
208 EXPECT_TRUE(list.Set(0, Value::CreateNullValue()));
209 EXPECT_TRUE(deletion_flag);
210 }
211}
212
[email protected]58b916e2011-06-28 22:56:33213TEST(ValuesTest, ListRemoval) {
initial.commitd7cae122008-07-26 21:49:38214 bool deletion_flag = true;
215 Value* removed_item = NULL;
216
217 {
218 ListValue list;
219 list.Append(new DeletionTestValue(&deletion_flag));
220 EXPECT_FALSE(deletion_flag);
[email protected]f297d182008-08-21 16:24:51221 EXPECT_EQ(1U, list.GetSize());
[email protected]836061b2008-08-13 14:57:51222 EXPECT_FALSE(list.Remove(std::numeric_limits<size_t>::max(),
223 &removed_item));
initial.commitd7cae122008-07-26 21:49:38224 EXPECT_FALSE(list.Remove(1, &removed_item));
225 EXPECT_TRUE(list.Remove(0, &removed_item));
226 ASSERT_TRUE(removed_item);
[email protected]f297d182008-08-21 16:24:51227 EXPECT_EQ(0U, list.GetSize());
initial.commitd7cae122008-07-26 21:49:38228 }
229 EXPECT_FALSE(deletion_flag);
230 delete removed_item;
231 removed_item = NULL;
232 EXPECT_TRUE(deletion_flag);
233
234 {
235 ListValue list;
236 list.Append(new DeletionTestValue(&deletion_flag));
237 EXPECT_FALSE(deletion_flag);
238 EXPECT_TRUE(list.Remove(0, NULL));
239 EXPECT_TRUE(deletion_flag);
[email protected]f297d182008-08-21 16:24:51240 EXPECT_EQ(0U, list.GetSize());
initial.commitd7cae122008-07-26 21:49:38241 }
[email protected]6832cbe2009-11-30 19:59:11242
243 {
244 ListValue list;
245 DeletionTestValue* value = new DeletionTestValue(&deletion_flag);
246 list.Append(value);
247 EXPECT_FALSE(deletion_flag);
[email protected]4fc3c5642011-08-13 17:34:31248 size_t index = 0;
249 list.Remove(*value, &index);
250 EXPECT_EQ(0U, index);
[email protected]6832cbe2009-11-30 19:59:11251 EXPECT_TRUE(deletion_flag);
252 EXPECT_EQ(0U, list.GetSize());
253 }
initial.commitd7cae122008-07-26 21:49:38254}
255
[email protected]58b916e2011-06-28 22:56:33256TEST(ValuesTest, DictionaryDeletion) {
[email protected]9e4cda7332010-07-31 04:56:14257 std::string key = "test";
258 bool deletion_flag = true;
259
260 {
261 DictionaryValue dict;
262 dict.Set(key, new DeletionTestValue(&deletion_flag));
263 EXPECT_FALSE(deletion_flag);
264 }
265 EXPECT_TRUE(deletion_flag);
266
267 {
268 DictionaryValue dict;
269 dict.Set(key, new DeletionTestValue(&deletion_flag));
270 EXPECT_FALSE(deletion_flag);
271 dict.Clear();
272 EXPECT_TRUE(deletion_flag);
273 }
274
275 {
276 DictionaryValue dict;
277 dict.Set(key, new DeletionTestValue(&deletion_flag));
278 EXPECT_FALSE(deletion_flag);
279 dict.Set(key, Value::CreateNullValue());
280 EXPECT_TRUE(deletion_flag);
281 }
282}
283
[email protected]58b916e2011-06-28 22:56:33284TEST(ValuesTest, DictionaryRemoval) {
[email protected]9e4cda7332010-07-31 04:56:14285 std::string key = "test";
286 bool deletion_flag = true;
287 Value* removed_item = NULL;
288
289 {
290 DictionaryValue dict;
291 dict.Set(key, new DeletionTestValue(&deletion_flag));
292 EXPECT_FALSE(deletion_flag);
293 EXPECT_TRUE(dict.HasKey(key));
294 EXPECT_FALSE(dict.Remove("absent key", &removed_item));
295 EXPECT_TRUE(dict.Remove(key, &removed_item));
296 EXPECT_FALSE(dict.HasKey(key));
297 ASSERT_TRUE(removed_item);
298 }
299 EXPECT_FALSE(deletion_flag);
300 delete removed_item;
301 removed_item = NULL;
302 EXPECT_TRUE(deletion_flag);
303
304 {
305 DictionaryValue dict;
306 dict.Set(key, new DeletionTestValue(&deletion_flag));
307 EXPECT_FALSE(deletion_flag);
308 EXPECT_TRUE(dict.HasKey(key));
309 EXPECT_TRUE(dict.Remove(key, NULL));
310 EXPECT_TRUE(deletion_flag);
311 EXPECT_FALSE(dict.HasKey(key));
312 }
313}
314
[email protected]58b916e2011-06-28 22:56:33315TEST(ValuesTest, DictionaryWithoutPathExpansion) {
[email protected]4dad9ad82009-11-25 20:47:52316 DictionaryValue dict;
[email protected]9e4cda7332010-07-31 04:56:14317 dict.Set("this.is.expanded", Value::CreateNullValue());
318 dict.SetWithoutPathExpansion("this.isnt.expanded", Value::CreateNullValue());
319
320 EXPECT_FALSE(dict.HasKey("this.is.expanded"));
321 EXPECT_TRUE(dict.HasKey("this"));
322 Value* value1;
323 EXPECT_TRUE(dict.Get("this", &value1));
324 DictionaryValue* value2;
325 ASSERT_TRUE(dict.GetDictionaryWithoutPathExpansion("this", &value2));
326 EXPECT_EQ(value1, value2);
327 EXPECT_EQ(1U, value2->size());
328
329 EXPECT_TRUE(dict.HasKey("this.isnt.expanded"));
330 Value* value3;
331 EXPECT_FALSE(dict.Get("this.isnt.expanded", &value3));
332 Value* value4;
333 ASSERT_TRUE(dict.GetWithoutPathExpansion("this.isnt.expanded", &value4));
334 EXPECT_EQ(Value::TYPE_NULL, value4->GetType());
335}
336
[email protected]58b916e2011-06-28 22:56:33337TEST(ValuesTest, DeepCopy) {
initial.commitd7cae122008-07-26 21:49:38338 DictionaryValue original_dict;
339 Value* original_null = Value::CreateNullValue();
[email protected]9e4cda7332010-07-31 04:56:14340 original_dict.Set("null", original_null);
[email protected]16f47e082011-01-18 02:16:59341 FundamentalValue* original_bool = Value::CreateBooleanValue(true);
[email protected]9e4cda7332010-07-31 04:56:14342 original_dict.Set("bool", original_bool);
[email protected]16f47e082011-01-18 02:16:59343 FundamentalValue* original_int = Value::CreateIntegerValue(42);
[email protected]9e4cda7332010-07-31 04:56:14344 original_dict.Set("int", original_int);
[email protected]fb534c92011-02-01 01:02:07345 FundamentalValue* original_double = Value::CreateDoubleValue(3.14);
346 original_dict.Set("double", original_double);
[email protected]16f47e082011-01-18 02:16:59347 StringValue* original_string = Value::CreateStringValue("hello");
[email protected]9e4cda7332010-07-31 04:56:14348 original_dict.Set("string", original_string);
[email protected]16f47e082011-01-18 02:16:59349 StringValue* original_string16 =
350 Value::CreateStringValue(ASCIIToUTF16("hello16"));
[email protected]e2e593d2010-08-03 15:42:58351 original_dict.Set("string16", original_string16);
[email protected]9e4cda7332010-07-31 04:56:14352
353 char* original_buffer = new char[42];
354 memset(original_buffer, '!', 42);
[email protected]51914092011-08-11 17:43:01355 BinaryValue* original_binary = BinaryValue::Create(original_buffer, 42);
[email protected]9e4cda7332010-07-31 04:56:14356 original_dict.Set("binary", original_binary);
357
358 ListValue* original_list = new ListValue();
[email protected]16f47e082011-01-18 02:16:59359 FundamentalValue* original_list_element_0 = Value::CreateIntegerValue(0);
[email protected]9e4cda7332010-07-31 04:56:14360 original_list->Append(original_list_element_0);
[email protected]16f47e082011-01-18 02:16:59361 FundamentalValue* original_list_element_1 = Value::CreateIntegerValue(1);
[email protected]9e4cda7332010-07-31 04:56:14362 original_list->Append(original_list_element_1);
363 original_dict.Set("list", original_list);
364
[email protected]16f47e082011-01-18 02:16:59365 scoped_ptr<DictionaryValue> copy_dict(original_dict.DeepCopy());
[email protected]9e4cda7332010-07-31 04:56:14366 ASSERT_TRUE(copy_dict.get());
367 ASSERT_NE(copy_dict.get(), &original_dict);
368
369 Value* copy_null = NULL;
370 ASSERT_TRUE(copy_dict->Get("null", &copy_null));
371 ASSERT_TRUE(copy_null);
372 ASSERT_NE(copy_null, original_null);
373 ASSERT_TRUE(copy_null->IsType(Value::TYPE_NULL));
374
375 Value* copy_bool = NULL;
376 ASSERT_TRUE(copy_dict->Get("bool", &copy_bool));
377 ASSERT_TRUE(copy_bool);
378 ASSERT_NE(copy_bool, original_bool);
379 ASSERT_TRUE(copy_bool->IsType(Value::TYPE_BOOLEAN));
380 bool copy_bool_value = false;
381 ASSERT_TRUE(copy_bool->GetAsBoolean(&copy_bool_value));
382 ASSERT_TRUE(copy_bool_value);
383
384 Value* copy_int = NULL;
385 ASSERT_TRUE(copy_dict->Get("int", &copy_int));
386 ASSERT_TRUE(copy_int);
387 ASSERT_NE(copy_int, original_int);
388 ASSERT_TRUE(copy_int->IsType(Value::TYPE_INTEGER));
389 int copy_int_value = 0;
390 ASSERT_TRUE(copy_int->GetAsInteger(&copy_int_value));
391 ASSERT_EQ(42, copy_int_value);
392
[email protected]fb534c92011-02-01 01:02:07393 Value* copy_double = NULL;
394 ASSERT_TRUE(copy_dict->Get("double", &copy_double));
395 ASSERT_TRUE(copy_double);
396 ASSERT_NE(copy_double, original_double);
397 ASSERT_TRUE(copy_double->IsType(Value::TYPE_DOUBLE));
398 double copy_double_value = 0;
399 ASSERT_TRUE(copy_double->GetAsDouble(&copy_double_value));
400 ASSERT_EQ(3.14, copy_double_value);
[email protected]9e4cda7332010-07-31 04:56:14401
402 Value* copy_string = NULL;
403 ASSERT_TRUE(copy_dict->Get("string", &copy_string));
404 ASSERT_TRUE(copy_string);
405 ASSERT_NE(copy_string, original_string);
406 ASSERT_TRUE(copy_string->IsType(Value::TYPE_STRING));
407 std::string copy_string_value;
[email protected]e2e593d2010-08-03 15:42:58408 string16 copy_string16_value;
[email protected]9e4cda7332010-07-31 04:56:14409 ASSERT_TRUE(copy_string->GetAsString(&copy_string_value));
[email protected]e2e593d2010-08-03 15:42:58410 ASSERT_TRUE(copy_string->GetAsString(&copy_string16_value));
[email protected]9e4cda7332010-07-31 04:56:14411 ASSERT_EQ(std::string("hello"), copy_string_value);
[email protected]e2e593d2010-08-03 15:42:58412 ASSERT_EQ(ASCIIToUTF16("hello"), copy_string16_value);
[email protected]9e4cda7332010-07-31 04:56:14413
[email protected]e2e593d2010-08-03 15:42:58414 Value* copy_string16 = NULL;
415 ASSERT_TRUE(copy_dict->Get("string16", &copy_string16));
416 ASSERT_TRUE(copy_string16);
417 ASSERT_NE(copy_string16, original_string16);
418 ASSERT_TRUE(copy_string16->IsType(Value::TYPE_STRING));
419 ASSERT_TRUE(copy_string16->GetAsString(&copy_string_value));
420 ASSERT_TRUE(copy_string16->GetAsString(&copy_string16_value));
[email protected]9e4cda7332010-07-31 04:56:14421 ASSERT_EQ(std::string("hello16"), copy_string_value);
[email protected]e2e593d2010-08-03 15:42:58422 ASSERT_EQ(ASCIIToUTF16("hello16"), copy_string16_value);
[email protected]9e4cda7332010-07-31 04:56:14423
424 Value* copy_binary = NULL;
425 ASSERT_TRUE(copy_dict->Get("binary", &copy_binary));
426 ASSERT_TRUE(copy_binary);
427 ASSERT_NE(copy_binary, original_binary);
[email protected]0de764e2011-08-26 01:54:00428 ASSERT_TRUE(copy_binary->IsType(Value::TYPE_BINARY));
429 ASSERT_NE(original_binary->GetBuffer(),
430 static_cast<BinaryValue*>(copy_binary)->GetBuffer());
431 ASSERT_EQ(original_binary->GetSize(),
432 static_cast<BinaryValue*>(copy_binary)->GetSize());
[email protected]9e4cda7332010-07-31 04:56:14433 ASSERT_EQ(0, memcmp(original_binary->GetBuffer(),
[email protected]0de764e2011-08-26 01:54:00434 static_cast<BinaryValue*>(copy_binary)->GetBuffer(),
435 original_binary->GetSize()));
[email protected]9e4cda7332010-07-31 04:56:14436
437 Value* copy_value = NULL;
438 ASSERT_TRUE(copy_dict->Get("list", &copy_value));
439 ASSERT_TRUE(copy_value);
440 ASSERT_NE(copy_value, original_list);
[email protected]0de764e2011-08-26 01:54:00441 ASSERT_TRUE(copy_value->IsType(Value::TYPE_LIST));
442 ListValue* copy_list = static_cast<ListValue*>(copy_value);
[email protected]9e4cda7332010-07-31 04:56:14443 ASSERT_EQ(2U, copy_list->GetSize());
444
445 Value* copy_list_element_0;
446 ASSERT_TRUE(copy_list->Get(0, &copy_list_element_0));
447 ASSERT_TRUE(copy_list_element_0);
448 ASSERT_NE(copy_list_element_0, original_list_element_0);
449 int copy_list_element_0_value;
450 ASSERT_TRUE(copy_list_element_0->GetAsInteger(&copy_list_element_0_value));
451 ASSERT_EQ(0, copy_list_element_0_value);
452
453 Value* copy_list_element_1;
454 ASSERT_TRUE(copy_list->Get(1, &copy_list_element_1));
455 ASSERT_TRUE(copy_list_element_1);
456 ASSERT_NE(copy_list_element_1, original_list_element_1);
457 int copy_list_element_1_value;
458 ASSERT_TRUE(copy_list_element_1->GetAsInteger(&copy_list_element_1_value));
459 ASSERT_EQ(1, copy_list_element_1_value);
460}
461
[email protected]58b916e2011-06-28 22:56:33462TEST(ValuesTest, Equals) {
initial.commitd7cae122008-07-26 21:49:38463 Value* null1 = Value::CreateNullValue();
464 Value* null2 = Value::CreateNullValue();
465 EXPECT_NE(null1, null2);
466 EXPECT_TRUE(null1->Equals(null2));
467
468 Value* boolean = Value::CreateBooleanValue(false);
469 EXPECT_FALSE(null1->Equals(boolean));
470 delete null1;
471 delete null2;
472 delete boolean;
473
474 DictionaryValue dv;
[email protected]9e4cda7332010-07-31 04:56:14475 dv.SetBoolean("a", false);
476 dv.SetInteger("b", 2);
[email protected]fb534c92011-02-01 01:02:07477 dv.SetDouble("c", 2.5);
[email protected]9e4cda7332010-07-31 04:56:14478 dv.SetString("d1", "string");
[email protected]ff4c1d82010-08-04 16:58:12479 dv.SetString("d2", ASCIIToUTF16("http://google.com"));
[email protected]9e4cda7332010-07-31 04:56:14480 dv.Set("e", Value::CreateNullValue());
481
[email protected]dc1f2442010-08-18 16:23:40482 scoped_ptr<DictionaryValue> copy;
[email protected]16f47e082011-01-18 02:16:59483 copy.reset(dv.DeepCopy());
[email protected]dc1f2442010-08-18 16:23:40484 EXPECT_TRUE(dv.Equals(copy.get()));
[email protected]9e4cda7332010-07-31 04:56:14485
486 ListValue* list = new ListValue;
487 list->Append(Value::CreateNullValue());
488 list->Append(new DictionaryValue);
489 dv.Set("f", list);
490
[email protected]dc1f2442010-08-18 16:23:40491 EXPECT_FALSE(dv.Equals(copy.get()));
[email protected]9e4cda7332010-07-31 04:56:14492 copy->Set("f", list->DeepCopy());
[email protected]dc1f2442010-08-18 16:23:40493 EXPECT_TRUE(dv.Equals(copy.get()));
[email protected]9e4cda7332010-07-31 04:56:14494
495 list->Append(Value::CreateBooleanValue(true));
[email protected]dc1f2442010-08-18 16:23:40496 EXPECT_FALSE(dv.Equals(copy.get()));
497
498 // Check if Equals detects differences in only the keys.
[email protected]16f47e082011-01-18 02:16:59499 copy.reset(dv.DeepCopy());
[email protected]dc1f2442010-08-18 16:23:40500 EXPECT_TRUE(dv.Equals(copy.get()));
501 copy->Remove("a", NULL);
502 copy->SetBoolean("aa", false);
503 EXPECT_FALSE(dv.Equals(copy.get()));
[email protected]9e4cda7332010-07-31 04:56:14504}
505
[email protected]58b916e2011-06-28 22:56:33506TEST(ValuesTest, StaticEquals) {
[email protected]73c47932010-12-06 18:13:43507 scoped_ptr<Value> null1(Value::CreateNullValue());
508 scoped_ptr<Value> null2(Value::CreateNullValue());
509 EXPECT_TRUE(Value::Equals(null1.get(), null2.get()));
510 EXPECT_TRUE(Value::Equals(NULL, NULL));
511
512 scoped_ptr<Value> i42(Value::CreateIntegerValue(42));
513 scoped_ptr<Value> j42(Value::CreateIntegerValue(42));
514 scoped_ptr<Value> i17(Value::CreateIntegerValue(17));
515 EXPECT_TRUE(Value::Equals(i42.get(), i42.get()));
516 EXPECT_TRUE(Value::Equals(j42.get(), i42.get()));
517 EXPECT_TRUE(Value::Equals(i42.get(), j42.get()));
518 EXPECT_FALSE(Value::Equals(i42.get(), i17.get()));
519 EXPECT_FALSE(Value::Equals(i42.get(), NULL));
520 EXPECT_FALSE(Value::Equals(NULL, i42.get()));
521
522 // NULL and Value::CreateNullValue() are intentionally different: We need
523 // support for NULL as a return value for "undefined" without caring for
524 // ownership of the pointer.
525 EXPECT_FALSE(Value::Equals(null1.get(), NULL));
526 EXPECT_FALSE(Value::Equals(NULL, null1.get()));
527}
528
[email protected]58b916e2011-06-28 22:56:33529TEST(ValuesTest, DeepCopyCovariantReturnTypes) {
[email protected]16f47e082011-01-18 02:16:59530 DictionaryValue original_dict;
531 Value* original_null = Value::CreateNullValue();
532 original_dict.Set("null", original_null);
533 FundamentalValue* original_bool = Value::CreateBooleanValue(true);
534 original_dict.Set("bool", original_bool);
535 FundamentalValue* original_int = Value::CreateIntegerValue(42);
536 original_dict.Set("int", original_int);
[email protected]fb534c92011-02-01 01:02:07537 FundamentalValue* original_double = Value::CreateDoubleValue(3.14);
538 original_dict.Set("double", original_double);
[email protected]16f47e082011-01-18 02:16:59539 StringValue* original_string = Value::CreateStringValue("hello");
540 original_dict.Set("string", original_string);
541 StringValue* original_string16 =
542 Value::CreateStringValue(ASCIIToUTF16("hello16"));
543 original_dict.Set("string16", original_string16);
544
545 char* original_buffer = new char[42];
546 memset(original_buffer, '!', 42);
[email protected]51914092011-08-11 17:43:01547 BinaryValue* original_binary = BinaryValue::Create(original_buffer, 42);
[email protected]16f47e082011-01-18 02:16:59548 original_dict.Set("binary", original_binary);
549
550 ListValue* original_list = new ListValue();
551 FundamentalValue* original_list_element_0 = Value::CreateIntegerValue(0);
552 original_list->Append(original_list_element_0);
553 FundamentalValue* original_list_element_1 = Value::CreateIntegerValue(1);
554 original_list->Append(original_list_element_1);
555 original_dict.Set("list", original_list);
556
557 Value* original_dict_value = &original_dict;
558 Value* original_bool_value = original_bool;
559 Value* original_int_value = original_int;
[email protected]fb534c92011-02-01 01:02:07560 Value* original_double_value = original_double;
[email protected]16f47e082011-01-18 02:16:59561 Value* original_string_value = original_string;
562 Value* original_string16_value = original_string16;
563 Value* original_binary_value = original_binary;
564 Value* original_list_value = original_list;
565
566 scoped_ptr<Value> copy_dict_value(original_dict_value->DeepCopy());
567 scoped_ptr<Value> copy_bool_value(original_bool_value->DeepCopy());
568 scoped_ptr<Value> copy_int_value(original_int_value->DeepCopy());
[email protected]fb534c92011-02-01 01:02:07569 scoped_ptr<Value> copy_double_value(original_double_value->DeepCopy());
[email protected]16f47e082011-01-18 02:16:59570 scoped_ptr<Value> copy_string_value(original_string_value->DeepCopy());
571 scoped_ptr<Value> copy_string16_value(original_string16_value->DeepCopy());
572 scoped_ptr<Value> copy_binary_value(original_binary_value->DeepCopy());
573 scoped_ptr<Value> copy_list_value(original_list_value->DeepCopy());
574
575 EXPECT_TRUE(original_dict_value->Equals(copy_dict_value.get()));
576 EXPECT_TRUE(original_bool_value->Equals(copy_bool_value.get()));
577 EXPECT_TRUE(original_int_value->Equals(copy_int_value.get()));
[email protected]fb534c92011-02-01 01:02:07578 EXPECT_TRUE(original_double_value->Equals(copy_double_value.get()));
[email protected]16f47e082011-01-18 02:16:59579 EXPECT_TRUE(original_string_value->Equals(copy_string_value.get()));
580 EXPECT_TRUE(original_string16_value->Equals(copy_string16_value.get()));
581 EXPECT_TRUE(original_binary_value->Equals(copy_binary_value.get()));
582 EXPECT_TRUE(original_list_value->Equals(copy_list_value.get()));
583}
584
[email protected]58b916e2011-06-28 22:56:33585TEST(ValuesTest, RemoveEmptyChildren) {
[email protected]ec330b52009-12-02 00:20:32586 scoped_ptr<DictionaryValue> root(new DictionaryValue);
587 // Remove empty lists and dictionaries.
[email protected]9e4cda7332010-07-31 04:56:14588 root->Set("empty_dict", new DictionaryValue);
589 root->Set("empty_list", new ListValue);
590 root->SetWithoutPathExpansion("a.b.c.d.e", new DictionaryValue);
591 root.reset(root->DeepCopyWithoutEmptyChildren());
592 EXPECT_TRUE(root->empty());
593
594 // Make sure we don't prune too much.
595 root->SetBoolean("bool", true);
596 root->Set("empty_dict", new DictionaryValue);
597 root->SetString("empty_string", "");
598 root.reset(root->DeepCopyWithoutEmptyChildren());
599 EXPECT_EQ(2U, root->size());
600
601 // Should do nothing.
602 root.reset(root->DeepCopyWithoutEmptyChildren());
603 EXPECT_EQ(2U, root->size());
604
605 // Nested test cases. These should all reduce back to the bool and string
606 // set above.
607 {
608 root->Set("a.b.c.d.e", new DictionaryValue);
609 root.reset(root->DeepCopyWithoutEmptyChildren());
610 EXPECT_EQ(2U, root->size());
611 }
612 {
613 DictionaryValue* inner = new DictionaryValue;
614 root->Set("dict_with_emtpy_children", inner);
615 inner->Set("empty_dict", new DictionaryValue);
616 inner->Set("empty_list", new ListValue);
617 root.reset(root->DeepCopyWithoutEmptyChildren());
618 EXPECT_EQ(2U, root->size());
619 }
620 {
621 ListValue* inner = new ListValue;
622 root->Set("list_with_empty_children", inner);
623 inner->Append(new DictionaryValue);
624 inner->Append(new ListValue);
625 root.reset(root->DeepCopyWithoutEmptyChildren());
626 EXPECT_EQ(2U, root->size());
627 }
628
629 // Nested with siblings.
630 {
631 ListValue* inner = new ListValue;
632 root->Set("list_with_empty_children", inner);
633 inner->Append(new DictionaryValue);
634 inner->Append(new ListValue);
635 DictionaryValue* inner2 = new DictionaryValue;
636 root->Set("dict_with_empty_children", inner2);
637 inner2->Set("empty_dict", new DictionaryValue);
638 inner2->Set("empty_list", new ListValue);
639 root.reset(root->DeepCopyWithoutEmptyChildren());
640 EXPECT_EQ(2U, root->size());
641 }
642
643 // Make sure nested values don't get pruned.
644 {
645 ListValue* inner = new ListValue;
646 root->Set("list_with_empty_children", inner);
647 ListValue* inner2 = new ListValue;
648 inner->Append(new DictionaryValue);
649 inner->Append(inner2);
650 inner2->Append(Value::CreateStringValue("hello"));
651 root.reset(root->DeepCopyWithoutEmptyChildren());
652 EXPECT_EQ(3U, root->size());
653 EXPECT_TRUE(root->GetList("list_with_empty_children", &inner));
654 EXPECT_EQ(1U, inner->GetSize()); // Dictionary was pruned.
655 EXPECT_TRUE(inner->GetList(0, &inner2));
656 EXPECT_EQ(1U, inner2->GetSize());
657 }
658}
659
[email protected]58b916e2011-06-28 22:56:33660TEST(ValuesTest, MergeDictionary) {
[email protected]c378cca2010-05-14 13:17:40661 scoped_ptr<DictionaryValue> base(new DictionaryValue);
[email protected]9e4cda7332010-07-31 04:56:14662 base->SetString("base_key", "base_key_value_base");
663 base->SetString("collide_key", "collide_key_value_base");
664 DictionaryValue* base_sub_dict = new DictionaryValue;
665 base_sub_dict->SetString("sub_base_key", "sub_base_key_value_base");
666 base_sub_dict->SetString("sub_collide_key", "sub_collide_key_value_base");
667 base->Set("sub_dict_key", base_sub_dict);
668
669 scoped_ptr<DictionaryValue> merge(new DictionaryValue);
670 merge->SetString("merge_key", "merge_key_value_merge");
671 merge->SetString("collide_key", "collide_key_value_merge");
672 DictionaryValue* merge_sub_dict = new DictionaryValue;
673 merge_sub_dict->SetString("sub_merge_key", "sub_merge_key_value_merge");
674 merge_sub_dict->SetString("sub_collide_key", "sub_collide_key_value_merge");
675 merge->Set("sub_dict_key", merge_sub_dict);
676
677 base->MergeDictionary(merge.get());
678
679 EXPECT_EQ(4U, base->size());
680 std::string base_key_value;
681 EXPECT_TRUE(base->GetString("base_key", &base_key_value));
682 EXPECT_EQ("base_key_value_base", base_key_value); // Base value preserved.
683 std::string collide_key_value;
684 EXPECT_TRUE(base->GetString("collide_key", &collide_key_value));
685 EXPECT_EQ("collide_key_value_merge", collide_key_value); // Replaced.
686 std::string merge_key_value;
687 EXPECT_TRUE(base->GetString("merge_key", &merge_key_value));
688 EXPECT_EQ("merge_key_value_merge", merge_key_value); // Merged in.
689
690 DictionaryValue* res_sub_dict;
691 EXPECT_TRUE(base->GetDictionary("sub_dict_key", &res_sub_dict));
692 EXPECT_EQ(3U, res_sub_dict->size());
693 std::string sub_base_key_value;
694 EXPECT_TRUE(res_sub_dict->GetString("sub_base_key", &sub_base_key_value));
695 EXPECT_EQ("sub_base_key_value_base", sub_base_key_value); // Preserved.
696 std::string sub_collide_key_value;
697 EXPECT_TRUE(res_sub_dict->GetString("sub_collide_key",
698 &sub_collide_key_value));
699 EXPECT_EQ("sub_collide_key_value_merge", sub_collide_key_value); // Replaced.
700 std::string sub_merge_key_value;
701 EXPECT_TRUE(res_sub_dict->GetString("sub_merge_key", &sub_merge_key_value));
702 EXPECT_EQ("sub_merge_key_value_merge", sub_merge_key_value); // Merged in.
703}
[email protected]58b916e2011-06-28 22:56:33704
[email protected]32c0e002011-11-08 21:26:41705TEST(ValuesTest, DictionaryIterator) {
706 DictionaryValue dict;
707 for (DictionaryValue::Iterator it(dict); it.HasNext(); it.Advance()) {
708 ADD_FAILURE();
709 }
710
711 StringValue value1("value1");
712 dict.Set("key1", value1.DeepCopy());
713 bool seen1 = false;
714 for (DictionaryValue::Iterator it(dict); it.HasNext(); it.Advance()) {
715 EXPECT_FALSE(seen1);
716 EXPECT_EQ("key1", it.key());
717 EXPECT_TRUE(value1.Equals(&it.value()));
718 seen1 = true;
719 }
720 EXPECT_TRUE(seen1);
721
722 StringValue value2("value2");
723 dict.Set("key2", value2.DeepCopy());
724 bool seen2 = seen1 = false;
725 for (DictionaryValue::Iterator it(dict); it.HasNext(); it.Advance()) {
726 if (it.key() == "key1") {
727 EXPECT_FALSE(seen1);
728 EXPECT_TRUE(value1.Equals(&it.value()));
729 seen1 = true;
730 } else if (it.key() == "key2") {
731 EXPECT_FALSE(seen2);
732 EXPECT_TRUE(value2.Equals(&it.value()));
733 seen2 = true;
734 } else {
735 ADD_FAILURE();
736 }
737 }
738 EXPECT_TRUE(seen1);
739 EXPECT_TRUE(seen2);
740}
741
[email protected]58b916e2011-06-28 22:56:33742} // namespace base