Coverage Report

Created: 2024-10-29 12:10

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