json-as 0.8.6 → 0.9.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (52) hide show
  1. package/.github/workflows/nodejs.yml +7 -1
  2. package/CHANGELOG +4 -1
  3. package/README.md +26 -16
  4. package/assembly/__tests__/deserialize.spec.ts +298 -0
  5. package/assembly/__tests__/serialize.spec.ts +375 -0
  6. package/assembly/deserialize/array/array.ts +31 -0
  7. package/assembly/deserialize/array/bool.ts +19 -0
  8. package/assembly/deserialize/array/float.ts +24 -0
  9. package/assembly/deserialize/array/integer.ts +24 -0
  10. package/assembly/deserialize/array/map.ts +27 -0
  11. package/assembly/deserialize/array/object.ts +27 -0
  12. package/assembly/deserialize/array/string.ts +29 -0
  13. package/assembly/deserialize/array.ts +37 -0
  14. package/assembly/deserialize/bool.ts +18 -0
  15. package/assembly/deserialize/box.ts +17 -0
  16. package/assembly/deserialize/date.ts +11 -0
  17. package/assembly/deserialize/float.ts +9 -0
  18. package/assembly/deserialize/integer.ts +7 -0
  19. package/assembly/deserialize/map.ts +182 -0
  20. package/assembly/deserialize/object.ts +136 -0
  21. package/assembly/deserialize/string.ts +88 -0
  22. package/assembly/index.d.ts +7 -1
  23. package/assembly/index.ts +129 -1
  24. package/assembly/serialize/array.ts +52 -0
  25. package/assembly/serialize/bool.ts +4 -0
  26. package/assembly/serialize/box.ts +10 -0
  27. package/assembly/serialize/date.ts +4 -0
  28. package/assembly/serialize/float.ts +4 -0
  29. package/assembly/serialize/integer.ts +5 -0
  30. package/assembly/serialize/map.ts +24 -0
  31. package/assembly/serialize/object.ts +7 -0
  32. package/assembly/serialize/string.ts +64 -0
  33. package/assembly/src/sink.ts +286 -0
  34. package/assembly/src/util.ts +6 -0
  35. package/assembly/test.ts +34 -16
  36. package/bench/benchmark.ts +7 -3
  37. package/bench.js +14 -3
  38. package/index.ts +1 -1
  39. package/package.json +6 -8
  40. package/transform/lib/index.js +301 -183
  41. package/transform/lib/index.old.js +257 -0
  42. package/transform/lib/types.js +17 -0
  43. package/transform/package.json +1 -1
  44. package/transform/src/index.old.ts +312 -0
  45. package/transform/src/index.ts +301 -215
  46. package/transform/tsconfig.json +2 -2
  47. package/tsconfig.json +94 -102
  48. package/assembly/__benches__/as-json.ts +0 -88
  49. package/assembly/__benches__/as-tral.d.ts +0 -1
  50. package/assembly/__tests__/as-json.spec.ts +0 -673
  51. package/assembly/__tests__/as-pect.d.ts +0 -1
  52. package/assembly/src/json.ts +0 -941
