mirror of
https://github.com/klzgrad/naiveproxy.git
synced 2024-11-28 16:26:10 +03:00
178 lines
4.6 KiB
C++
178 lines
4.6 KiB
C++
|
// 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 <stdint.h>
|
||
|
|
||
|
#include <cmath>
|
||
|
#include <limits>
|
||
|
|
||
|
#include "base/json/string_escape.h"
|
||
|
#include "base/logging.h"
|
||
|
#include "base/strings/string_number_conversions.h"
|
||
|
#include "base/strings/utf_string_conversions.h"
|
||
|
#include "base/values.h"
|
||
|
#include "util/build_config.h"
|
||
|
|
||
|
namespace base {
|
||
|
|
||
|
#if defined(OS_WIN)
|
||
|
const char kPrettyPrintLineEnding[] = "\r\n";
|
||
|
#else
|
||
|
const char kPrettyPrintLineEnding[] = "\n";
|
||
|
#endif
|
||
|
|
||
|
// static
|
||
|
bool JSONWriter::Write(const Value& node, std::string* json) {
|
||
|
return WriteWithOptions(node, 0, json);
|
||
|
}
|
||
|
|
||
|
// static
|
||
|
bool JSONWriter::WriteWithOptions(const Value& node,
|
||
|
int options,
|
||
|
std::string* json) {
|
||
|
json->clear();
|
||
|
// Is there a better way to estimate the size of the output?
|
||
|
json->reserve(1024);
|
||
|
|
||
|
JSONWriter writer(options, json);
|
||
|
bool result = writer.BuildJSONString(node, 0U);
|
||
|
|
||
|
if (options & OPTIONS_PRETTY_PRINT)
|
||
|
json->append(kPrettyPrintLineEnding);
|
||
|
|
||
|
return result;
|
||
|
}
|
||
|
|
||
|
JSONWriter::JSONWriter(int options, std::string* json)
|
||
|
: omit_binary_values_((options & OPTIONS_OMIT_BINARY_VALUES) != 0),
|
||
|
omit_double_type_preservation_(
|
||
|
(options & OPTIONS_OMIT_DOUBLE_TYPE_PRESERVATION) != 0),
|
||
|
pretty_print_((options & OPTIONS_PRETTY_PRINT) != 0),
|
||
|
json_string_(json) {
|
||
|
DCHECK(json);
|
||
|
}
|
||
|
|
||
|
bool JSONWriter::BuildJSONString(const Value& node, size_t depth) {
|
||
|
switch (node.type()) {
|
||
|
case Value::Type::NONE: {
|
||
|
json_string_->append("null");
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
case Value::Type::BOOLEAN: {
|
||
|
bool value;
|
||
|
bool result = node.GetAsBoolean(&value);
|
||
|
DCHECK(result);
|
||
|
json_string_->append(value ? "true" : "false");
|
||
|
return result;
|
||
|
}
|
||
|
|
||
|
case Value::Type::INTEGER: {
|
||
|
int value;
|
||
|
bool result = node.GetAsInteger(&value);
|
||
|
DCHECK(result);
|
||
|
json_string_->append(IntToString(value));
|
||
|
return result;
|
||
|
}
|
||
|
|
||
|
case Value::Type::STRING: {
|
||
|
std::string value;
|
||
|
bool result = node.GetAsString(&value);
|
||
|
DCHECK(result);
|
||
|
EscapeJSONString(value, true, json_string_);
|
||
|
return result;
|
||
|
}
|
||
|
|
||
|
case Value::Type::LIST: {
|
||
|
json_string_->push_back('[');
|
||
|
if (pretty_print_)
|
||
|
json_string_->push_back(' ');
|
||
|
|
||
|
const ListValue* list = nullptr;
|
||
|
bool first_value_has_been_output = false;
|
||
|
bool result = node.GetAsList(&list);
|
||
|
DCHECK(result);
|
||
|
for (const auto& value : *list) {
|
||
|
if (omit_binary_values_ && value.type() == Value::Type::BINARY)
|
||
|
continue;
|
||
|
|
||
|
if (first_value_has_been_output) {
|
||
|
json_string_->push_back(',');
|
||
|
if (pretty_print_)
|
||
|
json_string_->push_back(' ');
|
||
|
}
|
||
|
|
||
|
if (!BuildJSONString(value, depth))
|
||
|
result = false;
|
||
|
|
||
|
first_value_has_been_output = true;
|
||
|
}
|
||
|
|
||
|
if (pretty_print_)
|
||
|
json_string_->push_back(' ');
|
||
|
json_string_->push_back(']');
|
||
|
return result;
|
||
|
}
|
||
|
|
||
|
case Value::Type::DICTIONARY: {
|
||
|
json_string_->push_back('{');
|
||
|
if (pretty_print_)
|
||
|
json_string_->append(kPrettyPrintLineEnding);
|
||
|
|
||
|
const DictionaryValue* dict = nullptr;
|
||
|
bool first_value_has_been_output = false;
|
||
|
bool result = node.GetAsDictionary(&dict);
|
||
|
DCHECK(result);
|
||
|
for (DictionaryValue::Iterator itr(*dict); !itr.IsAtEnd();
|
||
|
itr.Advance()) {
|
||
|
if (omit_binary_values_ && itr.value().type() == Value::Type::BINARY) {
|
||
|
continue;
|
||
|
}
|
||
|
|
||
|
if (first_value_has_been_output) {
|
||
|
json_string_->push_back(',');
|
||
|
if (pretty_print_)
|
||
|
json_string_->append(kPrettyPrintLineEnding);
|
||
|
}
|
||
|
|
||
|
if (pretty_print_)
|
||
|
IndentLine(depth + 1U);
|
||
|
|
||
|
EscapeJSONString(itr.key(), true, json_string_);
|
||
|
json_string_->push_back(':');
|
||
|
if (pretty_print_)
|
||
|
json_string_->push_back(' ');
|
||
|
|
||
|
if (!BuildJSONString(itr.value(), depth + 1U))
|
||
|
result = false;
|
||
|
|
||
|
first_value_has_been_output = true;
|
||
|
}
|
||
|
|
||
|
if (pretty_print_) {
|
||
|
json_string_->append(kPrettyPrintLineEnding);
|
||
|
IndentLine(depth);
|
||
|
}
|
||
|
|
||
|
json_string_->push_back('}');
|
||
|
return result;
|
||
|
}
|
||
|
|
||
|
case Value::Type::BINARY:
|
||
|
// Successful only if we're allowed to omit it.
|
||
|
DLOG_IF(ERROR, !omit_binary_values_) << "Cannot serialize binary value.";
|
||
|
return omit_binary_values_;
|
||
|
}
|
||
|
NOTREACHED();
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
void JSONWriter::IndentLine(size_t depth) {
|
||
|
json_string_->append(depth * 3U, ' ');
|
||
|
}
|
||
|
|
||
|
} // namespace base
|