Finished most of delete functions

* Fixed some template errors
* Added IN helpers
This commit is contained in:
Patrick O'brien 2016-04-23 21:54:24 +10:00
parent 6d8706c07d
commit 9c493810ec
11 changed files with 172 additions and 30 deletions

View file

@ -9,6 +9,49 @@ import (
"unicode" "unicode"
) )
// WherePrimaryKeyIn generates a "in" string for where queries
// For example: (col1, col2) IN (($1, $2), ($3, $4))
func WherePrimaryKeyIn(numRows int, keyNames ...string) string {
in := &bytes.Buffer{}
if len(keyNames) == 0 {
return ""
}
in.WriteByte('(')
for i := 0; i < len(keyNames); i++ {
in.WriteString(`"` + keyNames[i] + `"`)
if i < len(keyNames)-1 {
in.WriteByte(',')
}
}
in.WriteString(") IN (")
c := 1
for i := 0; i < numRows; i++ {
for y := 0; y < len(keyNames); y++ {
if len(keyNames) > 1 && y == 0 {
in.WriteByte('(')
}
in.WriteString(fmt.Sprintf("$%d", c))
c++
if len(keyNames) > 1 && y == len(keyNames)-1 {
in.WriteByte(')')
}
if i != numRows-1 || y != len(keyNames)-1 {
in.WriteByte(',')
}
}
}
in.WriteByte(')')
return in.String()
}
// SelectNames returns the column names for a select statement // SelectNames returns the column names for a select statement
// Eg: col1, col2, col3 // Eg: col1, col2, col3
func SelectNames(results interface{}) string { func SelectNames(results interface{}) string {

View file

@ -11,6 +11,58 @@ type testObj struct {
HeadSize int HeadSize int
} }
func TestWherePrimaryKeyIn(t *testing.T) {
t.Parallel()
x := WherePrimaryKeyIn(1, "aa")
expect := `("aa") IN ($1)`
if x != expect {
t.Errorf("Expected %s, got %s\n", expect, x)
}
x = WherePrimaryKeyIn(2, "aa")
expect = `("aa") IN ($1,$2)`
if x != expect {
t.Errorf("Expected %s, got %s\n", expect, x)
}
x = WherePrimaryKeyIn(3, "aa")
expect = `("aa") IN ($1,$2,$3)`
if x != expect {
t.Errorf("Expected %s, got %s\n", expect, x)
}
x = WherePrimaryKeyIn(1, "aa", "bb")
expect = `("aa","bb") IN (($1,$2))`
if x != expect {
t.Errorf("Expected %s, got %s\n", expect, x)
}
x = WherePrimaryKeyIn(2, "aa", "bb")
expect = `("aa","bb") IN (($1,$2),($3,$4))`
if x != expect {
t.Errorf("Expected %s, got %s\n", expect, x)
}
x = WherePrimaryKeyIn(3, "aa", "bb")
expect = `("aa","bb") IN (($1,$2),($3,$4),($5,$6))`
if x != expect {
t.Errorf("Expected %s, got %s\n", expect, x)
}
x = WherePrimaryKeyIn(4, "aa", "bb")
expect = `("aa","bb") IN (($1,$2),($3,$4),($5,$6),($7,$8))`
if x != expect {
t.Errorf("Expected %s, got %s\n", expect, x)
}
x = WherePrimaryKeyIn(4, "aa", "bb", "cc")
expect = `("aa","bb","cc") IN (($1,$2,$3),($4,$5,$6),($7,$8,$9),($10,$11,$12))`
if x != expect {
t.Errorf("Expected %s, got %s\n", expect, x)
}
}
func TestGoVarToSQLName(t *testing.T) { func TestGoVarToSQLName(t *testing.T) {
t.Parallel() t.Parallel()

View file

@ -56,18 +56,6 @@ func TestBuildQuery(t *testing.T) {
} }
} }
func TestExecQuery(t *testing.T) {
t.Parallel()
}
func TestExecQueryOne(t *testing.T) {
t.Parallel()
}
func TestExecQueryAll(t *testing.T) {
t.Parallel()
}
func TestSetLimit(t *testing.T) { func TestSetLimit(t *testing.T) {
t.Parallel() t.Parallel()

View file

@ -39,6 +39,7 @@ var sqlBoilerImports = imports{
}, },
thirdparty: importList{ thirdparty: importList{
`"github.com/pobri19/sqlboiler/boil"`, `"github.com/pobri19/sqlboiler/boil"`,
`"github.com/pobri19/sqlboiler/boil/qs"`,
}, },
} }
@ -47,6 +48,7 @@ var sqlBoilerSinglesImports = map[string]imports{
standard: importList{}, standard: importList{},
thirdparty: importList{ thirdparty: importList{
`"github.com/pobri19/sqlboiler/boil"`, `"github.com/pobri19/sqlboiler/boil"`,
`"github.com/pobri19/sqlboiler/boil/qs"`,
}, },
}, },
} }
@ -102,6 +104,7 @@ var sqlBoilerTemplateFuncs = template.FuncMap{
"primaryKeyFlagIndex": primaryKeyFlagIndex, "primaryKeyFlagIndex": primaryKeyFlagIndex,
"updateParamNames": updateParamNames, "updateParamNames": updateParamNames,
"updateParamVariables": updateParamVariables, "updateParamVariables": updateParamVariables,
"primaryKeyStrList": primaryKeyStrList,
} }
// LoadConfigFile loads the toml config file into the cfg object // LoadConfigFile loads the toml config file into the cfg object

