sqlingo/database.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
}