blob: 6b9fce1ae915976be64a81ed5e6f24767e5bdce5 [file] [log] [blame]
henrike@webrtc.org0e118e72013-07-10 00:45:36 +00001/*
2 * libjingle
3 * Copyright 2004--2005, Google Inc.
4 *
5 * Redistribution and use in source and binary forms, with or without
6 * modification, are permitted provided that the following conditions are met:
7 *
8 * 1. Redistributions of source code must retain the above copyright notice,
9 * this list of conditions and the following disclaimer.
10 * 2. Redistributions in binary form must reproduce the above copyright notice,
11 * this list of conditions and the following disclaimer in the documentation
12 * and/or other materials provided with the distribution.
13 * 3. The name of the author may not be used to endorse or promote products
14 * derived from this software without specific prior written permission.
15 *
16 * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR IMPLIED
17 * WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
18 * MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO
19 * EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
20 * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
21 * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS;
22 * OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY,
23 * WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR
24 * OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF
25 * ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
26 */
27
28#include "talk/base/json.h"
29
30#include <errno.h>
pbos@webrtc.orgb9518272014-03-07 15:22:04 +000031#include <limits.h>
32#include <stdlib.h>
henrike@webrtc.org0e118e72013-07-10 00:45:36 +000033
henrike@webrtc.org0e118e72013-07-10 00:45:36 +000034#include <sstream>
35
36bool GetStringFromJson(const Json::Value& in, std::string* out) {
37 if (!in.isString()) {
38 std::ostringstream s;
39 if (in.isBool()) {
40 s << std::boolalpha << in.asBool();
41 } else if (in.isInt()) {
42 s << in.asInt();
43 } else if (in.isUInt()) {
44 s << in.asUInt();
45 } else if (in.isDouble()) {
46 s << in.asDouble();
47 } else {
48 return false;
49 }
50 *out = s.str();
51 } else {
52 *out = in.asString();
53 }
54 return true;
55}
56
57bool GetIntFromJson(const Json::Value& in, int* out) {
58 bool ret;
59 if (!in.isString()) {
60 ret = in.isConvertibleTo(Json::intValue);
61 if (ret) {
62 *out = in.asInt();
63 }
64 } else {
65 long val; // NOLINT
66 const char* c_str = in.asCString();
67 char* end_ptr;
68 errno = 0;
69 val = strtol(c_str, &end_ptr, 10); // NOLINT
70 ret = (end_ptr != c_str && *end_ptr == '\0' && !errno &&
71 val >= INT_MIN && val <= INT_MAX);
72 *out = val;
73 }
74 return ret;
75}
76
77bool GetUIntFromJson(const Json::Value& in, unsigned int* out) {
78 bool ret;
79 if (!in.isString()) {
80 ret = in.isConvertibleTo(Json::uintValue);
81 if (ret) {
82 *out = in.asUInt();
83 }
84 } else {
85 unsigned long val; // NOLINT
86 const char* c_str = in.asCString();
87 char* end_ptr;
88 errno = 0;
89 val = strtoul(c_str, &end_ptr, 10); // NOLINT
90 ret = (end_ptr != c_str && *end_ptr == '\0' && !errno &&
91 val <= UINT_MAX);
92 *out = val;
93 }
94 return ret;
95}
96
97bool GetBoolFromJson(const Json::Value& in, bool* out) {
98 bool ret;
99 if (!in.isString()) {
100 ret = in.isConvertibleTo(Json::booleanValue);
101 if (ret) {
102 *out = in.asBool();
103 }
104 } else {
105 if (in.asString() == "true") {
106 *out = true;
107 ret = true;
108 } else if (in.asString() == "false") {
109 *out = false;
110 ret = true;
111 } else {
112 ret = false;
113 }
114 }
115 return ret;
116}
117
118bool GetDoubleFromJson(const Json::Value& in, double* out) {
119 bool ret;
120 if (!in.isString()) {
121 ret = in.isConvertibleTo(Json::realValue);
122 if (ret) {
123 *out = in.asDouble();
124 }
125 } else {
126 double val;
127 const char* c_str = in.asCString();
128 char* end_ptr;
129 errno = 0;
130 val = strtod(c_str, &end_ptr);
131 ret = (end_ptr != c_str && *end_ptr == '\0' && !errno);
132 *out = val;
133 }
134 return ret;
135}
136
137namespace {
138template<typename T>
139bool JsonArrayToVector(const Json::Value& value,
140 bool (*getter)(const Json::Value& in, T* out),
141 std::vector<T> *vec) {
142 vec->clear();
143 if (!value.isArray()) {
144 return false;
145 }
146
147 for (Json::Value::ArrayIndex i = 0; i < value.size(); ++i) {
148 T val;
149 if (!getter(value[i], &val)) {
150 return false;
151 }
152 vec->push_back(val);
153 }
154
155 return true;
156}
157// Trivial getter helper
158bool GetValueFromJson(const Json::Value& in, Json::Value* out) {
159 *out = in;
160 return true;
161}
162} // unnamed namespace
163
164bool JsonArrayToValueVector(const Json::Value& in,
165 std::vector<Json::Value>* out) {
166 return JsonArrayToVector(in, GetValueFromJson, out);
167}
168
169bool JsonArrayToIntVector(const Json::Value& in,
170 std::vector<int>* out) {
171 return JsonArrayToVector(in, GetIntFromJson, out);
172}
173
174bool JsonArrayToUIntVector(const Json::Value& in,
175 std::vector<unsigned int>* out) {
176 return JsonArrayToVector(in, GetUIntFromJson, out);
177}
178
179bool JsonArrayToStringVector(const Json::Value& in,
180 std::vector<std::string>* out) {
181 return JsonArrayToVector(in, GetStringFromJson, out);
182}
183
184bool JsonArrayToBoolVector(const Json::Value& in,
185 std::vector<bool>* out) {
186 return JsonArrayToVector(in, GetBoolFromJson, out);
187}
188
189bool JsonArrayToDoubleVector(const Json::Value& in,
190 std::vector<double>* out) {
191 return JsonArrayToVector(in, GetDoubleFromJson, out);
192}
193
194namespace {
195template<typename T>
196Json::Value VectorToJsonArray(const std::vector<T>& vec) {
197 Json::Value result(Json::arrayValue);
198 for (size_t i = 0; i < vec.size(); ++i) {
199 result.append(Json::Value(vec[i]));
200 }
201 return result;
202}
203} // unnamed namespace
204
205Json::Value ValueVectorToJsonArray(const std::vector<Json::Value>& in) {
206 return VectorToJsonArray(in);
207}
208
209Json::Value IntVectorToJsonArray(const std::vector<int>& in) {
210 return VectorToJsonArray(in);
211}
212
213Json::Value UIntVectorToJsonArray(const std::vector<unsigned int>& in) {
214 return VectorToJsonArray(in);
215}
216
217Json::Value StringVectorToJsonArray(const std::vector<std::string>& in) {
218 return VectorToJsonArray(in);
219}
220
221Json::Value BoolVectorToJsonArray(const std::vector<bool>& in) {
222 return VectorToJsonArray(in);
223}
224
225Json::Value DoubleVectorToJsonArray(const std::vector<double>& in) {
226 return VectorToJsonArray(in);
227}
228
229bool GetValueFromJsonArray(const Json::Value& in, size_t n,
230 Json::Value* out) {
231 if (!in.isArray() || !in.isValidIndex(static_cast<int>(n))) {
232 return false;
233 }
234
235 *out = in[static_cast<Json::Value::ArrayIndex>(n)];
236 return true;
237}
238
239bool GetIntFromJsonArray(const Json::Value& in, size_t n,
240 int* out) {
241 Json::Value x;
242 return GetValueFromJsonArray(in, n, &x) && GetIntFromJson(x, out);
243}
244
245bool GetUIntFromJsonArray(const Json::Value& in, size_t n,
246 unsigned int* out) {
247 Json::Value x;
248 return GetValueFromJsonArray(in, n, &x) && GetUIntFromJson(x, out);
249}
250
251bool GetStringFromJsonArray(const Json::Value& in, size_t n,
252 std::string* out) {
253 Json::Value x;
254 return GetValueFromJsonArray(in, n, &x) && GetStringFromJson(x, out);
255}
256
257bool GetBoolFromJsonArray(const Json::Value& in, size_t n,
258 bool* out) {
259 Json::Value x;
260 return GetValueFromJsonArray(in, n, &x) && GetBoolFromJson(x, out);
261}
262
263bool GetDoubleFromJsonArray(const Json::Value& in, size_t n,
264 double* out) {
265 Json::Value x;
266 return GetValueFromJsonArray(in, n, &x) && GetDoubleFromJson(x, out);
267}
268
269bool GetValueFromJsonObject(const Json::Value& in, const std::string& k,
270 Json::Value* out) {
271 if (!in.isObject() || !in.isMember(k)) {
272 return false;
273 }
274
275 *out = in[k];
276 return true;
277}
278
279bool GetIntFromJsonObject(const Json::Value& in, const std::string& k,
280 int* out) {
281 Json::Value x;
282 return GetValueFromJsonObject(in, k, &x) && GetIntFromJson(x, out);
283}
284
285bool GetUIntFromJsonObject(const Json::Value& in, const std::string& k,
286 unsigned int* out) {
287 Json::Value x;
288 return GetValueFromJsonObject(in, k, &x) && GetUIntFromJson(x, out);
289}
290
291bool GetStringFromJsonObject(const Json::Value& in, const std::string& k,
292 std::string* out) {
293 Json::Value x;
294 return GetValueFromJsonObject(in, k, &x) && GetStringFromJson(x, out);
295}
296
297bool GetBoolFromJsonObject(const Json::Value& in, const std::string& k,
298 bool* out) {
299 Json::Value x;
300 return GetValueFromJsonObject(in, k, &x) && GetBoolFromJson(x, out);
301}
302
303bool GetDoubleFromJsonObject(const Json::Value& in, const std::string& k,
304 double* out) {
305 Json::Value x;
306 return GetValueFromJsonObject(in, k, &x) && GetDoubleFromJson(x, out);
307}
308
309std::string JsonValueToString(const Json::Value& json) {
310 Json::FastWriter w;
311 std::string value = w.write(json);
312 return value.substr(0, value.size() - 1); // trim trailing newline
313}