aboutsummaryrefslogtreecommitdiff
path: root/core/encoding/json
diff options
context:
space:
mode:
authorIllusionMan1212 <hisham.abourgheba@gmail.com>2025-10-24 00:49:33 +0200
committerIllusionMan1212 <hisham.abourgheba@gmail.com>2025-12-02 22:46:31 +0200
commit4b9e15786d8db44741cd6fd5c89a033e18548d66 (patch)
treef1a7fd52ef8edae40e59c1f2351ab01408aa5185 /core/encoding/json
parentb277d66e008e15d640208f3e11515b047f78be48 (diff)
encoding/json: custom json (un)marshalling
Diffstat (limited to 'core/encoding/json')
-rw-r--r--core/encoding/json/marshal.odin49
-rw-r--r--core/encoding/json/unmarshal.odin61
2 files changed, 66 insertions, 44 deletions
diff --git a/core/encoding/json/marshal.odin b/core/encoding/json/marshal.odin
index e563c326a..81884b71b 100644
--- a/core/encoding/json/marshal.odin
+++ b/core/encoding/json/marshal.odin
@@ -62,7 +62,10 @@ Marshal_Options :: struct {
mjson_skipped_first_braces_end: bool,
}
-marshal :: proc(v: any, opt: Marshal_Options = {}, allocator := context.allocator, loc := #caller_location) -> (data: []byte, err: Marshal_Error) {
+User_Marshaller :: #type proc(w: io.Writer, v: any)
+User_Marshaller_Map :: map[typeid]User_Marshaller
+
+marshal :: proc(v: any, opt: Marshal_Options = {}, user_marshallers: User_Marshaller_Map = nil, allocator := context.allocator, loc := #caller_location) -> (data: []byte, err: Marshal_Error) {
b := strings.builder_make(allocator, loc)
defer if err != nil {
strings.builder_destroy(&b)
@@ -72,7 +75,7 @@ marshal :: proc(v: any, opt: Marshal_Options = {}, allocator := context.allocato
runtime.DEFAULT_TEMP_ALLOCATOR_TEMP_GUARD(ignore = allocator == context.temp_allocator)
opt := opt
- marshal_to_builder(&b, v, &opt) or_return
+ marshal_to_builder(&b, v, &opt, user_marshallers) or_return
if len(b.buf) != 0 {
data = b.buf[:]
@@ -81,16 +84,24 @@ marshal :: proc(v: any, opt: Marshal_Options = {}, allocator := context.allocato
return data, nil
}
-marshal_to_builder :: proc(b: ^strings.Builder, v: any, opt: ^Marshal_Options) -> Marshal_Error {
- return marshal_to_writer(strings.to_writer(b), v, opt)
+marshal_to_builder :: proc(b: ^strings.Builder, v: any, opt: ^Marshal_Options, user_marshallers: User_Marshaller_Map = nil) -> Marshal_Error {
+ return marshal_to_writer(strings.to_writer(b), v, opt, user_marshallers)
}
-marshal_to_writer :: proc(w: io.Writer, v: any, opt: ^Marshal_Options) -> (err: Marshal_Error) {
+marshal_to_writer :: proc(w: io.Writer, v: any, opt: ^Marshal_Options, user_marshallers: User_Marshaller_Map = nil) -> (err: Marshal_Error) {
if v == nil {
io.write_string(w, "null") or_return
return
}
+ if user_marshallers != nil {
+ marshaller := user_marshallers[v.id]
+ if marshaller != nil {
+ marshaller(w, v)
+ return
+ }
+ }
+
ti := runtime.type_info_base(type_info_of(v.id))
a := any{v.data, ti.id}
@@ -208,7 +219,7 @@ marshal_to_writer :: proc(w: io.Writer, v: any, opt: ^Marshal_Options) -> (err:
for i in 0..<info.count {
opt_write_iteration(w, opt, i == 0) or_return
data := uintptr(v.data) + uintptr(i*info.elem_size)
- marshal_to_writer(w, any{rawptr(data), info.elem.id}, opt) or_return
+ marshal_to_writer(w, any{rawptr(data), info.elem.id}, opt, user_marshallers) or_return
}
opt_write_end(w, opt, ']') or_return
@@ -227,7 +238,7 @@ marshal_to_writer :: proc(w: io.Writer, v: any, opt: ^Marshal_Options) -> (err:
opt_write_iteration(w, opt, i == 0) or_return
opt_write_key(w, opt, enum_type.names[index]) or_return
data := uintptr(v.data) + uintptr(i*info.elem_size)
- marshal_to_writer(w, any{rawptr(data), info.elem.id}, opt) or_return
+ marshal_to_writer(w, any{rawptr(data), info.elem.id}, opt, user_marshallers) or_return
}
opt_write_end(w, opt, '}') or_return
@@ -237,7 +248,7 @@ marshal_to_writer :: proc(w: io.Writer, v: any, opt: ^Marshal_Options) -> (err:
for i in 0..<array.len {
opt_write_iteration(w, opt, i == 0) or_return
data := uintptr(array.data) + uintptr(i*info.elem_size)
- marshal_to_writer(w, any{rawptr(data), info.elem.id}, opt) or_return
+ marshal_to_writer(w, any{rawptr(data), info.elem.id}, opt, user_marshallers) or_return
}
opt_write_end(w, opt, ']') or_return
@@ -247,7 +258,7 @@ marshal_to_writer :: proc(w: io.Writer, v: any, opt: ^Marshal_Options) -> (err:
for i in 0..<slice.len {
opt_write_iteration(w, opt, i == 0) or_return
data := uintptr(slice.data) + uintptr(i*info.elem_size)
- marshal_to_writer(w, any{rawptr(data), info.elem.id}, opt) or_return
+ marshal_to_writer(w, any{rawptr(data), info.elem.id}, opt, user_marshallers) or_return
}
opt_write_end(w, opt, ']') or_return
@@ -297,7 +308,7 @@ marshal_to_writer :: proc(w: io.Writer, v: any, opt: ^Marshal_Options) -> (err:
}
}
- marshal_to_writer(w, any{value, info.value.id}, opt) or_return
+ marshal_to_writer(w, any{value, info.value.id}, opt, user_marshallers) or_return
}
} else {
Entry :: struct {
@@ -340,7 +351,7 @@ marshal_to_writer :: proc(w: io.Writer, v: any, opt: ^Marshal_Options) -> (err:
for s, i in sorted {
opt_write_iteration(w, opt, i == 0) or_return
opt_write_key(w, opt, s.key) or_return
- marshal_to_writer(w, s.value, opt) or_return
+ marshal_to_writer(w, s.value, opt, user_marshallers) or_return
}
}
}
@@ -384,7 +395,7 @@ marshal_to_writer :: proc(w: io.Writer, v: any, opt: ^Marshal_Options) -> (err:
return false
}
- marshal_struct_fields :: proc(w: io.Writer, v: any, opt: ^Marshal_Options) -> (err: Marshal_Error) {
+ marshal_struct_fields :: proc(w: io.Writer, v: any, opt: ^Marshal_Options, user_marshallers: User_Marshaller_Map) -> (err: Marshal_Error) {
ti := runtime.type_info_base(type_info_of(v.id))
info := ti.variant.(runtime.Type_Info_Struct)
first_iteration := true
@@ -419,7 +430,7 @@ marshal_to_writer :: proc(w: io.Writer, v: any, opt: ^Marshal_Options) -> (err:
} else {
// Marshal the fields of 'using _: T' fields directly into the parent struct
if info.usings[i] && name == "_" {
- marshal_struct_fields(w, the_value, opt) or_return
+ marshal_struct_fields(w, the_value, opt, user_marshallers) or_return
continue
} else {
opt_write_key(w, opt, name) or_return
@@ -427,13 +438,13 @@ marshal_to_writer :: proc(w: io.Writer, v: any, opt: ^Marshal_Options) -> (err:
}
- marshal_to_writer(w, the_value, opt) or_return
+ marshal_to_writer(w, the_value, opt, user_marshallers) or_return
}
return
}
opt_write_start(w, opt, '{') or_return
- marshal_struct_fields(w, v, opt) or_return
+ marshal_struct_fields(w, v, opt, user_marshallers) or_return
opt_write_end(w, opt, '}') or_return
case runtime.Type_Info_Union:
@@ -466,17 +477,17 @@ marshal_to_writer :: proc(w: io.Writer, v: any, opt: ^Marshal_Options) -> (err:
tag -= 1
}
id := info.variants[tag].id
- return marshal_to_writer(w, any{v.data, id}, opt)
+ return marshal_to_writer(w, any{v.data, id}, opt, user_marshallers)
case runtime.Type_Info_Enum:
if !opt.use_enum_names || len(info.names) == 0 {
- return marshal_to_writer(w, any{v.data, info.base.id}, opt)
+ return marshal_to_writer(w, any{v.data, info.base.id}, opt, user_marshallers)
} else {
name, found := reflect.enum_name_from_value_any(v)
if found {
- return marshal_to_writer(w, name, opt)
+ return marshal_to_writer(w, name, opt, user_marshallers)
} else {
- return marshal_to_writer(w, any{v.data, info.base.id}, opt)
+ return marshal_to_writer(w, any{v.data, info.base.id}, opt, user_marshallers)
}
}
diff --git a/core/encoding/json/unmarshal.odin b/core/encoding/json/unmarshal.odin
index 3cdc6429d..c1a2725a4 100644
--- a/core/encoding/json/unmarshal.odin
+++ b/core/encoding/json/unmarshal.odin
@@ -26,7 +26,10 @@ Unmarshal_Error :: union {
Unsupported_Type_Error,
}
-unmarshal_any :: proc(data: []byte, v: any, spec := DEFAULT_SPECIFICATION, allocator := context.allocator) -> Unmarshal_Error {
+User_Unmarshaller :: #type proc(p: ^Parser, v: any)
+User_Unmarshaller_Map :: map[typeid]User_Unmarshaller
+
+unmarshal_any :: proc(data: []byte, v: any, spec := DEFAULT_SPECIFICATION, user_unmarshallers: User_Unmarshaller_Map, allocator := context.allocator) -> Unmarshal_Error {
v := v
if v == nil || v.id == nil {
return .Invalid_Parameter
@@ -53,20 +56,20 @@ unmarshal_any :: proc(data: []byte, v: any, spec := DEFAULT_SPECIFICATION, alloc
if p.spec == .MJSON {
#partial switch p.curr_token.kind {
case .Ident, .String:
- return unmarshal_object(&p, data, .EOF)
+ return unmarshal_object(&p, data, .EOF, user_unmarshallers)
}
}
- return unmarshal_value(&p, data)
+ return unmarshal_value(&p, data, user_unmarshallers)
}
-unmarshal :: proc(data: []byte, ptr: ^$T, spec := DEFAULT_SPECIFICATION, allocator := context.allocator) -> Unmarshal_Error {
- return unmarshal_any(data, ptr, spec, allocator)
+unmarshal :: proc(data: []byte, ptr: ^$T, spec := DEFAULT_SPECIFICATION, user_unmarshallers: User_Unmarshaller_Map = nil, allocator := context.allocator) -> Unmarshal_Error {
+ return unmarshal_any(data, ptr, spec, user_unmarshallers, allocator)
}
-unmarshal_string :: proc(data: string, ptr: ^$T, spec := DEFAULT_SPECIFICATION, allocator := context.allocator) -> Unmarshal_Error {
- return unmarshal_any(transmute([]byte)data, ptr, spec, allocator)
+unmarshal_string :: proc(data: string, ptr: ^$T, spec := DEFAULT_SPECIFICATION, user_unmarshallers: User_Unmarshaller_Map = nil, allocator := context.allocator) -> Unmarshal_Error {
+ return unmarshal_any(transmute([]byte)data, ptr, spec, user_unmarshallers, allocator)
}
@@ -267,10 +270,18 @@ unmarshal_string_token :: proc(p: ^Parser, val: any, token: Token, ti: ^reflect.
@(private)
-unmarshal_value :: proc(p: ^Parser, v: any) -> (err: Unmarshal_Error) {
+unmarshal_value :: proc(p: ^Parser, v: any, user_unmarshallers: User_Unmarshaller_Map = nil) -> (err: Unmarshal_Error) {
UNSUPPORTED_TYPE := Unsupported_Type_Error{v.id, p.curr_token}
token := p.curr_token
+ if user_unmarshallers != nil {
+ unmarshaller := user_unmarshallers[v.id]
+ if unmarshaller != nil {
+ unmarshaller(p, v)
+ return
+ }
+ }
+
v := v
ti := reflect.type_info_base(type_info_of(v.id))
if u, ok := ti.variant.(reflect.Type_Info_Union); ok && token.kind != .Null {
@@ -287,7 +298,7 @@ unmarshal_value :: proc(p: ^Parser, v: any) -> (err: Unmarshal_Error) {
for variant, i in u.variants {
variant_any := any{v.data, variant.id}
variant_p := p^
- if err = unmarshal_value(&variant_p, variant_any); err == nil {
+ if err = unmarshal_value(&variant_p, variant_any, user_unmarshallers); err == nil {
p^ = variant_p
raw_tag := i
@@ -363,10 +374,10 @@ unmarshal_value :: proc(p: ^Parser, v: any) -> (err: Unmarshal_Error) {
return UNSUPPORTED_TYPE
case .Open_Brace:
- return unmarshal_object(p, v, .Close_Brace)
+ return unmarshal_object(p, v, .Close_Brace, user_unmarshallers)
case .Open_Bracket:
- return unmarshal_array(p, v)
+ return unmarshal_array(p, v, user_unmarshallers)
case:
if p.spec != .JSON {
@@ -423,7 +434,7 @@ json_name_from_tag_value :: proc(value: string) -> (json_name, extra: string) {
@(private)
-unmarshal_object :: proc(p: ^Parser, v: any, end_token: Token_Kind) -> (err: Unmarshal_Error) {
+unmarshal_object :: proc(p: ^Parser, v: any, end_token: Token_Kind, user_unmarshallers: User_Unmarshaller_Map = nil) -> (err: Unmarshal_Error) {
UNSUPPORTED_TYPE := Unsupported_Type_Error{v.id, p.curr_token}
if end_token == .Close_Brace {
@@ -522,7 +533,7 @@ unmarshal_object :: proc(p: ^Parser, v: any, end_token: Token_Kind) -> (err: Unm
field_ptr := rawptr(uintptr(v.data) + offset)
field := any{field_ptr, type.id}
- unmarshal_value(p, field) or_return
+ unmarshal_value(p, field, user_unmarshallers) or_return
if parse_comma(p) {
break struct_loop
@@ -564,7 +575,7 @@ unmarshal_object :: proc(p: ^Parser, v: any, end_token: Token_Kind) -> (err: Unm
mem.zero_slice(elem_backing)
- if uerr := unmarshal_value(p, map_backing_value); uerr != nil {
+ if uerr := unmarshal_value(p, map_backing_value, user_unmarshallers); uerr != nil {
delete(key, p.allocator)
return uerr
}
@@ -626,7 +637,7 @@ unmarshal_object :: proc(p: ^Parser, v: any, end_token: Token_Kind) -> (err: Unm
index_ptr := rawptr(uintptr(v.data) + uintptr(index*t.elem_size))
index_any := any{index_ptr, t.elem.id}
- unmarshal_value(p, index_any) or_return
+ unmarshal_value(p, index_any, user_unmarshallers) or_return
if parse_comma(p) {
break enumerated_array_loop
@@ -662,8 +673,8 @@ unmarshal_count_array :: proc(p: ^Parser) -> (length: uintptr) {
}
@(private)
-unmarshal_array :: proc(p: ^Parser, v: any) -> (err: Unmarshal_Error) {
- assign_array :: proc(p: ^Parser, base: rawptr, elem: ^reflect.Type_Info, length: uintptr) -> Unmarshal_Error {
+unmarshal_array :: proc(p: ^Parser, v: any, user_unmarshallers: User_Unmarshaller_Map = nil) -> (err: Unmarshal_Error) {
+ assign_array :: proc(p: ^Parser, base: rawptr, elem: ^reflect.Type_Info, length: uintptr, user_unmarshallers: User_Unmarshaller_Map = nil) -> Unmarshal_Error {
unmarshal_expect_token(p, .Open_Bracket)
for idx: uintptr = 0; p.curr_token.kind != .Close_Bracket; idx += 1 {
@@ -672,7 +683,7 @@ unmarshal_array :: proc(p: ^Parser, v: any) -> (err: Unmarshal_Error) {
elem_ptr := rawptr(uintptr(base) + idx*uintptr(elem.size))
elem := any{elem_ptr, elem.id}
- unmarshal_value(p, elem) or_return
+ unmarshal_value(p, elem, user_unmarshallers) or_return
if parse_comma(p) {
break
@@ -698,7 +709,7 @@ unmarshal_array :: proc(p: ^Parser, v: any) -> (err: Unmarshal_Error) {
raw.data = raw_data(data)
raw.len = int(length)
- return assign_array(p, raw.data, t.elem, length)
+ return assign_array(p, raw.data, t.elem, length, user_unmarshallers)
case reflect.Type_Info_Dynamic_Array:
raw := (^mem.Raw_Dynamic_Array)(v.data)
@@ -708,7 +719,7 @@ unmarshal_array :: proc(p: ^Parser, v: any) -> (err: Unmarshal_Error) {
raw.cap = int(length)
raw.allocator = p.allocator
- return assign_array(p, raw.data, t.elem, length)
+ return assign_array(p, raw.data, t.elem, length, user_unmarshallers)
case reflect.Type_Info_Array:
// NOTE(bill): Allow lengths which are less than the dst array
@@ -716,7 +727,7 @@ unmarshal_array :: proc(p: ^Parser, v: any) -> (err: Unmarshal_Error) {
return UNSUPPORTED_TYPE
}
- return assign_array(p, v.data, t.elem, length)
+ return assign_array(p, v.data, t.elem, length, user_unmarshallers)
case reflect.Type_Info_Enumerated_Array:
// NOTE(bill): Allow lengths which are less than the dst array
@@ -724,7 +735,7 @@ unmarshal_array :: proc(p: ^Parser, v: any) -> (err: Unmarshal_Error) {
return UNSUPPORTED_TYPE
}
- return assign_array(p, v.data, t.elem, length)
+ return assign_array(p, v.data, t.elem, length, user_unmarshallers)
case reflect.Type_Info_Complex:
// NOTE(bill): Allow lengths which are less than the dst array
@@ -733,9 +744,9 @@ unmarshal_array :: proc(p: ^Parser, v: any) -> (err: Unmarshal_Error) {
}
switch ti.id {
- case complex32: return assign_array(p, v.data, type_info_of(f16), 2)
- case complex64: return assign_array(p, v.data, type_info_of(f32), 2)
- case complex128: return assign_array(p, v.data, type_info_of(f64), 2)
+ case complex32: return assign_array(p, v.data, type_info_of(f16), 2, user_unmarshallers)
+ case complex64: return assign_array(p, v.data, type_info_of(f32), 2, user_unmarshallers)
+ case complex128: return assign_array(p, v.data, type_info_of(f64), 2, user_unmarshallers)
}
return UNSUPPORTED_TYPE