• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 // Copyright (c) 2012 The Chromium Authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file.
4 
5 #include "base/json/json_writer.h"
6 
7 #include <stdint.h>
8 
9 #include <cmath>
10 #include <limits>
11 
12 #include "base/json/string_escape.h"
13 #include "base/logging.h"
14 #include "base/strings/string_number_conversions.h"
15 #include "base/strings/utf_string_conversions.h"
16 #include "base/values.h"
17 #include "build/build_config.h"
18 
19 namespace base {
20 
21 #if defined(OS_WIN)
22 const char kPrettyPrintLineEnding[] = "\r\n";
23 #else
24 const char kPrettyPrintLineEnding[] = "\n";
25 #endif
26 
27 // static
Write(const Value & node,std::string * json)28 bool JSONWriter::Write(const Value& node, std::string* json) {
29   return WriteWithOptions(node, 0, json);
30 }
31 
32 // static
WriteWithOptions(const Value & node,int options,std::string * json)33 bool JSONWriter::WriteWithOptions(const Value& node,
34                                   int options,
35                                   std::string* json) {
36   json->clear();
37   // Is there a better way to estimate the size of the output?
38   json->reserve(1024);
39 
40   JSONWriter writer(options, json);
41   bool result = writer.BuildJSONString(node, 0U);
42 
43   if (options & OPTIONS_PRETTY_PRINT)
44     json->append(kPrettyPrintLineEnding);
45 
46   return result;
47 }
48 
JSONWriter(int options,std::string * json)49 JSONWriter::JSONWriter(int options, std::string* json)
50     : omit_binary_values_((options & OPTIONS_OMIT_BINARY_VALUES) != 0),
51       omit_double_type_preservation_(
52           (options & OPTIONS_OMIT_DOUBLE_TYPE_PRESERVATION) != 0),
53       pretty_print_((options & OPTIONS_PRETTY_PRINT) != 0),
54       json_string_(json) {
55   DCHECK(json);
56 }
57 
BuildJSONString(const Value & node,size_t depth)58 bool JSONWriter::BuildJSONString(const Value& node, size_t depth) {
59   switch (node.GetType()) {
60     case Value::TYPE_NULL: {
61       json_string_->append("null");
62       return true;
63     }
64 
65     case Value::TYPE_BOOLEAN: {
66       bool value;
67       bool result = node.GetAsBoolean(&value);
68       DCHECK(result);
69       json_string_->append(value ? "true" : "false");
70       return result;
71     }
72 
73     case Value::TYPE_INTEGER: {
74       int value;
75       bool result = node.GetAsInteger(&value);
76       DCHECK(result);
77       json_string_->append(IntToString(value));
78       return result;
79     }
80 
81     case Value::TYPE_DOUBLE: {
82       double value;
83       bool result = node.GetAsDouble(&value);
84       DCHECK(result);
85       if (omit_double_type_preservation_ &&
86           value <= std::numeric_limits<int64_t>::max() &&
87           value >= std::numeric_limits<int64_t>::min() &&
88           std::floor(value) == value) {
89         json_string_->append(Int64ToString(static_cast<int64_t>(value)));
90         return result;
91       }
92       std::string real = DoubleToString(value);
93       // Ensure that the number has a .0 if there's no decimal or 'e'.  This
94       // makes sure that when we read the JSON back, it's interpreted as a
95       // real rather than an int.
96       if (real.find('.') == std::string::npos &&
97           real.find('e') == std::string::npos &&
98           real.find('E') == std::string::npos) {
99         real.append(".0");
100       }
101       // The JSON spec requires that non-integer values in the range (-1,1)
102       // have a zero before the decimal point - ".52" is not valid, "0.52" is.
103       if (real[0] == '.') {
104         real.insert(static_cast<size_t>(0), static_cast<size_t>(1), '0');
105       } else if (real.length() > 1 && real[0] == '-' && real[1] == '.') {
106         // "-.1" bad "-0.1" good
107         real.insert(static_cast<size_t>(1), static_cast<size_t>(1), '0');
108       }
109       json_string_->append(real);
110       return result;
111     }
112 
113     case Value::TYPE_STRING: {
114       std::string value;
115       bool result = node.GetAsString(&value);
116       DCHECK(result);
117       EscapeJSONString(value, true, json_string_);
118       return result;
119     }
120 
121     case Value::TYPE_LIST: {
122       json_string_->push_back('[');
123       if (pretty_print_)
124         json_string_->push_back(' ');
125 
126       const ListValue* list = NULL;
127       bool first_value_has_been_output = false;
128       bool result = node.GetAsList(&list);
129       DCHECK(result);
130       for (ListValue::const_iterator it = list->begin(); it != list->end();
131            ++it) {
132         const Value* value = *it;
133         if (omit_binary_values_ && value->GetType() == Value::TYPE_BINARY)
134           continue;
135 
136         if (first_value_has_been_output) {
137           json_string_->push_back(',');
138           if (pretty_print_)
139             json_string_->push_back(' ');
140         }
141 
142         if (!BuildJSONString(*value, depth))
143           result = false;
144 
145         first_value_has_been_output = true;
146       }
147 
148       if (pretty_print_)
149         json_string_->push_back(' ');
150       json_string_->push_back(']');
151       return result;
152     }
153 
154     case Value::TYPE_DICTIONARY: {
155       json_string_->push_back('{');
156       if (pretty_print_)
157         json_string_->append(kPrettyPrintLineEnding);
158 
159       const DictionaryValue* dict = NULL;
160       bool first_value_has_been_output = false;
161       bool result = node.GetAsDictionary(&dict);
162       DCHECK(result);
163       for (DictionaryValue::Iterator itr(*dict); !itr.IsAtEnd();
164            itr.Advance()) {
165         if (omit_binary_values_ &&
166             itr.value().GetType() == Value::TYPE_BINARY) {
167           continue;
168         }
169 
170         if (first_value_has_been_output) {
171           json_string_->push_back(',');
172           if (pretty_print_)
173             json_string_->append(kPrettyPrintLineEnding);
174         }
175 
176         if (pretty_print_)
177           IndentLine(depth + 1U);
178 
179         EscapeJSONString(itr.key(), true, json_string_);
180         json_string_->push_back(':');
181         if (pretty_print_)
182           json_string_->push_back(' ');
183 
184         if (!BuildJSONString(itr.value(), depth + 1U))
185           result = false;
186 
187         first_value_has_been_output = true;
188       }
189 
190       if (pretty_print_) {
191         json_string_->append(kPrettyPrintLineEnding);
192         IndentLine(depth);
193       }
194 
195       json_string_->push_back('}');
196       return result;
197     }
198 
199     case Value::TYPE_BINARY:
200       // Successful only if we're allowed to omit it.
201       DLOG_IF(ERROR, !omit_binary_values_) << "Cannot serialize binary value.";
202       return omit_binary_values_;
203   }
204   NOTREACHED();
205   return false;
206 }
207 
IndentLine(size_t depth)208 void JSONWriter::IndentLine(size_t depth) {
209   json_string_->append(depth * 3U, ' ');
210 }
211 
212 }  // namespace base
213