2019-08-10 23:02:48 +02:00
|
|
|
// Copyright (c) 2019 Alexander Medvednikov. All rights reserved.
|
|
|
|
// Use of this source code is governed by an MIT license
|
|
|
|
// that can be found in the LICENSE file.
|
|
|
|
|
2019-10-13 15:37:43 +02:00
|
|
|
module compiler
|
2019-08-09 18:10:59 +02:00
|
|
|
|
2019-09-17 21:41:58 +02:00
|
|
|
import strings
|
2019-08-09 18:10:59 +02:00
|
|
|
|
2019-08-20 17:05:56 +02:00
|
|
|
fn sql_params2params_gen(sql_params []string, sql_types []string, qprefix string) string {
|
2019-08-20 16:32:39 +02:00
|
|
|
mut params_gen := ''
|
|
|
|
for i, mparam in sql_params {
|
2019-08-27 06:53:56 +02:00
|
|
|
param := mparam.trim_space()
|
2019-08-20 17:05:56 +02:00
|
|
|
paramtype := sql_types[ i ]
|
2019-08-20 16:32:39 +02:00
|
|
|
if param[0].is_digit() {
|
|
|
|
params_gen += '${qprefix}params[$i] = int_str($param).str;\n'
|
2019-08-20 17:05:56 +02:00
|
|
|
}else if param[0] == `\'` {
|
2019-08-27 06:53:56 +02:00
|
|
|
sparam := param.trim('\'')
|
2019-08-20 16:32:39 +02:00
|
|
|
params_gen += '${qprefix}params[$i] = "$sparam";\n'
|
2019-08-20 17:05:56 +02:00
|
|
|
} else {
|
|
|
|
// A variable like q.nr_orders
|
|
|
|
if paramtype == 'int' {
|
|
|
|
params_gen += '${qprefix}params[$i] = int_str( $param ).str;\n'
|
|
|
|
}else if paramtype == 'string' {
|
|
|
|
params_gen += '${qprefix}params[$i] = ${param}.str;\n'
|
|
|
|
}else{
|
2019-09-23 23:40:34 +02:00
|
|
|
verror('orm: only int and string variable types are supported in queries')
|
2019-08-20 17:05:56 +02:00
|
|
|
}
|
2019-08-20 16:32:39 +02:00
|
|
|
}
|
|
|
|
}
|
2019-08-20 17:05:56 +02:00
|
|
|
//println('>>>>>>>> params_gen')
|
|
|
|
//println( params_gen )
|
2019-08-20 16:32:39 +02:00
|
|
|
return params_gen
|
|
|
|
}
|
|
|
|
|
2019-09-17 21:41:58 +02:00
|
|
|
// `db.select from User where id == 1 && nr_bookings > 0`
|
2019-08-09 18:10:59 +02:00
|
|
|
fn (p mut Parser) select_query(fn_ph int) string {
|
2019-08-20 17:05:56 +02:00
|
|
|
// NB: qprefix and { p.sql_i, p.sql_params, p.sql_types } SHOULD be reset for each query,
|
2019-08-20 14:34:34 +02:00
|
|
|
// because we can have many queries in the _same_ scope.
|
|
|
|
qprefix := p.get_tmp().replace('tmp','sql') + '_'
|
|
|
|
p.sql_i = 0
|
2019-08-20 16:32:39 +02:00
|
|
|
p.sql_params = []string
|
2019-08-20 17:05:56 +02:00
|
|
|
p.sql_types = []string
|
2019-09-17 21:41:58 +02:00
|
|
|
|
|
|
|
mut q := 'select '
|
|
|
|
p.check(.key_select)
|
|
|
|
n := p.check_name()
|
2019-08-09 18:10:59 +02:00
|
|
|
if n == 'count' {
|
2019-09-17 21:41:58 +02:00
|
|
|
q += 'count(*) from '
|
|
|
|
p.check_name()
|
|
|
|
}
|
|
|
|
table_name := p.check_name()
|
|
|
|
// Register this type's fields as variables so they can be used in where expressions
|
|
|
|
typ := p.table.find_type(table_name)
|
2019-08-09 18:10:59 +02:00
|
|
|
if typ.name == '' {
|
2019-09-17 21:41:58 +02:00
|
|
|
p.error('unknown type `$table_name`')
|
|
|
|
}
|
|
|
|
//fields := typ.fields.filter(typ == 'string' || typ == 'int')
|
|
|
|
// get only string and int fields
|
|
|
|
mut fields := []Var
|
2019-08-12 17:54:28 +02:00
|
|
|
for i, field in typ.fields {
|
2019-09-17 21:41:58 +02:00
|
|
|
if field.typ != 'string' && field.typ != 'int' {
|
2019-08-12 17:54:28 +02:00
|
|
|
continue
|
2019-09-17 21:41:58 +02:00
|
|
|
}
|
|
|
|
fields << field
|
|
|
|
}
|
2019-08-12 17:54:28 +02:00
|
|
|
if fields.len == 0 {
|
2019-09-17 21:41:58 +02:00
|
|
|
p.error('V orm: select: empty fields in `$table_name`')
|
|
|
|
}
|
2019-08-12 17:54:28 +02:00
|
|
|
if fields[0].name != 'id' {
|
2019-09-17 21:41:58 +02:00
|
|
|
p.error('V orm: `id int` must be the first field in `$table_name`')
|
|
|
|
}
|
|
|
|
// 'select id, name, age from...'
|
2019-08-09 18:10:59 +02:00
|
|
|
if n == 'from' {
|
2019-08-12 17:54:28 +02:00
|
|
|
for i, field in fields {
|
2019-09-17 21:41:58 +02:00
|
|
|
q += field.name
|
2019-08-12 17:54:28 +02:00
|
|
|
if i < fields.len - 1 {
|
2019-09-17 21:41:58 +02:00
|
|
|
q += ', '
|
|
|
|
}
|
|
|
|
}
|
|
|
|
q += ' from '
|
|
|
|
}
|
2019-08-12 17:54:28 +02:00
|
|
|
for field in fields {
|
2019-09-17 21:41:58 +02:00
|
|
|
//println('registering sql field var $field.name')
|
|
|
|
if field.typ != 'string' && field.typ != 'int' {
|
2019-08-12 17:54:28 +02:00
|
|
|
continue
|
2019-09-17 21:41:58 +02:00
|
|
|
}
|
2019-09-23 19:34:08 +02:00
|
|
|
p.register_var({ field | is_used:true })
|
2019-09-17 21:41:58 +02:00
|
|
|
}
|
|
|
|
q += table_name
|
|
|
|
// `where` statement
|
|
|
|
if p.tok == .name && p.lit == 'where' {
|
|
|
|
p.next()
|
|
|
|
p.cgen.start_tmp()
|
|
|
|
p.is_sql = true
|
|
|
|
p.bool_expression()
|
|
|
|
p.is_sql = false
|
|
|
|
q += ' where ' + p.cgen.end_tmp()
|
|
|
|
}
|
|
|
|
// limit?
|
|
|
|
mut query_one := false
|
|
|
|
if p.tok == .name && p.lit == 'limit' {
|
|
|
|
p.next()
|
|
|
|
p.cgen.start_tmp()
|
|
|
|
p.is_sql = true
|
|
|
|
p.bool_expression()
|
|
|
|
p.is_sql = false
|
|
|
|
limit := p.cgen.end_tmp()
|
|
|
|
q += ' limit ' + limit
|
|
|
|
// `limit 1` means we are getting `?User`, not `[]User`
|
|
|
|
if limit.trim_space() == '1' {
|
|
|
|
query_one = true
|
|
|
|
}
|
|
|
|
}
|
2019-08-20 14:34:34 +02:00
|
|
|
println('sql query="$q"')
|
|
|
|
p.cgen.insert_before('// DEBUG_SQL prefix: $qprefix | fn_ph: $fn_ph | query: "$q" ')
|
|
|
|
|
2019-08-09 18:10:59 +02:00
|
|
|
if n == 'count' {
|
|
|
|
p.cgen.set_placeholder(fn_ph, 'pg__DB_q_int(')
|
2019-09-17 21:41:58 +02:00
|
|
|
p.gen(', tos2("$q"))')
|
|
|
|
} else {
|
|
|
|
// Build an object, assign each field.
|
|
|
|
tmp := p.get_tmp()
|
|
|
|
mut obj_gen := strings.new_builder(300)
|
2019-08-12 17:54:28 +02:00
|
|
|
for i, field in fields {
|
2019-09-17 21:41:58 +02:00
|
|
|
mut cast := ''
|
2019-08-09 18:10:59 +02:00
|
|
|
if field.typ == 'int' {
|
2019-09-17 21:41:58 +02:00
|
|
|
cast = 'v_string_int'
|
|
|
|
}
|
2019-10-10 00:07:00 +02:00
|
|
|
obj_gen.writeln('${qprefix}$tmp . $field.name = $cast( *(string*)array_get(${qprefix}row.vals, $i) );')
|
2019-09-17 21:41:58 +02:00
|
|
|
}
|
|
|
|
// One object
|
|
|
|
if query_one {
|
2019-08-20 17:05:56 +02:00
|
|
|
mut params_gen := sql_params2params_gen( p.sql_params, p.sql_types, qprefix )
|
2019-08-09 18:10:59 +02:00
|
|
|
p.cgen.insert_before('
|
|
|
|
|
2019-08-20 14:34:34 +02:00
|
|
|
char* ${qprefix}params[$p.sql_i];
|
|
|
|
$params_gen
|
|
|
|
|
|
|
|
Option_${table_name} opt_${qprefix}$tmp;
|
|
|
|
void* ${qprefix}res = PQexecParams(db.conn, "$q", $p.sql_i, 0, ${qprefix}params, 0, 0, 0) ;
|
|
|
|
array_pg__Row ${qprefix}rows = pg__res_to_rows ( ${qprefix}res ) ;
|
|
|
|
Option_pg__Row opt_${qprefix}row = pg__rows_first_or_empty( ${qprefix}rows );
|
|
|
|
if (! opt_${qprefix}row . ok ) {
|
|
|
|
opt_${qprefix}$tmp = v_error( opt_${qprefix}row . error );
|
|
|
|
}else{
|
|
|
|
$table_name ${qprefix}$tmp;
|
2019-09-17 21:41:58 +02:00
|
|
|
pg__Row ${qprefix}row = *(pg__Row*) opt_${qprefix}row . data;
|
2019-08-20 14:34:34 +02:00
|
|
|
${obj_gen.str()}
|
|
|
|
opt_${qprefix}$tmp = opt_ok( & ${qprefix}$tmp, sizeof($table_name) );
|
|
|
|
}
|
2019-08-09 18:10:59 +02:00
|
|
|
|
|
|
|
')
|
2019-08-20 14:34:34 +02:00
|
|
|
p.cgen.resetln('opt_${qprefix}$tmp')
|
2019-09-17 21:41:58 +02:00
|
|
|
}
|
|
|
|
// Array
|
2019-08-09 18:10:59 +02:00
|
|
|
else {
|
2019-08-20 16:32:39 +02:00
|
|
|
q += ' order by id'
|
2019-08-20 17:05:56 +02:00
|
|
|
params_gen := sql_params2params_gen( p.sql_params, p.sql_types, qprefix )
|
2019-08-20 14:34:34 +02:00
|
|
|
p.cgen.insert_before('char* ${qprefix}params[$p.sql_i];
|
2019-09-17 21:41:58 +02:00
|
|
|
$params_gen
|
2019-08-13 13:50:19 +02:00
|
|
|
|
2019-08-20 14:34:34 +02:00
|
|
|
void* ${qprefix}res = PQexecParams(db.conn, "$q", $p.sql_i, 0, ${qprefix}params, 0, 0, 0) ;
|
|
|
|
array_pg__Row ${qprefix}rows = pg__res_to_rows(${qprefix}res);
|
2019-08-09 18:10:59 +02:00
|
|
|
|
2019-09-17 21:41:58 +02:00
|
|
|
// TODO preallocate
|
2019-08-20 14:34:34 +02:00
|
|
|
array ${qprefix}arr_$tmp = new_array(0, 0, sizeof($table_name));
|
|
|
|
for (int i = 0; i < ${qprefix}rows.len; i++) {
|
2019-10-10 00:07:00 +02:00
|
|
|
pg__Row ${qprefix}row = *(pg__Row*)array_get(${qprefix}rows, i);
|
2019-08-20 14:34:34 +02:00
|
|
|
$table_name ${qprefix}$tmp;
|
2019-09-17 21:41:58 +02:00
|
|
|
${obj_gen.str()}
|
2019-08-20 14:34:34 +02:00
|
|
|
_PUSH(&${qprefix}arr_$tmp, ${qprefix}$tmp, ${tmp}2, $table_name);
|
2019-09-17 21:41:58 +02:00
|
|
|
}
|
2019-08-09 18:10:59 +02:00
|
|
|
')
|
2019-08-20 14:34:34 +02:00
|
|
|
p.cgen.resetln('${qprefix}arr_$tmp')
|
2019-09-17 21:41:58 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
}
|
2019-08-09 18:10:59 +02:00
|
|
|
if n == 'count' {
|
2019-09-17 21:41:58 +02:00
|
|
|
return 'int'
|
2019-08-20 16:32:39 +02:00
|
|
|
} else if query_one {
|
|
|
|
opt_type := 'Option_$table_name'
|
|
|
|
p.cgen.typedefs << 'typedef Option $opt_type;'
|
2019-10-24 11:47:21 +02:00
|
|
|
p.table.register_builtin( opt_type )
|
2019-08-20 16:32:39 +02:00
|
|
|
return opt_type
|
2019-08-09 18:10:59 +02:00
|
|
|
} else {
|
2019-09-17 21:41:58 +02:00
|
|
|
p.register_array('array_$table_name')
|
|
|
|
return 'array_$table_name'
|
|
|
|
}
|
|
|
|
}
|
2019-08-09 18:10:59 +02:00
|
|
|
|
2019-09-17 21:41:58 +02:00
|
|
|
// `db.insert(user)`
|
|
|
|
fn (p mut Parser) insert_query(fn_ph int) {
|
|
|
|
p.check_name()
|
|
|
|
p.check(.lpar)
|
|
|
|
var_name := p.check_name()
|
|
|
|
p.check(.rpar)
|
2019-09-23 19:34:08 +02:00
|
|
|
var := p.find_var(var_name) or { return }
|
2019-09-17 21:41:58 +02:00
|
|
|
typ := p.table.find_type(var.typ)
|
|
|
|
mut fields := []Var
|
2019-08-12 17:54:28 +02:00
|
|
|
for i, field in typ.fields {
|
2019-09-17 21:41:58 +02:00
|
|
|
if field.typ != 'string' && field.typ != 'int' {
|
2019-08-12 17:54:28 +02:00
|
|
|
continue
|
2019-09-17 21:41:58 +02:00
|
|
|
}
|
|
|
|
fields << field
|
|
|
|
}
|
2019-08-12 17:54:28 +02:00
|
|
|
if fields.len == 0 {
|
2019-09-17 21:41:58 +02:00
|
|
|
p.error('V orm: insert: empty fields in `$var.typ`')
|
|
|
|
}
|
2019-08-12 17:54:28 +02:00
|
|
|
if fields[0].name != 'id' {
|
2019-09-17 21:41:58 +02:00
|
|
|
p.error('V orm: `id int` must be the first field in `$var.typ`')
|
|
|
|
}
|
|
|
|
table_name := var.typ
|
|
|
|
mut sfields := '' // 'name, city, country'
|
|
|
|
mut params := '' // params[0] = 'bob'; params[1] = 'Vienna';
|
|
|
|
mut vals := '' // $1, $2, $3...
|
|
|
|
mut nr_vals := 0
|
2019-08-12 17:54:28 +02:00
|
|
|
for i, field in fields {
|
2019-08-09 18:10:59 +02:00
|
|
|
if field.name == 'id' {
|
2019-09-17 21:41:58 +02:00
|
|
|
continue
|
|
|
|
}
|
|
|
|
sfields += field.name
|
|
|
|
vals += '$' + i.str()
|
|
|
|
nr_vals++
|
2019-08-09 18:10:59 +02:00
|
|
|
params += 'params[${i-1}] = '
|
2019-09-17 21:41:58 +02:00
|
|
|
if field.typ == 'string' {
|
|
|
|
params += '$var_name . $field.name .str;\n'
|
2019-08-09 18:10:59 +02:00
|
|
|
} else if field.typ == 'int' {
|
2019-09-17 21:41:58 +02:00
|
|
|
params += 'int_str($var_name . $field.name).str;\n'
|
2019-08-09 18:10:59 +02:00
|
|
|
} else {
|
2019-09-17 21:41:58 +02:00
|
|
|
p.error('V ORM: unsupported type `$field.typ`')
|
|
|
|
}
|
|
|
|
if i < fields.len - 1 {
|
|
|
|
sfields += ', '
|
|
|
|
vals += ', '
|
|
|
|
}
|
|
|
|
}
|
|
|
|
p.cgen.insert_before('char* params[$nr_vals];' + params)
|
2019-08-09 18:10:59 +02:00
|
|
|
p.cgen.set_placeholder(fn_ph, 'PQexecParams( ')
|
2019-08-12 17:54:28 +02:00
|
|
|
p.genln('.conn, "insert into $table_name ($sfields) values ($vals)", $nr_vals,
|
2019-09-17 21:41:58 +02:00
|
|
|
0, params, 0, 0, 0)')
|
|
|
|
}
|
2019-08-09 18:10:59 +02:00
|
|
|
|