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 : 8400592 : void UniValue::clear()
19 : : {
20 : 8400592 : typ = VNULL;
21 : 8400592 : val.clear();
22 : 8400592 : keys.clear();
23 : 8400592 : values.clear();
24 : 8400592 : }
25 : :
26 : 0 : void UniValue::setNull()
27 : : {
28 : 0 : clear();
29 : 0 : }
30 : :
31 : 54758 : void UniValue::setBool(bool val_)
32 : : {
33 : 54758 : clear();
34 : 54758 : typ = VBOOL;
35 [ + + ]: 54758 : if (val_)
36 : 24151 : val = "1";
37 : 54758 : }
38 : :
39 : 3139188 : static bool validNumStr(const std::string& s)
40 : : {
41 [ + - ]: 3139188 : std::string tokenVal;
42 : 3139188 : unsigned int consumed;
43 [ + - ]: 3139188 : enum jtokentype tt = getJsonToken(tokenVal, consumed, s.data(), s.data() + s.size());
44 : 3139188 : return (tt == JTOK_NUMBER);
45 : 3139188 : }
46 : :
47 : 3139188 : void UniValue::setNumStr(std::string str)
48 : : {
49 [ - + ]: 3139188 : if (!validNumStr(str)) {
50 [ # # # # ]: 0 : throw std::runtime_error{"The string '" + str + "' is not a valid JSON number"};
51 : : }
52 : :
53 : 3139188 : clear();
54 : 3139188 : typ = VNUM;
55 : 3139188 : val = std::move(str);
56 : 3139188 : }
57 : :
58 : 19820 : void UniValue::setInt(uint64_t val_)
59 : : {
60 : 19820 : std::ostringstream oss;
61 : :
62 [ + - ]: 19820 : oss << val_;
63 : :
64 [ + - + - ]: 39640 : return setNumStr(oss.str());
65 : 19820 : }
66 : :
67 : 3119303 : void UniValue::setInt(int64_t val_)
68 : : {
69 : 3119303 : std::ostringstream oss;
70 : :
71 [ + - ]: 3119303 : oss << val_;
72 : :
73 [ + - + - ]: 6238606 : return setNumStr(oss.str());
74 : 3119303 : }
75 : :
76 : 65 : void UniValue::setFloat(double val_)
77 : : {
78 : 65 : std::ostringstream oss;
79 : :
80 [ + - ]: 65 : oss << std::setprecision(16) << val_;
81 : :
82 [ + - + - ]: 130 : return setNumStr(oss.str());
83 : 65 : }
84 : :
85 : 5188578 : void UniValue::setStr(std::string str)
86 : : {
87 : 5188578 : clear();
88 : 5188578 : typ = VSTR;
89 : 5188578 : val = std::move(str);
90 : 5188578 : }
91 : :
92 : 4115 : void UniValue::setArray()
93 : : {
94 : 4115 : clear();
95 : 4115 : typ = VARR;
96 : 4115 : }
97 : :
98 : 2241 : void UniValue::setObject()
99 : : {
100 : 2241 : clear();
101 : 2241 : typ = VOBJ;
102 : 2241 : }
103 : :
104 : 2096842 : void UniValue::push_back(UniValue val)
105 : : {
106 : 2096842 : checkType(VARR);
107 : :
108 : 2096842 : values.push_back(std::move(val));
109 : 2096842 : }
110 : :
111 : 0 : void UniValue::push_backV(const std::vector<UniValue>& vec)
112 : : {
113 : 0 : checkType(VARR);
114 : :
115 : 0 : values.insert(values.end(), vec.begin(), vec.end());
116 : 0 : }
117 : :
118 : 9587192 : void UniValue::pushKVEnd(std::string key, UniValue val)
119 : : {
120 : 9587192 : checkType(VOBJ);
121 : :
122 : 9587192 : keys.push_back(std::move(key));
123 : 9587192 : values.push_back(std::move(val));
124 : 9587192 : }
125 : :
126 : 9565818 : void UniValue::pushKV(std::string key, UniValue val)
127 : : {
128 : 9565818 : checkType(VOBJ);
129 : :
130 : 9565818 : size_t idx;
131 [ + + ]: 9565818 : if (findKey(key, idx))
132 : 508 : values[idx] = std::move(val);
133 : : else
134 [ + - ]: 19130620 : pushKVEnd(std::move(key), std::move(val));
135 : 9565818 : }
136 : :
137 : 8 : void UniValue::pushKVs(UniValue obj)
138 : : {
139 : 8 : checkType(VOBJ);
140 : 8 : obj.checkType(VOBJ);
141 : :
142 [ + + ]: 32 : for (size_t i = 0; i < obj.keys.size(); i++)
143 [ + - + - ]: 48 : pushKVEnd(std::move(obj.keys.at(i)), std::move(obj.values.at(i)));
144 : 8 : }
145 : :
146 : 0 : void UniValue::getObjMap(std::map<std::string,UniValue>& kv) const
147 : : {
148 [ # # ]: 0 : if (typ != VOBJ)
149 : : return;
150 : :
151 : 0 : kv.clear();
152 [ # # ]: 0 : for (size_t i = 0; i < keys.size(); i++)
153 : 0 : kv[keys[i]] = values[i];
154 : : }
155 : :
156 : 9637066 : bool UniValue::findKey(const std::string& key, size_t& retIdx) const
157 : : {
158 [ + + ]: 23267075 : for (size_t i = 0; i < keys.size(); i++) {
159 [ + + ]: 13701533 : if (keys[i] == key) {
160 : 71524 : retIdx = i;
161 : 71524 : return true;
162 : : }
163 : : }
164 : :
165 : : return false;
166 : : }
167 : :
168 : 0 : bool UniValue::checkObject(const std::map<std::string,UniValue::VType>& t) const
169 : : {
170 [ # # ]: 0 : if (typ != VOBJ) {
171 : : return false;
172 : : }
173 : :
174 [ # # ]: 0 : for (const auto& object: t) {
175 : 0 : size_t idx = 0;
176 [ # # ]: 0 : if (!findKey(object.first, idx)) {
177 : : return false;
178 : : }
179 : :
180 [ # # ]: 0 : if (values.at(idx).getType() != object.second) {
181 : : return false;
182 : : }
183 : : }
184 : :
185 : : return true;
186 : : }
187 : :
188 : 71114 : const UniValue& UniValue::operator[](const std::string& key) const
189 : : {
190 [ + + ]: 71114 : if (typ != VOBJ)
191 : : return NullUniValue;
192 : :
193 : 71106 : size_t index = 0;
194 [ + + ]: 71106 : if (!findKey(key, index))
195 : : return NullUniValue;
196 : :
197 : 71016 : return values.at(index);
198 : : }
199 : :
200 : 45928 : const UniValue& UniValue::operator[](size_t index) const
201 : : {
202 [ + - ]: 45928 : if (typ != VOBJ && typ != VARR)
203 : : return NullUniValue;
204 [ + + ]: 45928 : if (index >= values.size())
205 : : return NullUniValue;
206 : :
207 : 39626 : return values.at(index);
208 : : }
209 : :
210 : 21426172 : void UniValue::checkType(const VType& expected) const
211 : : {
212 [ + + ]: 21426172 : if (typ != expected) {
213 [ + - + - : 13878 : throw type_error{"JSON value of type " + std::string{uvTypeName(typ)} + " is not of expected type " +
+ - ]
214 [ + - + - : 27756 : std::string{uvTypeName(expected)}};
+ - ]
215 : : }
216 : 21419233 : }
217 : :
218 : 14014 : const char *uvTypeName(UniValue::VType t)
219 : : {
220 [ + + + + : 14014 : switch (t) {
+ - + ]
221 : : case UniValue::VNULL: return "null";
222 : 41 : case UniValue::VBOOL: return "bool";
223 : 166 : case UniValue::VOBJ: return "object";
224 : 206 : case UniValue::VARR: return "array";
225 : 6794 : case UniValue::VSTR: return "string";
226 : 2374 : case UniValue::VNUM: return "number";
227 : : }
228 : :
229 : : // not reached
230 : 0 : return nullptr;
231 : : }
232 : :
233 : 11782 : const UniValue& UniValue::find_value(std::string_view key) const
234 : : {
235 [ + + ]: 104003 : for (unsigned int i = 0; i < keys.size(); ++i) {
236 [ + + ]: 95201 : if (keys[i] == key) {
237 : 2980 : return values.at(i);
238 : : }
239 : : }
240 : : return NullUniValue;
241 : : }
242 : :
|