2019-06-23 23:22:43 +08:00
|
|
|
// Copyright 2019 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 core
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
"database/sql"
|
|
|
|
"errors"
|
|
|
|
"reflect"
|
|
|
|
)
|
|
|
|
|
2019-08-27 10:17:23 +08:00
|
|
|
// Stmt reprents a stmt objects
|
2019-06-23 23:22:43 +08:00
|
|
|
type Stmt struct {
|
|
|
|
*sql.Stmt
|
|
|
|
db *DB
|
|
|
|
names map[string]int
|
|
|
|
}
|
|
|
|
|
|
|
|
func (db *DB) PrepareContext(ctx context.Context, query string) (*Stmt, error) {
|
|
|
|
names := make(map[string]int)
|
|
|
|
var i int
|
|
|
|
query = re.ReplaceAllStringFunc(query, func(src string) string {
|
|
|
|
names[src[1:]] = i
|
|
|
|
i += 1
|
|
|
|
return "?"
|
|
|
|
})
|
|
|
|
|
|
|
|
stmt, err := db.DB.PrepareContext(ctx, query)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
return &Stmt{stmt, db, names}, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (db *DB) Prepare(query string) (*Stmt, error) {
|
|
|
|
return db.PrepareContext(context.Background(), query)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *Stmt) ExecMapContext(ctx context.Context, mp interface{}) (sql.Result, error) {
|
|
|
|
vv := reflect.ValueOf(mp)
|
|
|
|
if vv.Kind() != reflect.Ptr || vv.Elem().Kind() != reflect.Map {
|
|
|
|
return nil, errors.New("mp should be a map's pointer")
|
|
|
|
}
|
|
|
|
|
|
|
|
args := make([]interface{}, len(s.names))
|
|
|
|
for k, i := range s.names {
|
|
|
|
args[i] = vv.Elem().MapIndex(reflect.ValueOf(k)).Interface()
|
|
|
|
}
|
|
|
|
return s.Stmt.ExecContext(ctx, args...)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *Stmt) ExecMap(mp interface{}) (sql.Result, error) {
|
|
|
|
return s.ExecMapContext(context.Background(), mp)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *Stmt) ExecStructContext(ctx context.Context, st interface{}) (sql.Result, error) {
|
|
|
|
vv := reflect.ValueOf(st)
|
|
|
|
if vv.Kind() != reflect.Ptr || vv.Elem().Kind() != reflect.Struct {
|
|
|
|
return nil, errors.New("mp should be a map's pointer")
|
|
|
|
}
|
|
|
|
|
|
|
|
args := make([]interface{}, len(s.names))
|
|
|
|
for k, i := range s.names {
|
|
|
|
args[i] = vv.Elem().FieldByName(k).Interface()
|
|
|
|
}
|
|
|
|
return s.Stmt.ExecContext(ctx, args...)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *Stmt) ExecStruct(st interface{}) (sql.Result, error) {
|
|
|
|
return s.ExecStructContext(context.Background(), st)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *Stmt) QueryContext(ctx context.Context, args ...interface{}) (*Rows, error) {
|
|
|
|
rows, err := s.Stmt.QueryContext(ctx, args...)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
return &Rows{rows, s.db}, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *Stmt) Query(args ...interface{}) (*Rows, error) {
|
|
|
|
return s.QueryContext(context.Background(), args...)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *Stmt) QueryMapContext(ctx context.Context, mp interface{}) (*Rows, error) {
|
|
|
|
vv := reflect.ValueOf(mp)
|
|
|
|
if vv.Kind() != reflect.Ptr || vv.Elem().Kind() != reflect.Map {
|
|
|
|
return nil, errors.New("mp should be a map's pointer")
|
|
|
|
}
|
|
|
|
|
|
|
|
args := make([]interface{}, len(s.names))
|
|
|
|
for k, i := range s.names {
|
|
|
|
args[i] = vv.Elem().MapIndex(reflect.ValueOf(k)).Interface()
|
|
|
|
}
|
|
|
|
|
|
|
|
return s.QueryContext(ctx, args...)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *Stmt) QueryMap(mp interface{}) (*Rows, error) {
|
|
|
|
return s.QueryMapContext(context.Background(), mp)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *Stmt) QueryStructContext(ctx context.Context, st interface{}) (*Rows, error) {
|
|
|
|
vv := reflect.ValueOf(st)
|
|
|
|
if vv.Kind() != reflect.Ptr || vv.Elem().Kind() != reflect.Struct {
|
|
|
|
return nil, errors.New("mp should be a map's pointer")
|
|
|
|
}
|
|
|
|
|
|
|
|
args := make([]interface{}, len(s.names))
|
|
|
|
for k, i := range s.names {
|
|
|
|
args[i] = vv.Elem().FieldByName(k).Interface()
|
|
|
|
}
|
|
|
|
|
|
|
|
return s.Query(args...)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *Stmt) QueryStruct(st interface{}) (*Rows, error) {
|
|
|
|
return s.QueryStructContext(context.Background(), st)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *Stmt) QueryRowContext(ctx context.Context, args ...interface{}) *Row {
|
|
|
|
rows, err := s.QueryContext(ctx, args...)
|
|
|
|
return &Row{rows, err}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *Stmt) QueryRow(args ...interface{}) *Row {
|
|
|
|
return s.QueryRowContext(context.Background(), args...)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *Stmt) QueryRowMapContext(ctx context.Context, mp interface{}) *Row {
|
|
|
|
vv := reflect.ValueOf(mp)
|
|
|
|
if vv.Kind() != reflect.Ptr || vv.Elem().Kind() != reflect.Map {
|
|
|
|
return &Row{nil, errors.New("mp should be a map's pointer")}
|
|
|
|
}
|
|
|
|
|
|
|
|
args := make([]interface{}, len(s.names))
|
|
|
|
for k, i := range s.names {
|
|
|
|
args[i] = vv.Elem().MapIndex(reflect.ValueOf(k)).Interface()
|
|
|
|
}
|
|
|
|
|
|
|
|
return s.QueryRowContext(ctx, args...)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *Stmt) QueryRowMap(mp interface{}) *Row {
|
|
|
|
return s.QueryRowMapContext(context.Background(), mp)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *Stmt) QueryRowStructContext(ctx context.Context, st interface{}) *Row {
|
|
|
|
vv := reflect.ValueOf(st)
|
|
|
|
if vv.Kind() != reflect.Ptr || vv.Elem().Kind() != reflect.Struct {
|
|
|
|
return &Row{nil, errors.New("st should be a struct's pointer")}
|
|
|
|
}
|
|
|
|
|
|
|
|
args := make([]interface{}, len(s.names))
|
|
|
|
for k, i := range s.names {
|
|
|
|
args[i] = vv.Elem().FieldByName(k).Interface()
|
|
|
|
}
|
|
|
|
|
|
|
|
return s.QueryRowContext(ctx, args...)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *Stmt) QueryRowStruct(st interface{}) *Row {
|
|
|
|
return s.QueryRowStructContext(context.Background(), st)
|
|
|
|
}
|