View file

@ -176,12 +176,12 @@ func outputCompileErrors(buf *bytes.Buffer, outFolder string) {
} }
scanner := bufio.NewScanner(fh) scanner := bufio.NewScanner(fh)
throwaway := eObj.lineNumber - 2 throwaway := eObj.lineNumber - 5
for throwaway > 0 && scanner.Scan() { for throwaway > 0 && scanner.Scan() {
throwaway-- throwaway--
} }
for i := 0; i < 3; i++ { for i := 0; i < 6; i++ {
if scanner.Scan() { if scanner.Scan() {
b := scanner.Bytes() b := scanner.Bytes()
if len(b) != 0 { if len(b) != 0 {

View file

@ -234,6 +234,16 @@ func wherePrimaryKey(pkeyCols []string, start int) string {
return output return output
} }
// primaryKeyStrList returns a list of primary key column names in strings
// For example: "col1", "col2", "col3"
func primaryKeyStrList(pkeyCols []string) string {
for i, c := range pkeyCols {
pkeyCols[i] = fmt.Sprintf(`"%s"`, c)
}
return strings.Join(pkeyCols, ", ")
}
// commaList returns a comma seperated list: "col1, col2, col3" // commaList returns a comma seperated list: "col1, col2, col3"
func commaList(cols []string) string { func commaList(cols []string) string {
return strings.Join(cols, ", ") return strings.Join(cols, ", ")

View file

@ -1,17 +1,16 @@
{{- $tableNameSingular := titleCaseSingular .Table.Name -}} {{- $tableNameSingular := titleCaseSingular .Table.Name -}}
{{- $tableNamePlural := titleCasePlural .Table.Name -}} {{- $tableNamePlural := titleCasePlural .Table.Name -}}
{{- $varNameSingular := camelCaseSingular .Table.Name -}} {{- $varNameSingular := camelCaseSingular .Table.Name -}}
{{- $varNamePlural := camelCasePlural .Table.Name -}}
type {{$varNameSingular}}Query struct { type {{$varNameSingular}}Query struct {
*boil.Query *boil.Query
} }
// {{$tableNamePlural}}All retrieves all records. // {{$tableNamePlural}}All retrieves all records.
func {{$tableNamePlural}}(mods ...QueryMod) {{$varNameSingular}}Query { func {{$tableNamePlural}}(mods ...qs.QueryMod) {{$varNameSingular}}Query {
return {{$tableNamePlural}}X(boil.GetDB(), mods...) return {{$tableNamePlural}}X(boil.GetDB(), mods...)
} }
func {{$tableNamePlural}}X(exec boil.Executor, mods ...QueryMod) {{$tableNameSingular}}Query { func {{$tableNamePlural}}X(exec boil.Executor, mods ...qs.QueryMod) {{$varNameSingular}}Query {
mods = append(mods, boil.From("{{.Table.Name}}")) mods = append(mods, qs.From("{{.Table.Name}}"))
return NewQueryX(exec, mods...) return {{$varNameSingular}}Query{NewQueryX(exec, mods...)}
} }

View file

@ -3,19 +3,22 @@
{{- $varNameSingular := camelCaseSingular .Table.Name -}} {{- $varNameSingular := camelCaseSingular .Table.Name -}}
// Delete deletes a single {{$tableNameSingular}} record. // Delete deletes a single {{$tableNameSingular}} record.
// Delete will match against the primary key column to find the record to delete. // Delete will match against the primary key column to find the record to delete.
func (o *{{$tableNameSingular}}) Delete(mods ...QueryMod) error { func (o *{{$tableNameSingular}}) Delete() error {
return o.DeleteX(boil.GetDB(), mods...) return o.DeleteX(boil.GetDB())
} }
func (o *{{$tableNameSingular}}) DeleteX(exec boil.Executor, mods ...QueryMod) error { func (o *{{$tableNameSingular}}) DeleteX(exec boil.Executor) error {
var mods []qs.QueryMod
mods = append(mods, mods = append(mods,
boil.From("{{.table.Name}}"), qs.From("{{.Table.Name}}"),
boil.Where("{{wherePrimaryKey .Table.Pkey.Columns 1}}", {{paramsPrimaryKey "o." .Table.PKey.Columns true}}), qs.Where("{{wherePrimaryKey .Table.PKey.Columns 1}}", {{paramsPrimaryKey "o." .Table.PKey.Columns true}}),
) )
query := NewQueryX(exec, mods...) query := NewQueryX(exec, mods...)
boil.SetDelete(query)
_, err := exec.Exec("DELETE FROM {{.Table.Name}} WHERE {{wherePrimaryKey .Table.PKey.Columns 1}}", {{paramsPrimaryKey "o." .Table.PKey.Columns true}}) _, err := boil.ExecQuery(query)
if err != nil { if err != nil {
return fmt.Errorf("{{.PkgName}}: unable to delete from {{.Table.Name}}: %s", err) return fmt.Errorf("{{.PkgName}}: unable to delete from {{.Table.Name}}: %s", err)
} }
@ -24,9 +27,37 @@ func (o *{{$tableNameSingular}}) DeleteX(exec boil.Executor, mods ...QueryMod) e
} }
func (o {{$varNameSingular}}Query) DeleteAll() error { func (o {{$varNameSingular}}Query) DeleteAll() error {
_, err := db.Exec("DELETE FROM {{.Table.Name}} WHERE {{wherePrimaryKey .Table.PKey.Columns 1}}", {{paramsPrimaryKey "o." .Table.PKey.Columns true}}) boil.SetDelete(o)
_, err := boil.ExecQuery(query)
if err != nil { if err != nil {
return fmt.Errorf("{{.PkgName}}: unable to delete from {{.Table.Name}}: %s", err) return fmt.Errorf("{{.PkgName}}: unable to delete all from {{.Table.Name}}: %s", err)
}
return nil
}
func (o {{$varNameSingular}}Slice) DeleteAll() error {
return DeleteAllX(boil.GetDB())
}
func (o {{$varNameSingular}}Slice) DeleteAllX(exec boil.Executor) error {
var mods []qs.QueryMod
args := o.inPrimaryKeyArgs()
in := boil.WherePrimaryKeyIn(len(o), {{primaryKeyStrList .Table.PKey.Columns}})
mods = append(mods,
qs.From("{{.Table.Name}}"),
qs.Where(in, args...),
)
query := NewQueryX(exec, mods...)
boil.SetDelete(query)
_, err := boil.ExecQuery(query)
if err != nil {
return fmt.Errorf("{{.PkgName}}: unable to delete all from {{$varNameSingular}} slice: %s", err)
} }
return nil return nil

View file

@ -21,7 +21,7 @@ func {{$tableNameSingular}}FindX(exec boil.Executor, id int64, selectList ...str
return nil, errors.New("{{.PkgName}}: no id provided for {{.Table.Name}} select") return nil, errors.New("{{.PkgName}}: no id provided for {{.Table.Name}} select")
} }
var {{$varNameSingular}} *{{$tableNameSingular}} var {{$varNameSingular}} *{{$tableNameSingular}}
err := boil.GetDB().Select(&{{$varNameSingular}}, `SELECT {{selectParamNames $dbName .Table.Columns}} WHERE id=$1`, id) //err := boil.GetDB().Select(&{{$varNameSingular}}, `SELECT {{selectParamNames $dbName .Table.Columns}} WHERE id=$1`, id)
if err != nil { if err != nil {
return nil, fmt.Errorf("{{.PkgName}}: unable to select from {{.Table.Name}}: %s", err) return nil, fmt.Errorf("{{.PkgName}}: unable to select from {{.Table.Name}}: %s", err)

View file

@ -0,0 +1,15 @@
{{if hasPrimaryKey .Table.PKey -}}
{{- $tableNameSingular := titleCaseSingular .Table.Name -}}
{{- $varNameSingular := camelCaseSingular .Table.Name -}}
func (o {{$varNameSingular}}Slice) inPrimaryKeyArgs() []interface{} {
var args []interface{}
for i := 0; i < len(o); i++ {
{{- range $key, $value := .Table.PKey.Columns }}
args = append(args, o.{{titleCase $value}})
{{ end -}}
}
return args
}
{{- end}}

View file

@ -7,8 +7,9 @@ func NewQuery(mods ...qs.QueryMod) *boil.Query {
} }
// NewQueryX initializes a new Query using the passed in QueryMods // NewQueryX initializes a new Query using the passed in QueryMods
func NewQueryX(executor boil.Executor, mods ...qs.QueryMod) *boil.Query { func NewQueryX(exec boil.Executor, mods ...qs.QueryMod) *boil.Query {
q := &boil.Query{executor: executor} q := &boil.Query{}
boil.SetExecutor(q, exec)
qs.Apply(q, mods...) qs.Apply(q, mods...)
return q return q