checker: do not allow {} for aliases; orm: update stmt

pull/5483/head
Alexander Medvednikov 2020-06-24 16:35:18 +02:00
parent 0ea2e687d1
commit 9df29d0dd2
5 changed files with 83 additions and 32 deletions

View File

@ -126,10 +126,9 @@ fn test_orm_sqlite() {
// //
/* /*
sql db { sql db {
update User set age = 31 where name = 'Kate' update User set age = 31 where name == 'Kate'
} }
*/ */
} }

View File

@ -343,6 +343,10 @@ pub fn (mut c Checker) struct_init(mut struct_init ast.StructInit) table.Type {
if type_sym.kind == .interface_ { if type_sym.kind == .interface_ {
c.error('cannot instantiate interface `$type_sym.name`', struct_init.pos) c.error('cannot instantiate interface `$type_sym.name`', struct_init.pos)
} }
if type_sym.kind == .alias {
c.error('cannot instantiate type alias `$type_sym.name`', struct_init.pos)
return table.void_type
}
if !type_sym.is_public && type_sym.kind != .placeholder && type_sym.mod != c.mod { if !type_sym.is_public && type_sym.kind != .placeholder && type_sym.mod != c.mod {
c.error('type `$type_sym.name` is private', struct_init.pos) c.error('type `$type_sym.name` is private', struct_init.pos)
} }

View File

@ -501,7 +501,7 @@ pub fn (mut p Parser) stmt(is_top_level bool) ast.Stmt {
.name, .key_mut, .key_static, .mul { .name, .key_mut, .key_static, .mul {
if p.tok.kind == .name { if p.tok.kind == .name {
if p.tok.lit == 'sql' { if p.tok.lit == 'sql' {
return p.sql_insert_expr() return p.sql_stmt()
} }
if p.peek_tok.kind == .colon { if p.peek_tok.kind == .colon {
// `label:` // `label:`

View File

@ -73,7 +73,8 @@ fn (mut p Parser) sql_expr() ast.Expr {
// get only string and int fields // get only string and int fields
// mut fields := []Var // mut fields := []Var
info := sym.info as table.Struct info := sym.info as table.Struct
fields := info.fields.filter(it.typ in [table.string_type, table.int_type, table.bool_type] && 'skip' !in it.attrs) fields := info.fields.filter(it.typ in [table.string_type, table.int_type, table.bool_type] &&
'skip' !in it.attrs)
if fields.len == 0 { if fields.len == 0 {
p.error('V orm: select: empty fields in `$table_name`') p.error('V orm: select: empty fields in `$table_name`')
} }
@ -103,9 +104,13 @@ fn (mut p Parser) sql_expr() ast.Expr {
} }
} }
fn (mut p Parser) sql_insert_expr() ast.SqlStmt{ // insert user into User
// update User set nr_oders=nr_orders+1 where id == user_id
fn (mut p Parser) sql_stmt() ast.SqlStmt {
p.inside_match = true p.inside_match = true
defer { p.inside_match = false } defer {
p.inside_match = false
}
// `sql db {` // `sql db {`
p.check_name() p.check_name()
db_expr := p.expr(0) db_expr := p.expr(0)
@ -113,27 +118,65 @@ fn (mut p Parser) sql_insert_expr() ast.SqlStmt{
p.check(.lcbr) p.check(.lcbr)
// kind := ast.SqlExprKind.select_ // kind := ast.SqlExprKind.select_
// //
p.check_name() // insert mut n := p.check_name() // insert
mut object_var_name := '' mut kind := ast.SqlStmtKind.insert
if n == 'delete' {
kind = .delete
} else if n == 'update' {
kind = .update
}
mut inserted_var_name := ''
mut table_name := ''
expr := p.expr(0) expr := p.expr(0)
match expr { match expr {
ast.Ident { object_var_name = expr.name } ast.Ident {
else { p.error('can only insert variables') } if kind == .insert {
inserted_var_name = expr.name
} else if kind == .update {
table_name = expr.name
} }
n := p.check_name() // into }
if n != 'into' { else {
p.error('can only insert variables')
}
}
n = p.check_name() // into
mut updated_columns := []string{}
if kind == .insert && n != 'into' {
p.error('expecting `into`') p.error('expecting `into`')
} else if kind == .update {
if n != 'set' {
p.error('expecting `set`')
} }
table_type := p.parse_type() // `User` column := p.check_name()
updated_columns << column
p.check(.assign)
p.expr(0)
}
mut table_type := table.Type(0)
if kind == .insert {
table_type = p.parse_type() // `User`
sym := p.table.get_type_symbol(table_type) sym := p.table.get_type_symbol(table_type)
// info := sym.info as table.Struct // info := sym.info as table.Struct
// fields := info.fields.filter(it.typ in [table.string_type, table.int_type, table.bool_type]) // fields := info.fields.filter(it.typ in [table.string_type, table.int_type, table.bool_type])
table_name := sym.name table_name = sym.name
} else if kind == .update {
idx := p.table.find_type_idx(table_name)
table_type = table.new_type(idx)
p.check_sql_keyword('where')
p.expr(0)
}
p.check(.rcbr) p.check(.rcbr)
return ast.SqlStmt{ return ast.SqlStmt{
db_expr: db_expr db_expr: db_expr
table_name: table_name table_name: table_name
table_type: table_type table_type: table_type
object_var_name: object_var_name object_var_name: inserted_var_name
}
}
fn (mut p Parser) check_sql_keyword(name string) {
if p.check_name() != name {
p.error('orm: expecting `$name`')
} }
} }

View File

@ -62,7 +62,9 @@ fn (mut p Parser) struct_decl() ast.StructDecl {
mut comments := []ast.Comment{} mut comments := []ast.Comment{}
for p.tok.kind == .comment { for p.tok.kind == .comment {
comments << p.comment() comments << p.comment()
if p.tok.kind == .rcbr {break} if p.tok.kind == .rcbr {
break
}
} }
if p.tok.kind == .rcbr { if p.tok.kind == .rcbr {
end_comments = comments end_comments = comments
@ -112,17 +114,19 @@ fn (mut p Parser) struct_decl() ast.StructDecl {
} }
for p.tok.kind == .comment { for p.tok.kind == .comment {
comments << p.comment() comments << p.comment()
if p.tok.kind == .rcbr {break} if p.tok.kind == .rcbr {
break
}
} }
field_start_pos := p.tok.position() field_start_pos := p.tok.position()
field_name := p.check_name() field_name := p.check_name()
// p.warn('field $field_name') // p.warn('field $field_name')
for p.tok.kind == .comment { for p.tok.kind == .comment {
comments << p.comment() comments << p.comment()
if p.tok.kind == .rcbr {break} if p.tok.kind == .rcbr {
break
}
} }
// println(p.tok.position()) // println(p.tok.position())
typ := p.parse_type() typ := p.parse_type()
// field_pos := field_start_pos.extend(p.tok.position()) // field_pos := field_start_pos.extend(p.tok.position())
@ -144,9 +148,10 @@ fn (mut p Parser) struct_decl() ast.StructDecl {
break break
} }
comments << p.comment() comments << p.comment()
if p.tok.kind == .rcbr {break} if p.tok.kind == .rcbr {
break
}
} }
mut attrs := []string{} mut attrs := []string{}
if p.tok.kind == .lsbr { if p.tok.kind == .lsbr {
parsed_attrs := p.attributes(false) parsed_attrs := p.attributes(false)
@ -163,7 +168,7 @@ fn (mut p Parser) struct_decl() ast.StructDecl {
// p.expr(0) // p.expr(0)
default_expr = p.expr(0) default_expr = p.expr(0)
match default_expr { match default_expr {
ast.EnumVal { it.typ = typ } ast.EnumVal { default_expr.typ = typ }
// TODO: implement all types?? // TODO: implement all types??
else {} else {}
} }
@ -340,7 +345,7 @@ fn (mut p Parser) interface_decl() ast.InterfaceDecl {
p.error('interface methods cannot contain uppercase letters, use snake_case instead') p.error('interface methods cannot contain uppercase letters, use snake_case instead')
} }
// field_names << name // field_names << name
args2, _ := p.fn_args() args2, _ := p.fn_args() // TODO merge table.Arg and ast.Arg to avoid this
mut args := [table.Arg{ mut args := [table.Arg{
name: 'x' name: 'x'
typ: typ typ: typ