xref: /aosp_15_r20/external/webrtc/rtc_base/strings/json.cc (revision d9f758449e529ab9291ac668be2861e7a55c2422)
1 /*
2  *  Copyright 2004 The WebRTC Project Authors. All rights reserved.
3  *
4  *  Use of this source code is governed by a BSD-style license
5  *  that can be found in the LICENSE file in the root of the source
6  *  tree. An additional intellectual property rights grant can be found
7  *  in the file PATENTS.  All contributing project authors may
8  *  be found in the AUTHORS file in the root of the source tree.
9  */
10 
11 #include "rtc_base/strings/json.h"
12 
13 #include <errno.h>
14 #include <limits.h>
15 #include <stdlib.h>
16 
17 #include "absl/strings/string_view.h"
18 #include "rtc_base/string_encode.h"
19 
20 namespace rtc {
21 
GetStringFromJson(const Json::Value & in,std::string * out)22 bool GetStringFromJson(const Json::Value& in, std::string* out) {
23   if (!in.isString()) {
24     if (in.isBool()) {
25       *out = rtc::ToString(in.asBool());
26     } else if (in.isInt()) {
27       *out = rtc::ToString(in.asInt());
28     } else if (in.isUInt()) {
29       *out = rtc::ToString(in.asUInt());
30     } else if (in.isDouble()) {
31       *out = rtc::ToString(in.asDouble());
32     } else {
33       return false;
34     }
35   } else {
36     *out = in.asString();
37   }
38   return true;
39 }
40 
GetIntFromJson(const Json::Value & in,int * out)41 bool GetIntFromJson(const Json::Value& in, int* out) {
42   bool ret;
43   if (!in.isString()) {
44     ret = in.isConvertibleTo(Json::intValue);
45     if (ret) {
46       *out = in.asInt();
47     }
48   } else {
49     long val;  // NOLINT
50     const char* c_str = in.asCString();
51     char* end_ptr;
52     errno = 0;
53     val = strtol(c_str, &end_ptr, 10);  // NOLINT
54     ret = (end_ptr != c_str && *end_ptr == '\0' && !errno && val >= INT_MIN &&
55            val <= INT_MAX);
56     *out = val;
57   }
58   return ret;
59 }
60 
GetUIntFromJson(const Json::Value & in,unsigned int * out)61 bool GetUIntFromJson(const Json::Value& in, unsigned int* out) {
62   bool ret;
63   if (!in.isString()) {
64     ret = in.isConvertibleTo(Json::uintValue);
65     if (ret) {
66       *out = in.asUInt();
67     }
68   } else {
69     unsigned long val;  // NOLINT
70     const char* c_str = in.asCString();
71     char* end_ptr;
72     errno = 0;
73     val = strtoul(c_str, &end_ptr, 10);  // NOLINT
74     ret = (end_ptr != c_str && *end_ptr == '\0' && !errno && val <= UINT_MAX);
75     *out = val;
76   }
77   return ret;
78 }
79 
GetBoolFromJson(const Json::Value & in,bool * out)80 bool GetBoolFromJson(const Json::Value& in, bool* out) {
81   bool ret;
82   if (!in.isString()) {
83     ret = in.isConvertibleTo(Json::booleanValue);
84     if (ret) {
85       *out = in.asBool();
86     }
87   } else {
88     if (in.asString() == "true") {
89       *out = true;
90       ret = true;
91     } else if (in.asString() == "false") {
92       *out = false;
93       ret = true;
94     } else {
95       ret = false;
96     }
97   }
98   return ret;
99 }
100 
GetDoubleFromJson(const Json::Value & in,double * out)101 bool GetDoubleFromJson(const Json::Value& in, double* out) {
102   bool ret;
103   if (!in.isString()) {
104     ret = in.isConvertibleTo(Json::realValue);
105     if (ret) {
106       *out = in.asDouble();
107     }
108   } else {
109     double val;
110     const char* c_str = in.asCString();
111     char* end_ptr;
112     errno = 0;
113     val = strtod(c_str, &end_ptr);
114     ret = (end_ptr != c_str && *end_ptr == '\0' && !errno);
115     *out = val;
116   }
117   return ret;
118 }
119 
120 namespace {
121 template <typename T>
JsonArrayToVector(const Json::Value & value,bool (* getter)(const Json::Value & in,T * out),std::vector<T> * vec)122 bool JsonArrayToVector(const Json::Value& value,
123                        bool (*getter)(const Json::Value& in, T* out),
124                        std::vector<T>* vec) {
125   vec->clear();
126   if (!value.isArray()) {
127     return false;
128   }
129 
130   for (Json::Value::ArrayIndex i = 0; i < value.size(); ++i) {
131     T val;
132     if (!getter(value[i], &val)) {
133       return false;
134     }
135     vec->push_back(val);
136   }
137 
138   return true;
139 }
140 // Trivial getter helper
GetValueFromJson(const Json::Value & in,Json::Value * out)141 bool GetValueFromJson(const Json::Value& in, Json::Value* out) {
142   *out = in;
143   return true;
144 }
145 }  // unnamed namespace
146 
JsonArrayToValueVector(const Json::Value & in,std::vector<Json::Value> * out)147 bool JsonArrayToValueVector(const Json::Value& in,
148                             std::vector<Json::Value>* out) {
149   return JsonArrayToVector(in, GetValueFromJson, out);
150 }
151 
JsonArrayToIntVector(const Json::Value & in,std::vector<int> * out)152 bool JsonArrayToIntVector(const Json::Value& in, std::vector<int>* out) {
153   return JsonArrayToVector(in, GetIntFromJson, out);
154 }
155 
JsonArrayToUIntVector(const Json::Value & in,std::vector<unsigned int> * out)156 bool JsonArrayToUIntVector(const Json::Value& in,
157                            std::vector<unsigned int>* out) {
158   return JsonArrayToVector(in, GetUIntFromJson, out);
159 }
160 
JsonArrayToStringVector(const Json::Value & in,std::vector<std::string> * out)161 bool JsonArrayToStringVector(const Json::Value& in,
162                              std::vector<std::string>* out) {
163   return JsonArrayToVector(in, GetStringFromJson, out);
164 }
165 
JsonArrayToBoolVector(const Json::Value & in,std::vector<bool> * out)166 bool JsonArrayToBoolVector(const Json::Value& in, std::vector<bool>* out) {
167   return JsonArrayToVector(in, GetBoolFromJson, out);
168 }
169 
JsonArrayToDoubleVector(const Json::Value & in,std::vector<double> * out)170 bool JsonArrayToDoubleVector(const Json::Value& in, std::vector<double>* out) {
171   return JsonArrayToVector(in, GetDoubleFromJson, out);
172 }
173 
174 namespace {
175 template <typename T>
VectorToJsonArray(const std::vector<T> & vec)176 Json::Value VectorToJsonArray(const std::vector<T>& vec) {
177   Json::Value result(Json::arrayValue);
178   for (size_t i = 0; i < vec.size(); ++i) {
179     result.append(Json::Value(vec[i]));
180   }
181   return result;
182 }
183 }  // unnamed namespace
184 
ValueVectorToJsonArray(const std::vector<Json::Value> & in)185 Json::Value ValueVectorToJsonArray(const std::vector<Json::Value>& in) {
186   return VectorToJsonArray(in);
187 }
188 
IntVectorToJsonArray(const std::vector<int> & in)189 Json::Value IntVectorToJsonArray(const std::vector<int>& in) {
190   return VectorToJsonArray(in);
191 }
192 
UIntVectorToJsonArray(const std::vector<unsigned int> & in)193 Json::Value UIntVectorToJsonArray(const std::vector<unsigned int>& in) {
194   return VectorToJsonArray(in);
195 }
196 
StringVectorToJsonArray(const std::vector<std::string> & in)197 Json::Value StringVectorToJsonArray(const std::vector<std::string>& in) {
198   return VectorToJsonArray(in);
199 }
200 
BoolVectorToJsonArray(const std::vector<bool> & in)201 Json::Value BoolVectorToJsonArray(const std::vector<bool>& in) {
202   return VectorToJsonArray(in);
203 }
204 
DoubleVectorToJsonArray(const std::vector<double> & in)205 Json::Value DoubleVectorToJsonArray(const std::vector<double>& in) {
206   return VectorToJsonArray(in);
207 }
208 
GetValueFromJsonArray(const Json::Value & in,size_t n,Json::Value * out)209 bool GetValueFromJsonArray(const Json::Value& in, size_t n, Json::Value* out) {
210   if (!in.isArray() || !in.isValidIndex(static_cast<int>(n))) {
211     return false;
212   }
213 
214   *out = in[static_cast<Json::Value::ArrayIndex>(n)];
215   return true;
216 }
217 
GetIntFromJsonArray(const Json::Value & in,size_t n,int * out)218 bool GetIntFromJsonArray(const Json::Value& in, size_t n, int* out) {
219   Json::Value x;
220   return GetValueFromJsonArray(in, n, &x) && GetIntFromJson(x, out);
221 }
222 
GetUIntFromJsonArray(const Json::Value & in,size_t n,unsigned int * out)223 bool GetUIntFromJsonArray(const Json::Value& in, size_t n, unsigned int* out) {
224   Json::Value x;
225   return GetValueFromJsonArray(in, n, &x) && GetUIntFromJson(x, out);
226 }
227 
GetStringFromJsonArray(const Json::Value & in,size_t n,std::string * out)228 bool GetStringFromJsonArray(const Json::Value& in, size_t n, std::string* out) {
229   Json::Value x;
230   return GetValueFromJsonArray(in, n, &x) && GetStringFromJson(x, out);
231 }
232 
GetBoolFromJsonArray(const Json::Value & in,size_t n,bool * out)233 bool GetBoolFromJsonArray(const Json::Value& in, size_t n, bool* out) {
234   Json::Value x;
235   return GetValueFromJsonArray(in, n, &x) && GetBoolFromJson(x, out);
236 }
237 
GetDoubleFromJsonArray(const Json::Value & in,size_t n,double * out)238 bool GetDoubleFromJsonArray(const Json::Value& in, size_t n, double* out) {
239   Json::Value x;
240   return GetValueFromJsonArray(in, n, &x) && GetDoubleFromJson(x, out);
241 }
242 
GetValueFromJsonObject(const Json::Value & in,absl::string_view k,Json::Value * out)243 bool GetValueFromJsonObject(const Json::Value& in,
244                             absl::string_view k,
245                             Json::Value* out) {
246   std::string k_str(k);
247   if (!in.isObject() || !in.isMember(k_str)) {
248     return false;
249   }
250 
251   *out = in[k_str];
252   return true;
253 }
254 
GetIntFromJsonObject(const Json::Value & in,absl::string_view k,int * out)255 bool GetIntFromJsonObject(const Json::Value& in,
256                           absl::string_view k,
257                           int* out) {
258   Json::Value x;
259   return GetValueFromJsonObject(in, k, &x) && GetIntFromJson(x, out);
260 }
261 
GetUIntFromJsonObject(const Json::Value & in,absl::string_view k,unsigned int * out)262 bool GetUIntFromJsonObject(const Json::Value& in,
263                            absl::string_view k,
264                            unsigned int* out) {
265   Json::Value x;
266   return GetValueFromJsonObject(in, k, &x) && GetUIntFromJson(x, out);
267 }
268 
GetStringFromJsonObject(const Json::Value & in,absl::string_view k,std::string * out)269 bool GetStringFromJsonObject(const Json::Value& in,
270                              absl::string_view k,
271                              std::string* out) {
272   Json::Value x;
273   return GetValueFromJsonObject(in, k, &x) && GetStringFromJson(x, out);
274 }
275 
GetBoolFromJsonObject(const Json::Value & in,absl::string_view k,bool * out)276 bool GetBoolFromJsonObject(const Json::Value& in,
277                            absl::string_view k,
278                            bool* out) {
279   Json::Value x;
280   return GetValueFromJsonObject(in, k, &x) && GetBoolFromJson(x, out);
281 }
282 
GetDoubleFromJsonObject(const Json::Value & in,absl::string_view k,double * out)283 bool GetDoubleFromJsonObject(const Json::Value& in,
284                              absl::string_view k,
285                              double* out) {
286   Json::Value x;
287   return GetValueFromJsonObject(in, k, &x) && GetDoubleFromJson(x, out);
288 }
289 
JsonValueToString(const Json::Value & json)290 std::string JsonValueToString(const Json::Value& json) {
291   Json::StreamWriterBuilder builder;
292   std::string output = Json::writeString(builder, json);
293   return output.substr(0, output.size() - 1);  // trim trailing newline
294 }
295 
296 }  // namespace rtc
297