// 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 #include "base/json/string_escape.h" #include "base/logging.h" #include "base/strings/string_number_conversions.h" #include "base/strings/stringprintf.h" #include "base/strings/utf_string_conversions.h" #include "base/values.h" namespace base { #if defined(OS_WIN) static const char kPrettyPrintLineEnding[] = "\r\n"; #else static const char kPrettyPrintLineEnding[] = "\n"; #endif /* static */ const char* JSONWriter::kEmptyArray = "[]"; /* static */ void JSONWriter::Write(const Value* const node, std::string* json) { WriteWithOptions(node, 0, json); } /* static */ void JSONWriter::WriteWithOptions(const Value* const node, int options, std::string* json) { json->clear(); // Is there a better way to estimate the size of the output? json->reserve(1024); bool escape = !(options & OPTIONS_DO_NOT_ESCAPE); bool omit_binary_values = !!(options & OPTIONS_OMIT_BINARY_VALUES); bool omit_double_type_preservation = !!(options & OPTIONS_OMIT_DOUBLE_TYPE_PRESERVATION); bool pretty_print = !!(options & OPTIONS_PRETTY_PRINT); JSONWriter writer(escape, omit_binary_values, omit_double_type_preservation, pretty_print, json); writer.BuildJSONString(node, 0); if (pretty_print) json->append(kPrettyPrintLineEnding); } JSONWriter::JSONWriter(bool escape, bool omit_binary_values, bool omit_double_type_preservation, bool pretty_print, std::string* json) : escape_(escape), omit_binary_values_(omit_binary_values), omit_double_type_preservation_(omit_double_type_preservation), pretty_print_(pretty_print), json_string_(json) { DCHECK(json); } void JSONWriter::BuildJSONString(const Value* const node, int depth) { switch (node->GetType()) { case Value::TYPE_NULL: json_string_->append("null"); break; case Value::TYPE_BOOLEAN: { bool value; bool result = node->GetAsBoolean(&value); DCHECK(result); json_string_->append(value ? "true" : "false"); break; } case Value::TYPE_INTEGER: { int value; bool result = node->GetAsInteger(&value); DCHECK(result); base::StringAppendF(json_string_, "%d", value); break; } case Value::TYPE_DOUBLE: { double value; bool result = node->GetAsDouble(&value); DCHECK(result); if (omit_double_type_preservation_ && value <= kint64max && value >= kint64min && std::floor(value) == value) { json_string_->append(Int64ToString(static_cast(value))); break; } std::string real = DoubleToString(value); // Ensure that the number has a .0 if there's no decimal or 'e'. This // makes sure that when we read the JSON back, it's interpreted as a // real rather than an int. if (real.find('.') == std::string::npos && real.find('e') == std::string::npos && real.find('E') == std::string::npos) { real.append(".0"); } // The JSON spec requires that non-integer values in the range (-1,1) // have a zero before the decimal point - ".52" is not valid, "0.52" is. if (real[0] == '.') { real.insert(0, "0"); } else if (real.length() > 1 && real[0] == '-' && real[1] == '.') { // "-.1" bad "-0.1" good real.insert(1, "0"); } json_string_->append(real); break; } case Value::TYPE_STRING: { std::string value; bool result = node->GetAsString(&value); DCHECK(result); if (escape_) { JsonDoubleQuote(UTF8ToUTF16(value), true, json_string_); } else { JsonDoubleQuote(value, true, json_string_); } break; } case Value::TYPE_LIST: { json_string_->append("["); if (pretty_print_) json_string_->append(" "); const ListValue* list = static_cast(node); for (size_t i = 0; i < list->GetSize(); ++i) { const Value* value = NULL; bool result = list->Get(i, &value); DCHECK(result); if (omit_binary_values_ && value->GetType() == Value::TYPE_BINARY) { continue; } if (i != 0) { json_string_->append(","); if (pretty_print_) json_string_->append(" "); } BuildJSONString(value, depth); } if (pretty_print_) json_string_->append(" "); json_string_->append("]"); break; } case Value::TYPE_DICTIONARY: { json_string_->append("{"); if (pretty_print_) json_string_->append(kPrettyPrintLineEnding); const DictionaryValue* dict = static_cast(node); bool first_entry = true; for (DictionaryValue::Iterator itr(*dict); !itr.IsAtEnd(); itr.Advance(), first_entry = false) { if (omit_binary_values_ && itr.value().GetType() == Value::TYPE_BINARY) { continue; } if (!first_entry) { json_string_->append(","); if (pretty_print_) json_string_->append(kPrettyPrintLineEnding); } if (pretty_print_) IndentLine(depth + 1); AppendQuotedString(itr.key()); if (pretty_print_) { json_string_->append(": "); } else { json_string_->append(":"); } BuildJSONString(&itr.value(), depth + 1); } if (pretty_print_) { json_string_->append(kPrettyPrintLineEnding); IndentLine(depth); json_string_->append("}"); } else { json_string_->append("}"); } break; } case Value::TYPE_BINARY: { if (!omit_binary_values_) { NOTREACHED() << "Cannot serialize binary value."; } break; } default: NOTREACHED() << "unknown json type"; } } void JSONWriter::AppendQuotedString(const std::string& str) { // TODO(viettrungluu): |str| is UTF-8, not ASCII, so to properly escape it we // have to convert it to UTF-16. This round-trip is suboptimal. JsonDoubleQuote(UTF8ToUTF16(str), true, json_string_); } void JSONWriter::IndentLine(int depth) { // It may be faster to keep an indent string so we don't have to keep // reallocating. json_string_->append(std::string(depth * 3, ' ')); } } // namespace base