blob: ae46800dde686f9a3255f57dc8a99bc5b28fd445 [file] [log] [blame]
// Copyright (c) 2012 The Chromium Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include "base/json/json_writer.h"
#include "base/values.h"
#include "testing/gtest/include/gtest/gtest.h"
namespace base {
TEST(JSONWriterTest, BasicTypes) {
std::string output_js;
// Test null.
Value* root = Value::CreateNullValue();
EXPECT_TRUE(JSONWriter::Write(root, &output_js));
EXPECT_EQ("null", output_js);
delete root;
// Test empty dict.
root = new DictionaryValue;
EXPECT_TRUE(JSONWriter::Write(root, &output_js));
EXPECT_EQ("{}", output_js);
delete root;
// Test empty list.
root = new ListValue;
EXPECT_TRUE(JSONWriter::Write(root, &output_js));
EXPECT_EQ("[]", output_js);
delete root;
// Test integer values.
root = new FundamentalValue(42);
EXPECT_TRUE(JSONWriter::Write(root, &output_js));
EXPECT_EQ("42", output_js);
delete root;
// Test boolean values.
root = new FundamentalValue(true);
EXPECT_TRUE(JSONWriter::Write(root, &output_js));
EXPECT_EQ("true", output_js);
delete root;
// Test Real values should always have a decimal or an 'e'.
root = new FundamentalValue(1.0);
EXPECT_TRUE(JSONWriter::Write(root, &output_js));
EXPECT_EQ("1.0", output_js);
delete root;
// Test Real values in the the range (-1, 1) must have leading zeros
root = new FundamentalValue(0.2);
EXPECT_TRUE(JSONWriter::Write(root, &output_js));
EXPECT_EQ("0.2", output_js);
delete root;
// Test Real values in the the range (-1, 1) must have leading zeros
root = new FundamentalValue(-0.8);
EXPECT_TRUE(JSONWriter::Write(root, &output_js));
EXPECT_EQ("-0.8", output_js);
delete root;
// Test String values.
root = new StringValue("foo");
EXPECT_TRUE(JSONWriter::Write(root, &output_js));
EXPECT_EQ("\"foo\"", output_js);
delete root;
}
TEST(JSONWriterTest, NestedTypes) {
std::string output_js;
// Writer unittests like empty list/dict nesting,
// list list nesting, etc.
DictionaryValue root_dict;
ListValue* list = new ListValue;
root_dict.Set("list", list);
DictionaryValue* inner_dict = new DictionaryValue;
list->Append(inner_dict);
inner_dict->SetInteger("inner int", 10);
ListValue* inner_list = new ListValue;
list->Append(inner_list);
list->Append(new FundamentalValue(true));
// Test the pretty-printer.
EXPECT_TRUE(JSONWriter::Write(&root_dict, &output_js));
EXPECT_EQ("{\"list\":[{\"inner int\":10},[],true]}", output_js);
EXPECT_TRUE(JSONWriter::WriteWithOptions(&root_dict,
JSONWriter::OPTIONS_PRETTY_PRINT,
&output_js));
// The pretty-printer uses a different newline style on Windows than on
// other platforms.
#if defined(OS_WIN)
#define JSON_NEWLINE "\r\n"
#else
#define JSON_NEWLINE "\n"
#endif
EXPECT_EQ("{" JSON_NEWLINE
" \"list\": [ {" JSON_NEWLINE
" \"inner int\": 10" JSON_NEWLINE
" }, [ ], true ]" JSON_NEWLINE
"}" JSON_NEWLINE,
output_js);
#undef JSON_NEWLINE
}
TEST(JSONWriterTest, KeysWithPeriods) {
std::string output_js;
DictionaryValue period_dict;
period_dict.SetWithoutPathExpansion("a.b", new FundamentalValue(3));
period_dict.SetWithoutPathExpansion("c", new FundamentalValue(2));
DictionaryValue* period_dict2 = new DictionaryValue;
period_dict2->SetWithoutPathExpansion("g.h.i.j", new FundamentalValue(1));
period_dict.SetWithoutPathExpansion("d.e.f", period_dict2);
EXPECT_TRUE(JSONWriter::Write(&period_dict, &output_js));
EXPECT_EQ("{\"a.b\":3,\"c\":2,\"d.e.f\":{\"g.h.i.j\":1}}", output_js);
DictionaryValue period_dict3;
period_dict3.Set("a.b", new FundamentalValue(2));
period_dict3.SetWithoutPathExpansion("a.b", new FundamentalValue(1));
EXPECT_TRUE(JSONWriter::Write(&period_dict3, &output_js));
EXPECT_EQ("{\"a\":{\"b\":2},\"a.b\":1}", output_js);
}
TEST(JSONWriterTest, BinaryValues) {
std::string output_js;
// Binary values should return errors unless suppressed via the
// OPTIONS_OMIT_BINARY_VALUES flag.
Value* root = BinaryValue::CreateWithCopiedBuffer("asdf", 4);
EXPECT_FALSE(JSONWriter::Write(root, &output_js));
EXPECT_TRUE(JSONWriter::WriteWithOptions(
root, JSONWriter::OPTIONS_OMIT_BINARY_VALUES, &output_js));
EXPECT_TRUE(output_js.empty());
delete root;
ListValue binary_list;
binary_list.Append(BinaryValue::CreateWithCopiedBuffer("asdf", 4));
binary_list.Append(new FundamentalValue(5));
binary_list.Append(BinaryValue::CreateWithCopiedBuffer("asdf", 4));
binary_list.Append(new FundamentalValue(2));
binary_list.Append(BinaryValue::CreateWithCopiedBuffer("asdf", 4));
EXPECT_FALSE(JSONWriter::Write(&binary_list, &output_js));
EXPECT_TRUE(JSONWriter::WriteWithOptions(
&binary_list, JSONWriter::OPTIONS_OMIT_BINARY_VALUES, &output_js));
EXPECT_EQ("[5,2]", output_js);
DictionaryValue binary_dict;
binary_dict.Set("a", BinaryValue::CreateWithCopiedBuffer("asdf", 4));
binary_dict.Set("b", new FundamentalValue(5));
binary_dict.Set("c", BinaryValue::CreateWithCopiedBuffer("asdf", 4));
binary_dict.Set("d", new FundamentalValue(2));
binary_dict.Set("e", BinaryValue::CreateWithCopiedBuffer("asdf", 4));
EXPECT_FALSE(JSONWriter::Write(&binary_dict, &output_js));
EXPECT_TRUE(JSONWriter::WriteWithOptions(
&binary_dict, JSONWriter::OPTIONS_OMIT_BINARY_VALUES, &output_js));
EXPECT_EQ("{\"b\":5,\"d\":2}", output_js);
}
TEST(JSONWriterTest, DoublesAsInts) {
std::string output_js;
// Test allowing a double with no fractional part to be written as an integer.
FundamentalValue double_value(1e10);
EXPECT_TRUE(JSONWriter::WriteWithOptions(
&double_value,
JSONWriter::OPTIONS_OMIT_DOUBLE_TYPE_PRESERVATION,
&output_js));
EXPECT_EQ("10000000000", output_js);
}
} // namespace base