Line data Source code
1 : /******************************************************************************
2 : *
3 : * Project: CPL - Common Portability Library
4 : * Purpose: JSon streaming writer
5 : * Author: Even Rouault, even.rouault at spatialys.com
6 : *
7 : ******************************************************************************
8 : * Copyright (c) 2019, Even Rouault <even.rouault at spatialys.com>
9 : *
10 : * SPDX-License-Identifier: MIT
11 : ****************************************************************************/
12 :
13 : /*! @cond Doxygen_Suppress */
14 :
15 : #include <cmath>
16 : #include <vector>
17 : #include <string>
18 :
19 : #include "cpl_conv.h"
20 : #include "cpl_float.h"
21 : #include "cpl_string.h"
22 : #include "cpl_json_streaming_writer.h"
23 :
24 277 : CPLJSonStreamingWriter::CPLJSonStreamingWriter(
25 277 : SerializationFuncType pfnSerializationFunc, void *pUserData)
26 277 : : m_pfnSerializationFunc(pfnSerializationFunc), m_pUserData(pUserData)
27 : {
28 277 : }
29 :
30 277 : CPLJSonStreamingWriter::~CPLJSonStreamingWriter()
31 : {
32 277 : CPLAssert(m_nLevel == 0);
33 277 : CPLAssert(m_states.empty());
34 277 : }
35 :
36 200 : void CPLJSonStreamingWriter::clear()
37 : {
38 200 : m_nLevel = 0;
39 200 : m_osStr.clear();
40 200 : m_osIndentAcc.clear();
41 200 : m_states.clear();
42 200 : m_bWaitForValue = false;
43 200 : }
44 :
45 22103 : void CPLJSonStreamingWriter::Serialize(const std::string_view &str)
46 : {
47 22103 : if (m_pfnSerializationFunc)
48 : {
49 5274 : m_osTmpForSerialize = str;
50 5274 : m_pfnSerializationFunc(m_osTmpForSerialize.c_str(), m_pUserData);
51 : }
52 : else
53 : {
54 16829 : m_osStr.append(str);
55 : }
56 22103 : }
57 :
58 17026 : void CPLJSonStreamingWriter::Serialize(const char *pszStr, size_t nLength)
59 : {
60 17026 : Serialize(std::string_view(pszStr, nLength));
61 17026 : }
62 :
63 0 : void CPLJSonStreamingWriter::SetIndentationSize(int nSpaces)
64 : {
65 0 : CPLAssert(m_nLevel == 0);
66 0 : m_osIndent.clear();
67 0 : m_osIndent.resize(nSpaces, ' ');
68 0 : }
69 :
70 1376 : void CPLJSonStreamingWriter::IncIndent()
71 : {
72 1376 : m_nLevel++;
73 1376 : if (m_bPretty)
74 1242 : m_osIndentAcc += m_osIndent;
75 1376 : }
76 :
77 1366 : void CPLJSonStreamingWriter::DecIndent()
78 : {
79 1366 : CPLAssert(m_nLevel > 0);
80 1366 : m_nLevel--;
81 1366 : if (m_bPretty)
82 1232 : m_osIndentAcc.resize(m_osIndentAcc.size() - m_osIndent.size());
83 1366 : }
84 :
85 : const std::string &
86 4300 : CPLJSonStreamingWriter::FormatString(const std::string_view &str)
87 : {
88 4300 : m_osTmpForFormatString.clear();
89 4300 : m_osTmpForFormatString += '"';
90 61909 : for (char ch : str)
91 : {
92 57609 : switch (ch)
93 : {
94 884 : case '"':
95 884 : m_osTmpForFormatString += "\\\"";
96 884 : break;
97 2 : case '\\':
98 2 : m_osTmpForFormatString += "\\\\";
99 2 : break;
100 2 : case '\b':
101 2 : m_osTmpForFormatString += "\\b";
102 2 : break;
103 2 : case '\f':
104 2 : m_osTmpForFormatString += "\\f";
105 2 : break;
106 165 : case '\n':
107 165 : m_osTmpForFormatString += "\\n";
108 165 : break;
109 2 : case '\r':
110 2 : m_osTmpForFormatString += "\\r";
111 2 : break;
112 2 : case '\t':
113 2 : m_osTmpForFormatString += "\\t";
114 2 : break;
115 56550 : default:
116 56550 : if (static_cast<unsigned char>(ch) < ' ')
117 2 : m_osTmpForFormatString += CPLSPrintf("\\u%04X", ch);
118 : else
119 56548 : m_osTmpForFormatString += ch;
120 56550 : break;
121 : }
122 : }
123 4300 : m_osTmpForFormatString += '"';
124 4300 : return m_osTmpForFormatString;
125 : }
126 :
127 6535 : void CPLJSonStreamingWriter::EmitCommaIfNeeded()
128 : {
129 6535 : if (m_bWaitForValue)
130 : {
131 2567 : m_bWaitForValue = false;
132 : }
133 3968 : else if (!m_states.empty())
134 : {
135 3773 : if (!m_states.back().bFirstChild)
136 : {
137 2443 : Serialize(",", 1);
138 2443 : if (m_bPretty && !m_bNewLineEnabled)
139 132 : Serialize(" ", 1);
140 : }
141 3773 : if (m_bPretty && m_bNewLineEnabled)
142 : {
143 3381 : Serialize("\n", 1);
144 3381 : Serialize(m_osIndentAcc.c_str(), m_osIndentAcc.size());
145 : }
146 3773 : m_states.back().bFirstChild = false;
147 : }
148 6535 : }
149 :
150 870 : void CPLJSonStreamingWriter::StartObj()
151 : {
152 870 : EmitCommaIfNeeded();
153 870 : Serialize("{", 1);
154 870 : IncIndent();
155 870 : m_states.emplace_back(State(true));
156 870 : }
157 :
158 860 : void CPLJSonStreamingWriter::EndObj()
159 : {
160 860 : CPLAssert(!m_bWaitForValue);
161 860 : CPLAssert(!m_states.empty());
162 860 : CPLAssert(m_states.back().bIsObj);
163 860 : DecIndent();
164 860 : if (!m_states.back().bFirstChild)
165 : {
166 832 : if (m_bPretty && m_bNewLineEnabled)
167 : {
168 737 : Serialize("\n", 1);
169 737 : Serialize(m_osIndentAcc.c_str(), m_osIndentAcc.size());
170 : }
171 : }
172 860 : m_states.pop_back();
173 860 : Serialize("}", 1);
174 860 : }
175 :
176 506 : void CPLJSonStreamingWriter::StartArray()
177 : {
178 506 : EmitCommaIfNeeded();
179 506 : Serialize("[", 1);
180 506 : IncIndent();
181 506 : m_states.emplace_back(State(false));
182 506 : }
183 :
184 506 : void CPLJSonStreamingWriter::EndArray()
185 : {
186 506 : CPLAssert(!m_states.empty());
187 506 : CPLAssert(!m_states.back().bIsObj);
188 506 : DecIndent();
189 506 : if (!m_states.back().bFirstChild)
190 : {
191 488 : if (m_bPretty && m_bNewLineEnabled)
192 : {
193 412 : Serialize("\n", 1);
194 412 : Serialize(m_osIndentAcc.c_str(), m_osIndentAcc.size());
195 : }
196 : }
197 506 : m_states.pop_back();
198 506 : Serialize("]", 1);
199 506 : }
200 :
201 2567 : void CPLJSonStreamingWriter::AddObjKey(const std::string_view &key)
202 : {
203 2567 : CPLAssert(!m_states.empty());
204 2567 : CPLAssert(m_states.back().bIsObj);
205 2567 : CPLAssert(!m_bWaitForValue);
206 2567 : EmitCommaIfNeeded();
207 2567 : Serialize(FormatString(key));
208 2567 : if (m_bPretty)
209 2402 : Serialize(": ", 2);
210 : else
211 165 : Serialize(":", 1);
212 2567 : m_bWaitForValue = true;
213 2567 : }
214 :
215 7 : void CPLJSonStreamingWriter::Add(bool bVal)
216 : {
217 7 : EmitCommaIfNeeded();
218 7 : Serialize(bVal ? "true" : "false", bVal ? 4 : 5);
219 7 : }
220 :
221 559 : void CPLJSonStreamingWriter::Add(const char *pszStr)
222 : {
223 559 : EmitCommaIfNeeded();
224 559 : Serialize(FormatString(std::string_view(pszStr)));
225 559 : }
226 :
227 25 : void CPLJSonStreamingWriter::Add(const std::string_view &str)
228 : {
229 25 : EmitCommaIfNeeded();
230 25 : Serialize(FormatString(str));
231 25 : }
232 :
233 1149 : void CPLJSonStreamingWriter::Add(const std::string &str)
234 : {
235 1149 : EmitCommaIfNeeded();
236 1149 : Serialize(FormatString(str));
237 1149 : }
238 :
239 135 : void CPLJSonStreamingWriter::AddSerializedValue(const std::string_view &str)
240 : {
241 135 : EmitCommaIfNeeded();
242 135 : Serialize(str);
243 135 : }
244 :
245 180 : void CPLJSonStreamingWriter::Add(std::int64_t nVal)
246 : {
247 180 : EmitCommaIfNeeded();
248 180 : Serialize(CPLSPrintf(CPL_FRMT_GIB, static_cast<GIntBig>(nVal)));
249 180 : }
250 :
251 393 : void CPLJSonStreamingWriter::Add(std::uint64_t nVal)
252 : {
253 393 : EmitCommaIfNeeded();
254 393 : Serialize(CPLSPrintf(CPL_FRMT_GUIB, static_cast<GUIntBig>(nVal)));
255 393 : }
256 :
257 0 : void CPLJSonStreamingWriter::Add(GFloat16 hfVal, int nPrecision)
258 : {
259 0 : EmitCommaIfNeeded();
260 0 : if (CPLIsNan(hfVal))
261 : {
262 0 : Serialize("\"NaN\"", 5);
263 : }
264 0 : else if (CPLIsInf(hfVal))
265 : {
266 0 : Serialize(hfVal > 0 ? "\"Infinity\"" : "\"-Infinity\"",
267 0 : hfVal > 0 ? 10 : 11);
268 : }
269 : else
270 : {
271 : char szFormatting[10];
272 0 : snprintf(szFormatting, sizeof(szFormatting), "%%.%dg", nPrecision);
273 0 : Serialize(CPLSPrintf(szFormatting, double(hfVal)));
274 : }
275 0 : }
276 :
277 26 : void CPLJSonStreamingWriter::Add(float fVal, int nPrecision)
278 : {
279 26 : EmitCommaIfNeeded();
280 26 : if (std::isnan(fVal))
281 : {
282 11 : Serialize("\"NaN\"", 5);
283 : }
284 15 : else if (std::isinf(fVal))
285 : {
286 2 : Serialize(fVal > 0 ? "\"Infinity\"" : "\"-Infinity\"",
287 : fVal > 0 ? 10 : 11);
288 : }
289 : else
290 : {
291 : char szFormatting[10];
292 13 : snprintf(szFormatting, sizeof(szFormatting), "%%.%dg", nPrecision);
293 13 : Serialize(CPLSPrintf(szFormatting, static_cast<double>(fVal)));
294 : }
295 26 : }
296 :
297 91 : void CPLJSonStreamingWriter::Add(double dfVal, int nPrecision)
298 : {
299 91 : EmitCommaIfNeeded();
300 91 : if (std::isnan(dfVal))
301 : {
302 35 : Serialize("\"NaN\"", 5);
303 : }
304 56 : else if (std::isinf(dfVal))
305 : {
306 2 : Serialize(dfVal > 0 ? "\"Infinity\"" : "\"-Infinity\"");
307 : }
308 : else
309 : {
310 : char szFormatting[10];
311 54 : snprintf(szFormatting, sizeof(szFormatting), "%%.%dg", nPrecision);
312 54 : Serialize(CPLSPrintf(szFormatting, dfVal));
313 : }
314 91 : }
315 :
316 27 : void CPLJSonStreamingWriter::AddNull()
317 : {
318 27 : EmitCommaIfNeeded();
319 27 : Serialize("null", 4);
320 27 : }
321 :
322 : /*! @endcond */
|