1 | // Copyright (C) 2016 The Qt Company Ltd. |
---|---|
2 | // SPDX-License-Identifier: LicenseRef-Qt-Commercial OR LGPL-3.0-only OR GPL-2.0-only OR GPL-3.0-only |
3 | #include <qv4jsonobject_p.h> |
4 | #include <qv4objectproto_p.h> |
5 | #include <qv4numberobject_p.h> |
6 | #include <qv4stringobject_p.h> |
7 | #include <qv4booleanobject_p.h> |
8 | #include <qv4objectiterator_p.h> |
9 | #include <qv4scopedvalue_p.h> |
10 | #include <qv4runtime_p.h> |
11 | #include <qv4variantobject_p.h> |
12 | #include "qv4jscall_p.h" |
13 | #include <qv4symbol_p.h> |
14 | |
15 | #include <qstack.h> |
16 | #include <qstringlist.h> |
17 | |
18 | #include <wtf/MathExtras.h> |
19 | |
20 | using namespace QV4; |
21 | |
22 | //#define PARSER_DEBUG |
23 | #ifdef PARSER_DEBUG |
24 | static int indent = 0; |
25 | #define BEGIN qDebug() << QByteArray(4*indent++, ' ').constData() |
26 | #define END --indent |
27 | #define DEBUG qDebug() << QByteArray(4*indent, ' ').constData() |
28 | #else |
29 | #define BEGIN if (1) ; else qDebug() |
30 | #define END do {} while (0) |
31 | #define DEBUG if (1) ; else qDebug() |
32 | #endif |
33 | |
34 | |
35 | DEFINE_OBJECT_VTABLE(JsonObject); |
36 | |
37 | static const int nestingLimit = 1024; |
38 | |
39 | |
40 | JsonParser::JsonParser(ExecutionEngine *engine, const QChar *json, int length) |
41 | : engine(engine), head(json), json(json), nestingLevel(0), lastError(QJsonParseError::NoError) |
42 | { |
43 | end = json + length; |
44 | } |
45 | |
46 | |
47 | |
48 | /* |
49 | |
50 | begin-array = ws %x5B ws ; [ left square bracket |
51 | |
52 | begin-object = ws %x7B ws ; { left curly bracket |
53 | |
54 | end-array = ws %x5D ws ; ] right square bracket |
55 | |
56 | end-object = ws %x7D ws ; } right curly bracket |
57 | |
58 | name-separator = ws %x3A ws ; : colon |
59 | |
60 | value-separator = ws %x2C ws ; , comma |
61 | |
62 | Insignificant whitespace is allowed before or after any of the six |
63 | structural characters. |
64 | |
65 | ws = *( |
66 | %x20 / ; Space |
67 | %x09 / ; Horizontal tab |
68 | %x0A / ; Line feed or New line |
69 | %x0D ; Carriage return |
70 | ) |
71 | |
72 | */ |
73 | |
74 | enum { |
75 | Space = 0x20, |
76 | Tab = 0x09, |
77 | LineFeed = 0x0a, |
78 | Return = 0x0d, |
79 | BeginArray = 0x5b, |
80 | BeginObject = 0x7b, |
81 | EndArray = 0x5d, |
82 | EndObject = 0x7d, |
83 | NameSeparator = 0x3a, |
84 | ValueSeparator = 0x2c, |
85 | Quote = 0x22 |
86 | }; |
87 | |
88 | bool JsonParser::eatSpace() |
89 | { |
90 | while (json < end) { |
91 | const char16_t ch = json->unicode(); |
92 | if (ch > Space) |
93 | break; |
94 | if (ch != Space && |
95 | ch != Tab && |
96 | ch != LineFeed && |
97 | ch != Return) |
98 | break; |
99 | ++json; |
100 | } |
101 | return (json < end); |
102 | } |
103 | |
104 | QChar JsonParser::nextToken() |
105 | { |
106 | if (!eatSpace()) |
107 | return u'\0'; |
108 | QChar token = *json++; |
109 | switch (token.unicode()) { |
110 | case BeginArray: |
111 | case BeginObject: |
112 | case NameSeparator: |
113 | case ValueSeparator: |
114 | case EndArray: |
115 | case EndObject: |
116 | eatSpace(); |
117 | break; |
118 | case Quote: |
119 | break; |
120 | default: |
121 | token = u'\0'; |
122 | break; |
123 | } |
124 | return token; |
125 | } |
126 | |
127 | /* |
128 | JSON-text = object / array |
129 | */ |
130 | ReturnedValue JsonParser::parse(QJsonParseError *error) |
131 | { |
132 | #ifdef PARSER_DEBUG |
133 | indent = 0; |
134 | qDebug() << ">>>>> parser begin"; |
135 | #endif |
136 | |
137 | eatSpace(); |
138 | |
139 | Scope scope(engine); |
140 | ScopedValue v(scope); |
141 | if (!parseValue(val: v)) { |
142 | #ifdef PARSER_DEBUG |
143 | qDebug() << ">>>>> parser error"; |
144 | #endif |
145 | if (lastError == QJsonParseError::NoError) |
146 | lastError = QJsonParseError::IllegalValue; |
147 | error->offset = json - head; |
148 | error->error = lastError; |
149 | return Encode::undefined(); |
150 | } |
151 | |
152 | // some input left... |
153 | if (eatSpace()) { |
154 | lastError = QJsonParseError::IllegalValue; |
155 | error->offset = json - head; |
156 | error->error = lastError; |
157 | return Encode::undefined(); |
158 | } |
159 | |
160 | END; |
161 | error->offset = 0; |
162 | error->error = QJsonParseError::NoError; |
163 | return v->asReturnedValue(); |
164 | } |
165 | |
166 | /* |
167 | object = begin-object [ member *( value-separator member ) ] |
168 | end-object |
169 | */ |
170 | |
171 | ReturnedValue JsonParser::parseObject() |
172 | { |
173 | if (++nestingLevel > nestingLimit) { |
174 | lastError = QJsonParseError::DeepNesting; |
175 | return Encode::undefined(); |
176 | } |
177 | |
178 | BEGIN << "parseObject pos="<< json; |
179 | Scope scope(engine); |
180 | |
181 | ScopedObject o(scope, engine->newObject()); |
182 | |
183 | QChar token = nextToken(); |
184 | while (token.unicode() == Quote) { |
185 | if (!parseMember(o)) |
186 | return Encode::undefined(); |
187 | token = nextToken(); |
188 | if (token.unicode() != ValueSeparator) |
189 | break; |
190 | token = nextToken(); |
191 | if (token.unicode() == EndObject) { |
192 | lastError = QJsonParseError::MissingObject; |
193 | return Encode::undefined(); |
194 | } |
195 | } |
196 | |
197 | DEBUG << "end token="<< token; |
198 | if (token.unicode() != EndObject) { |
199 | lastError = QJsonParseError::UnterminatedObject; |
200 | return Encode::undefined(); |
201 | } |
202 | |
203 | END; |
204 | |
205 | --nestingLevel; |
206 | return o.asReturnedValue(); |
207 | } |
208 | |
209 | /* |
210 | member = string name-separator value |
211 | */ |
212 | bool JsonParser::parseMember(Object *o) |
213 | { |
214 | BEGIN << "parseMember"; |
215 | Scope scope(engine); |
216 | |
217 | QString key; |
218 | if (!parseString(string: &key)) |
219 | return false; |
220 | QChar token = nextToken(); |
221 | if (token.unicode() != NameSeparator) { |
222 | lastError = QJsonParseError::MissingNameSeparator; |
223 | return false; |
224 | } |
225 | ScopedValue val(scope); |
226 | if (!parseValue(val)) |
227 | return false; |
228 | |
229 | ScopedString s(scope, engine->newString(s: key)); |
230 | PropertyKey skey = s->toPropertyKey(); |
231 | if (skey.isArrayIndex()) { |
232 | o->put(idx: skey.asArrayIndex(), v: val); |
233 | } else { |
234 | // avoid trouble with properties named __proto__ |
235 | o->insertMember(s, v: val); |
236 | } |
237 | |
238 | END; |
239 | return true; |
240 | } |
241 | |
242 | /* |
243 | array = begin-array [ value *( value-separator value ) ] end-array |
244 | */ |
245 | ReturnedValue JsonParser::parseArray() |
246 | { |
247 | Scope scope(engine); |
248 | BEGIN << "parseArray"; |
249 | ScopedArrayObject array(scope, engine->newArrayObject()); |
250 | |
251 | if (++nestingLevel > nestingLimit) { |
252 | lastError = QJsonParseError::DeepNesting; |
253 | return Encode::undefined(); |
254 | } |
255 | |
256 | if (!eatSpace()) { |
257 | lastError = QJsonParseError::UnterminatedArray; |
258 | return Encode::undefined(); |
259 | } |
260 | if (json->unicode() == EndArray) { |
261 | nextToken(); |
262 | } else { |
263 | uint index = 0; |
264 | while (1) { |
265 | ScopedValue val(scope); |
266 | if (!parseValue(val)) |
267 | return Encode::undefined(); |
268 | array->arraySet(index, value: val); |
269 | QChar token = nextToken(); |
270 | if (token.unicode() == EndArray) |
271 | break; |
272 | else if (token.unicode() != ValueSeparator) { |
273 | if (!eatSpace()) |
274 | lastError = QJsonParseError::UnterminatedArray; |
275 | else |
276 | lastError = QJsonParseError::MissingValueSeparator; |
277 | return Encode::undefined(); |
278 | } |
279 | ++index; |
280 | } |
281 | } |
282 | |
283 | DEBUG << "size ="<< array->getLength(); |
284 | END; |
285 | |
286 | --nestingLevel; |
287 | return array.asReturnedValue(); |
288 | } |
289 | |
290 | /* |
291 | value = false / null / true / object / array / number / string |
292 | |
293 | */ |
294 | |
295 | bool JsonParser::parseValue(Value *val) |
296 | { |
297 | BEGIN << "parse Value"<< *json; |
298 | |
299 | switch ((json++)->unicode()) { |
300 | case u'n': |
301 | if (end - json < 3) { |
302 | lastError = QJsonParseError::IllegalValue; |
303 | return false; |
304 | } |
305 | if (*json++ == u'u' && |
306 | *json++ == u'l' && |
307 | *json++ == u'l') { |
308 | *val = Value::nullValue(); |
309 | DEBUG << "value: null"; |
310 | END; |
311 | return true; |
312 | } |
313 | lastError = QJsonParseError::IllegalValue; |
314 | return false; |
315 | case u't': |
316 | if (end - json < 3) { |
317 | lastError = QJsonParseError::IllegalValue; |
318 | return false; |
319 | } |
320 | if (*json++ == u'r' && |
321 | *json++ == u'u' && |
322 | *json++ == u'e') { |
323 | *val = Value::fromBoolean(b: true); |
324 | DEBUG << "value: true"; |
325 | END; |
326 | return true; |
327 | } |
328 | lastError = QJsonParseError::IllegalValue; |
329 | return false; |
330 | case u'f': |
331 | if (end - json < 4) { |
332 | lastError = QJsonParseError::IllegalValue; |
333 | return false; |
334 | } |
335 | if (*json++ == u'a' && |
336 | *json++ == u'l' && |
337 | *json++ == u's' && |
338 | *json++ == u'e') { |
339 | *val = Value::fromBoolean(b: false); |
340 | DEBUG << "value: false"; |
341 | END; |
342 | return true; |
343 | } |
344 | lastError = QJsonParseError::IllegalValue; |
345 | return false; |
346 | case Quote: { |
347 | QString value; |
348 | if (!parseString(string: &value)) |
349 | return false; |
350 | DEBUG << "value: string"; |
351 | END; |
352 | *val = Value::fromHeapObject(m: engine->newString(s: value)); |
353 | return true; |
354 | } |
355 | case BeginArray: { |
356 | *val = parseArray(); |
357 | if (val->isUndefined()) |
358 | return false; |
359 | DEBUG << "value: array"; |
360 | END; |
361 | return true; |
362 | } |
363 | case BeginObject: { |
364 | *val = parseObject(); |
365 | if (val->isUndefined()) |
366 | return false; |
367 | DEBUG << "value: object"; |
368 | END; |
369 | return true; |
370 | } |
371 | case EndArray: |
372 | lastError = QJsonParseError::MissingObject; |
373 | return false; |
374 | default: |
375 | --json; |
376 | if (!parseNumber(val)) |
377 | return false; |
378 | DEBUG << "value: number"; |
379 | END; |
380 | } |
381 | |
382 | return true; |
383 | } |
384 | |
385 | |
386 | |
387 | |
388 | |
389 | /* |
390 | number = [ minus ] int [ frac ] [ exp ] |
391 | decimal-point = %x2E ; . |
392 | digit1-9 = %x31-39 ; 1-9 |
393 | e = %x65 / %x45 ; e E |
394 | exp = e [ minus / plus ] 1*DIGIT |
395 | frac = decimal-point 1*DIGIT |
396 | int = zero / ( digit1-9 *DIGIT ) |
397 | minus = %x2D ; - |
398 | plus = %x2B ; + |
399 | zero = %x30 ; 0 |
400 | |
401 | */ |
402 | |
403 | bool JsonParser::parseNumber(Value *val) |
404 | { |
405 | BEGIN << "parseNumber"<< *json; |
406 | |
407 | const QChar *start = json; |
408 | bool isInt = true; |
409 | |
410 | // minus |
411 | if (json < end && *json == u'-') |
412 | ++json; |
413 | |
414 | // int = zero / ( digit1-9 *DIGIT ) |
415 | if (json < end && *json == u'0') { |
416 | ++json; |
417 | } else { |
418 | while (json < end && *json >= u'0' && *json <= u'9') |
419 | ++json; |
420 | } |
421 | |
422 | // frac = decimal-point 1*DIGIT |
423 | if (json < end && *json == u'.') { |
424 | isInt = false; |
425 | ++json; |
426 | while (json < end && *json >= u'0' && *json <= u'9') |
427 | ++json; |
428 | } |
429 | |
430 | // exp = e [ minus / plus ] 1*DIGIT |
431 | if (json < end && (*json == u'e' || *json == u'E')) { |
432 | isInt = false; |
433 | ++json; |
434 | if (json < end && (*json == u'-' || *json == u'+')) |
435 | ++json; |
436 | while (json < end && *json >= u'0' && *json <= u'9') |
437 | ++json; |
438 | } |
439 | |
440 | QString number(start, json - start); |
441 | DEBUG << "numberstring"<< number; |
442 | |
443 | if (isInt) { |
444 | bool ok; |
445 | int n = number.toInt(ok: &ok); |
446 | if (ok && n < (1<<25) && n > -(1<<25)) { |
447 | *val = Value::fromInt32(i: n); |
448 | END; |
449 | return true; |
450 | } |
451 | } |
452 | |
453 | bool ok; |
454 | double d; |
455 | d = number.toDouble(ok: &ok); |
456 | |
457 | if (!ok) { |
458 | lastError = QJsonParseError::IllegalNumber; |
459 | return false; |
460 | } |
461 | |
462 | * val = Value::fromDouble(d); |
463 | |
464 | END; |
465 | return true; |
466 | } |
467 | |
468 | /* |
469 | |
470 | string = quotation-mark *char quotation-mark |
471 | |
472 | char = unescaped / |
473 | escape ( |
474 | %x22 / ; " quotation mark U+0022 |
475 | %x5C / ; \ reverse solidus U+005C |
476 | %x2F / ; / solidus U+002F |
477 | %x62 / ; b backspace U+0008 |
478 | %x66 / ; f form feed U+000C |
479 | %x6E / ; n line feed U+000A |
480 | %x72 / ; r carriage return U+000D |
481 | %x74 / ; t tab U+0009 |
482 | %x75 4HEXDIG ) ; uXXXX U+XXXX |
483 | |
484 | escape = %x5C ; \ |
485 | |
486 | quotation-mark = %x22 ; " |
487 | |
488 | unescaped = %x20-21 / %x23-5B / %x5D-10FFFF |
489 | */ |
490 | static inline bool addHexDigit(QChar digit, uint *result) |
491 | { |
492 | ushort d = digit.unicode(); |
493 | *result <<= 4; |
494 | if (d >= u'0' && d <= u'9') |
495 | *result |= (d - u'0'); |
496 | else if (d >= u'a' && d <= u'f') |
497 | *result |= (d - u'a') + 10; |
498 | else if (d >= u'A' && d <= u'F') |
499 | *result |= (d - u'A') + 10; |
500 | else |
501 | return false; |
502 | return true; |
503 | } |
504 | |
505 | static inline bool scanEscapeSequence(const QChar *&json, const QChar *end, uint *ch) |
506 | { |
507 | ++json; |
508 | if (json >= end) |
509 | return false; |
510 | |
511 | DEBUG << "scan escape"; |
512 | uint escaped = (json++)->unicode(); |
513 | switch (escaped) { |
514 | case u'"': |
515 | *ch = '"'; break; |
516 | case u'\\': |
517 | *ch = '\\'; break; |
518 | case u'/': |
519 | *ch = '/'; break; |
520 | case u'b': |
521 | *ch = 0x8; break; |
522 | case u'f': |
523 | *ch = 0xc; break; |
524 | case u'n': |
525 | *ch = 0xa; break; |
526 | case u'r': |
527 | *ch = 0xd; break; |
528 | case u't': |
529 | *ch = 0x9; break; |
530 | case u'u': { |
531 | *ch = 0; |
532 | if (json > end - 4) |
533 | return false; |
534 | for (int i = 0; i < 4; ++i) { |
535 | if (!addHexDigit(digit: *json, result: ch)) |
536 | return false; |
537 | ++json; |
538 | } |
539 | return true; |
540 | } |
541 | default: |
542 | return false; |
543 | } |
544 | return true; |
545 | } |
546 | |
547 | |
548 | bool JsonParser::parseString(QString *string) |
549 | { |
550 | BEGIN << "parse string stringPos="<< json; |
551 | |
552 | while (json < end) { |
553 | if (*json == u'"') |
554 | break; |
555 | else if (*json == u'\\') { |
556 | uint ch = 0; |
557 | if (!scanEscapeSequence(json, end, ch: &ch)) { |
558 | lastError = QJsonParseError::IllegalEscapeSequence; |
559 | return false; |
560 | } |
561 | if (QChar::requiresSurrogates(ucs4: ch)) { |
562 | *string += QChar(QChar::highSurrogate(ucs4: ch)) + QChar(QChar::lowSurrogate(ucs4: ch)); |
563 | } else { |
564 | *string += QChar(ch); |
565 | } |
566 | } else { |
567 | if (json->unicode() <= 0x1f) { |
568 | lastError = QJsonParseError::IllegalEscapeSequence; |
569 | return false; |
570 | } |
571 | *string += *json; |
572 | ++json; |
573 | } |
574 | } |
575 | ++json; |
576 | |
577 | if (json > end) { |
578 | lastError = QJsonParseError::UnterminatedString; |
579 | return false; |
580 | } |
581 | |
582 | END; |
583 | return true; |
584 | } |
585 | |
586 | |
587 | struct Stringify |
588 | { |
589 | ExecutionEngine *v4; |
590 | FunctionObject *replacerFunction; |
591 | QV4::String *propertyList; |
592 | int propertyListSize; |
593 | QString gap; |
594 | QString indent; |
595 | QStack<Object *> stack; |
596 | |
597 | bool stackContains(Object *o) { |
598 | for (int i = 0; i < stack.size(); ++i) |
599 | if (stack.at(i)->d() == o->d()) |
600 | return true; |
601 | return false; |
602 | } |
603 | |
604 | Stringify(ExecutionEngine *e) : v4(e), replacerFunction(nullptr), propertyList(nullptr), propertyListSize(0) {} |
605 | |
606 | QString Str(const QString &key, const Value &v); |
607 | QString JA(Object *a); |
608 | QString JO(Object *o); |
609 | |
610 | QString makeMember(const QString &key, const Value &v); |
611 | }; |
612 | |
613 | class [[nodiscard]] CallDepthAndCycleChecker |
614 | { |
615 | Q_DISABLE_COPY_MOVE(CallDepthAndCycleChecker); |
616 | |
617 | public: |
618 | CallDepthAndCycleChecker(Stringify *stringify, Object *o) |
619 | : m_callDepthRecorder(stringify->v4) |
620 | { |
621 | if (stringify->stackContains(o)) { |
622 | stringify->v4->throwTypeError( |
623 | QStringLiteral("Cannot convert circular structure to JSON")); |
624 | } |
625 | |
626 | stringify->v4->checkStackLimits(); |
627 | } |
628 | |
629 | bool foundProblem() const { return m_callDepthRecorder.ee->hasException; } |
630 | |
631 | private: |
632 | ExecutionEngineCallDepthRecorder<1> m_callDepthRecorder; |
633 | }; |
634 | |
635 | static QString quote(const QString &str) |
636 | { |
637 | QString product; |
638 | const int length = str.size(); |
639 | product.reserve(asize: length + 2); |
640 | product += u'"'; |
641 | for (int i = 0; i < length; ++i) { |
642 | QChar c = str.at(i); |
643 | switch (c.unicode()) { |
644 | case u'"': |
645 | product += QLatin1String("\\\""); |
646 | break; |
647 | case u'\\': |
648 | product += QLatin1String("\\\\"); |
649 | break; |
650 | case u'\b': |
651 | product += QLatin1String("\\b"); |
652 | break; |
653 | case u'\f': |
654 | product += QLatin1String("\\f"); |
655 | break; |
656 | case u'\n': |
657 | product += QLatin1String("\\n"); |
658 | break; |
659 | case u'\r': |
660 | product += QLatin1String("\\r"); |
661 | break; |
662 | case u'\t': |
663 | product += QLatin1String("\\t"); |
664 | break; |
665 | default: |
666 | if (c.unicode() <= 0x1f) { |
667 | product += QLatin1String("\\u00"); |
668 | product += (c.unicode() > 0xf ? u'1' : u'0') + |
669 | QLatin1Char("0123456789abcdef"[c.unicode() & 0xf]); |
670 | } else { |
671 | product += c; |
672 | } |
673 | } |
674 | } |
675 | product += u'"'; |
676 | return product; |
677 | } |
678 | |
679 | QString Stringify::Str(const QString &key, const Value &v) |
680 | { |
681 | Scope scope(v4); |
682 | |
683 | ScopedValue value(scope, v); |
684 | ScopedObject o(scope, value); |
685 | if (o) { |
686 | ScopedString s(scope, v4->newString(QStringLiteral("toJSON"))); |
687 | ScopedFunctionObject toJSON(scope, o->get(name: s)); |
688 | if (!!toJSON) { |
689 | JSCallArguments jsCallData(scope, 1); |
690 | *jsCallData.thisObject = value; |
691 | jsCallData.args[0] = v4->newString(s: key); |
692 | value = toJSON->call(data: jsCallData); |
693 | if (v4->hasException) |
694 | return QString(); |
695 | } |
696 | } |
697 | |
698 | if (replacerFunction) { |
699 | JSCallArguments jsCallData(scope, 2); |
700 | jsCallData.args[0] = v4->newString(s: key); |
701 | jsCallData.args[1] = value; |
702 | |
703 | if (stack.isEmpty()) { |
704 | ScopedObject holder(scope, v4->newObject()); |
705 | holder->put(name: scope.engine->id_empty(), v); |
706 | *jsCallData.thisObject = holder; |
707 | } else { |
708 | *jsCallData.thisObject = stack.top(); |
709 | } |
710 | |
711 | value = replacerFunction->call(data: jsCallData); |
712 | if (v4->hasException) |
713 | return QString(); |
714 | } |
715 | |
716 | o = value->asReturnedValue(); |
717 | if (o) { |
718 | if (NumberObject *n = o->as<NumberObject>()) |
719 | value = Encode(n->value()); |
720 | else if (StringObject *so = o->as<StringObject>()) |
721 | value = so->d()->string; |
722 | else if (BooleanObject *b = o->as<BooleanObject>()) |
723 | value = Encode(b->value()); |
724 | } |
725 | |
726 | if (value->isNull()) |
727 | return QStringLiteral("null"); |
728 | if (value->isBoolean()) |
729 | return value->booleanValue() ? QStringLiteral("true") : QStringLiteral( "false"); |
730 | if (value->isString()) |
731 | return quote(str: value->stringValue()->toQString()); |
732 | |
733 | if (value->isNumber()) { |
734 | double d = value->toNumber(); |
735 | return std::isfinite(x: d) ? value->toQString() : QStringLiteral("null"); |
736 | } |
737 | |
738 | if (const QV4::VariantObject *v = value->as<QV4::VariantObject>()) { |
739 | return quote(str: v->d()->data().toString()); |
740 | } |
741 | |
742 | o = value->asReturnedValue(); |
743 | if (o) { |
744 | if (!o->as<FunctionObject>()) { |
745 | if (o->isArrayLike()) { |
746 | return JA(a: o.getPointer()); |
747 | } else { |
748 | return JO(o); |
749 | } |
750 | } |
751 | } |
752 | |
753 | return QString(); |
754 | } |
755 | |
756 | QString Stringify::makeMember(const QString &key, const Value &v) |
757 | { |
758 | QString strP = Str(key, v); |
759 | if (!strP.isEmpty()) { |
760 | QString member = quote(str: key) + u':'; |
761 | if (!gap.isEmpty()) |
762 | member += u' '; |
763 | member += strP; |
764 | return member; |
765 | } |
766 | return QString(); |
767 | } |
768 | |
769 | QString Stringify::JO(Object *o) |
770 | { |
771 | CallDepthAndCycleChecker check(this, o); |
772 | if (check.foundProblem()) |
773 | return QString(); |
774 | |
775 | Scope scope(v4); |
776 | |
777 | QString result; |
778 | stack.push(t: o); |
779 | QString stepback = indent; |
780 | indent += gap; |
781 | |
782 | QStringList partial; |
783 | if (!propertyListSize) { |
784 | ObjectIterator it(scope, o, ObjectIterator::EnumerableOnly); |
785 | ScopedValue name(scope); |
786 | |
787 | ScopedValue val(scope); |
788 | while (1) { |
789 | name = it.nextPropertyNameAsString(value: val); |
790 | if (name->isNull()) |
791 | break; |
792 | QString key = name->toQString(); |
793 | QString member = makeMember(key, v: val); |
794 | if (!member.isEmpty()) |
795 | partial += member; |
796 | } |
797 | } else { |
798 | ScopedValue v(scope); |
799 | for (int i = 0; i < propertyListSize; ++i) { |
800 | bool exists; |
801 | String *s = propertyList + i; |
802 | if (!s) |
803 | continue; |
804 | v = o->get(name: s, hasProperty: &exists); |
805 | if (!exists) |
806 | continue; |
807 | QString member = makeMember(key: s->toQString(), v); |
808 | if (!member.isEmpty()) |
809 | partial += member; |
810 | } |
811 | } |
812 | |
813 | if (partial.isEmpty()) { |
814 | result = QStringLiteral("{}"); |
815 | } else if (gap.isEmpty()) { |
816 | result = u'{' + partial.join(sep: u',') + u'}'; |
817 | } else { |
818 | QString separator = QLatin1String(",\n") + indent; |
819 | result = QLatin1String("{\n") + indent + partial.join(sep: separator) + u'\n' |
820 | + stepback + u'}'; |
821 | } |
822 | |
823 | indent = stepback; |
824 | stack.pop(); |
825 | return result; |
826 | } |
827 | |
828 | QString Stringify::JA(Object *a) |
829 | { |
830 | CallDepthAndCycleChecker check(this, a); |
831 | if (check.foundProblem()) |
832 | return QString(); |
833 | |
834 | Scope scope(a->engine()); |
835 | |
836 | QString result; |
837 | stack.push(t: a); |
838 | QString stepback = indent; |
839 | indent += gap; |
840 | |
841 | QStringList partial; |
842 | uint len = a->getLength(); |
843 | ScopedValue v(scope); |
844 | for (uint i = 0; i < len; ++i) { |
845 | bool exists; |
846 | v = a->get(idx: i, hasProperty: &exists); |
847 | if (!exists) { |
848 | partial += QStringLiteral("null"); |
849 | continue; |
850 | } |
851 | QString strP = Str(key: QString::number(i), v); |
852 | if (!strP.isEmpty()) |
853 | partial += strP; |
854 | else |
855 | partial += QStringLiteral("null"); |
856 | } |
857 | |
858 | if (partial.isEmpty()) { |
859 | result = QStringLiteral("[]"); |
860 | } else if (gap.isEmpty()) { |
861 | result = u'[' + partial.join(sep: u',') + u']'; |
862 | } else { |
863 | QString separator = QLatin1String(",\n") + indent; |
864 | result = QLatin1String("[\n") + indent + partial.join(sep: separator) + u'\n' + stepback + u']'; |
865 | } |
866 | |
867 | indent = stepback; |
868 | stack.pop(); |
869 | return result; |
870 | } |
871 | |
872 | |
873 | void Heap::JsonObject::init() |
874 | { |
875 | Object::init(); |
876 | Scope scope(internalClass->engine); |
877 | ScopedObject o(scope, this); |
878 | |
879 | o->defineDefaultProperty(QStringLiteral("parse"), code: QV4::JsonObject::method_parse, argumentCount: 2); |
880 | o->defineDefaultProperty(QStringLiteral("stringify"), code: QV4::JsonObject::method_stringify, argumentCount: 3); |
881 | ScopedString json(scope, scope.engine->newString(QStringLiteral("JSON"))); |
882 | o->defineReadonlyConfigurableProperty(name: scope.engine->symbol_toStringTag(), value: json); |
883 | } |
884 | |
885 | |
886 | ReturnedValue JsonObject::method_parse(const FunctionObject *b, const Value *, const Value *argv, int argc) |
887 | { |
888 | ExecutionEngine *v4 = b->engine(); |
889 | QString jtext; |
890 | if (argc > 0) |
891 | jtext = argv[0].toQString(); |
892 | |
893 | DEBUG << "parsing source = "<< jtext; |
894 | JsonParser parser(v4, jtext.constData(), jtext.size()); |
895 | QJsonParseError error; |
896 | ReturnedValue result = parser.parse(error: &error); |
897 | if (error.error != QJsonParseError::NoError) { |
898 | DEBUG << "parse error"<< error.errorString(); |
899 | RETURN_RESULT(v4->throwSyntaxError(QStringLiteral("JSON.parse: Parse error"))); |
900 | } |
901 | |
902 | return result; |
903 | } |
904 | |
905 | ReturnedValue JsonObject::method_stringify(const FunctionObject *b, const Value *, const Value *argv, int argc) |
906 | { |
907 | Scope scope(b); |
908 | Stringify stringify(scope.engine); |
909 | |
910 | ScopedObject o(scope, argc > 1 ? argv[1] : Value::undefinedValue()); |
911 | if (o) { |
912 | stringify.replacerFunction = o->as<FunctionObject>(); |
913 | if (o->isArrayObject()) { |
914 | int arrayLen = scope.engine->safeForAllocLength(len64: o->getLength()); |
915 | CHECK_EXCEPTION(); |
916 | stringify.propertyList = static_cast<QV4::String *>(scope.alloc(nValues: arrayLen)); |
917 | for (int i = 0; i < arrayLen; ++i) { |
918 | Value *v = stringify.propertyList + i; |
919 | *v = o->get(idx: i); |
920 | if (v->as<NumberObject>() || v->as<StringObject>() || v->isNumber()) |
921 | *v = v->toString(e: scope.engine); |
922 | if (!v->isString()) { |
923 | v->setM(nullptr); |
924 | } else { |
925 | for (int j = 0; j <i; ++j) { |
926 | if (stringify.propertyList[j].m() == v->m()) { |
927 | v->setM(nullptr); |
928 | break; |
929 | } |
930 | } |
931 | } |
932 | } |
933 | } |
934 | } |
935 | |
936 | ScopedValue s(scope, argc > 2 ? argv[2] : Value::undefinedValue()); |
937 | if (NumberObject *n = s->as<NumberObject>()) |
938 | s = Encode(n->value()); |
939 | else if (StringObject *so = s->as<StringObject>()) |
940 | s = so->d()->string; |
941 | |
942 | if (s->isNumber()) { |
943 | stringify.gap = QString(qMin(a: 10, b: (int)s->toInteger()), u' '); |
944 | } else if (String *str = s->stringValue()) { |
945 | stringify.gap = str->toQString().left(n: 10); |
946 | } |
947 | |
948 | |
949 | ScopedValue arg0(scope, argc ? argv[0] : Value::undefinedValue()); |
950 | QString result = stringify.Str(key: QString(), v: arg0); |
951 | if (result.isEmpty() || scope.hasException()) |
952 | RETURN_UNDEFINED(); |
953 | return Encode(scope.engine->newString(s: result)); |
954 | } |
955 | |
956 | |
957 | |
958 | ReturnedValue JsonObject::fromJsonValue(ExecutionEngine *engine, const QJsonValue &value) |
959 | { |
960 | if (value.isString()) |
961 | return engine->newString(s: value.toString())->asReturnedValue(); |
962 | else if (value.isDouble()) |
963 | return Encode(value.toDouble()); |
964 | else if (value.isBool()) |
965 | return Encode(value.toBool()); |
966 | else if (value.isArray()) |
967 | return fromJsonArray(engine, array: value.toArray()); |
968 | else if (value.isObject()) |
969 | return fromJsonObject(engine, object: value.toObject()); |
970 | else if (value.isNull()) |
971 | return Encode::null(); |
972 | else |
973 | return Encode::undefined(); |
974 | } |
975 | |
976 | QJsonValue JsonObject::toJsonValue(const Value &value, V4ObjectSet &visitedObjects) |
977 | { |
978 | if (value.isNumber()) |
979 | return QJsonValue(value.toNumber()); |
980 | else if (value.isBoolean()) |
981 | return QJsonValue((bool)value.booleanValue()); |
982 | else if (value.isNull()) |
983 | return QJsonValue(QJsonValue::Null); |
984 | else if (value.isUndefined()) |
985 | return QJsonValue(QJsonValue::Undefined); |
986 | else if (String *s = value.stringValue()) |
987 | return QJsonValue(s->toQString()); |
988 | |
989 | Q_ASSERT(value.isObject()); |
990 | Scope scope(value.as<Object>()->engine()); |
991 | if (ScopedArrayObject a{ scope, value }) { |
992 | return toJsonArray(o: a, visitedObjects); |
993 | } else if (Scoped<QV4::Sequence> a{ scope, value }) { |
994 | return toJsonArray(o: a, visitedObjects); |
995 | } else if (Scoped<QmlListWrapper> lw{ scope, value }) { |
996 | return toJsonArray(o: lw, visitedObjects); |
997 | } else if (ScopedObject o{ scope, value }) { |
998 | return toJsonObject(o, visitedObjects); |
999 | } |
1000 | |
1001 | return QJsonValue(value.toQString()); |
1002 | } |
1003 | |
1004 | QV4::ReturnedValue JsonObject::fromJsonObject(ExecutionEngine *engine, const QJsonObject &object) |
1005 | { |
1006 | Scope scope(engine); |
1007 | ScopedObject o(scope, engine->newObject()); |
1008 | ScopedString s(scope); |
1009 | ScopedValue v(scope); |
1010 | for (QJsonObject::const_iterator it = object.begin(), cend = object.end(); it != cend; ++it) { |
1011 | v = fromJsonValue(engine, value: it.value()); |
1012 | o->put(name: (s = engine->newString(s: it.key())), v); |
1013 | } |
1014 | return o.asReturnedValue(); |
1015 | } |
1016 | |
1017 | QJsonObject JsonObject::toJsonObject(const Object *o, V4ObjectSet &visitedObjects) |
1018 | { |
1019 | QJsonObject result; |
1020 | if (!o || o->as<FunctionObject>()) |
1021 | return result; |
1022 | |
1023 | Scope scope(o->engine()); |
1024 | |
1025 | if (visitedObjects.contains(value: ObjectItem(o))) { |
1026 | // Avoid recursion. |
1027 | // For compatibility with QVariant{List,Map} conversion, we return an |
1028 | // empty object (and no error is thrown). |
1029 | return result; |
1030 | } |
1031 | |
1032 | visitedObjects.insert(value: ObjectItem(o)); |
1033 | |
1034 | ObjectIterator it(scope, o, ObjectIterator::EnumerableOnly); |
1035 | ScopedValue name(scope); |
1036 | QV4::ScopedValue val(scope); |
1037 | while (1) { |
1038 | name = it.nextPropertyNameAsString(value: val); |
1039 | if (name->isNull()) |
1040 | break; |
1041 | |
1042 | QString key = name->toQStringNoThrow(); |
1043 | if (!val->as<FunctionObject>()) |
1044 | result.insert(key, value: toJsonValue(value: val, visitedObjects)); |
1045 | } |
1046 | |
1047 | visitedObjects.remove(value: ObjectItem(o)); |
1048 | |
1049 | return result; |
1050 | } |
1051 | |
1052 | QV4::ReturnedValue JsonObject::fromJsonArray(ExecutionEngine *engine, const QJsonArray &array) |
1053 | { |
1054 | Scope scope(engine); |
1055 | int size = array.size(); |
1056 | ScopedArrayObject a(scope, engine->newArrayObject()); |
1057 | a->arrayReserve(n: size); |
1058 | ScopedValue v(scope); |
1059 | for (int i = 0; i < size; i++) |
1060 | a->arrayPut(index: i, value: (v = fromJsonValue(engine, value: array.at(i)))); |
1061 | a->setArrayLengthUnchecked(size); |
1062 | return a.asReturnedValue(); |
1063 | } |
1064 | |
1065 | QJsonArray JsonObject::toJsonArray(const Object *a, V4ObjectSet &visitedObjects) |
1066 | { |
1067 | QJsonArray result; |
1068 | if (!a) |
1069 | return result; |
1070 | |
1071 | Scope scope(a->engine()); |
1072 | |
1073 | if (visitedObjects.contains(value: ObjectItem(a))) { |
1074 | // Avoid recursion. |
1075 | // For compatibility with QVariant{List,Map} conversion, we return an |
1076 | // empty array (and no error is thrown). |
1077 | return result; |
1078 | } |
1079 | |
1080 | visitedObjects.insert(value: ObjectItem(a)); |
1081 | |
1082 | ScopedValue v(scope); |
1083 | quint32 length = a->getLength(); |
1084 | for (quint32 i = 0; i < length; ++i) { |
1085 | v = a->get(idx: i); |
1086 | if (v->as<FunctionObject>()) |
1087 | v = Encode::null(); |
1088 | result.append(value: toJsonValue(value: v, visitedObjects)); |
1089 | } |
1090 | |
1091 | visitedObjects.remove(value: ObjectItem(a)); |
1092 | |
1093 | return result; |
1094 | } |
1095 |
Definitions
- nestingLimit
- JsonParser
- eatSpace
- nextToken
- parse
- parseObject
- parseMember
- parseArray
- parseValue
- parseNumber
- addHexDigit
- scanEscapeSequence
- parseString
- Stringify
- stackContains
- Stringify
- CallDepthAndCycleChecker
- CallDepthAndCycleChecker
- CallDepthAndCycleChecker
- foundProblem
- quote
- Str
- makeMember
- JO
- JA
- init
- method_parse
- method_stringify
- fromJsonValue
- toJsonValue
- fromJsonObject
- toJsonObject
- fromJsonArray
Learn to use CMake with our Intro Training
Find out more