annotate C++/modules/Json/Json.cpp @ 445:f5e62f6c1475

Json: add escape free function
author David Demelier <markand@malikania.fr>
date Mon, 26 Oct 2015 19:29:34 +0100
parents d485d36a8de1
children 902b034df6e3
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
rev   line source
244
777bc3cb665a Add Json, a jansson wrapper
David Demelier <markand@malikania.fr>
parents:
diff changeset
1 /*
405
David Demelier <markand@malikania.fr>
parents: 404
diff changeset
2 * Json.cpp -- C++14 JSON manipulation using jansson parser
244
777bc3cb665a Add Json, a jansson wrapper
David Demelier <markand@malikania.fr>
parents:
diff changeset
3 *
403
d5ec1174b707 Massive cleanup
David Demelier <markand@malikania.fr>
parents: 334
diff changeset
4 * Copyright (c) 2013-2015 David Demelier <markand@malikania.fr>
244
777bc3cb665a Add Json, a jansson wrapper
David Demelier <markand@malikania.fr>
parents:
diff changeset
5 *
777bc3cb665a Add Json, a jansson wrapper
David Demelier <markand@malikania.fr>
parents:
diff changeset
6 * Permission to use, copy, modify, and/or distribute this software for any
777bc3cb665a Add Json, a jansson wrapper
David Demelier <markand@malikania.fr>
parents:
diff changeset
7 * purpose with or without fee is hereby granted, provided that the above
777bc3cb665a Add Json, a jansson wrapper
David Demelier <markand@malikania.fr>
parents:
diff changeset
8 * copyright notice and this permission notice appear in all copies.
777bc3cb665a Add Json, a jansson wrapper
David Demelier <markand@malikania.fr>
parents:
diff changeset
9 *
777bc3cb665a Add Json, a jansson wrapper
David Demelier <markand@malikania.fr>
parents:
diff changeset
10 * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
777bc3cb665a Add Json, a jansson wrapper
David Demelier <markand@malikania.fr>
parents:
diff changeset
11 * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
777bc3cb665a Add Json, a jansson wrapper
David Demelier <markand@malikania.fr>
parents:
diff changeset
12 * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
777bc3cb665a Add Json, a jansson wrapper
David Demelier <markand@malikania.fr>
parents:
diff changeset
13 * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
777bc3cb665a Add Json, a jansson wrapper
David Demelier <markand@malikania.fr>
parents:
diff changeset
14 * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
777bc3cb665a Add Json, a jansson wrapper
David Demelier <markand@malikania.fr>
parents:
diff changeset
15 * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
777bc3cb665a Add Json, a jansson wrapper
David Demelier <markand@malikania.fr>
parents:
diff changeset
16 * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
777bc3cb665a Add Json, a jansson wrapper
David Demelier <markand@malikania.fr>
parents:
diff changeset
17 */
777bc3cb665a Add Json, a jansson wrapper
David Demelier <markand@malikania.fr>
parents:
diff changeset
18
405
David Demelier <markand@malikania.fr>
parents: 404
diff changeset
19 #include "Json.h"
314
4c3019385769 Json: add iterators
David Demelier <markand@malikania.fr>
parents: 312
diff changeset
20
405
David Demelier <markand@malikania.fr>
parents: 404
diff changeset
21 #include <jansson.h>
244
777bc3cb665a Add Json, a jansson wrapper
David Demelier <markand@malikania.fr>
parents:
diff changeset
22
404
9ab878fb9fa2 Json: bring classes into json namespace
David Demelier <markand@malikania.fr>
parents: 403
diff changeset
23 namespace json {
9ab878fb9fa2 Json: bring classes into json namespace
David Demelier <markand@malikania.fr>
parents: 403
diff changeset
24
405
David Demelier <markand@malikania.fr>
parents: 404
diff changeset
25 namespace {
David Demelier <markand@malikania.fr>
parents: 404
diff changeset
26
David Demelier <markand@malikania.fr>
parents: 404
diff changeset
27 void readObject(Value &parent, json_t *object);
David Demelier <markand@malikania.fr>
parents: 404
diff changeset
28 void readArray(Value &parent, json_t *array);
David Demelier <markand@malikania.fr>
parents: 404
diff changeset
29
David Demelier <markand@malikania.fr>
parents: 404
diff changeset
30 Value readValue(json_t *v)
David Demelier <markand@malikania.fr>
parents: 404
diff changeset
31 {
David Demelier <markand@malikania.fr>
parents: 404
diff changeset
32 if (json_is_null(v)) {
David Demelier <markand@malikania.fr>
parents: 404
diff changeset
33 return Value{nullptr};
David Demelier <markand@malikania.fr>
parents: 404
diff changeset
34 }
David Demelier <markand@malikania.fr>
parents: 404
diff changeset
35 if (json_is_string(v)) {
David Demelier <markand@malikania.fr>
parents: 404
diff changeset
36 return Value{json_string_value(v)};
David Demelier <markand@malikania.fr>
parents: 404
diff changeset
37 }
406
David Demelier <markand@malikania.fr>
parents: 405
diff changeset
38 if (json_is_real(v)) {
405
David Demelier <markand@malikania.fr>
parents: 404
diff changeset
39 return Value{json_number_value(v)};
David Demelier <markand@malikania.fr>
parents: 404
diff changeset
40 }
406
David Demelier <markand@malikania.fr>
parents: 405
diff changeset
41 if (json_is_integer(v)) {
David Demelier <markand@malikania.fr>
parents: 405
diff changeset
42 return Value{static_cast<int>(json_integer_value(v))};
David Demelier <markand@malikania.fr>
parents: 405
diff changeset
43 }
405
David Demelier <markand@malikania.fr>
parents: 404
diff changeset
44 if (json_is_boolean(v)) {
David Demelier <markand@malikania.fr>
parents: 404
diff changeset
45 return Value{json_boolean_value(v)};
David Demelier <markand@malikania.fr>
parents: 404
diff changeset
46 }
David Demelier <markand@malikania.fr>
parents: 404
diff changeset
47 if (json_is_object(v)) {
David Demelier <markand@malikania.fr>
parents: 404
diff changeset
48 Object object;
David Demelier <markand@malikania.fr>
parents: 404
diff changeset
49
David Demelier <markand@malikania.fr>
parents: 404
diff changeset
50 readObject(object, v);
David Demelier <markand@malikania.fr>
parents: 404
diff changeset
51
David Demelier <markand@malikania.fr>
parents: 404
diff changeset
52 return object;
David Demelier <markand@malikania.fr>
parents: 404
diff changeset
53 }
David Demelier <markand@malikania.fr>
parents: 404
diff changeset
54 if (json_is_array(v)) {
David Demelier <markand@malikania.fr>
parents: 404
diff changeset
55 Array array;
David Demelier <markand@malikania.fr>
parents: 404
diff changeset
56
David Demelier <markand@malikania.fr>
parents: 404
diff changeset
57 readArray(array, v);
David Demelier <markand@malikania.fr>
parents: 404
diff changeset
58
David Demelier <markand@malikania.fr>
parents: 404
diff changeset
59 return array;
David Demelier <markand@malikania.fr>
parents: 404
diff changeset
60 }
David Demelier <markand@malikania.fr>
parents: 404
diff changeset
61
David Demelier <markand@malikania.fr>
parents: 404
diff changeset
62 return Value{};
David Demelier <markand@malikania.fr>
parents: 404
diff changeset
63 }
David Demelier <markand@malikania.fr>
parents: 404
diff changeset
64
David Demelier <markand@malikania.fr>
parents: 404
diff changeset
65 void readObject(Value &parent, json_t *object)
David Demelier <markand@malikania.fr>
parents: 404
diff changeset
66 {
David Demelier <markand@malikania.fr>
parents: 404
diff changeset
67 const char *key;
David Demelier <markand@malikania.fr>
parents: 404
diff changeset
68 json_t *value;
David Demelier <markand@malikania.fr>
parents: 404
diff changeset
69
David Demelier <markand@malikania.fr>
parents: 404
diff changeset
70 json_object_foreach(object, key, value) {
David Demelier <markand@malikania.fr>
parents: 404
diff changeset
71 static_cast<Object &>(parent).insert(key, readValue(value));
David Demelier <markand@malikania.fr>
parents: 404
diff changeset
72 }
David Demelier <markand@malikania.fr>
parents: 404
diff changeset
73 }
David Demelier <markand@malikania.fr>
parents: 404
diff changeset
74
David Demelier <markand@malikania.fr>
parents: 404
diff changeset
75 void readArray(Value &parent, json_t *array)
David Demelier <markand@malikania.fr>
parents: 404
diff changeset
76 {
David Demelier <markand@malikania.fr>
parents: 404
diff changeset
77 size_t index;
David Demelier <markand@malikania.fr>
parents: 404
diff changeset
78 json_t *value;
David Demelier <markand@malikania.fr>
parents: 404
diff changeset
79
David Demelier <markand@malikania.fr>
parents: 404
diff changeset
80 json_array_foreach(array, index, value) {
David Demelier <markand@malikania.fr>
parents: 404
diff changeset
81 static_cast<Array &>(parent).append(readValue(value));
David Demelier <markand@malikania.fr>
parents: 404
diff changeset
82 }
David Demelier <markand@malikania.fr>
parents: 404
diff changeset
83 }
David Demelier <markand@malikania.fr>
parents: 404
diff changeset
84
David Demelier <markand@malikania.fr>
parents: 404
diff changeset
85 template <typename Func, typename... Args>
David Demelier <markand@malikania.fr>
parents: 404
diff changeset
86 Value convert(Func fn, Args&&... args)
David Demelier <markand@malikania.fr>
parents: 404
diff changeset
87 {
David Demelier <markand@malikania.fr>
parents: 404
diff changeset
88 json_error_t error;
David Demelier <markand@malikania.fr>
parents: 404
diff changeset
89 json_t *json = fn(std::forward<Args>(args)..., &error);
David Demelier <markand@malikania.fr>
parents: 404
diff changeset
90
David Demelier <markand@malikania.fr>
parents: 404
diff changeset
91 if (json == nullptr) {
David Demelier <markand@malikania.fr>
parents: 404
diff changeset
92 throw Error{error.text, error.source, error.line, error.column, error.position};
David Demelier <markand@malikania.fr>
parents: 404
diff changeset
93 }
David Demelier <markand@malikania.fr>
parents: 404
diff changeset
94
David Demelier <markand@malikania.fr>
parents: 404
diff changeset
95 Value value;
David Demelier <markand@malikania.fr>
parents: 404
diff changeset
96
David Demelier <markand@malikania.fr>
parents: 404
diff changeset
97 if (json_is_object(json)) {
David Demelier <markand@malikania.fr>
parents: 404
diff changeset
98 value = Object{};
David Demelier <markand@malikania.fr>
parents: 404
diff changeset
99 readObject(value, json);
David Demelier <markand@malikania.fr>
parents: 404
diff changeset
100 } else {
David Demelier <markand@malikania.fr>
parents: 404
diff changeset
101 value = Array{};
David Demelier <markand@malikania.fr>
parents: 404
diff changeset
102 readArray(value, json);
David Demelier <markand@malikania.fr>
parents: 404
diff changeset
103 }
David Demelier <markand@malikania.fr>
parents: 404
diff changeset
104
David Demelier <markand@malikania.fr>
parents: 404
diff changeset
105 json_decref(json);
David Demelier <markand@malikania.fr>
parents: 404
diff changeset
106
David Demelier <markand@malikania.fr>
parents: 404
diff changeset
107 return value;
David Demelier <markand@malikania.fr>
parents: 404
diff changeset
108 }
David Demelier <markand@malikania.fr>
parents: 404
diff changeset
109
David Demelier <markand@malikania.fr>
parents: 404
diff changeset
110 } // !namespace
David Demelier <markand@malikania.fr>
parents: 404
diff changeset
111
David Demelier <markand@malikania.fr>
parents: 404
diff changeset
112 bool Value::toBool() const noexcept
David Demelier <markand@malikania.fr>
parents: 404
diff changeset
113 {
406
David Demelier <markand@malikania.fr>
parents: 405
diff changeset
114 if (m_type != Type::Boolean) {
405
David Demelier <markand@malikania.fr>
parents: 404
diff changeset
115 return false;
406
David Demelier <markand@malikania.fr>
parents: 405
diff changeset
116 }
405
David Demelier <markand@malikania.fr>
parents: 404
diff changeset
117
David Demelier <markand@malikania.fr>
parents: 404
diff changeset
118 return m_boolean;
David Demelier <markand@malikania.fr>
parents: 404
diff changeset
119 }
David Demelier <markand@malikania.fr>
parents: 404
diff changeset
120
406
David Demelier <markand@malikania.fr>
parents: 405
diff changeset
121 double Value::toReal() const noexcept
405
David Demelier <markand@malikania.fr>
parents: 404
diff changeset
122 {
406
David Demelier <markand@malikania.fr>
parents: 405
diff changeset
123 if (m_type != Type::Real) {
405
David Demelier <markand@malikania.fr>
parents: 404
diff changeset
124 return 0;
406
David Demelier <markand@malikania.fr>
parents: 405
diff changeset
125 }
405
David Demelier <markand@malikania.fr>
parents: 404
diff changeset
126
David Demelier <markand@malikania.fr>
parents: 404
diff changeset
127 return m_number;
David Demelier <markand@malikania.fr>
parents: 404
diff changeset
128 }
David Demelier <markand@malikania.fr>
parents: 404
diff changeset
129
406
David Demelier <markand@malikania.fr>
parents: 405
diff changeset
130 int Value::toInt() const noexcept
David Demelier <markand@malikania.fr>
parents: 405
diff changeset
131 {
David Demelier <markand@malikania.fr>
parents: 405
diff changeset
132 if (m_type != Type::Int) {
David Demelier <markand@malikania.fr>
parents: 405
diff changeset
133 return 0;
David Demelier <markand@malikania.fr>
parents: 405
diff changeset
134 }
David Demelier <markand@malikania.fr>
parents: 405
diff changeset
135
David Demelier <markand@malikania.fr>
parents: 405
diff changeset
136 return m_integer;
David Demelier <markand@malikania.fr>
parents: 405
diff changeset
137 }
David Demelier <markand@malikania.fr>
parents: 405
diff changeset
138
405
David Demelier <markand@malikania.fr>
parents: 404
diff changeset
139 std::string Value::toString() const noexcept
David Demelier <markand@malikania.fr>
parents: 404
diff changeset
140 {
David Demelier <markand@malikania.fr>
parents: 404
diff changeset
141 if (m_type != Type::String) {
David Demelier <markand@malikania.fr>
parents: 404
diff changeset
142 return "";
David Demelier <markand@malikania.fr>
parents: 404
diff changeset
143 }
David Demelier <markand@malikania.fr>
parents: 404
diff changeset
144
David Demelier <markand@malikania.fr>
parents: 404
diff changeset
145 return m_string;
David Demelier <markand@malikania.fr>
parents: 404
diff changeset
146 }
244
777bc3cb665a Add Json, a jansson wrapper
David Demelier <markand@malikania.fr>
parents:
diff changeset
147
404
9ab878fb9fa2 Json: bring classes into json namespace
David Demelier <markand@malikania.fr>
parents: 403
diff changeset
148 Object Value::toObject() const noexcept
244
777bc3cb665a Add Json, a jansson wrapper
David Demelier <markand@malikania.fr>
parents:
diff changeset
149 {
405
David Demelier <markand@malikania.fr>
parents: 404
diff changeset
150 if (m_type != Type::Object) {
David Demelier <markand@malikania.fr>
parents: 404
diff changeset
151 return Object{};
David Demelier <markand@malikania.fr>
parents: 404
diff changeset
152 }
311
David Demelier <markand@malikania.fr>
parents: 244
diff changeset
153
405
David Demelier <markand@malikania.fr>
parents: 404
diff changeset
154 return Object(*this);
311
David Demelier <markand@malikania.fr>
parents: 244
diff changeset
155 }
David Demelier <markand@malikania.fr>
parents: 244
diff changeset
156
404
9ab878fb9fa2 Json: bring classes into json namespace
David Demelier <markand@malikania.fr>
parents: 403
diff changeset
157 Array Value::toArray() const noexcept
311
David Demelier <markand@malikania.fr>
parents: 244
diff changeset
158 {
405
David Demelier <markand@malikania.fr>
parents: 404
diff changeset
159 if (m_type != Type::Array) {
David Demelier <markand@malikania.fr>
parents: 404
diff changeset
160 return Array{};
David Demelier <markand@malikania.fr>
parents: 404
diff changeset
161 }
314
4c3019385769 Json: add iterators
David Demelier <markand@malikania.fr>
parents: 312
diff changeset
162
405
David Demelier <markand@malikania.fr>
parents: 404
diff changeset
163 return Array(*this);
311
David Demelier <markand@malikania.fr>
parents: 244
diff changeset
164 }
David Demelier <markand@malikania.fr>
parents: 244
diff changeset
165
405
David Demelier <markand@malikania.fr>
parents: 404
diff changeset
166 Document::Document(Buffer buffer)
311
David Demelier <markand@malikania.fr>
parents: 244
diff changeset
167 {
405
David Demelier <markand@malikania.fr>
parents: 404
diff changeset
168 m_value = convert(json_loads, buffer.text.c_str(), 0);
244
777bc3cb665a Add Json, a jansson wrapper
David Demelier <markand@malikania.fr>
parents:
diff changeset
169 }
777bc3cb665a Add Json, a jansson wrapper
David Demelier <markand@malikania.fr>
parents:
diff changeset
170
405
David Demelier <markand@malikania.fr>
parents: 404
diff changeset
171 Document::Document(File file)
314
4c3019385769 Json: add iterators
David Demelier <markand@malikania.fr>
parents: 312
diff changeset
172 {
405
David Demelier <markand@malikania.fr>
parents: 404
diff changeset
173 m_value = convert(json_load_file, file.path.c_str(), 0);
404
9ab878fb9fa2 Json: bring classes into json namespace
David Demelier <markand@malikania.fr>
parents: 403
diff changeset
174 }
9ab878fb9fa2 Json: bring classes into json namespace
David Demelier <markand@malikania.fr>
parents: 403
diff changeset
175
445
f5e62f6c1475 Json: add escape free function
David Demelier <markand@malikania.fr>
parents: 406
diff changeset
176 std::string escape(std::string value) noexcept
f5e62f6c1475 Json: add escape free function
David Demelier <markand@malikania.fr>
parents: 406
diff changeset
177 {
f5e62f6c1475 Json: add escape free function
David Demelier <markand@malikania.fr>
parents: 406
diff changeset
178 for (auto it = value.begin(); it != value.end(); ++it) {
f5e62f6c1475 Json: add escape free function
David Demelier <markand@malikania.fr>
parents: 406
diff changeset
179 switch (*it) {
f5e62f6c1475 Json: add escape free function
David Demelier <markand@malikania.fr>
parents: 406
diff changeset
180 case '\\':
f5e62f6c1475 Json: add escape free function
David Demelier <markand@malikania.fr>
parents: 406
diff changeset
181 case '/':
f5e62f6c1475 Json: add escape free function
David Demelier <markand@malikania.fr>
parents: 406
diff changeset
182 case '"':
f5e62f6c1475 Json: add escape free function
David Demelier <markand@malikania.fr>
parents: 406
diff changeset
183 it = value.insert(it, '\\');
f5e62f6c1475 Json: add escape free function
David Demelier <markand@malikania.fr>
parents: 406
diff changeset
184 it++;
f5e62f6c1475 Json: add escape free function
David Demelier <markand@malikania.fr>
parents: 406
diff changeset
185 break;
f5e62f6c1475 Json: add escape free function
David Demelier <markand@malikania.fr>
parents: 406
diff changeset
186 case '\b':
f5e62f6c1475 Json: add escape free function
David Demelier <markand@malikania.fr>
parents: 406
diff changeset
187 value.replace(it, it + 1, "\\b");
f5e62f6c1475 Json: add escape free function
David Demelier <markand@malikania.fr>
parents: 406
diff changeset
188 it += 1;
f5e62f6c1475 Json: add escape free function
David Demelier <markand@malikania.fr>
parents: 406
diff changeset
189 break;
f5e62f6c1475 Json: add escape free function
David Demelier <markand@malikania.fr>
parents: 406
diff changeset
190 case '\f':
f5e62f6c1475 Json: add escape free function
David Demelier <markand@malikania.fr>
parents: 406
diff changeset
191 value.replace(it, it + 1, "\\f");
f5e62f6c1475 Json: add escape free function
David Demelier <markand@malikania.fr>
parents: 406
diff changeset
192 it += 1;
f5e62f6c1475 Json: add escape free function
David Demelier <markand@malikania.fr>
parents: 406
diff changeset
193 break;
f5e62f6c1475 Json: add escape free function
David Demelier <markand@malikania.fr>
parents: 406
diff changeset
194 case '\n':
f5e62f6c1475 Json: add escape free function
David Demelier <markand@malikania.fr>
parents: 406
diff changeset
195 value.replace(it, it + 1, "\\n");
f5e62f6c1475 Json: add escape free function
David Demelier <markand@malikania.fr>
parents: 406
diff changeset
196 it += 1;
f5e62f6c1475 Json: add escape free function
David Demelier <markand@malikania.fr>
parents: 406
diff changeset
197 break;
f5e62f6c1475 Json: add escape free function
David Demelier <markand@malikania.fr>
parents: 406
diff changeset
198 case '\r':
f5e62f6c1475 Json: add escape free function
David Demelier <markand@malikania.fr>
parents: 406
diff changeset
199 value.replace(it, it + 1, "\\r");
f5e62f6c1475 Json: add escape free function
David Demelier <markand@malikania.fr>
parents: 406
diff changeset
200 it += 1;
f5e62f6c1475 Json: add escape free function
David Demelier <markand@malikania.fr>
parents: 406
diff changeset
201 break;
f5e62f6c1475 Json: add escape free function
David Demelier <markand@malikania.fr>
parents: 406
diff changeset
202 case '\t':
f5e62f6c1475 Json: add escape free function
David Demelier <markand@malikania.fr>
parents: 406
diff changeset
203 value.replace(it, it + 1, "\\t");
f5e62f6c1475 Json: add escape free function
David Demelier <markand@malikania.fr>
parents: 406
diff changeset
204 it += 1;
f5e62f6c1475 Json: add escape free function
David Demelier <markand@malikania.fr>
parents: 406
diff changeset
205 break;
f5e62f6c1475 Json: add escape free function
David Demelier <markand@malikania.fr>
parents: 406
diff changeset
206 default:
f5e62f6c1475 Json: add escape free function
David Demelier <markand@malikania.fr>
parents: 406
diff changeset
207 break;
f5e62f6c1475 Json: add escape free function
David Demelier <markand@malikania.fr>
parents: 406
diff changeset
208 }
f5e62f6c1475 Json: add escape free function
David Demelier <markand@malikania.fr>
parents: 406
diff changeset
209 }
f5e62f6c1475 Json: add escape free function
David Demelier <markand@malikania.fr>
parents: 406
diff changeset
210
f5e62f6c1475 Json: add escape free function
David Demelier <markand@malikania.fr>
parents: 406
diff changeset
211 return value;
f5e62f6c1475 Json: add escape free function
David Demelier <markand@malikania.fr>
parents: 406
diff changeset
212 }
f5e62f6c1475 Json: add escape free function
David Demelier <markand@malikania.fr>
parents: 406
diff changeset
213
406
David Demelier <markand@malikania.fr>
parents: 405
diff changeset
214 } // !json