Branch data Line data Source code
1 : : // Copyright 2014 BitPay Inc.
2 : : // Copyright 2015 Bitcoin Core Developers
3 : : // Distributed under the MIT software license, see the accompanying
4 : : // file COPYING or https://opensource.org/licenses/mit-license.php.
5 : :
6 : : #include <univalue.h>
7 : :
8 : : #include <iomanip>
9 : : #include <map>
10 : : #include <memory>
11 : : #include <sstream>
12 : : #include <string>
13 : : #include <utility>
14 : : #include <vector>
15 : :
16 : : const UniValue NullUniValue;
17 : :
18 : 442439 : void UniValue::clear()
19 : : {
20 : 442439 : typ = VNULL;
21 : 442439 : val.clear();
22 : 442439 : keys.clear();
23 : 442439 : values.clear();
24 : 442439 : }
25 : :
26 : 1 : void UniValue::setNull()
27 : : {
28 : 1 : clear();
29 : 1 : }
30 : :
31 : 168179 : void UniValue::setBool(bool val_)
32 : : {
33 : 168179 : clear();
34 : 168179 : typ = VBOOL;
35 [ + + ]: 168179 : if (val_)
36 : 1471 : val = "1";
37 : 168179 : }
38 : :
39 : 2402 : static bool validNumStr(const std::string& s)
40 : : {
41 [ + - ]: 2402 : std::string tokenVal;
42 : 2402 : unsigned int consumed;
43 [ + - ]: 2402 : enum jtokentype tt = getJsonToken(tokenVal, consumed, s.data(), s.data() + s.size());
44 : 2402 : return (tt == JTOK_NUMBER);
45 : 2402 : }
46 : :
47 : 2402 : void UniValue::setNumStr(std::string str)
48 : : {
49 [ + + ]: 2402 : if (!validNumStr(str)) {
50 [ + - + - ]: 3 : throw std::runtime_error{"The string '" + str + "' is not a valid JSON number"};
51 : : }
52 : :
53 : 2401 : clear();
54 : 2401 : typ = VNUM;
55 : 2401 : val = std::move(str);
56 : 2401 : }
57 : :
58 : 298 : void UniValue::setInt(uint64_t val_)
59 : : {
60 : 298 : std::ostringstream oss;
61 : :
62 [ + - ]: 298 : oss << val_;
63 : :
64 [ + - + - ]: 596 : return setNumStr(oss.str());
65 : 298 : }
66 : :
67 : 1983 : void UniValue::setInt(int64_t val_)
68 : : {
69 : 1983 : std::ostringstream oss;
70 : :
71 [ + - ]: 1983 : oss << val_;
72 : :
73 [ + - + - ]: 3966 : return setNumStr(oss.str());
74 : 1983 : }
75 : :
76 : 85 : void UniValue::setFloat(double val_)
77 : : {
78 : 85 : std::ostringstream oss;
79 : :
80 [ + - ]: 85 : oss << std::setprecision(16) << val_;
81 : :
82 [ + - + - ]: 170 : return setNumStr(oss.str());
83 : 85 : }
84 : :
85 : 271509 : void UniValue::setStr(std::string str)
86 : : {
87 : 271509 : clear();
88 : 271509 : typ = VSTR;
89 : 271509 : val = std::move(str);
90 : 271509 : }
91 : :
92 : 96 : void UniValue::setArray()
93 : : {
94 : 96 : clear();
95 : 96 : typ = VARR;
96 : 96 : }
97 : :
98 : 61 : void UniValue::setObject()
99 : : {
100 : 61 : clear();
101 : 61 : typ = VOBJ;
102 : 61 : }
103 : :
104 : 1181 : void UniValue::push_back(UniValue val)
105 : : {
106 : 1181 : checkType(VARR);
107 : :
108 : 1180 : values.push_back(std::move(val));
109 : 1180 : }
110 : :
111 : 2 : void UniValue::push_backV(const std::vector<UniValue>& vec)
112 : : {
113 : 2 : checkType(VARR);
114 : :
115 : 1 : values.insert(values.end(), vec.begin(), vec.end());
116 : 1 : }
117 : :
118 : 1936 : void UniValue::pushKVEnd(std::string key, UniValue val)
119 : : {
120 : 1936 : checkType(VOBJ);
121 : :
122 : 1935 : keys.push_back(std::move(key));
123 : 1935 : values.push_back(std::move(val));
124 : 1935 : }
125 : :
126 : 1732 : void UniValue::pushKV(std::string key, UniValue val)
127 : : {
128 : 1732 : checkType(VOBJ);
129 : :
130 : 1731 : size_t idx;
131 [ + + ]: 1731 : if (findKey(key, idx))
132 : 1 : values[idx] = std::move(val);
133 : : else
134 [ + - ]: 3460 : pushKVEnd(std::move(key), std::move(val));
135 : 1731 : }
136 : :
137 : 2 : void UniValue::pushKVs(UniValue obj)
138 : : {
139 : 2 : checkType(VOBJ);
140 : 1 : obj.checkType(VOBJ);
141 : :
142 [ + + ]: 3 : for (size_t i = 0; i < obj.keys.size(); i++)
143 [ + - + - ]: 4 : pushKVEnd(std::move(obj.keys.at(i)), std::move(obj.values.at(i)));
144 : 1 : }
145 : :
146 : 1 : void UniValue::getObjMap(std::map<std::string,UniValue>& kv) const
147 : : {
148 [ + - ]: 1 : if (typ != VOBJ)
149 : : return;
150 : :
151 : 1 : kv.clear();
152 [ + + ]: 3 : for (size_t i = 0; i < keys.size(); i++)
153 : 2 : kv[keys[i]] = values[i];
154 : : }
155 : :
156 : 2235 : bool UniValue::findKey(const std::string& key, size_t& retIdx) const
157 : : {
158 [ + + ]: 6432 : for (size_t i = 0; i < keys.size(); i++) {
159 [ + + ]: 4667 : if (keys[i] == key) {
160 : 470 : retIdx = i;
161 : 470 : return true;
162 : : }
163 : : }
164 : :
165 : : return false;
166 : : }
167 : :
168 : 11 : bool UniValue::checkObject(const std::map<std::string,UniValue::VType>& t) const
169 : : {
170 [ + - ]: 11 : if (typ != VOBJ) {
171 : : return false;
172 : : }
173 : :
174 [ + + ]: 51 : for (const auto& object: t) {
175 : 42 : size_t idx = 0;
176 [ + - ]: 42 : if (!findKey(object.first, idx)) {
177 : : return false;
178 : : }
179 : :
180 [ + + ]: 42 : if (values.at(idx).getType() != object.second) {
181 : : return false;
182 : : }
183 : : }
184 : :
185 : : return true;
186 : : }
187 : :
188 : 403 : const UniValue& UniValue::operator[](const std::string& key) const
189 : : {
190 [ + - ]: 403 : if (typ != VOBJ)
191 : : return NullUniValue;
192 : :
193 : 403 : size_t index = 0;
194 [ + + ]: 403 : if (!findKey(key, index))
195 : : return NullUniValue;
196 : :
197 : 400 : return values.at(index);
198 : : }
199 : :
200 : 17084 : const UniValue& UniValue::operator[](size_t index) const
201 : : {
202 [ + - ]: 17084 : if (typ != VOBJ && typ != VARR)
203 : : return NullUniValue;
204 [ + + ]: 17084 : if (index >= values.size())
205 : : return NullUniValue;
206 : :
207 : 16996 : return values.at(index);
208 : : }
209 : :
210 : 142985 : void UniValue::checkType(const VType& expected) const
211 : : {
212 [ + + ]: 142985 : if (typ != expected) {
213 [ + - + - : 54 : throw type_error{"JSON value of type " + std::string{uvTypeName(typ)} + " is not of expected type " +
+ - ]
214 [ + - + - : 108 : std::string{uvTypeName(expected)}};
+ - ]
215 : : }
216 : 142958 : }
217 : :
218 : 58 : const char *uvTypeName(UniValue::VType t)
219 : : {
220 [ + + + + : 58 : switch (t) {
+ - + ]
221 : : case UniValue::VNULL: return "null";
222 : 4 : case UniValue::VBOOL: return "bool";
223 : 10 : case UniValue::VOBJ: return "object";
224 : 9 : case UniValue::VARR: return "array";
225 : 15 : case UniValue::VSTR: return "string";
226 : 14 : case UniValue::VNUM: return "number";
227 : : }
228 : :
229 : : // not reached
230 : 0 : return nullptr;
231 : : }
232 : :
233 : 493 : const UniValue& UniValue::find_value(std::string_view key) const
234 : : {
235 [ + + ]: 1050 : for (unsigned int i = 0; i < keys.size(); ++i) {
236 [ + + ]: 1002 : if (keys[i] == key) {
237 : 445 : return values.at(i);
238 : : }
239 : : }
240 : : return NullUniValue;
241 : : }
242 : :
|