@@ -1,941 +0,0 @@
1
- import { StringSink } from "as-string-sink/assembly";
2
- import { isSpace } from "util/string";
3
- import {
4
- aCode,
5
- bCode,
6
- eCode,
7
- fCode,
8
- lCode,
9
- nCode,
10
- rCode,
11
- sCode,
12
- tCode,
13
- uCode,
14
-
15
- backSlashCode,
16
- colonCode,
17
- commaCode,
18
- forwardSlashCode,
19
- leftBraceCode,
20
- leftBracketCode,
21
- quoteCode,
22
- rightBraceCode,
23
- rightBracketCode,
24
-
25
- backspaceCode,
26
- carriageReturnCode,
27
- tabCode,
28
- formFeedCode,
29
- newLineCode,
30
-
31
- commaWord,
32
- quoteWord,
33
-
34
- leftBraceWord,
35
- leftBracketWord,
36
- rightBracketWord,
37
- emptyArrayWord,
38
-
39
- trueWord,
40
- falseWord,
41
- nullWord,
42
- } from "./chars";
43
- import { snip_fast, unsafeCharCodeAt, containsCodePoint } from "./util";
44
- import { Virtual } from "as-virtual/assembly";
45
- import { Box } from "as-container/assembly";
46
-
47
- /**
48
- * JSON Encoder/Decoder for AssemblyScript
49
- */
50
- export namespace JSON {
51
- /**
52
- * Stringifies valid JSON data.
53
- * ```js
54
- * __JSON_Stringify<T>(data)
55
- * ```
56
- * @param data T
57
- * @returns string
58
- */
59
- // @ts-ignore: Decorator
60
- @inline export function stringify<T>(data: T): string {
61
- // String
62
- if (isString<T>() && data != null) {
63
- return serializeString(data as string);
64
- } else if (isBoolean<T>()) {
65
- return data ? "true" : "false";
66
- } else if (data instanceof Box) {
67
- if (isNullable<T>() && (changetype<usize>(data._val) == <usize>0 || changetype<usize>(data) == <usize>0)) {
68
- return nullWord;
69
- }
70
- return JSON.stringify(data.unwrap());
71
- } else if (isNullable<T>() && changetype<usize>(data) == <usize>0) {
72
- return nullWord;
73
- // @ts-ignore
74
- } else if ((isInteger<T>() || isFloat<T>()) && isFinite(data)) {
75
- // @ts-ignore
76
- return data.toString();
77
- // @ts-ignore: Hidden function
78
- } else if (isDefined(data.__JSON_Serialize)) {
79
- // @ts-ignore: Hidden function
80
- return data.__JSON_Serialize();
81
- } else if (data instanceof Date) {
82
- return `"${data.toISOString()}"`;
83
- } else if (isArrayLike<T>()) {
84
- // @ts-ignore
85
- if (data.length == 0) {
86
- return emptyArrayWord;
87
- // @ts-ignore
88
- } else if (isString<valueof<T>>()) {
89
- let result = leftBracketWord;
90
- // @ts-ignore
91
- for (let i = 0; i < data.length - 1; i++) {
92
- // @ts-ignore
93
- result += serializeString(unchecked(data[i]));
94
- result += commaWord;
95
- }
96
- // @ts-ignore
97
- result += serializeString(unchecked(data[data.length - 1]));
98
- result += rightBracketWord;
99
- return result;
100
- // @ts-ignore
101
- } else if (isBoolean<valueof<T>>()) {
102
- // @ts-ignore
103
- return leftBracketWord + data.join(commaWord) + rightBracketWord;
104
- // @ts-ignore
105
- } else if (isFloat<valueof<T>>() || isInteger<valueof<T>>()) {
106
- // @ts-ignore
107
- return leftBracketWord + data.join(commaWord) + rightBracketWord;
108
- } else {
109
- let result = new StringSink(leftBracketWord);
110
- // @ts-ignore
111
- for (let i = 0; i < data.length - 1; i++) {
112
- // @ts-ignore
113
- result.write(__JSON_Stringify(unchecked(data[i])));
114
- result.writeCodePoint(commaCode);
115
- }
116
- // @ts-ignore
117
- result.write(__JSON_Stringify(unchecked(data[data.length - 1])));
118
- result.writeCodePoint(rightBracketCode);
119
- return result.toString();
120
- }
121
- } else if (data instanceof Map) {
122
- let result = new StringSink(leftBraceWord);
123
- let keys = data.keys();
124
- let values = data.values();
125
- const end = data.size - 1;
126
- for (let i = 0; i < end; i++) {
127
- result.write(serializeString(unchecked(keys[i]).toString()));
128
- result.writeCodePoint(colonCode);
129
- result.write(__JSON_Stringify(unchecked(values[i])));
130
- result.writeCodePoint(commaCode);
131
- }
132
- result.write(serializeString(unchecked(keys[end]).toString()));
133
- result.writeCodePoint(colonCode);
134
- result.write(__JSON_Stringify(unchecked(values[end])));
135
-
136
- result.writeCodePoint(rightBraceCode);
137
- return result.toString();
138
- } else {
139
- throw new Error(
140
- `Could not serialize data of type ${nameof<T>()}. Make sure to add the correct decorators to classes.`
141
- );
142
- }
143
- }
144
- // @ts-ignore: Decorator
145
- @unsafe
146
- @inline export function stringifyTo<T>(data: T, out: string): void {
147
- throw new Error("Method is deprecated");
148
- }
149
- /**
150
- * Parses valid JSON strings into their original format.
151
- * ```js
152
- * JSON.parse<T>(data)
153
- * ```
154
- * @param data string
155
- * @returns T
156
- */
157
-
158
- // @ts-ignore: Decorator
159
- @inline export function parse<T>(data: string, initializeDefaultValues: boolean = false): T {
160
- if (isString<T>()) {
161
- // @ts-ignore
162
- return parseString(data);
163
- } else if (isBoolean<T>()) {
164
- // @ts-ignore
165
- return parseBoolean<T>(data);
166
- } else if (isFloat<T>() || isInteger<T>()) {
167
- return parseNumber<T>(data);
168
- } else if (isArrayLike<T>()) {
169
- // @ts-ignore
170
- return parseArray<T>(data.trimStart());
171
- // @ts-ignore
172
- }
173
- let type: nonnull<T> = changetype<nonnull<T>>(0);
174
- if (type instanceof Box) {
175
- const instance = changetype<nonnull<T>>(__new(offsetof<nonnull<T>>(), idof<nonnull<T>>()))// as Box<usize>;
176
- const val = instance._val;
177
- instance._val = parseDirectInference(val, data);
178
- // @ts-ignore
179
- return changetype<T>(instance);
180
- } else if (isNullable<T>() && data == nullWord) {
181
- // @ts-ignore
182
- return null;
183
- // @ts-ignore
184
- } else if (isDefined(type.__JSON_Set_Key)) {
185
- return parseObject<T>(data.trimStart(), initializeDefaultValues);
186
- } else if (isMap<T>()) {
187
- return parseMap<T>(data.trimStart(), initializeDefaultValues);
188
- } else if (idof<nonnull<T>>() == idof<Date>()) {
189
- // @ts-ignore
190
- return parseDate(data);
191
- } else {
192
- throw new Error(
193
- `Could not deserialize data ${data} to type ${nameof<T>()}. Make sure to add the correct decorators to classes.`
194
- );
195
- }
196
- }
197
- }
198
-
199
- // @ts-ignore: Decorator
200
- @global @inline function __parseObjectValue<T>(data: string, initializeDefaultValues: boolean): T {
201
- if (isString<T>()) {
202
- // @ts-ignore
203
- return data;
204
- } else if (isBoolean<T>()) {
205
- // @ts-ignore
206
- return parseBoolean<T>(data);
207
- } else if (isFloat<T>() || isInteger<T>()) {
208
- return parseNumber<T>(data);
209
- } else if (isArrayLike<T>()) {
210
- // @ts-ignore
211
- return parseArray<T>(data);
212
- // @ts-ignore
213
- }
214
- let type: nonnull<T> = changetype<nonnull<T>>(0);
215
- if (type instanceof Box) {
216
- const instance = changetype<nonnull<T>>(__new(offsetof<nonnull<T>>(), idof<nonnull<T>>()))// as Box<usize>;
217
- const val = instance._val;
218
- instance._val = parseDirectInference(val, data);
219
- // @ts-ignore
220
- return changetype<T>(instance);
221
- } else if (isNullable<T>() && data == nullWord) {
222
- // @ts-ignore
223
- return null;
224
- // @ts-ignore
225
- } else if (isDefined(type.__JSON_Set_Key)) {
226
- return parseObject<T>(data.trimStart(), initializeDefaultValues);
227
- } else if (isMap<T>()) {
228
- return parseMap<T>(data.trimStart(), initializeDefaultValues);
229
- } else if (idof<nonnull<T>>() == idof<Date>()) {
230
- // @ts-ignore
231
- return parseDate(data);
232
- } else {
233
- throw new Error(
234
- `Could not deserialize data ${data} to type ${nameof<T>()}. Make sure to add the correct decorators to classes.`
235
- );
236
- }
237
- }
238
-
239
- // @ts-ignore: Decorator
240
- @inline function serializeString(data: string): string {
241
- if (data.length === 0) {
242
- return quoteWord + quoteWord;
243
- }
244
-
245
- let result = new StringSink(quoteWord);
246
-
247
- let last: i32 = 0;
248
- for (let i = 0; i < data.length; i++) {
249
- const char = unsafeCharCodeAt(<string>data, i);
250
- if (char === quoteCode || char === backSlashCode) {
251
- result.write(<string>data, last, i);
252
- result.writeCodePoint(backSlashCode);
253
- last = i;
254
- } else if (char < 16) {
255
- result.write(<string>data, last, i);
256
- last = i + 1;
257
- switch (char) {
258
- case backspaceCode: {
259
- result.write("\\b");
260
- break;
261
- }
262
- case tabCode: {
263
- result.write("\\t");
264
- break;
265
- }
266
- case newLineCode: {
267
- result.write("\\n");
268
- break;
269
- }
270
- case formFeedCode: {
271
- result.write("\\f");
272
- break;
273
- }
274
- case carriageReturnCode: {
275
- result.write("\\r");
276
- break;
277
- }
278
- default: {
279
- // all chars 0-31 must be encoded as a four digit unicode escape sequence
280
- // \u0000 to \u000f handled here
281
- result.write("\\u000");
282
- result.write(char.toString(16));
283
- break;
284
- }
285
- }
286
- } else if (char < 32) {
287
- result.write(<string>data, last, i);
288
- last = i + 1;
289
- // all chars 0-31 must be encoded as a four digit unicode escape sequence
290
- // \u0010 to \u001f handled here
291
- result.write("\\u00");
292
- result.write(char.toString(16));
293
- }
294
- }
295
- result.write(<string>data, last);
296
- result.writeCodePoint(quoteCode);
297
- return result.toString();
298
- }
299
-
300
- // @ts-ignore: Decorator
301
- @inline function parseString(data: string, start: i32 = 0, end: i32 = 0): string {
302
- end = end || data.length - 1;
303
- let result = StringSink.withCapacity(end - start - 1);
304
- let last = start + 1;
305
- for (let i = last; i < end; i++) {
306
- if (unsafeCharCodeAt(data, i) !== backSlashCode) {
307
- continue;
308
- }
309
- const char = unsafeCharCodeAt(data, ++i);
310
- result.write(data, last, i - 1);
311
- switch (char) {
312
- case quoteCode: {
313
- result.writeCodePoint(quoteCode);
314
- last = i + 1;
315
- break;
316
- }
317
- case backSlashCode: {
318
- result.writeCodePoint(backSlashCode);
319
- last = i + 1;
320
- break;
321
- }
322
- case forwardSlashCode: {
323
- result.writeCodePoint(forwardSlashCode);
324
- last = i + 1;
325
- break;
326
- }
327
- case bCode: {
328
- result.writeCodePoint(backspaceCode);
329
- last = i + 1;
330
- break;
331
- }
332
- case fCode: {
333
- result.writeCodePoint(formFeedCode);
334
- last = i + 1;
335
- break;
336
- }
337
- case nCode: {
338
- result.writeCodePoint(newLineCode);
339
- last = i + 1;
340
- break;
341
- }
342
- case rCode: {
343
- result.writeCodePoint(carriageReturnCode);
344
- last = i + 1;
345
- break;
346
- }
347
- case tCode: {
348
- result.writeCodePoint(tabCode);
349
- last = i + 1;
350
- break;
351
- }
352
- case uCode: {
353
- const code = u16.parse(data.slice(i + 1, i + 5), 16);
354
- result.writeCodePoint(code);
355
- i += 4;
356
- last = i + 1;
357
- break;
358
- }
359
- default: {
360
- throw new Error(`JSON: Cannot parse "${data}" as string. Invalid escape sequence: \\${data.charAt(i)}`);
361
- }
362
- }
363
- }
364
- if (end > last) {
365
- result.write(data, last, end);
366
- }
367
- return result.toString()
368
- }
369
-
370
- // @ts-ignore: Decorator
371
- @inline function parseBoolean<T extends boolean>(data: string): T {
372
- if (data.length > 3 && data.startsWith(trueWord)) return <T>true;
373
- else if (data.length > 4 && data.startsWith(falseWord)) return <T>false;
374
- else throw new Error(`JSON: Cannot parse "${data}" as boolean`);
375
- }
376
-
377
- // @ts-ignore: Decorator
378
- @inline function parseNumber<T>(data: string): T {
379
- if (isInteger<T>()) {
380
- // @ts-ignore
381
- return snip_fast<T>(data);
382
- }
383
- // @ts-ignore
384
- const type: T = 0;
385
- // @ts-ignore
386
- if (type instanceof f64) return f64.parse(data);
387
- // @ts-ignore
388
- else if (type instanceof f32) return f32.parse(data);
389
- }
390
-
391
- // @ts-ignore: Decorator
392
- @inline function parseNumberDirectInference<T>(type: T, data: string): T {
393
- if (isInteger(type)) {
394
- // @ts-ignore
395
- return snip_fast<T>(data);
396
- }
397
- // @ts-ignore
398
- if (type instanceof f64) return f64.parse(data);
399
- // @ts-ignore
400
- else if (type instanceof f32) return f32.parse(data);
401
- }
402
-
403
- // @ts-ignore: Decorator
404
- @inline function parseObject<T>(data: string, initializeDefaultValues: boolean): T {
405
- const schema: nonnull<T> = changetype<nonnull<T>>(
406
- __new(offsetof<nonnull<T>>(), idof<nonnull<T>>())
407
- );
408
-
409
- // @ts-ignore
410
- if (initializeDefaultValues) schema.__JSON_Initialize();
411
-
412
- const key = Virtual.createEmpty<string>();
413
- let isKey = false;
414
- let depth = 0;
415
- let outerLoopIndex = 1;
416
- for (; outerLoopIndex < data.length - 1; outerLoopIndex++) {
417
- const char = unsafeCharCodeAt(data, outerLoopIndex);
418
- if (char === leftBracketCode) {
419
- for (
420
- let arrayValueIndex = outerLoopIndex;
421
- arrayValueIndex < data.length - 1;
422
- arrayValueIndex++
423
- ) {
424
- const char = unsafeCharCodeAt(data, arrayValueIndex);
425
- if (char === leftBracketCode) {
426
- depth++;
427
- } else if (char === rightBracketCode) {
428
- depth--;
429
- if (depth === 0) {
430
- ++arrayValueIndex;
431
- // @ts-ignore
432
- schema.__JSON_Set_Key(key, data, outerLoopIndex, arrayValueIndex, initializeDefaultValues);
433
- outerLoopIndex = arrayValueIndex;
434
- isKey = false;
435
- break;
436
- }
437
- }
438
- }
439
- } else if (char === leftBraceCode) {
440
- for (
441
- let objectValueIndex = outerLoopIndex;
442
- objectValueIndex < data.length - 1;
443
- objectValueIndex++
444
- ) {
445
- const char = unsafeCharCodeAt(data, objectValueIndex);
446
- if (char === leftBraceCode) {
447
- depth++;
448
- } else if (char === rightBraceCode) {
449
- depth--;
450
- if (depth === 0) {
451
- ++objectValueIndex;
452
- // @ts-ignore
453
- schema.__JSON_Set_Key(key, data, outerLoopIndex, objectValueIndex, initializeDefaultValues);
454
- outerLoopIndex = objectValueIndex;
455
- isKey = false;
456
- break;
457
- }
458
- }
459
- }
460
- } else if (char === quoteCode) {
461
- let escaping = false;
462
- for (
463
- let stringValueIndex = ++outerLoopIndex;
464
- stringValueIndex < data.length - 1;
465
- stringValueIndex++
466
- ) {
467
- const char = unsafeCharCodeAt(data, stringValueIndex);
468
- if (char === backSlashCode && !escaping) {
469
- escaping = true;
470
- } else {
471
- if (char === quoteCode && !escaping) {
472
- if (isKey === false) {
473
- // perf: we can avoid creating a new string here if the key doesn't contain any escape sequences
474
- if (containsCodePoint(data, backSlashCode, outerLoopIndex, stringValueIndex)) {
475
- key.reinst(parseString(data, outerLoopIndex - 1, stringValueIndex));
476
- } else {
477
- key.reinst(data, outerLoopIndex, stringValueIndex);
478
- }
479
- isKey = true;
480
- } else {
481
- const value = parseString(data, outerLoopIndex - 1, stringValueIndex);
482
- // @ts-ignore
483
- schema.__JSON_Set_Key(key, value, 0, value.length, initializeDefaultValues);
484
- isKey = false;
485
- }
486
- outerLoopIndex = ++stringValueIndex;
487
- break;
488
- }
489
- escaping = false;
490
- }
491
- }
492
- } else if (
493
- char == nCode &&
494
- unsafeCharCodeAt(data, ++outerLoopIndex) === uCode &&
495
- unsafeCharCodeAt(data, ++outerLoopIndex) === lCode &&
496
- unsafeCharCodeAt(data, ++outerLoopIndex) === lCode
497
- ) {
498
- // @ts-ignore
499
- schema.__JSON_Set_Key(key, nullWord, 0, 4, initializeDefaultValues);
500
- isKey = false;
501
- } else if (
502
- char === tCode &&
503
- unsafeCharCodeAt(data, ++outerLoopIndex) === rCode &&
504
- unsafeCharCodeAt(data, ++outerLoopIndex) === uCode &&
505
- unsafeCharCodeAt(data, ++outerLoopIndex) === eCode
506
- ) {
507
- // @ts-ignore
508
- schema.__JSON_Set_Key(key, trueWord, 0, 4, initializeDefaultValues);
509
- isKey = false;
510
- } else if (
511
- char === fCode &&
512
- unsafeCharCodeAt(data, ++outerLoopIndex) === aCode &&
513
- unsafeCharCodeAt(data, ++outerLoopIndex) === lCode &&
514
- unsafeCharCodeAt(data, ++outerLoopIndex) === sCode &&
515
- unsafeCharCodeAt(data, ++outerLoopIndex) === eCode
516
- ) {
517
- // @ts-ignore
518
- schema.__JSON_Set_Key(key, falseWord, 0, 5, initializeDefaultValues);
519
- isKey = false;
520
- } else if ((char >= 48 && char <= 57) || char === 45) {
521
- let numberValueIndex = ++outerLoopIndex;
522
- for (; numberValueIndex < data.length; numberValueIndex++) {
523
- const char = unsafeCharCodeAt(data, numberValueIndex);
524
- if (char === commaCode || char === rightBraceCode || isSpace(char)) {
525
- // @ts-ignore
526
- schema.__JSON_Set_Key(key, data, outerLoopIndex - 1, numberValueIndex, initializeDefaultValues);
527
- outerLoopIndex = numberValueIndex;
528
- isKey = false;
529
- break;
530
- }
531
- }
532
- }
533
- }
534
- return schema;
535
- }
536
-
537
- // @ts-ignore: Decorator
538
- @inline function parseMap<T extends Map>(data: string, initializeDefaultValues: boolean): T {
539
-
540
- const map: nonnull<T> = changetype<nonnull<T>>(
541
- __new(offsetof<nonnull<T>>(), idof<nonnull<T>>())
542
- );
543
-
544
- if (!isDefined(map.set)) {
545
- throw new Error("Tried to parse a map, but the types did not match!")
546
- }
547
-
548
- const key = Virtual.createEmpty<string>();
549
- let isKey = false;
550
- let depth = 0;
551
- let outerLoopIndex = 1;
552
- for (; outerLoopIndex < data.length - 1; outerLoopIndex++) {
553
- const char = unsafeCharCodeAt(data, outerLoopIndex);
554
- if (char === leftBracketCode) {
555
- for (
556
- let arrayValueIndex = outerLoopIndex;
557
- arrayValueIndex < data.length - 1;
558
- arrayValueIndex++
559
- ) {
560
- const char = unsafeCharCodeAt(data, arrayValueIndex);
561
- if (char === leftBracketCode) {
562
- depth++;
563
- } else if (char === rightBracketCode) {
564
- depth--;
565
- if (depth === 0) {
566
- ++arrayValueIndex;
567
- map.set(parseMapKey<indexof<T>>(key), JSON.parse<valueof<T>>(data.slice(outerLoopIndex, arrayValueIndex), initializeDefaultValues));
568
- outerLoopIndex = arrayValueIndex;
569
- isKey = false;
570
- break;
571
- }
572
- }
573
- }
574
- } else if (char === leftBraceCode) {
575
- for (
576
- let objectValueIndex = outerLoopIndex;
577
- objectValueIndex < data.length - 1;
578
- objectValueIndex++
579
- ) {
580
- const char = unsafeCharCodeAt(data, objectValueIndex);
581
- if (char === leftBraceCode) {
582
- depth++;
583
- } else if (char === rightBraceCode) {
584
- depth--;
585
- if (depth === 0) {
586
- ++objectValueIndex;
587
- map.set(parseMapKey<indexof<T>>(key), JSON.parse<valueof<T>>(data.slice(outerLoopIndex, objectValueIndex), initializeDefaultValues));
588
- outerLoopIndex = objectValueIndex;
589
- isKey = false;
590
- break;
591
- }
592
- }
593
- }
594
- } else if (char === quoteCode) {
595
- let escaping = false;
596
- for (
597
- let stringValueIndex = ++outerLoopIndex;
598
- stringValueIndex < data.length - 1;
599
- stringValueIndex++
600
- ) {
601
- const char = unsafeCharCodeAt(data, stringValueIndex);
602
- if (char === backSlashCode && !escaping) {
603
- escaping = true;
604
- } else {
605
- if (
606
- char === quoteCode && !escaping
607
- ) {
608
- if (isKey === false) {
609
- // perf: we can avoid creating a new string here if the key doesn't contain any escape sequences
610
- if (containsCodePoint(data, backSlashCode, outerLoopIndex, stringValueIndex)) {
611
- key.reinst(parseString(data, outerLoopIndex - 1, stringValueIndex));
612
- } else {
613
- key.reinst(data, outerLoopIndex, stringValueIndex);
614
- }
615
- isKey = true;
616
- } else {
617
- if (isString<valueof<T>>()) {
618
- const value = parseString(data, outerLoopIndex - 1, stringValueIndex);
619
- map.set(parseMapKey<indexof<T>>(key), value);
620
- }
621
- isKey = false;
622
- }
623
- outerLoopIndex = ++stringValueIndex;
624
- break;
625
- }
626
- escaping = false;
627
- }
628
- }
629
- } else if (
630
- char == nCode &&
631
- unsafeCharCodeAt(data, ++outerLoopIndex) === uCode &&
632
- unsafeCharCodeAt(data, ++outerLoopIndex) === lCode &&
633
- unsafeCharCodeAt(data, ++outerLoopIndex) === lCode) {
634
- if (isNullable<valueof<T>>()) {
635
- map.set(parseMapKey<indexof<T>>(key), null);
636
- }
637
- isKey = false;
638
- } else if (
639
- char === tCode &&
640
- unsafeCharCodeAt(data, ++outerLoopIndex) === rCode &&
641
- unsafeCharCodeAt(data, ++outerLoopIndex) === uCode &&
642
- unsafeCharCodeAt(data, ++outerLoopIndex) === eCode
643
- ) {
644
- if (isBoolean<valueof<T>>()) {
645
- map.set(parseMapKey<indexof<T>>(key), true);
646
- }
647
- isKey = false;
648
- } else if (
649
- char === fCode &&
650
- unsafeCharCodeAt(data, ++outerLoopIndex) === aCode &&
651
- unsafeCharCodeAt(data, ++outerLoopIndex) === lCode &&
652
- unsafeCharCodeAt(data, ++outerLoopIndex) === sCode &&
653
- unsafeCharCodeAt(data, ++outerLoopIndex) === eCode
654
- ) {
655
- if (isBoolean<valueof<T>>()) {
656
- map.set(parseMapKey<indexof<T>>(key), false);
657
- }
658
- isKey = false;
659
- } else if ((char >= 48 && char <= 57) || char === 45) {
660
- let numberValueIndex = ++outerLoopIndex;
661
- for (; numberValueIndex < data.length; numberValueIndex++) {
662
- const char = unsafeCharCodeAt(data, numberValueIndex);
663
- if (char === colonCode || char === commaCode || char === rightBraceCode || isSpace(char)) {
664
- if (isFloat<valueof<T>>() || isInteger<valueof<T>>()) {
665
- map.set(parseMapKey<indexof<T>>(key), parseNumber<valueof<T>>(data.slice(outerLoopIndex - 1, numberValueIndex)));
666
- }
667
- outerLoopIndex = numberValueIndex;
668
- isKey = false;
669
- break;
670
- }
671
- }
672
- }
673
- }
674
-
675
- return map;
676
- }
677
-
678
- //@ts-ignore: Decorator
679
- @inline function parseMapKey<T>(key: Virtual<string>): T {
680
- const k = key.copyOut();
681
- if (isString<T>()) {
682
- return k as T;
683
- } else if (isBoolean<T>()) {
684
- // @ts-ignore
685
- return parseBoolean<T>(k) as T;
686
- } else if (isInteger<T>() || isFloat<T>()) {
687
- return parseNumber<T>(k);
688
- }
689
-
690
- throw new Error(`JSON: Cannot parse JSON object to a Map with a key of type ${nameof<T>()}`);
691
- }
692
-
693
- // @ts-ignore: Decorator
694
- @inline function parseArray<T extends unknown[]>(data: string): T {
695
- if (isString<valueof<T>>()) {
696
- return <T>parseStringArray(data);
697
- } else if (isBoolean<valueof<T>>()) {
698
- // @ts-ignore
699
- return parseBooleanArray<T>(data);
700
- } else if (isFloat<valueof<T>>() || isInteger<valueof<T>>()) {
701
- // @ts-ignore
702
- return parseNumberArray<T>(data);
703
- } else if (isArrayLike<valueof<T>>()) {
704
- // @ts-ignore
705
- return parseArrayArray<T>(data);
706
- } else if (isMap<valueof<T>>()) {
707
- return parseObjectArray<T>(data);
708
- } else if (isManaged<valueof<T>>() || isReference<valueof<T>>()) {
709
- // We instantiate the required memory for the class and fill it. This is extremely unsafe and uses "a bit of magic".
710
- const type = changetype<nonnull<valueof<T>>>(
711
- __new(offsetof<nonnull<valueof<T>>>(), idof<nonnull<valueof<T>>>())
712
- );
713
- // @ts-ignore
714
- if (isDefined(type.__JSON_Set_Key)) {
715
- return parseObjectArray<T>(data);
716
- }
717
- }
718
-
719
- throw new Error("Tried to parse array, but failed!")
720
- }
721
-
722
- // @ts-ignore: Decorator
723
- @inline function parseStringArray(data: string): string[] {
724
- const result: string[] = [];
725
- let lastPos = 0;
726
- let instr = false;
727
- let escaping = false;
728
- for (let i = 1; i < data.length - 1; i++) {
729
- const char = unsafeCharCodeAt(data, i);
730
- if (char === backSlashCode && !escaping) {
731
- escaping = true;
732
- } else {
733
- if (char === quoteCode && !escaping) {
734
- if (instr === false) {
735
- instr = true;
736
- lastPos = i;
737
- } else {
738
- instr = false;
739
- result.push(parseString(data, lastPos, i));
740
- }
741
- }
742
- escaping = false;
743
- }
744
- }
745
- return result;
746
- }
747
-
748
- // @ts-ignore: Decorator
749
- @inline function parseBooleanArray<T extends boolean[]>(data: string): T {
750
- const result = instantiate<T>();
751
- let lastPos = 1;
752
- for (let i = 1; i < data.length - 1; i++) {
753
- const char = unsafeCharCodeAt(data, i);
754
- if (char === tCode || char === fCode) {
755
- lastPos = i;
756
- } else if (char === eCode) {
757
- i++;
758
- result.push(parseBoolean<valueof<T>>(data.slice(lastPos, i)));
759
- }
760
- }
761
- return result;
762
- }
763
-
764
- // @ts-ignore: Decorator
765
- @inline function parseNumberArray<T extends number[]>(data: string): T {
766
- const result = instantiate<T>();
767
- let lastPos = 0;
768
- let i = 1;
769
- let awaitingParse = false;
770
- for (; i < data.length; i++) {
771
- const char = unsafeCharCodeAt(data, i);
772
- if (lastPos === 0 && ((char >= 48 && char <= 57) || char === 45)) {
773
- awaitingParse = true;
774
- lastPos = i;
775
- } else if (awaitingParse && (isSpace(char) || char == commaCode || char == rightBracketCode) && lastPos > 0) {
776
- awaitingParse = false;
777
- result.push(parseNumber<valueof<T>>(data.slice(lastPos, i)));
778
- lastPos = 0;
779
- }
780
- }
781
- return result;
782
- }
783
-
784
- // @ts-ignore: Decorator
785
- @inline function parseArrayArray<T extends unknown[][]>(data: string): T {
786
- const result = instantiate<T>();
787
- let lastPos = 0;
788
- let depth = 0;
789
- let i = 1;
790
- // Find start of bracket
791
- //for (; unsafeCharCodeAt(data, i) !== leftBracketCode; i++) {}
792
- //i++;
793
- for (; i < data.length - 1; i++) {
794
- const char = unsafeCharCodeAt(data, i);
795
- if (char === leftBracketCode) {
796
- if (depth === 0) {
797
- lastPos = i;
798
- }
799
- // Shifting is 6% faster than incrementing
800
- depth++;
801
- } else if (char === rightBracketCode) {
802
- depth--;
803
- if (depth === 0) {
804
- i++;
805
- result.push(JSON.parse<valueof<T>>(data.slice(lastPos, i)));
806
- }
807
- }
808
- }
809
- return result;
810
- }
811
-
812
- // @ts-ignore: Decorator
813
- @inline function parseObjectArray<T extends unknown[]>(data: string): T {
814
- const result = instantiate<T>();
815
- let lastPos: u32 = 1;
816
- let depth: u32 = 0;
817
- for (let pos: u32 = 0; pos < <u32>data.length; pos++) {
818
- const char = unsafeCharCodeAt(data, pos);
819
- if (char === leftBraceCode) {
820
- if (depth === 0) {
821
- lastPos = pos;
822
- }
823
- depth++;
824
- } else if (char === rightBraceCode) {
825
- depth--;
826
- if (depth === 0) {
827
- pos++;
828
- result.push(JSON.parse<valueof<T>>(data.slice(lastPos, pos)));
829
- //lastPos = pos + 2;
830
- }
831
- }
832
- }
833
- return result;
834
- }
835
-
836
- function parseDate(dateTimeString: string): Date {
837
- // Use AssemblyScript's date parser
838
- const d = Date.fromString(dateTimeString);
839
-
840
- // Return a new object instead of the one that the parser returned.
841
- // This may seem redundant, but addreses the issue when Date
842
- // is globally aliased to wasi_Date (or some other superclass).
843
- return new Date(d.getTime());
844
- }
845
-
846
- // @ts-ignore: Decorator
847
- @inline function isMap<T>(): bool {
848
- let type = changetype<T>(0);
849
- return type instanceof Map;
850
- }
851
-
852
- // Dirty fix
853
- // @ts-ignore: Decorator
854
- @global @inline function __JSON_Stringify<T>(data: T): string {
855
- // String
856
- if (isString<T>() && data != null) {
857
- return serializeString(data as string);
858
- } else if (isBoolean<T>()) {
859
- return data ? "true" : "false";
860
- } else if (isNullable<T>() && changetype<usize>(data) == <usize>0) {
861
- return nullWord;
862
- // @ts-ignore
863
- } else if ((isInteger<T>() || isFloat<T>()) && isFinite(data)) {
864
- // @ts-ignore
865
- return data.toString();
866
- // @ts-ignore: Hidden function
867
- } else if (isDefined(data.__JSON_Serialize)) {
868
- // @ts-ignore: Hidden function
869
- return data.__JSON_Serialize();
870
- } else if (data instanceof Date) {
871
- return `"${data.toISOString()}"`;
872
- } else if (data instanceof Box) {
873
- if (isNullable<T>() && changetype<usize>(data) == <usize>0) {
874
- return nullWord;
875
- }
876
- return JSON.stringify(data.unwrap());
877
- } else if (isArrayLike<T>()) {
878
- // @ts-ignore
879
- if (data.length == 0) {
880
- return emptyArrayWord;
881
- // @ts-ignore
882
- } else if (isString<valueof<T>>()) {
883
- let result = leftBracketWord;
884
- // @ts-ignore
885
- for (let i = 0; i < data.length - 1; i++) {
886
- // @ts-ignore
887
- result += serializeString(unchecked(data[i]));
888
- result += commaWord;
889
- }
890
- // @ts-ignore
891
- result += serializeString(unchecked(data[data.length - 1]));
892
- result += rightBracketWord;
893
- return result;
894
- // @ts-ignore
895
- } else if (isBoolean<valueof<T>>()) {
896
- // @ts-ignore
897
- return leftBracketWord + data.join(commaWord) + rightBracketWord;
898
- // @ts-ignore
899
- } else if (isFloat<valueof<T>>() || isInteger<valueof<T>>()) {
900
- // @ts-ignore
901
- return leftBracketWord + data.join(commaWord) + rightBracketWord;
902
- } else {
903
- let result = new StringSink(leftBracketWord);
904
- // @ts-ignore
905
- for (let i = 0; i < data.length - 1; i++) {
906
- // @ts-ignore
907
- result.write(__JSON_Stringify(unchecked(data[i])));
908
- result.writeCodePoint(commaCode);
909
- }
910
- // @ts-ignore
911
- result.write(__JSON_Stringify(unchecked(data[data.length - 1])));
912
- result.writeCodePoint(rightBracketCode);
913
- return result.toString();
914
- }
915
- } else if (data instanceof Map) {
916
- let result = new StringSink(leftBraceWord);
917
- let keys = data.keys();
918
- let values = data.values();
919
- const end = data.size - 1;
920
- for (let i = 0; i < end; i++) {
921
- result.write(serializeString(unchecked(keys[i]).toString()));
922
- result.writeCodePoint(colonCode);
923
- result.write(__JSON_Stringify(unchecked(values[i])));
924
- result.writeCodePoint(commaCode);
925
- }
926
- result.write(serializeString(unchecked(keys[end]).toString()));
927
- result.writeCodePoint(colonCode);
928
- result.write(__JSON_Stringify(unchecked(values[end])));
929
-
930
- result.writeCodePoint(rightBraceCode);
931
- return result.toString();
932
- } else {
933
- throw new Error(
934
- `Could not serialize data of type ${nameof<T>()}. Make sure to add the correct decorators to classes.`
935
- );
936
- }
937
- }
938
-
939
- @inline function parseDirectInference<T>(type: T, data: string, initializeDefaultValues: boolean = false): T {
940
- return JSON.parse<T>(data, initializeDefaultValues)
941
- }