checker: do not allow {} for aliases; orm: update stmt
parent
0ea2e687d1
commit
9df29d0dd2
|
@ -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'
|
||||||
}
|
}
|
||||||
*/
|
*/
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
|
@ -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)
|
||||||
}
|
}
|
||||||
|
|
|
@ -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:`
|
||||||
|
|
|
@ -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`')
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -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
|
||||||
|
|
Loading…
Reference in New Issue