orm: `update` with selectors and other fixes

pull/5505/head
Alexander Medvednikov 2020-06-25 22:23:19 +02:00
parent 450c6e843c
commit 328cb7ed7e
5 changed files with 59 additions and 6 deletions

View File

@ -7,6 +7,7 @@
- C2V translator - C2V translator
- doom.v - doom.v
- rune type, replace ustring with []rune - rune type, replace ustring with []rune
- maps with non-string keys

View File

@ -106,13 +106,11 @@ fn test_orm_sqlite() {
kate := sql db { kate := sql db {
select from User where id == 3 select from User where id == 3
} }
println(kate)
assert kate.is_customer == true assert kate.is_customer == true
// //
customer := sql db { customer := sql db {
select from User where is_customer == true limit 1 select from User where is_customer == true limit 1
} }
println(customer)
assert customer.is_customer == true assert customer.is_customer == true
assert customer.name == 'Kate' assert customer.name == 'Kate'
// //
@ -122,19 +120,48 @@ fn test_orm_sqlite() {
kate2 := sql db { kate2 := sql db {
select from User where id == 3 select from User where id == 3
} }
println(kate2)
assert kate2.age == 31 assert kate2.age == 31
assert kate2.name == 'Kate' assert kate2.name == 'Kate'
// //
sql db { sql db {
update User set age = 32, name = 'Kate N' where name == 'Kate' update User set age = 32, name = 'Kate N' where name == 'Kate'
} }
kate3 := sql db { mut kate3 := sql db {
select from User where id == 3
}
assert kate3.age == 32
assert kate3.name == 'Kate N'
//
/*
sql db {
update User set age = age + 1, name = 'Kate N' where name == 'Kate'
}
kate3 = sql db {
select from User where id == 3 select from User where id == 3
} }
println(kate3) println(kate3)
assert kate3.age == 32 assert kate3.age == 32
assert kate3.name == 'Kate N' assert kate3.name == 'Kate N'
*/
new_age := 33
sql db {
update User set age = new_age, name = 'Kate N' where id == 3
}
kate3 = sql db {
select from User where id == 3
}
assert kate3.age == 33
assert kate3.name == 'Kate N'
//
foo := Foo{34}
sql db {
update User set age = foo.age, name = 'Kate N' where id == 3
}
kate3 = sql db {
select from User where id == 3
}
assert kate3.age == 34
assert kate3.name == 'Kate N'
} }
struct User { struct User {
@ -145,6 +172,10 @@ struct User {
skipped_string string [skip] skipped_string string [skip]
} }
struct Foo {
age int
}
fn test_orm_pg() { fn test_orm_pg() {

View File

@ -105,7 +105,8 @@ pub:
expr Expr expr Expr
field_name string field_name string
pub mut: pub mut:
expr_type table.Type expr_type table.Type // type of `Foo` in `Foo.bar`
typ table.Type // type of the entire thing (`Foo.bar`)
} }
// module declaration // module declaration

View File

@ -1252,6 +1252,7 @@ pub fn (mut c Checker) selector_expr(mut selector_expr ast.SelectorExpr) table.T
// variadic // variadic
if typ.has_flag(.variadic) { if typ.has_flag(.variadic) {
if field_name == 'len' { if field_name == 'len' {
selector_expr.typ = table.int_type
return table.int_type return table.int_type
} }
} }
@ -1259,6 +1260,7 @@ pub fn (mut c Checker) selector_expr(mut selector_expr ast.SelectorExpr) table.T
if sym.mod != c.mod && !field.is_pub { if sym.mod != c.mod && !field.is_pub {
c.error('field `${sym.name}.$field_name` is not public', selector_expr.pos) c.error('field `${sym.name}.$field_name` is not public', selector_expr.pos)
} }
selector_expr.typ = field.typ
return field.typ return field.typ
} }
if sym.kind != .struct_ { if sym.kind != .struct_ {
@ -2721,6 +2723,11 @@ fn (mut c Checker) sql_stmt(mut node ast.SqlStmt) table.Type {
fields := c.fetch_and_verify_orm_fields(info, node.pos, node.table_name) fields := c.fetch_and_verify_orm_fields(info, node.pos, node.table_name)
node.fields = fields node.fields = fields
c.expr(node.db_expr) c.expr(node.db_expr)
if node.kind== .update {
for expr in node.update_exprs {
c.expr(expr)
}
}
return table.void_type return table.void_type
} }

View File

@ -273,10 +273,23 @@ fn (mut g Gen) expr_to_sql(expr ast.Expr) {
} else if typ == table.int_type { } else if typ == table.int_type {
g.sql_bind_int(expr.name) g.sql_bind_int(expr.name)
} else { } else {
verror('bad sql type $typ') verror('bad sql type=$typ ident_name=$expr.name')
} }
} }
} }
ast.SelectorExpr {
g.inc_sql_i()
if expr.typ == table.int_type {
if expr.expr !is ast.Ident {
verror('orm selector not ident')
}
ident := expr.expr as ast.Ident
g.sql_bind_int(ident.name + '.' + expr.field_name)
}
else {
verror('bad sql type=$expr.typ selector expr=$expr.field_name')
}
}
else { else {
g.expr(expr) g.expr(expr)
} }