mirror of
https://github.com/odin-lang/Odin.git
synced 2026-04-19 13:00:28 +00:00
encoding/cbor: cleanup base64 tag
This commit is contained in:
@@ -213,20 +213,20 @@ tag_big_marshal :: proc(_: ^Tag_Implementation, e: Encoder, v: any) -> Marshal_E
|
||||
// is uninitialized (which we checked).
|
||||
|
||||
is_neg, err := big.is_negative(&vv, mem.panic_allocator())
|
||||
assert(err == nil, "only errors if not initialized, which has been checked")
|
||||
assert(err == nil, "should only error if not initialized, which has been checked")
|
||||
|
||||
tnr: u8 = TAG_NEGATIVE_BIG_NR if is_neg else TAG_UNSIGNED_BIG_NR
|
||||
_encode_u8(e.writer, tnr, .Tag) or_return
|
||||
|
||||
size_in_bytes, berr := big.int_to_bytes_size(&vv, false, mem.panic_allocator())
|
||||
assert(berr == nil, "only errors if not initialized, which has been checked")
|
||||
assert(berr == nil, "should only error if not initialized, which has been checked")
|
||||
assert(size_in_bytes >= 0)
|
||||
|
||||
err_conv(_encode_u64(e, u64(size_in_bytes), .Bytes)) or_return
|
||||
|
||||
for offset := (size_in_bytes*8)-8; offset >= 0; offset -= 8 {
|
||||
bits, derr := big.int_bitfield_extract(&vv, offset, 8, mem.panic_allocator())
|
||||
assert(derr == nil, "only errors if not initialized or invalid argument (offset and count), which won't happen")
|
||||
assert(derr == nil, "should only error if not initialized or invalid argument (offset and count), which won't happen")
|
||||
|
||||
io.write_full(e.writer, {u8(bits & 255)}) or_return
|
||||
}
|
||||
@@ -273,63 +273,75 @@ tag_cbor_marshal :: proc(_: ^Tag_Implementation, e: Encoder, v: any) -> Marshal_
|
||||
}
|
||||
}
|
||||
|
||||
// NOTE: this could probably be more efficient by decoding bytes from CBOR and then from base64 at the same time.
|
||||
@(private)
|
||||
tag_base64_unmarshal :: proc(_: ^Tag_Implementation, r: io.Reader, _: Tag_Number, v: any) -> (err: Unmarshal_Error) {
|
||||
hdr := _decode_header(r) or_return
|
||||
hdr := _decode_header(r) or_return
|
||||
major, add := _header_split(hdr)
|
||||
#partial switch major {
|
||||
case .Text:
|
||||
ti := reflect.type_info_base(type_info_of(v.id))
|
||||
_unmarshal_bytes(r, v, ti, hdr, add) or_return
|
||||
#partial switch t in ti.variant {
|
||||
case runtime.Type_Info_String:
|
||||
switch t.is_cstring {
|
||||
case true:
|
||||
str := string((^cstring)(v.data)^)
|
||||
decoded := base64.decode(str) or_return
|
||||
(^cstring)(v.data)^ = strings.clone_to_cstring(string(decoded)) or_return
|
||||
delete(decoded)
|
||||
delete(str)
|
||||
case false:
|
||||
str := (^string)(v.data)^
|
||||
decoded := base64.decode(str) or_return
|
||||
(^string)(v.data)^ = string(decoded)
|
||||
delete(str)
|
||||
}
|
||||
return
|
||||
ti := reflect.type_info_base(type_info_of(v.id))
|
||||
|
||||
case runtime.Type_Info_Array:
|
||||
raw := ([^]byte)(v.data)
|
||||
decoded := base64.decode(string(raw[:t.count])) or_return
|
||||
copy(raw[:t.count], decoded)
|
||||
delete(decoded)
|
||||
return
|
||||
if major != .Text && major != .Bytes {
|
||||
return .Bad_Tag_Value
|
||||
}
|
||||
|
||||
case runtime.Type_Info_Slice:
|
||||
raw := (^[]byte)(v.data)
|
||||
decoded := base64.decode(string(raw^)) or_return
|
||||
delete(raw^)
|
||||
raw^ = decoded
|
||||
return
|
||||
bytes: string; {
|
||||
context.allocator = context.temp_allocator
|
||||
bytes = string(err_conv(_decode_bytes(r, add)) or_return)
|
||||
}
|
||||
defer delete(bytes, context.temp_allocator)
|
||||
|
||||
case runtime.Type_Info_Dynamic_Array:
|
||||
raw := (^mem.Raw_Dynamic_Array)(v.data)
|
||||
str := string(((^[dynamic]byte)(v.data)^)[:])
|
||||
#partial switch t in ti.variant {
|
||||
case reflect.Type_Info_String:
|
||||
|
||||
decoded := base64.decode(str) or_return
|
||||
delete(str)
|
||||
if t.is_cstring {
|
||||
length := base64.decoded_len(bytes)
|
||||
builder := strings.builder_make(0, length+1)
|
||||
base64.decode_into(strings.to_stream(&builder), bytes) or_return
|
||||
|
||||
raw.data = raw_data(decoded)
|
||||
raw.len = len(decoded)
|
||||
raw.cap = len(decoded)
|
||||
return
|
||||
|
||||
case: unreachable()
|
||||
raw := (^cstring)(v.data)
|
||||
raw^ = cstring(raw_data(builder.buf))
|
||||
} else {
|
||||
raw := (^string)(v.data)
|
||||
raw^ = string(base64.decode(bytes) or_return)
|
||||
}
|
||||
|
||||
case: return .Bad_Tag_Value
|
||||
return
|
||||
|
||||
case reflect.Type_Info_Slice:
|
||||
elem_base := reflect.type_info_base(t.elem)
|
||||
|
||||
if elem_base.id != byte { return _unsupported(v, hdr) }
|
||||
|
||||
raw := (^[]byte)(v.data)
|
||||
raw^ = base64.decode(bytes) or_return
|
||||
return
|
||||
|
||||
case reflect.Type_Info_Dynamic_Array:
|
||||
elem_base := reflect.type_info_base(t.elem)
|
||||
|
||||
if elem_base.id != byte { return _unsupported(v, hdr) }
|
||||
|
||||
decoded := base64.decode(bytes) or_return
|
||||
|
||||
raw := (^mem.Raw_Dynamic_Array)(v.data)
|
||||
raw.data = raw_data(decoded)
|
||||
raw.len = len(decoded)
|
||||
raw.cap = len(decoded)
|
||||
raw.allocator = context.allocator
|
||||
return
|
||||
|
||||
case reflect.Type_Info_Array:
|
||||
elem_base := reflect.type_info_base(t.elem)
|
||||
|
||||
if elem_base.id != byte { return _unsupported(v, hdr) }
|
||||
|
||||
if base64.decoded_len(bytes) > t.count { return _unsupported(v, hdr) }
|
||||
|
||||
slice := ([^]byte)(v.data)[:len(bytes)]
|
||||
copy(slice, base64.decode(bytes) or_return)
|
||||
return
|
||||
}
|
||||
|
||||
return _unsupported(v, hdr)
|
||||
}
|
||||
|
||||
@(private)
|
||||
@@ -355,7 +367,7 @@ tag_base64_marshal :: proc(_: ^Tag_Implementation, e: Encoder, v: any) -> Marsha
|
||||
}
|
||||
}
|
||||
|
||||
out_len := base64.encoded_length(bytes)
|
||||
out_len := base64.encoded_len(bytes)
|
||||
err_conv(_encode_u64(e, u64(out_len), .Text)) or_return
|
||||
return base64.encode_into(e.writer, bytes)
|
||||
}
|
||||
|
||||
Reference in New Issue
Block a user