-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathqjsonpath.cpp
159 lines (149 loc) · 4.53 KB
/
qjsonpath.cpp
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
#include "qjsonpath.h"
#include <QJsonDocument>
#include <QJsonValue>
#include <QJsonArray>
#include <QJsonObject>
#include <QString>
#include <QRegExp>
#include <stack>
#include <memory>
namespace QJsonPath {
std::pair<int, QString> getKey(const QString &path, int lpos)
{
if (path.at(lpos) != '.')
throw std::logic_error("Path parsing error: getIndex, should be used with position of dot");
int pos = path.indexOf(QRegExp("[\\[\\]\\.]"), ++lpos);
if (pos == -1)
pos = path.size();
int size = pos - lpos;
if (size <= 0)
throw std::logic_error("Parsing error: zero-length index");
return {pos, path.mid(lpos, size)};
}
std::pair<int, QString> getIndex(const QString &path, int lpos)
{
if (path.at(lpos) != '[')
throw std::logic_error("Path parsing error: getIndex, should be used with position of left bracket");
int pos = path.indexOf(QRegExp("[\\[\\]\\.]"), ++lpos);
if (pos == -1)
throw std::logic_error("Path parsing error: unclosed brackets - end of path encountered");
if (path[pos] != ']')
throw std::logic_error("Parsing error: unexpected closing bracket");
int size = pos - lpos;
if (size <= 0)
throw std::logic_error("Parsing error: zero-length index");
return {++pos, path.mid(lpos, size)};
}
QJsonValue getValue(const QJsonValue &root, const QString &path)
{
QJsonValue new_root = root;
int pos = 0;
QString key;
while (pos != path.size()) {
switch (path.at(pos).toLatin1())
{
case '.':
std::tie(pos, key) = getKey(path, pos);
new_root = new_root[key];
break;
case '[':
std::tie(pos, key) = getIndex(path, pos);
new_root = new_root[key.toInt()];
break;
default:
throw std::logic_error("Parsing error: expected key or index");
}
}
return new_root;
}
QJsonValue getValue(const QJsonDocument &root, const QString &path)
{
QJsonValue new_root;
if (path.startsWith('['))
{
if (not root.isArray())
throw std::logic_error("Parsing error: expected array, but document is not an array");
new_root = root.array();
}
else if (path.startsWith('.'))
{
if (not root.isObject())
throw std::logic_error("Parsing error: expected object, but document is not an object");
new_root = root.object();
}
else {
return new_root;
}
return getValue(new_root, path);
}
void setValue(QJsonValue &root, const QString &path, const QJsonValue &value)
{
std::stack<QJsonValue> nodes;
nodes.push(root);
std::stack<QString> keys {};
int pos = 0;
QString key;
while (pos != path.size()) {
switch (path.at(pos).toLatin1())
{
case '.':
std::tie(pos, key) = getKey(path, pos);
nodes.push(nodes.top()[key]);
keys.push(key);
break;
case '[':
std::tie(pos, key) = getIndex(path, pos);
nodes.push(nodes.top()[key.toInt()]);
keys.push(key);
break;
default:
throw std::logic_error("Parsing error: expected key or index");
}
}
QJsonValue node = value;
nodes.pop();
while (nodes.size() >= 1)
{
if (nodes.top().isArray())
{
QJsonArray array = nodes.top().toArray();
array[keys.top().toInt()] = node;
node = array;
}
else if (nodes.top().isObject())
{
QJsonObject object = nodes.top().toObject();
object[keys.top()] = node;
node = object;
}
nodes.pop();
keys.pop();
}
root = node;
}
void setValue(QJsonDocument &root, const QString &path, const QJsonValue &value)
{
QJsonValue new_root;
if (path.startsWith('['))
{
if (not root.isArray())
throw std::logic_error("Parsing error: expected array, but document is not an array");
new_root = root.array();
}
else if (path.startsWith('.'))
{
if (not root.isObject())
throw std::logic_error("Parsing error: expected object, but document is not an object");
new_root = root.object();
}
else {
new_root = value;
}
setValue(new_root, path, value);
if (new_root.isArray())
root = QJsonDocument(new_root.toArray());
else
if (new_root.isObject())
root = QJsonDocument(new_root.toObject());
}
}