- // Copyright 2016 The Xorm Authors. All rights reserved.
- // Use of this source code is governed by a BSD-style
- // license that can be found in the LICENSE file.
-
- package builder
-
- import (
- "bytes"
- "fmt"
- "sort"
- )
-
- // Insert creates an insert Builder
- func Insert(eq ...interface{}) *Builder {
- builder := &Builder{cond: NewCond()}
- return builder.Insert(eq...)
- }
-
- func (b *Builder) insertSelectWriteTo(w Writer) error {
- if _, err := fmt.Fprintf(w, "INSERT INTO %s ", b.into); err != nil {
- return err
- }
-
- if len(b.insertCols) > 0 {
- fmt.Fprintf(w, "(")
- for _, col := range b.insertCols {
- fmt.Fprintf(w, col)
- }
- fmt.Fprintf(w, ") ")
- }
-
- return b.selectWriteTo(w)
- }
-
- func (b *Builder) insertWriteTo(w Writer) error {
- if len(b.into) <= 0 {
- return ErrNoTableName
- }
- if len(b.insertCols) <= 0 && b.from == "" {
- return ErrNoColumnToInsert
- }
-
- if b.into != "" && b.from != "" {
- return b.insertSelectWriteTo(w)
- }
-
- if _, err := fmt.Fprintf(w, "INSERT INTO %s (", b.into); err != nil {
- return err
- }
-
- var args = make([]interface{}, 0)
- var bs []byte
- var valBuffer = bytes.NewBuffer(bs)
-
- for i, col := range b.insertCols {
- value := b.insertVals[i]
- fmt.Fprint(w, col)
- if e, ok := value.(expr); ok {
- fmt.Fprintf(valBuffer, "(%s)", e.sql)
- args = append(args, e.args...)
- } else {
- fmt.Fprint(valBuffer, "?")
- args = append(args, value)
- }
-
- if i != len(b.insertCols)-1 {
- if _, err := fmt.Fprint(w, ","); err != nil {
- return err
- }
- if _, err := fmt.Fprint(valBuffer, ","); err != nil {
- return err
- }
- }
- }
-
- if _, err := fmt.Fprint(w, ") Values ("); err != nil {
- return err
- }
-
- if _, err := w.Write(valBuffer.Bytes()); err != nil {
- return err
- }
- if _, err := fmt.Fprint(w, ")"); err != nil {
- return err
- }
-
- w.Append(args...)
-
- return nil
- }
-
- type insertColsSorter struct {
- cols []string
- vals []interface{}
- }
-
- func (s insertColsSorter) Len() int {
- return len(s.cols)
- }
-
- func (s insertColsSorter) Swap(i, j int) {
- s.cols[i], s.cols[j] = s.cols[j], s.cols[i]
- s.vals[i], s.vals[j] = s.vals[j], s.vals[i]
- }
-
- func (s insertColsSorter) Less(i, j int) bool {
- return s.cols[i] < s.cols[j]
- }
-
- // Insert sets insert SQL
- func (b *Builder) Insert(eq ...interface{}) *Builder {
- if len(eq) > 0 {
- var paramType = -1
- for _, e := range eq {
- switch t := e.(type) {
- case Eq:
- if paramType == -1 {
- paramType = 0
- }
- if paramType != 0 {
- break
- }
- for k, v := range t {
- b.insertCols = append(b.insertCols, k)
- b.insertVals = append(b.insertVals, v)
- }
- case string:
- if paramType == -1 {
- paramType = 1
- }
- if paramType != 1 {
- break
- }
- b.insertCols = append(b.insertCols, t)
- }
- }
- }
-
- if len(b.insertCols) == len(b.insertVals) {
- sort.Sort(insertColsSorter{
- cols: b.insertCols,
- vals: b.insertVals,
- })
- }
- b.optype = insertType
- return b
- }
|