mirror of
https://gitee.com/Lyon1998/pikapython.git
synced 2025-01-22 17:12:55 +08:00
297 lines
9.2 KiB
C
297 lines
9.2 KiB
C
#include "_json.h"
|
|
#include "_pika_cJSON.h"
|
|
#include "jsmn.h"
|
|
|
|
#if !PIKASCRIPT_VERSION_REQUIRE_MINIMUN(1, 13, 0)
|
|
#error "pikapython version must be greater than 1.13.0"
|
|
#endif
|
|
|
|
cJSON* _pika_cJSON_decode(Arg* d);
|
|
|
|
#define USING_JSMN 0
|
|
|
|
int parse_json_jsmn(const char* json_string,
|
|
jsmn_parser* parser,
|
|
jsmntok_t** tokens) {
|
|
jsmn_init(parser);
|
|
|
|
int token_count =
|
|
jsmn_parse(parser, json_string, strlen(json_string), NULL, 0);
|
|
|
|
if (token_count < 0) {
|
|
// JSMN_ERROR_INVAL or JSMN_ERROR_PART
|
|
pika_platform_printf("Failed to parse JSON: %d\n", token_count);
|
|
return -1;
|
|
}
|
|
|
|
*tokens = pika_platform_malloc(sizeof(jsmntok_t) * token_count);
|
|
if (*tokens == NULL) {
|
|
pika_platform_printf("Failed to allocate memory: %d\n", token_count);
|
|
return -1;
|
|
}
|
|
|
|
jsmn_init(parser);
|
|
int result = jsmn_parse(parser, json_string, strlen(json_string), *tokens,
|
|
token_count);
|
|
if (result < 0) {
|
|
pika_platform_printf("Failed to parse JSON: %d\n", result);
|
|
pika_platform_free(*tokens);
|
|
*tokens = NULL;
|
|
return -1;
|
|
}
|
|
|
|
return result;
|
|
}
|
|
|
|
Arg* json_encode_jsmn(jsmntok_t* t,
|
|
int* index,
|
|
char* json_str,
|
|
int token_count) {
|
|
Arg* val;
|
|
pika_assert(*index >= 0);
|
|
pika_assert(*index < token_count);
|
|
jsmntype_t type = t[*index].type;
|
|
switch (type) {
|
|
case JSMN_PRIMITIVE: {
|
|
char buff[PIKA_NAME_BUFF_SIZE] = {0};
|
|
pika_assert(t[*index].end - t[*index].start < PIKA_NAME_BUFF_SIZE);
|
|
pika_platform_memcpy(buff, json_str + t[*index].start,
|
|
t[*index].end - t[*index].start);
|
|
if (strEqu(buff, "true")) {
|
|
val = arg_newBool(1);
|
|
} else if (strEqu(buff, "false")) {
|
|
val = arg_newBool(0);
|
|
} else if (strEqu(buff, "null")) {
|
|
val = arg_newNull();
|
|
} else {
|
|
/* float */
|
|
if (strIsContain(buff, '.') ||
|
|
(strIsContain(buff, 'e') || strIsContain(buff, 'E'))) {
|
|
val = arg_newFloat(strtod(buff, NULL));
|
|
} else {
|
|
/* int */
|
|
val = arg_newInt(fast_atoi(buff));
|
|
}
|
|
}
|
|
break;
|
|
}
|
|
case JSMN_STRING: {
|
|
val = arg_newStrN(json_str + t[*index].start,
|
|
t[*index].end - t[*index].start);
|
|
char* raw = arg_getStr(val);
|
|
if (strIsContain(arg_getStr(val), '\\')) {
|
|
Args buffs = {0};
|
|
size_t i = 0;
|
|
char* transfered_str = strsTransfer(&buffs, raw, &i);
|
|
Arg* val_transfered = arg_newStr(transfered_str);
|
|
arg_deinit(val);
|
|
val = val_transfered;
|
|
strsDeinit(&buffs);
|
|
}
|
|
break;
|
|
}
|
|
case JSMN_OBJECT: {
|
|
PikaObj* ret = obj_newDict(NULL);
|
|
int num_keys = t[*index].size;
|
|
for (int i = 0; i < num_keys; i++) {
|
|
(*index)++;
|
|
jsmntok_t* key_tok = &t[*index];
|
|
char* value = json_str + key_tok->start;
|
|
int size = key_tok->end - key_tok->start;
|
|
pika_assert(key_tok->type == JSMN_STRING);
|
|
char key[PIKA_NAME_BUFF_SIZE] = {0};
|
|
pika_assert(size < PIKA_NAME_BUFF_SIZE);
|
|
pika_platform_memcpy(key, value, size);
|
|
(*index)++;
|
|
Arg* val_nested =
|
|
json_encode_jsmn(t, index, json_str, token_count);
|
|
objDict_set(ret, key, val_nested);
|
|
arg_deinit(val_nested);
|
|
}
|
|
val = arg_newObj(ret);
|
|
break;
|
|
}
|
|
case JSMN_ARRAY: {
|
|
PikaObj* ret = obj_newList(NULL);
|
|
jsmntok_t* key_tok = &t[*index];
|
|
int num_elements = key_tok->size;
|
|
for (int i = 0; i < num_elements; i++) {
|
|
(*index)++;
|
|
Arg* val_nested =
|
|
json_encode_jsmn(t, index, json_str, token_count);
|
|
objList_append(ret, val_nested);
|
|
arg_deinit(val_nested);
|
|
}
|
|
val = arg_newObj(ret);
|
|
break;
|
|
}
|
|
default: /* Should not reach here */
|
|
val = NULL;
|
|
}
|
|
return val;
|
|
}
|
|
|
|
Arg* json_encode_cjson(cJSON* cjson) {
|
|
if (cjson == NULL) {
|
|
return NULL;
|
|
}
|
|
|
|
switch (cjson->type) {
|
|
case pika_cJSON_Invalid: {
|
|
return NULL;
|
|
}
|
|
case pika_cJSON_False: {
|
|
return arg_newBool(pika_false);
|
|
}
|
|
case pika_cJSON_True: {
|
|
return arg_newBool(pika_true);
|
|
}
|
|
case pika_cJSON_NULL: {
|
|
return arg_newNull();
|
|
}
|
|
case pika_cJSON_Number: {
|
|
pika_float num_f = cjson->valuedouble;
|
|
int num_i = cjson->valueint;
|
|
if (num_f == num_i) {
|
|
return arg_newInt(num_i);
|
|
} else {
|
|
return arg_newFloat(num_f);
|
|
}
|
|
}
|
|
case pika_cJSON_String: {
|
|
return arg_newStr(cjson->valuestring);
|
|
}
|
|
case pika_cJSON_Array: {
|
|
PikaObj* ret = obj_newList(NULL);
|
|
for (int i = 0; i < pika_cJSON_GetArraySize(cjson); i++) {
|
|
cJSON* item = pika_cJSON_GetArrayItem(cjson, i);
|
|
Arg* nested_arg = json_encode_cjson(item);
|
|
objList_append(ret, nested_arg);
|
|
arg_deinit(nested_arg);
|
|
}
|
|
return arg_newObj(ret);
|
|
}
|
|
case pika_cJSON_Object: {
|
|
PikaObj* ret = obj_newDict(NULL);
|
|
cJSON* child = cjson->child;
|
|
for (int i = 0; i < pika_cJSON_GetArraySize(cjson); i++) {
|
|
char* key = child->string;
|
|
Arg* nested_arg = json_encode_cjson(child);
|
|
objDict_set(ret, key, nested_arg);
|
|
arg_deinit(nested_arg);
|
|
child = child->next;
|
|
}
|
|
return arg_newObj(ret);
|
|
}
|
|
case pika_cJSON_Raw: {
|
|
return arg_newStr(cjson->valuestring);
|
|
}
|
|
default: {
|
|
return NULL;
|
|
}
|
|
}
|
|
}
|
|
|
|
Arg* _json_loads(PikaObj* self, char* json_str) {
|
|
#if USING_JSMN
|
|
jsmn_parser parser;
|
|
Arg* ret = NULL;
|
|
jsmntok_t* tokens = NULL;
|
|
int token_count = parse_json_jsmn(json_str, &parser, &tokens);
|
|
if (token_count < 0) {
|
|
ret = NULL;
|
|
goto __exit;
|
|
}
|
|
ret = json_encode_jsmn(tokens, &(int){0}, json_str, token_count);
|
|
__exit:
|
|
if (NULL != tokens) {
|
|
free(tokens);
|
|
}
|
|
return ret;
|
|
#else
|
|
cJSON* cjson = pika_cJSON_Parse(json_str);
|
|
if (cjson == NULL) {
|
|
pika_platform_printf("JSONDecodeError: at \'%s\'\r\n",
|
|
pika_cJSON_GetErrorPtr());
|
|
obj_setErrorCode(self, PIKA_RES_ERR_RUNTIME_ERROR);
|
|
return NULL;
|
|
}
|
|
Arg* ret = json_encode_cjson(cjson);
|
|
pika_cJSON_Delete(cjson);
|
|
return ret;
|
|
#endif
|
|
}
|
|
|
|
typedef struct {
|
|
cJSON* jsonArray;
|
|
} JsonListContext;
|
|
|
|
int32_t jsonListEachHandle(PikaObj* self,
|
|
int itemIndex,
|
|
Arg* itemEach,
|
|
void* context) {
|
|
JsonListContext* ctx = (JsonListContext*)context;
|
|
pika_cJSON_AddItemToArray(ctx->jsonArray, _pika_cJSON_decode(itemEach));
|
|
return 0;
|
|
}
|
|
|
|
typedef struct {
|
|
cJSON* jsonObject;
|
|
} JsonDictContext;
|
|
|
|
int32_t jsonDictEachHandle(PikaObj* self,
|
|
Arg* keyEach,
|
|
Arg* valEach,
|
|
void* context) {
|
|
JsonDictContext* ctx = (JsonDictContext*)context;
|
|
pika_cJSON_AddItemToObject(ctx->jsonObject, arg_getStr(keyEach),
|
|
_pika_cJSON_decode(valEach));
|
|
return 0;
|
|
}
|
|
|
|
cJSON* _pika_cJSON_decode(Arg* d) {
|
|
ArgType type = arg_getType(d);
|
|
switch (type) {
|
|
case ARG_TYPE_NONE:
|
|
return pika_cJSON_CreateNull();
|
|
|
|
case ARG_TYPE_INT:
|
|
return pika_cJSON_CreateNumber(arg_getInt(d));
|
|
|
|
case ARG_TYPE_FLOAT:
|
|
return pika_cJSON_CreateNumber(arg_getFloat(d));
|
|
|
|
case ARG_TYPE_BOOL:
|
|
return arg_getBool(d) ? pika_cJSON_CreateTrue()
|
|
: pika_cJSON_CreateFalse();
|
|
|
|
case ARG_TYPE_STRING:
|
|
return pika_cJSON_CreateString(arg_getStr(d));
|
|
|
|
default:
|
|
if (arg_isList(d)) {
|
|
JsonListContext context;
|
|
context.jsonArray = pika_cJSON_CreateArray();
|
|
objList_forEach(arg_getObj(d), jsonListEachHandle, &context);
|
|
return context.jsonArray;
|
|
}
|
|
|
|
if (arg_isDict(d)) {
|
|
JsonDictContext context;
|
|
context.jsonObject = pika_cJSON_CreateObject();
|
|
objDict_forEach(arg_getObj(d), jsonDictEachHandle, &context);
|
|
return context.jsonObject;
|
|
}
|
|
return pika_cJSON_CreateNull();
|
|
}
|
|
}
|
|
|
|
char* _json_dumps(PikaObj* self, Arg* d) {
|
|
cJSON* json = _pika_cJSON_decode(d);
|
|
char* ret = pika_cJSON_Print(json);
|
|
char* cached = obj_cacheStr(self, ret);
|
|
pika_cJSON_Delete(json);
|
|
pika_platform_free(ret);
|
|
return cached;
|
|
}
|