1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36 #ifndef UCX_JSON_H
37 #define UCX_JSON_H
38
39 #include "common.h"
40 #include "allocator.h"
41 #include "string.h"
42 #include "buffer.h"
43 #include "array_list.h"
44
45 #include <string.h>
46
47 #ifdef __cplusplus
48 extern "C" {
49 #endif
50
51
52
53
54
55 enum cx_json_token_type {
56
57
58
59 CX_JSON_NO_TOKEN,
60
61
62
63 CX_JSON_TOKEN_ERROR,
64
65
66
67 CX_JSON_TOKEN_BEGIN_ARRAY,
68
69
70
71 CX_JSON_TOKEN_BEGIN_OBJECT,
72
73
74
75 CX_JSON_TOKEN_END_ARRAY,
76
77
78
79 CX_JSON_TOKEN_END_OBJECT,
80
81
82
83 CX_JSON_TOKEN_NAME_SEPARATOR,
84
85
86
87 CX_JSON_TOKEN_VALUE_SEPARATOR,
88
89
90
91 CX_JSON_TOKEN_STRING,
92
93
94
95 CX_JSON_TOKEN_INTEGER,
96
97
98
99 CX_JSON_TOKEN_NUMBER,
100
101
102
103 CX_JSON_TOKEN_LITERAL,
104
105
106
107 CX_JSON_TOKEN_SPACE
108 };
109
110
111
112
113 enum cx_json_value_type {
114
115
116
117 CX_JSON_NOTHING,
118
119
120
121 CX_JSON_OBJECT,
122
123
124
125 CX_JSON_ARRAY,
126
127
128
129 CX_JSON_STRING,
130
131
132
133 CX_JSON_INTEGER,
134
135
136
137 CX_JSON_NUMBER,
138
139
140
141 CX_JSON_LITERAL
142 };
143
144
145
146
147 enum cx_json_literal {
148
149
150
151 CX_JSON_NULL,
152
153
154
155 CX_JSON_TRUE,
156
157
158
159 CX_JSON_FALSE
160 };
161
162
163
164
165 typedef enum cx_json_token_type CxJsonTokenType;
166
167
168
169 typedef enum cx_json_value_type CxJsonValueType;
170
171
172
173
174 typedef struct cx_json_s CxJson;
175
176
177
178
179 typedef struct cx_json_token_s CxJsonToken;
180
181
182
183
184 typedef struct cx_json_value_s CxJsonValue;
185
186
187
188
189 typedef struct cx_json_array_s CxJsonArray;
190
191
192
193 typedef struct cx_json_object_s CxJsonObject;
194
195
196
197 typedef struct cx_mutstr_s CxJsonString;
198
199
200
201 typedef int64_t CxJsonInteger;
202
203
204
205 typedef double CxJsonNumber;
206
207
208
209 typedef enum cx_json_literal CxJsonLiteral;
210
211
212
213
214 typedef struct cx_json_obj_value_s CxJsonObjValue;
215
216
217
218
219 struct cx_json_array_s {
220
221
222
223 CX_ARRAY_DECLARE(CxJsonValue*, array);
224 };
225
226
227
228
229 struct cx_json_object_s {
230
231
232
233 CX_ARRAY_DECLARE(CxJsonObjValue, values);
234
235
236
237 size_t *indices;
238 };
239
240
241
242
243 struct cx_json_obj_value_s {
244
245
246
247 cxmutstr name;
248
249
250
251 CxJsonValue *value;
252 };
253
254
255
256
257 struct cx_json_value_s {
258
259
260
261
262
263 const CxAllocator *allocator;
264
265
266
267
268
269 CxJsonValueType type;
270
271
272
273 union {
274
275
276
277 CxJsonArray array;
278
279
280
281 CxJsonObject object;
282
283
284
285 CxJsonString string;
286
287
288
289 CxJsonInteger integer;
290
291
292
293 CxJsonNumber number;
294
295
296
297 CxJsonLiteral literal;
298 } value;
299 };
300
301
302
303
304
305
306 struct cx_json_token_s {
307
308
309
310 CxJsonTokenType tokentype;
311
312
313
314 bool allocated;
315
316
317
318
319
320
321 cxmutstr content;
322 };
323
324
325
326
327 struct cx_json_s {
328
329
330
331 const CxAllocator *allocator;
332
333
334
335 CxBuffer buffer;
336
337
338
339
340
341
342 CxJsonToken uncompleted;
343
344
345
346
347
348
349 CxJsonValue *parsed;
350
351
352
353
354
355
356 CxJsonObjValue uncompleted_member;
357
358
359
360
361 CX_ARRAY_DECLARE_SIZED(
int, states,
unsigned);
362
363
364
365
366 CX_ARRAY_DECLARE_SIZED(CxJsonValue*, vbuf,
unsigned);
367
368
369
370
371 int states_internal[
8];
372
373
374
375
376 CxJsonValue* vbuf_internal[
8];
377 };
378
379
380
381
382 enum cx_json_status {
383
384
385
386 CX_JSON_NO_ERROR,
387
388
389
390 CX_JSON_NO_DATA,
391
392
393
394
395
396 CX_JSON_INCOMPLETE_DATA,
397
398
399
400
401
402
403
404
405 CX_JSON_OK,
406
407
408
409 CX_JSON_NULL_DATA,
410
411
412
413 CX_JSON_BUFFER_ALLOC_FAILED,
414
415
416
417 CX_JSON_VALUE_ALLOC_FAILED,
418
419
420
421 CX_JSON_FORMAT_ERROR_NUMBER,
422
423
424
425 CX_JSON_FORMAT_ERROR_UNEXPECTED_TOKEN
426 };
427
428
429
430
431 typedef enum cx_json_status CxJsonStatus;
432
433
434
435
436 struct cx_json_writer_s {
437
438
439
440 bool pretty;
441
442
443
444 bool sort_members;
445
446
447
448
449
450 uint8_t frac_max_digits;
451
452
453
454
455 bool indent_space;
456
457
458
459
460 uint8_t indent;
461
462
463
464 bool escape_slash;
465 };
466
467
468
469
470 typedef struct cx_json_writer_s CxJsonWriter;
471
472
473
474
475
476
477 cx_attr_nodiscard
478 cx_attr_export
479 CxJsonWriter cxJsonWriterCompact(
void);
480
481
482
483
484
485
486
487 cx_attr_nodiscard
488 cx_attr_export
489 CxJsonWriter cxJsonWriterPretty(bool use_spaces);
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510 cx_attr_nonnull_arg(
1,
2,
3)
511 cx_attr_export
512 int cxJsonWrite(
513 void* target,
514 const CxJsonValue* value,
515 cx_write_func wfunc,
516 const CxJsonWriter* settings
517 );
518
519
520
521
522
523
524
525
526 cx_attr_nonnull_arg(
1)
527 cx_attr_export
528 void cxJsonInit(CxJson *json,
const CxAllocator *allocator);
529
530
531
532
533
534
535
536 cx_attr_nonnull
537 cx_attr_export
538 void cxJsonDestroy(CxJson *json);
539
540
541
542
543
544
545
546
547
548 cx_attr_nonnull
549 static inline
void cxJsonReset(CxJson *json) {
550 const CxAllocator *allocator = json->allocator;
551 cxJsonDestroy(json);
552 cxJsonInit(json, allocator);
553 }
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573 cx_attr_nonnull
574 cx_attr_access_r(
2,
3)
575 cx_attr_export
576 int cxJsonFilln(CxJson *json,
const char *buf,
size_t len);
577
578 #ifdef __cplusplus
579 }
580
581 cx_attr_nonnull
582 static inline
int cxJsonFill(
583 CxJson *json,
584 cxstring str
585 ) {
586 return cxJsonFilln(json, str.ptr, str.length);
587 }
588
589 cx_attr_nonnull
590 static inline
int cxJsonFill(
591 CxJson *json,
592 cxmutstr str
593 ) {
594 return cxJsonFilln(json, str.ptr, str.length);
595 }
596
597 cx_attr_nonnull
598 cx_attr_cstr_arg(
2)
599 static inline
int cxJsonFill(
600 CxJson *json,
601 const char *str
602 ) {
603 return cxJsonFilln(json, str, strlen(str));
604 }
605
606 extern "C" {
607 #else
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625 #define cxJsonFill(json, str) _Generic((str), \
626 cxstring: cx_json_fill_cxstr, \
627 cxmutstr: cx_json_fill_mutstr, \
628 char*: cx_json_fill_str, \
629 const char*: cx_json_fill_str) \
630 (json, str)
631
632
633
634
635 cx_attr_nonnull
636 static inline
int cx_json_fill_cxstr(
637 CxJson *json,
638 cxstring str
639 ) {
640 return cxJsonFilln(json, str.ptr, str.length);
641 }
642
643
644
645
646 cx_attr_nonnull
647 static inline
int cx_json_fill_mutstr(
648 CxJson *json,
649 cxmutstr str
650 ) {
651 return cxJsonFilln(json, str.ptr, str.length);
652 }
653
654
655
656
657 cx_attr_nonnull
658 cx_attr_cstr_arg(
2)
659 static inline
int cx_json_fill_str(
660 CxJson *json,
661 const char *str
662 ) {
663 return cxJsonFilln(json, str, strlen(str));
664 }
665 #endif
666
667
668
669
670
671
672
673
674
675 cx_attr_nodiscard
676 cx_attr_export
677 CxJsonValue* cxJsonCreateObj(
const CxAllocator* allocator);
678
679
680
681
682
683
684
685
686
687 cx_attr_nodiscard
688 cx_attr_export
689 CxJsonValue* cxJsonCreateArr(
const CxAllocator* allocator);
690
691
692
693
694
695
696
697
698
699
700 cx_attr_nodiscard
701 cx_attr_export
702 CxJsonValue* cxJsonCreateNumber(
const CxAllocator* allocator,
double num);
703
704
705
706
707
708
709
710
711
712
713 cx_attr_nodiscard
714 cx_attr_export
715 CxJsonValue* cxJsonCreateInteger(
const CxAllocator* allocator,
int64_t num);
716
717
718
719
720
721
722
723
724
725
726
727 cx_attr_nodiscard
728 cx_attr_nonnull_arg(
2)
729 cx_attr_cstr_arg(
2)
730 cx_attr_export
731 CxJsonValue* cxJsonCreateString(
const CxAllocator* allocator,
const char *str);
732
733
734
735
736
737
738
739
740
741
742
743 cx_attr_nodiscard
744 cx_attr_export
745 CxJsonValue* cxJsonCreateCxString(
const CxAllocator* allocator, cxstring str);
746
747
748
749
750
751
752
753
754
755
756 cx_attr_nodiscard
757 cx_attr_export
758 CxJsonValue* cxJsonCreateLiteral(
const CxAllocator* allocator, CxJsonLiteral lit);
759
760
761
762
763
764
765
766
767
768
769 cx_attr_nonnull
770 cx_attr_access_r(
2,
3)
771 cx_attr_export
772 int cxJsonArrAddNumbers(CxJsonValue* arr,
const double* num,
size_t count);
773
774
775
776
777
778
779
780
781
782
783 cx_attr_nonnull
784 cx_attr_access_r(
2,
3)
785 cx_attr_export
786 int cxJsonArrAddIntegers(CxJsonValue* arr,
const int64_t* num,
size_t count);
787
788
789
790
791
792
793
794
795
796
797
798
799
800 cx_attr_nonnull
801 cx_attr_access_r(
2,
3)
802 cx_attr_export
803 int cxJsonArrAddStrings(CxJsonValue* arr,
const char*
const* str,
size_t count);
804
805
806
807
808
809
810
811
812
813
814
815
816
817 cx_attr_nonnull
818 cx_attr_access_r(
2,
3)
819 cx_attr_export
820 int cxJsonArrAddCxStrings(CxJsonValue* arr,
const cxstring* str,
size_t count);
821
822
823
824
825
826
827
828
829
830
831 cx_attr_nonnull
832 cx_attr_access_r(
2,
3)
833 cx_attr_export
834 int cxJsonArrAddLiterals(CxJsonValue* arr,
const CxJsonLiteral* lit,
size_t count);
835
836
837
838
839
840
841
842
843
844
845
846
847
848 cx_attr_nonnull
849 cx_attr_access_r(
2,
3)
850 cx_attr_export
851 int cxJsonArrAddValues(CxJsonValue* arr, CxJsonValue*
const* val,
size_t count);
852
853
854
855
856
857
858
859
860
861
862
863
864
865
866
867 cx_attr_nonnull
868 cx_attr_export
869 int cxJsonObjPut(CxJsonValue* obj, cxstring name, CxJsonValue* child);
870
871
872
873
874
875
876
877
878
879
880 cx_attr_nonnull
881 cx_attr_export
882 CxJsonValue* cxJsonObjPutObj(CxJsonValue* obj, cxstring name);
883
884
885
886
887
888
889
890
891
892
893 cx_attr_nonnull
894 cx_attr_export
895 CxJsonValue* cxJsonObjPutArr(CxJsonValue* obj, cxstring name);
896
897
898
899
900
901
902
903
904
905
906
907 cx_attr_nonnull
908 cx_attr_export
909 CxJsonValue* cxJsonObjPutNumber(CxJsonValue* obj, cxstring name,
double num);
910
911
912
913
914
915
916
917
918
919
920
921 cx_attr_nonnull
922 cx_attr_export
923 CxJsonValue* cxJsonObjPutInteger(CxJsonValue* obj, cxstring name,
int64_t num);
924
925
926
927
928
929
930
931
932
933
934
935
936
937 cx_attr_nonnull
938 cx_attr_cstr_arg(
3)
939 cx_attr_export
940 CxJsonValue* cxJsonObjPutString(CxJsonValue* obj, cxstring name,
const char* str);
941
942
943
944
945
946
947
948
949
950
951
952
953
954 cx_attr_nonnull
955 cx_attr_export
956 CxJsonValue* cxJsonObjPutCxString(CxJsonValue* obj, cxstring name, cxstring str);
957
958
959
960
961
962
963
964
965
966
967
968 cx_attr_nonnull
969 cx_attr_export
970 CxJsonValue* cxJsonObjPutLiteral(CxJsonValue* obj, cxstring name, CxJsonLiteral lit);
971
972
973
974
975
976
977
978
979
980
981
982
983 cx_attr_export
984 void cxJsonValueFree(CxJsonValue *value);
985
986
987
988
989
990
991
992
993
994
995
996
997
998
999
1000
1001
1002
1003
1004
1005
1006
1007
1008 cx_attr_nonnull
1009 cx_attr_access_w(
2)
1010 cx_attr_export
1011 CxJsonStatus cxJsonNext(CxJson *json, CxJsonValue **value);
1012
1013
1014
1015
1016
1017
1018
1019
1020 cx_attr_nonnull
1021 static inline bool cxJsonIsObject(
const CxJsonValue *value) {
1022 return value->type ==
CX_JSON_OBJECT;
1023 }
1024
1025
1026
1027
1028
1029
1030
1031
1032 cx_attr_nonnull
1033 static inline bool cxJsonIsArray(
const CxJsonValue *value) {
1034 return value->type ==
CX_JSON_ARRAY;
1035 }
1036
1037
1038
1039
1040
1041
1042
1043
1044 cx_attr_nonnull
1045 static inline bool cxJsonIsString(
const CxJsonValue *value) {
1046 return value->type ==
CX_JSON_STRING;
1047 }
1048
1049
1050
1051
1052
1053
1054
1055
1056
1057
1058
1059
1060 cx_attr_nonnull
1061 static inline bool cxJsonIsNumber(
const CxJsonValue *value) {
1062 return value->type ==
CX_JSON_NUMBER || value->type ==
CX_JSON_INTEGER;
1063 }
1064
1065
1066
1067
1068
1069
1070
1071
1072
1073 cx_attr_nonnull
1074 static inline bool cxJsonIsInteger(
const CxJsonValue *value) {
1075 return value->type ==
CX_JSON_INTEGER;
1076 }
1077
1078
1079
1080
1081
1082
1083
1084
1085
1086
1087
1088
1089
1090 cx_attr_nonnull
1091 static inline bool cxJsonIsLiteral(
const CxJsonValue *value) {
1092 return value->type ==
CX_JSON_LITERAL;
1093 }
1094
1095
1096
1097
1098
1099
1100
1101
1102
1103
1104 cx_attr_nonnull
1105 static inline bool cxJsonIsBool(
const CxJsonValue *value) {
1106 return cxJsonIsLiteral(value) && value->value.literal !=
CX_JSON_NULL;
1107 }
1108
1109
1110
1111
1112
1113
1114
1115
1116
1117
1118
1119
1120
1121 cx_attr_nonnull
1122 static inline bool cxJsonIsTrue(
const CxJsonValue *value) {
1123 return cxJsonIsLiteral(value) && value->value.literal ==
CX_JSON_TRUE;
1124 }
1125
1126
1127
1128
1129
1130
1131
1132
1133
1134
1135
1136
1137
1138 cx_attr_nonnull
1139 static inline bool cxJsonIsFalse(
const CxJsonValue *value) {
1140 return cxJsonIsLiteral(value) && value->value.literal ==
CX_JSON_FALSE;
1141 }
1142
1143
1144
1145
1146
1147
1148
1149
1150
1151 cx_attr_nonnull
1152 static inline bool cxJsonIsNull(
const CxJsonValue *value) {
1153 return cxJsonIsLiteral(value) && value->value.literal ==
CX_JSON_NULL;
1154 }
1155
1156
1157
1158
1159
1160
1161
1162
1163
1164
1165 cx_attr_nonnull
1166 cx_attr_returns_nonnull
1167 static inline
char *cxJsonAsString(
const CxJsonValue *value) {
1168 return value->value.string.ptr;
1169 }
1170
1171
1172
1173
1174
1175
1176
1177
1178
1179
1180 cx_attr_nonnull
1181 static inline cxstring cxJsonAsCxString(
const CxJsonValue *value) {
1182 return cx_strcast(value->value.string);
1183 }
1184
1185
1186
1187
1188
1189
1190
1191
1192
1193
1194 cx_attr_nonnull
1195 static inline cxmutstr cxJsonAsCxMutStr(
const CxJsonValue *value) {
1196 return value->value.string;
1197 }
1198
1199
1200
1201
1202
1203
1204
1205
1206
1207
1208 cx_attr_nonnull
1209 static inline
double cxJsonAsDouble(
const CxJsonValue *value) {
1210 if (value->type ==
CX_JSON_INTEGER) {
1211 return (
double) value->value.integer;
1212 }
else {
1213 return value->value.number;
1214 }
1215 }
1216
1217
1218
1219
1220
1221
1222
1223
1224
1225
1226
1227
1228
1229 cx_attr_nonnull
1230 static inline
int64_t cxJsonAsInteger(
const CxJsonValue *value) {
1231 if (value->type ==
CX_JSON_INTEGER) {
1232 return value->value.integer;
1233 }
else {
1234 return (
int64_t) value->value.number;
1235 }
1236 }
1237
1238
1239
1240
1241
1242
1243
1244
1245
1246
1247
1248 cx_attr_nonnull
1249 static inline bool cxJsonAsBool(
const CxJsonValue *value) {
1250 return value->value.literal ==
CX_JSON_TRUE;
1251 }
1252
1253
1254
1255
1256
1257
1258
1259
1260
1261
1262 cx_attr_nonnull
1263 static inline
size_t cxJsonArrSize(
const CxJsonValue *value) {
1264 return value->value.array.array_size;
1265 }
1266
1267
1268
1269
1270
1271
1272
1273
1274
1275
1276
1277
1278
1279
1280
1281 cx_attr_nonnull
1282 cx_attr_returns_nonnull
1283 cx_attr_export
1284 CxJsonValue *cxJsonArrGet(
const CxJsonValue *value,
size_t index);
1285
1286
1287
1288
1289
1290
1291
1292
1293
1294
1295
1296
1297 cx_attr_nonnull
1298 cx_attr_nodiscard
1299 cx_attr_export
1300 CxIterator cxJsonArrIter(
const CxJsonValue *value);
1301
1302
1303
1304
1305
1306
1307
1308
1309
1310
1311
1312
1313
1314 cx_attr_nonnull
1315 cx_attr_nodiscard
1316 cx_attr_export
1317 CxIterator cxJsonObjIter(
const CxJsonValue *value);
1318
1319
1320
1321
1322 cx_attr_nonnull
1323 cx_attr_returns_nonnull
1324 cx_attr_export
1325 CxJsonValue *cx_json_obj_get_cxstr(
const CxJsonValue *value, cxstring name);
1326
1327 #ifdef __cplusplus
1328 }
1329
1330 static inline CxJsonValue *cxJsonObjGet(
const CxJsonValue *value, cxstring name) {
1331 return cx_json_obj_get_cxstr(value, name);
1332 }
1333
1334 static inline CxJsonValue *cxJsonObjGet(
const CxJsonValue *value, cxmutstr name) {
1335 return cx_json_obj_get_cxstr(value, cx_strcast(name));
1336 }
1337
1338 static inline CxJsonValue *cxJsonObjGet(
const CxJsonValue *value,
const char *name) {
1339 return cx_json_obj_get_cxstr(value, cx_str(name));
1340 }
1341
1342 extern "C" {
1343 #else
1344
1345
1346
1347
1348
1349
1350
1351
1352
1353
1354
1355
1356
1357
1358 #define cxJsonObjGet(value, name) _Generic((name), \
1359 cxstring: cx_json_obj_get_cxstr, \
1360 cxmutstr: cx_json_obj_get_mutstr, \
1361 char*: cx_json_obj_get_str, \
1362 const char*: cx_json_obj_get_str) \
1363 (value, name)
1364
1365
1366
1367
1368 cx_attr_nonnull
1369 cx_attr_returns_nonnull
1370 static inline CxJsonValue *cx_json_obj_get_mutstr(
const CxJsonValue *value, cxmutstr name) {
1371 return cx_json_obj_get_cxstr(value, cx_strcast(name));
1372 }
1373
1374
1375
1376
1377 cx_attr_nonnull
1378 cx_attr_returns_nonnull
1379 cx_attr_cstr_arg(
2)
1380 static inline CxJsonValue *cx_json_obj_get_str(
const CxJsonValue *value,
const char *name) {
1381 return cx_json_obj_get_cxstr(value, cx_str(name));
1382 }
1383 #endif
1384
1385 #ifdef __cplusplus
1386 }
1387 #endif
1388
1389 #endif
1390
1391