json: wrap up decode()

pull/4708/head
Alexander Medvednikov 2020-05-04 16:46:36 +02:00
parent 5eb71c4a13
commit fd8a194df0
6 changed files with 68 additions and 38 deletions

View File

@ -6,6 +6,7 @@ module json
#flag -I @VROOT/thirdparty/cJSON #flag -I @VROOT/thirdparty/cJSON
#flag @VROOT/thirdparty/cJSON/cJSON.o #flag @VROOT/thirdparty/cJSON/cJSON.o
#include "cJSON.h" #include "cJSON.h"
#define js_get(object, key) cJSON_GetObjectItemCaseSensitive((object), (key))
struct C.cJSON { struct C.cJSON {
valueint int valueint int
valuedouble f32 valuedouble f32
@ -22,84 +23,84 @@ pub fn encode(x voidptr) string {
return '' return ''
} }
fn jsdecode_int(root &C.cJSON) int { fn decode_int(root &C.cJSON) int {
if isnil(root) { if isnil(root) {
return 0 return 0
} }
return root.valueint return root.valueint
} }
fn jsdecode_i8(root &C.cJSON) i8 { fn decode_i8(root &C.cJSON) i8 {
if isnil(root) { if isnil(root) {
return i8(0) return i8(0)
} }
return i8(root.valueint) return i8(root.valueint)
} }
fn jsdecode_i16(root &C.cJSON) i16 { fn decode_i16(root &C.cJSON) i16 {
if isnil(root) { if isnil(root) {
return i16(0) return i16(0)
} }
return i16(root.valueint) return i16(root.valueint)
} }
fn jsdecode_i64(root &C.cJSON) i64 { fn decode_i64(root &C.cJSON) i64 {
if isnil(root) { if isnil(root) {
return i64(0) return i64(0)
} }
return i64(root.valuedouble) // i64 is double in C return i64(root.valuedouble) // i64 is double in C
} }
fn jsdecode_byte(root &C.cJSON) byte { fn decode_byte(root &C.cJSON) byte {
if isnil(root) { if isnil(root) {
return byte(0) return byte(0)
} }
return byte(root.valueint) return byte(root.valueint)
} }
fn jsdecode_u16(root &C.cJSON) u16 { fn decode_u16(root &C.cJSON) u16 {
if isnil(root) { if isnil(root) {
return u16(0) return u16(0)
} }
return u16(root.valueint) return u16(root.valueint)
} }
fn jsdecode_u32(root &C.cJSON) u32 { fn decode_u32(root &C.cJSON) u32 {
if isnil(root) { if isnil(root) {
return u32(0) return u32(0)
} }
return u32(root.valueint) return u32(root.valueint)
} }
fn jsdecode_u64(root &C.cJSON) u64 { fn decode_u64(root &C.cJSON) u64 {
if isnil(root) { if isnil(root) {
return u64(0) return u64(0)
} }
return u64(root.valueint) return u64(root.valueint)
} }
fn jsdecode_f32(root &C.cJSON) f32 { fn decode_f32(root &C.cJSON) f32 {
if isnil(root) { if isnil(root) {
return f32(0) return f32(0)
} }
return root.valuedouble return root.valuedouble
} }
fn jsdecode_f64(root &C.cJSON) f64 { fn decode_f64(root &C.cJSON) f64 {
if isnil(root) { if isnil(root) {
return f64(0) return f64(0)
} }
return f64(root.valuedouble) return f64(root.valuedouble)
} }
fn jsdecode_string(root &C.cJSON) string { fn decode_string(root &C.cJSON) string {
if isnil(root) { if isnil(root) {
return '' return ''
} }
if isnil(root.valuestring) { if isnil(root.valuestring) {
return '' return ''
} }
// println('jsdecode string valuestring="$root.valuestring"') // println('decode string valuestring="$root.valuestring"')
// return tos(root.valuestring, _strlen(root.valuestring)) // return tos(root.valuestring, _strlen(root.valuestring))
return tos_clone(root.valuestring) // , _strlen(root.valuestring)) return tos_clone(root.valuestring) // , _strlen(root.valuestring))
} }
@ -122,7 +123,7 @@ fn C.cJSON_Parse() &C.cJSON
fn C.cJSON_PrintUnformatted() byteptr fn C.cJSON_PrintUnformatted() byteptr
fn jsdecode_bool(root &C.cJSON) bool { fn decode_bool(root &C.cJSON) bool {
if isnil(root) { if isnil(root) {
return false return false
} }

View File

@ -721,13 +721,14 @@ pub fn (mut c Checker) call_fn(call_expr mut ast.CallExpr) table.Type {
// println(fn_name) // println(fn_name)
// } // }
if fn_name == 'json.encode' { if fn_name == 'json.encode' {
} else if fn_name == 'json.decode' {
expr := call_expr.args[0].expr
if !(expr is ast.Type) {
c.error('json.decode: first argument needs to be a type', call_expr.pos)
return table.void_type
} }
if fn_name == 'json.decode' { typ := expr as ast.Type
ident := call_expr.args[0].expr as ast.Ident return typ.typ.set_flag(.optional)
// sym := c.table.find_type(ident.name)
idx := c.table.find_type_idx(ident.name)
println('js.decode t=$ident.name')
return table.Type(idx)
} }
// look for function in format `mod.fn` or `fn` (main/builtin) // look for function in format `mod.fn` or `fn` (main/builtin)
mut f := table.Fn{} mut f := table.Fn{}

View File

@ -383,12 +383,22 @@ fn (mut g Gen) fn_call(node ast.CallExpr) {
mut name := node.name mut name := node.name
is_print := name == 'println' || name == 'print' is_print := name == 'println' || name == 'print'
print_method := if name == 'println' { 'println' } else { 'print' } print_method := if name == 'println' { 'println' } else { 'print' }
g.is_json_fn = name == 'json.encode' is_json_encode := name == 'json.encode'
is_json_decode := name == 'json.decode'
g.is_json_fn = is_json_encode || is_json_decode
mut json_type_str := '' mut json_type_str := ''
if g.is_json_fn { if g.is_json_fn {
if name == 'json.encode' {
g.write('json__json_print(') g.write('json__json_print(')
g.gen_json_for_type(node.args[0].typ) g.gen_json_for_type(node.args[0].typ)
json_type_str = g.table.get_type_symbol(node.args[0].typ).name json_type_str = g.table.get_type_symbol(node.args[0].typ).name
} else {
g.insert_before('// json.decode')
ast_type := node.args[0].expr as ast.Type
// `json.decode(User, s)` => json.decode_User(s)
sym := g.table.get_type_symbol(ast_type.typ)
name += '_' + sym.name
}
} }
if node.is_c { if node.is_c {
// Skip "C." // Skip "C."
@ -397,8 +407,8 @@ fn (mut g Gen) fn_call(node ast.CallExpr) {
} else { } else {
name = c_name(name) name = c_name(name)
} }
if g.is_json_fn { if is_json_encode {
// `json__decode` => `json__decode_User` // `json__encode` => `json__encode_User`
name += '_' + json_type_str name += '_' + json_type_str
} }
// Generate tmp vars for values that have to be freed. // Generate tmp vars for values that have to be freed.
@ -476,14 +486,25 @@ fn (mut g Gen) fn_call(node ast.CallExpr) {
pafn := g.fn_decl.name.after('.') pafn := g.fn_decl.name.after('.')
mut pamod := g.fn_decl.name.all_before_last('.') mut pamod := g.fn_decl.name.all_before_last('.')
if pamod == pafn { if pamod == pafn {
pamod = if g.fn_decl.is_builtin { 'builtin' } else { 'main' } pamod = if g.fn_decl.is_builtin {
'builtin'
} else {
'main'
}
} }
g.write('panic_debug($paline, tos3("$pafile"), tos3("$pamod"), tos3("$pafn"), ') g.write('panic_debug($paline, tos3("$pafile"), tos3("$pamod"), tos3("$pafn"), ')
g.call_args(node.args, node.expected_arg_types) g.call_args(node.args, node.expected_arg_types)
g.write(')') g.write(')')
} else { } else {
g.write('${name}(') g.write('${name}(')
if is_json_decode {
g.write('json__json_parse(')
// Skip the first argument in json.decode which is a type
// its name was already used to generate the function call
g.call_args(node.args[1..], node.expected_arg_types)
} else {
g.call_args(node.args, node.expected_arg_types) g.call_args(node.args, node.expected_arg_types)
}
g.write(')') g.write(')')
} }
// if node.or_block.stmts.len > 0 { // if node.or_block.stmts.len > 0 {

View File

@ -31,8 +31,9 @@ fn (mut g Gen) gen_json_for_type(typ table.Type) {
// Code gen decoder // Code gen decoder
dec_fn_name := js_dec_name(sym.name) dec_fn_name := js_dec_name(sym.name)
dec.writeln(' dec.writeln('
Option ${dec_fn_name}(cJSON* root, $styp* res) { //Option ${dec_fn_name}(cJSON* root, $styp* res) {
// $styp res; Option ${dec_fn_name}(cJSON* root) {
$styp res;
if (!root) { if (!root) {
const char *error_ptr = cJSON_GetErrorPtr(); const char *error_ptr = cJSON_GetErrorPtr();
if (error_ptr != NULL) { if (error_ptr != NULL) {
@ -70,7 +71,7 @@ cJSON* ${enc_fn_name}($styp val) {
g.gen_json_for_type(field.typ) g.gen_json_for_type(field.typ)
dec_name := js_dec_name(field_type) dec_name := js_dec_name(field_type)
if is_js_prim(field_type) { if is_js_prim(field_type) {
dec.writeln(' res->$field.name = $dec_name (js_get(' + 'root, "$name"))') dec.writeln(' res . $field.name = $dec_name (js_get(' + 'root, "$name"))')
} else { } else {
dec.writeln(' $dec_name (js_get(root, "$name"), & (res->$field.name))') dec.writeln(' $dec_name (js_get(root, "$name"), & (res->$field.name))')
} }
@ -80,7 +81,7 @@ cJSON* ${enc_fn_name}($styp val) {
} }
// cJSON_delete // cJSON_delete
// p.cgen.fns << '$dec return opt_ok(res); \n}' // p.cgen.fns << '$dec return opt_ok(res); \n}'
dec.writeln('return opt_ok(res, sizeof(*res)); \n}') dec.writeln('return opt_ok(&res, sizeof(res)); \n}')
enc.writeln('\treturn o;\n}') enc.writeln('\treturn o;\n}')
g.definitions.writeln(dec.str()) g.definitions.writeln(dec.str())
g.gowrappers.writeln(enc.str()) g.gowrappers.writeln(enc.str())

View File

@ -20,6 +20,11 @@ pub fn (mut p Parser) call_expr(is_c, is_js bool, mod string) ast.CallExpr {
} else { } else {
name name
} }
if fn_name == 'json.decode' {
// Makes name_expr() parse the type (`User` in `json.decode(User, txt)`)`
p.inside_is = true
p.expr_mod = ''
}
p.check(.lpar) p.check(.lpar)
args := p.call_args() args := p.call_args()
last_pos := p.tok.position() last_pos := p.tok.position()

View File

@ -32,7 +32,7 @@ mut:
mod string // current module name mod string // current module name
attr string attr string
attr_ctdefine string attr_ctdefine string
expr_mod string expr_mod string // for constructing full type names in parse_type()
scope &ast.Scope scope &ast.Scope
global_scope &ast.Scope global_scope &ast.Scope
imports map[string]string imports map[string]string
@ -588,8 +588,9 @@ pub fn (mut p Parser) name_expr() ast.Expr {
p.inside_is = false p.inside_is = false
// get type position before moving to next // get type position before moving to next
type_pos := p.tok.position() type_pos := p.tok.position()
typ := p.parse_type()
return ast.Type{ return ast.Type{
typ: p.parse_type() typ: typ
pos: type_pos pos: type_pos
} }
} }
@ -1062,8 +1063,8 @@ fn (mut p Parser) return_stmt() ast.Return {
// left hand side of `=` or `:=` in `a,b,c := 1,2,3` // left hand side of `=` or `:=` in `a,b,c := 1,2,3`
fn (mut p Parser) global_decl() ast.GlobalDecl { fn (mut p Parser) global_decl() ast.GlobalDecl {
if !p.pref.translated && !p.pref.is_livemain && !p.builtin_mod && !p.pref.building_v && p.mod != if !p.pref.translated && !p.pref.is_livemain && !p.builtin_mod && !p.pref.building_v &&
'ui' && p.mod != 'gg2' && p.mod != 'uiold' && !os.getwd().contains('/volt') && !p.pref.enable_globals { p.mod != 'ui' && p.mod != 'gg2' && p.mod != 'uiold' && !os.getwd().contains('/volt') && !p.pref.enable_globals {
p.error('use `v --enable-globals ...` to enable globals') p.error('use `v --enable-globals ...` to enable globals')
} }
p.next() p.next()