2025-09-08 15:41:44 +08:00

192 lines
6.5 KiB
C

#include "pocketpy/pocketpy.h"
#include "pocketpy/common/utils.h"
#include "pocketpy/objects/object.h"
#include "pocketpy/common/sstream.h"
#include "pocketpy/interpreter/vm.h"
#include <math.h>
static bool json_loads(int argc, py_Ref argv) {
PY_CHECK_ARGC(1);
PY_CHECK_ARG_TYPE(0, tp_str);
const char* source = py_tostr(argv);
return py_json_loads(source);
}
static bool json_dumps(int argc, py_Ref argv) {
PY_CHECK_ARGC(2);
PY_CHECK_ARG_TYPE(1, tp_int);
int indent = py_toint(&argv[1]);
return py_json_dumps(argv, indent);
}
void pk__add_module_json() {
py_Ref mod = py_newmodule("json");
py_setdict(mod, py_name("null"), py_None());
py_setdict(mod, py_name("true"), py_True());
py_setdict(mod, py_name("false"), py_False());
py_TValue tmp;
py_newfloat(&tmp, NAN);
py_setdict(mod, py_name("NaN"), &tmp);
py_newfloat(&tmp, INFINITY);
py_setdict(mod, py_name("Infinity"), &tmp);
py_bindfunc(mod, "loads", json_loads);
py_bind(mod, "dumps(obj, indent=0)", json_dumps);
}
typedef struct {
c11_sbuf* buf;
bool first;
int indent;
int depth;
} json__write_dict_kv_ctx;
static bool json__write_object(c11_sbuf* buf, py_TValue* obj, int indent, int depth);
static void json__write_indent(c11_sbuf* buf, int n_spaces) {
for(int i = 0; i < n_spaces; i++) {
c11_sbuf__write_char(buf, ' ');
}
}
static bool json__write_array(c11_sbuf* buf, py_TValue* arr, int length, int indent, int depth) {
c11_sbuf__write_char(buf, '[');
if(length == 0) {
c11_sbuf__write_char(buf, ']');
return true;
}
if(indent > 0) c11_sbuf__write_char(buf, '\n');
int n_spaces = indent * depth;
const char* sep = indent > 0 ? ",\n" : ", ";
for(int i = 0; i < length; i++) {
if(i != 0) c11_sbuf__write_cstr(buf, sep);
json__write_indent(buf, n_spaces);
bool ok = json__write_object(buf, arr + i, indent, depth);
if(!ok) return false;
}
if(indent > 0) {
c11_sbuf__write_char(buf, '\n');
json__write_indent(buf, n_spaces - indent);
}
c11_sbuf__write_char(buf, ']');
return true;
}
static bool json__write_dict_kv(py_Ref k, py_Ref v, void* ctx_) {
json__write_dict_kv_ctx* ctx = ctx_;
int n_spaces = ctx->indent * ctx->depth;
const char* sep = ctx->indent > 0 ? ",\n" : ", ";
if(!ctx->first) c11_sbuf__write_cstr(ctx->buf, sep);
ctx->first = false;
if(!py_isstr(k)) return TypeError("keys must be strings");
json__write_indent(ctx->buf, n_spaces);
c11_sbuf__write_quoted(ctx->buf, py_tosv(k), '"');
c11_sbuf__write_cstr(ctx->buf, ": ");
return json__write_object(ctx->buf, v, ctx->indent, ctx->depth);
}
static bool json__write_namedict_kv(py_Name k, py_Ref v, void* ctx_) {
json__write_dict_kv_ctx* ctx = ctx_;
int n_spaces = ctx->indent * ctx->depth;
const char* sep = ctx->indent > 0 ? ",\n" : ", ";
if(!ctx->first) c11_sbuf__write_cstr(ctx->buf, sep);
ctx->first = false;
json__write_indent(ctx->buf, n_spaces);
c11_sbuf__write_quoted(ctx->buf, py_name2sv(k), '"');
c11_sbuf__write_cstr(ctx->buf, ": ");
return json__write_object(ctx->buf, v, ctx->indent, ctx->depth);
}
static bool json__write_object(c11_sbuf* buf, py_TValue* obj, int indent, int depth) {
switch(obj->type) {
case tp_NoneType: c11_sbuf__write_cstr(buf, "null"); return true;
case tp_int: c11_sbuf__write_int(buf, obj->_i64); return true;
case tp_float: {
if(isnan(obj->_f64)) {
c11_sbuf__write_cstr(buf, "NaN");
} else if(isinf(obj->_f64)) {
c11_sbuf__write_cstr(buf, obj->_f64 < 0 ? "-Infinity" : "Infinity");
} else {
c11_sbuf__write_f64(buf, obj->_f64, -1);
}
return true;
}
case tp_bool: {
c11_sbuf__write_cstr(buf, py_tobool(obj) ? "true" : "false");
return true;
}
case tp_str: {
c11_sbuf__write_quoted(buf, py_tosv(obj), '"');
return true;
}
case tp_list: {
return json__write_array(buf, py_list_data(obj), py_list_len(obj), indent, depth + 1);
}
case tp_tuple: {
return json__write_array(buf, py_tuple_data(obj), py_tuple_len(obj), indent, depth + 1);
}
case tp_dict: {
c11_sbuf__write_char(buf, '{');
if(py_dict_len(obj) == 0) {
c11_sbuf__write_char(buf, '}');
return true;
}
if(indent > 0) c11_sbuf__write_char(buf, '\n');
json__write_dict_kv_ctx ctx = {.buf = buf,
.first = true,
.indent = indent,
.depth = depth + 1};
bool ok = py_dict_apply(obj, json__write_dict_kv, &ctx);
if(!ok) return false;
if(indent > 0) {
c11_sbuf__write_char(buf, '\n');
json__write_indent(buf, indent * depth);
}
c11_sbuf__write_char(buf, '}');
return true;
}
case tp_namedict: {
py_Ref original = py_getslot(obj, 0);
c11_sbuf__write_char(buf, '{');
if(PyObject__dict(original->_obj)->length == 0) {
c11_sbuf__write_char(buf, '}');
return true;
}
if(indent > 0) c11_sbuf__write_char(buf, '\n');
json__write_dict_kv_ctx ctx = {.buf = buf,
.first = true,
.indent = indent,
.depth = depth + 1};
bool ok = py_applydict(original, json__write_namedict_kv, &ctx);
if(!ok) return false;
if(indent > 0) {
c11_sbuf__write_char(buf, '\n');
json__write_indent(buf, indent * depth);
}
c11_sbuf__write_char(buf, '}');
return true;
}
default: return TypeError("'%t' object is not JSON serializable", obj->type);
}
}
bool py_json_dumps(py_Ref val, int indent) {
c11_sbuf buf;
c11_sbuf__ctor(&buf);
bool ok = json__write_object(&buf, val, indent, 0);
if(!ok) {
c11_sbuf__dtor(&buf);
return false;
}
c11_sbuf__py_submit(&buf, py_retval());
return true;
}
bool py_json_loads(const char* source) {
py_GlobalRef mod = py_getmodule("json");
return py_exec(source, "<json>", EVAL_MODE, mod);
}