180 lines
3.5 KiB
V
180 lines
3.5 KiB
V
// Copyright (c) 2019-2021 Alexander Medvednikov. All rights reserved.
|
|
// Use of this source code is governed by an MIT license
|
|
// that can be found in the LICENSE file.
|
|
module json2
|
|
|
|
import strings
|
|
|
|
fn write_value(v Any, i int, len int, mut wr strings.Builder) {
|
|
str := v.json_str()
|
|
if v is string {
|
|
wr.write_string('"$str"')
|
|
} else {
|
|
wr.write_string(str)
|
|
}
|
|
if i >= len - 1 {
|
|
return
|
|
}
|
|
wr.write_b(`,`)
|
|
}
|
|
|
|
// str returns the string representation of the `map[string]Any`.
|
|
pub fn (flds map[string]Any) str() string {
|
|
mut wr := strings.new_builder(200)
|
|
wr.write_b(`{`)
|
|
mut i := 0
|
|
for k, v in flds {
|
|
wr.write_string('"$k":')
|
|
write_value(v, i, flds.len, mut wr)
|
|
i++
|
|
}
|
|
wr.write_b(`}`)
|
|
defer {
|
|
unsafe { wr.free() }
|
|
}
|
|
res := wr.str()
|
|
return res
|
|
}
|
|
|
|
// str returns the string representation of the `[]Any`.
|
|
pub fn (flds []Any) str() string {
|
|
mut wr := strings.new_builder(200)
|
|
wr.write_b(`[`)
|
|
for i, v in flds {
|
|
write_value(v, i, flds.len, mut wr)
|
|
}
|
|
wr.write_b(`]`)
|
|
defer {
|
|
unsafe { wr.free() }
|
|
}
|
|
res := wr.str()
|
|
return res
|
|
}
|
|
|
|
// str returns the string representation of the `Any` type. Use the `json_str` method
|
|
// if you want to use the escaped str() version of the `Any` type.
|
|
pub fn (f Any) str() string {
|
|
if f is string {
|
|
return f
|
|
} else {
|
|
return f.json_str()
|
|
}
|
|
}
|
|
|
|
// json_str returns the JSON string representation of the `Any` type.
|
|
pub fn (f Any) json_str() string {
|
|
match f {
|
|
string {
|
|
return json_string(f)
|
|
}
|
|
int {
|
|
return f.str()
|
|
}
|
|
u64, i64 {
|
|
return f.str()
|
|
}
|
|
f32 {
|
|
str_f32 := f.str()
|
|
if str_f32.ends_with('.') {
|
|
return '${str_f32}0'
|
|
}
|
|
return str_f32
|
|
}
|
|
f64 {
|
|
str_f64 := f.str()
|
|
if str_f64.ends_with('.') {
|
|
return '${str_f64}0'
|
|
}
|
|
return str_f64
|
|
}
|
|
bool {
|
|
return f.str()
|
|
}
|
|
map[string]Any {
|
|
return f.str()
|
|
}
|
|
[]Any {
|
|
return f.str()
|
|
}
|
|
Null {
|
|
return 'null'
|
|
}
|
|
}
|
|
}
|
|
|
|
// char_len_list is a modified version of builtin.utf8_str_len
|
|
// that returns an array of character lengths. (e.g "t✔" => [1,2])
|
|
fn char_len_list(s string) []int {
|
|
mut l := 1
|
|
mut ls := []int{}
|
|
for i := 0; i < s.len; i++ {
|
|
c := s[i]
|
|
if (c & (1 << 7)) != 0 {
|
|
for t := byte(1 << 6); (c & t) != 0; t >>= 1 {
|
|
l++
|
|
i++
|
|
}
|
|
}
|
|
ls << l
|
|
l = 1
|
|
}
|
|
return ls
|
|
}
|
|
|
|
const escaped_chars = [r'\b', r'\f', r'\n', r'\r', r'\t']
|
|
|
|
// json_string returns the JSON spec-compliant version of the string.
|
|
[manualfree]
|
|
fn json_string(s string) string {
|
|
// not the best implementation but will revisit it soon
|
|
char_lens := char_len_list(s)
|
|
mut sb := strings.new_builder(s.len)
|
|
mut i := 0
|
|
defer {
|
|
unsafe {
|
|
char_lens.free()
|
|
// freeing string builder on defer after
|
|
// returning .str() still isn't working :(
|
|
// sb.free()
|
|
}
|
|
}
|
|
for char_len in char_lens {
|
|
if char_len == 1 {
|
|
chr := s[i]
|
|
if chr in important_escapable_chars {
|
|
for j := 0; j < important_escapable_chars.len; j++ {
|
|
if chr == important_escapable_chars[j] {
|
|
sb.write_string(json2.escaped_chars[j])
|
|
break
|
|
}
|
|
}
|
|
} else if chr == `"` || chr == `/` || chr == `\\` {
|
|
sb.write_string('\\' + chr.ascii_str())
|
|
} else {
|
|
sb.write_b(chr)
|
|
}
|
|
} else {
|
|
slice := s[i..i + char_len]
|
|
hex_code := slice.utf32_code().hex()
|
|
if hex_code.len < 4 {
|
|
// an utf8 codepoint
|
|
sb.write_string(slice)
|
|
} else if hex_code.len == 4 {
|
|
sb.write_string('\\u$hex_code')
|
|
} else {
|
|
// TODO: still figuring out what
|
|
// to do with more than 4 chars
|
|
sb.write_b(` `)
|
|
}
|
|
unsafe {
|
|
slice.free()
|
|
hex_code.free()
|
|
}
|
|
}
|
|
i += char_len
|
|
}
|
|
str := sb.str()
|
|
unsafe { sb.free() }
|
|
return str
|
|
}
|