mirror of https://github.com/lqs/sqlingo
284 lines
7.5 KiB
Go
284 lines
7.5 KiB
Go
package sqlingo
|
|
|
|
import (
|
|
"context"
|
|
"database/sql"
|
|
"fmt"
|
|
"os"
|
|
"path/filepath"
|
|
"runtime"
|
|
"strings"
|
|
"sync"
|
|
"time"
|
|
)
|
|
|
|
const (
|
|
// for colorful terminal print
|
|
green = "\033[32m"
|
|
red = "\033[31m"
|
|
blue = "\033[34m"
|
|
reset = "\033[0m"
|
|
)
|
|
|
|
// Database is the interface of a database with underlying sql.DB object.
|
|
type Database interface {
|
|
// GetDB returns the underlying sql.DB object of the database
|
|
GetDB() *sql.DB
|
|
// BeginTx starts a transaction and executes the function f.
|
|
BeginTx(ctx context.Context, opts *sql.TxOptions, f func(tx Transaction) error) error
|
|
// Query executes a query and returns the cursor
|
|
Query(sql string) (Cursor, error)
|
|
// QueryContext executes a query with context and returns the cursor
|
|
QueryContext(ctx context.Context, sqlString string) (Cursor, error)
|
|
// Execute executes a statement
|
|
Execute(sql string) (sql.Result, error)
|
|
// ExecuteContext executes a statement with context
|
|
ExecuteContext(ctx context.Context, sql string) (sql.Result, error)
|
|
// SetLogger sets the logger function.
|
|
// Deprecated: use SetInterceptor instead
|
|
SetLogger(logger LoggerFunc)
|
|
// SetRetryPolicy sets the retry policy function.
|
|
// Deprecated: use SetInterceptor instead
|
|
SetRetryPolicy(retryPolicy func(err error) bool)
|
|
// EnableCallerInfo enable or disable the caller info in the log.
|
|
// Deprecated: use SetInterceptor instead
|
|
EnableCallerInfo(enableCallerInfo bool)
|
|
// SetInterceptor sets an interceptor function
|
|
SetInterceptor(interceptor InterceptorFunc)
|
|
|
|
// Select initiates a SELECT statement
|
|
Select(fields ...interface{}) selectWithFields
|
|
// SelectDistinct initiates a SELECT DISTINCT statement
|
|
SelectDistinct(fields ...interface{}) selectWithFields
|
|
// SelectFrom initiates a SELECT * FROM statement
|
|
SelectFrom(tables ...Table) selectWithTables
|
|
// InsertInto initiates a INSERT INTO statement
|
|
InsertInto(table Table) insertWithTable
|
|
// ReplaceInto initiates a REPLACE INTO statement
|
|
ReplaceInto(table Table) insertWithTable
|
|
// Update initiates a UPDATE statement
|
|
Update(table Table) updateWithSet
|
|
// DeleteFrom initiates a DELETE FROM statement
|
|
DeleteFrom(table Table) deleteWithTable
|
|
}
|
|
|
|
type txOrDB interface {
|
|
QueryContext(ctx context.Context, query string, args ...interface{}) (*sql.Rows, error)
|
|
ExecContext(ctx context.Context, query string, args ...interface{}) (sql.Result, error)
|
|
}
|
|
|
|
var (
|
|
once sync.Once
|
|
srcPrefix string
|
|
)
|
|
|
|
type database struct {
|
|
db *sql.DB
|
|
tx *sql.Tx
|
|
logger LoggerFunc
|
|
dialect dialect
|
|
retryPolicy func(error) bool
|
|
enableCallerInfo bool
|
|
interceptor InterceptorFunc
|
|
}
|
|
|
|
type LoggerFunc func(sql string, duration time.Duration, isTx bool, retry bool)
|
|
|
|
func (d *database) SetLogger(loggerFunc LoggerFunc) {
|
|
d.logger = loggerFunc
|
|
}
|
|
|
|
// DefaultLogger is sqlingo default logger,
|
|
// which print log to stderr and regard executing time gt 100ms as slow sql.
|
|
func DefaultLogger(sql string, duration time.Duration, isTx bool, retry bool) {
|
|
// for finding code position, try once is enough
|
|
once.Do(func() {
|
|
// $GOPATH/pkg/mod/github.com/lqs/sqlingo@vX.X.X/database.go
|
|
_, file, _, _ := runtime.Caller(0)
|
|
// $GOPATH/pkg/mod/github.com/lqs/sqlingo@vX.X.X
|
|
srcPrefix = filepath.Dir(file)
|
|
})
|
|
|
|
var file string
|
|
var line int
|
|
var ok bool
|
|
for i := 0; i < 16; i++ {
|
|
_, file, line, ok = runtime.Caller(i)
|
|
// `!strings.HasPrefix(file, srcPrefix)` jump out when using sqlingo as dependent package
|
|
// `strings.HasSuffix(file, "_test.go")` jump out when executing unit test cases
|
|
// `!ok` this is so terrible for something unexpected happened
|
|
if !ok || !strings.HasPrefix(file, srcPrefix) || strings.HasSuffix(file, "_test.go") {
|
|
break
|
|
}
|
|
}
|
|
|
|
// todo shouldn't append ';' here
|
|
if !strings.HasSuffix(sql, ";") {
|
|
sql += ";"
|
|
}
|
|
|
|
sb := strings.Builder{}
|
|
sb.Grow(32)
|
|
sb.WriteString("|")
|
|
sb.WriteString(duration.String())
|
|
if isTx {
|
|
sb.WriteString("|transaction") // todo using something traceable
|
|
}
|
|
if retry {
|
|
sb.WriteString("|retry")
|
|
}
|
|
sb.WriteString("|")
|
|
|
|
line1 := strings.Join(
|
|
[]string{
|
|
"[sqlingo]",
|
|
time.Now().Format("2006-01-02 15:04:05"),
|
|
sb.String(),
|
|
file + ":" + fmt.Sprint(line),
|
|
},
|
|
" ")
|
|
|
|
// print to stderr
|
|
fmt.Fprintln(os.Stderr, blue+line1+reset)
|
|
if duration < 100*time.Millisecond {
|
|
fmt.Fprintf(os.Stderr, "%s%s%s\n", green, sql, reset)
|
|
} else {
|
|
fmt.Fprintf(os.Stderr, "%s%s%s\n", red, sql, reset)
|
|
}
|
|
fmt.Fprintln(os.Stderr)
|
|
}
|
|
|
|
func (d *database) SetRetryPolicy(retryPolicy func(err error) bool) {
|
|
d.retryPolicy = retryPolicy
|
|
}
|
|
|
|
func (d *database) EnableCallerInfo(enableCallerInfo bool) {
|
|
d.enableCallerInfo = enableCallerInfo
|
|
}
|
|
|
|
func (d *database) SetInterceptor(interceptor InterceptorFunc) {
|
|
d.interceptor = interceptor
|
|
}
|
|
|
|
// Open a database, similar to sql.Open.
|
|
// `db` using a default logger, which print log to stderr and regard executing time gt 100ms as slow sql.
|
|
// To disable the default logger, use `db.SetLogger(nil)`.
|
|
func Open(driverName string, dataSourceName string) (db Database, err error) {
|
|
var sqlDB *sql.DB
|
|
if dataSourceName != "" {
|
|
sqlDB, err = sql.Open(driverName, dataSourceName)
|
|
if err != nil {
|
|
return
|
|
}
|
|
}
|
|
db = Use(driverName, sqlDB)
|
|
return
|
|
}
|
|
|
|
// Use an existing *sql.DB handle
|
|
func Use(driverName string, sqlDB *sql.DB) Database {
|
|
return &database{
|
|
dialect: getDialectFromDriverName(driverName),
|
|
db: sqlDB,
|
|
}
|
|
}
|
|
|
|
func (d database) GetDB() *sql.DB {
|
|
return d.db
|
|
}
|
|
|
|
func (d database) getTxOrDB() txOrDB {
|
|
if d.tx != nil {
|
|
return d.tx
|
|
}
|
|
return d.db
|
|
}
|
|
|
|
func (d database) Query(sqlString string) (Cursor, error) {
|
|
return d.QueryContext(context.Background(), sqlString)
|
|
}
|
|
|
|
func (d database) QueryContext(ctx context.Context, sqlString string) (Cursor, error) {
|
|
isRetry := false
|
|
for {
|
|
sqlStringWithCallerInfo := getCallerInfo(d, isRetry) + sqlString
|
|
rows, err := d.queryContextOnce(ctx, sqlStringWithCallerInfo, isRetry)
|
|
if err != nil {
|
|
isRetry = d.tx == nil && d.retryPolicy != nil && d.retryPolicy(err)
|
|
if isRetry {
|
|
continue
|
|
}
|
|
return nil, err
|
|
}
|
|
return cursor{rows: rows}, nil
|
|
}
|
|
}
|
|
|
|
func (d database) queryContextOnce(ctx context.Context, sqlString string, retry bool) (*sql.Rows, error) {
|
|
if ctx == nil {
|
|
ctx = context.Background()
|
|
}
|
|
startTime := time.Now()
|
|
defer func() {
|
|
endTime := time.Now()
|
|
if d.logger != nil {
|
|
d.logger(sqlString, endTime.Sub(startTime), false, retry)
|
|
}
|
|
}()
|
|
|
|
interceptor := d.interceptor
|
|
var rows *sql.Rows
|
|
invoker := func(ctx context.Context, sql string) (err error) {
|
|
rows, err = d.getTxOrDB().QueryContext(ctx, sql)
|
|
return
|
|
}
|
|
|
|
var err error
|
|
if interceptor == nil {
|
|
err = invoker(ctx, sqlString)
|
|
} else {
|
|
err = interceptor(ctx, sqlString, invoker)
|
|
}
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
return rows, nil
|
|
}
|
|
|
|
func (d database) Execute(sqlString string) (sql.Result, error) {
|
|
return d.ExecuteContext(context.Background(), sqlString)
|
|
}
|
|
|
|
// ExecuteContext todo Is there need retry?
|
|
func (d database) ExecuteContext(ctx context.Context, sqlString string) (sql.Result, error) {
|
|
if ctx == nil {
|
|
ctx = context.Background()
|
|
}
|
|
sqlStringWithCallerInfo := getCallerInfo(d, false) + sqlString
|
|
startTime := time.Now()
|
|
defer func() {
|
|
endTime := time.Now()
|
|
if d.logger != nil {
|
|
d.logger(sqlStringWithCallerInfo, endTime.Sub(startTime), false, false)
|
|
}
|
|
}()
|
|
|
|
var result sql.Result
|
|
invoker := func(ctx context.Context, sql string) (err error) {
|
|
result, err = d.getTxOrDB().ExecContext(ctx, sql)
|
|
return
|
|
}
|
|
var err error
|
|
if d.interceptor == nil {
|
|
err = invoker(ctx, sqlStringWithCallerInfo)
|
|
} else {
|
|
err = d.interceptor(ctx, sqlStringWithCallerInfo, invoker)
|
|
}
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
return result, err
|
|
}
|