mirror of
https://github.com/astaxie/beego.git
synced 2024-11-22 08:40:54 +00:00
Merge pull request #4098 from jianzhiyao/frt/stmt_config
make stmt cache size configurable & wrap kv
This commit is contained in:
commit
7312197732
@ -14,14 +14,29 @@
|
|||||||
|
|
||||||
package common
|
package common
|
||||||
|
|
||||||
// KV is common structure to store key-value data.
|
type KV interface {
|
||||||
|
GetKey() interface{}
|
||||||
|
GetValue() interface{}
|
||||||
|
}
|
||||||
|
|
||||||
|
// SimpleKV is common structure to store key-value data.
|
||||||
// when you need something like Pair, you can use this
|
// when you need something like Pair, you can use this
|
||||||
type KV struct {
|
type SimpleKV struct {
|
||||||
Key interface{}
|
Key interface{}
|
||||||
Value interface{}
|
Value interface{}
|
||||||
}
|
}
|
||||||
|
|
||||||
// KVs will store KV collection as map
|
var _ KV = new(SimpleKV)
|
||||||
|
|
||||||
|
func (s *SimpleKV) GetKey() interface{} {
|
||||||
|
return s.Key
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *SimpleKV) GetValue() interface{} {
|
||||||
|
return s.Value
|
||||||
|
}
|
||||||
|
|
||||||
|
// KVs will store SimpleKV collection as map
|
||||||
type KVs struct {
|
type KVs struct {
|
||||||
kvs map[interface{}]interface{}
|
kvs map[interface{}]interface{}
|
||||||
}
|
}
|
||||||
@ -63,7 +78,7 @@ func NewKVs(kvs ...KV) *KVs {
|
|||||||
kvs: make(map[interface{}]interface{}, len(kvs)),
|
kvs: make(map[interface{}]interface{}, len(kvs)),
|
||||||
}
|
}
|
||||||
for _, kv := range kvs {
|
for _, kv := range kvs {
|
||||||
res.kvs[kv.Key] = kv.Value
|
res.kvs[kv.GetKey()] = kv.GetValue()
|
||||||
}
|
}
|
||||||
return res
|
return res
|
||||||
}
|
}
|
||||||
|
@ -22,7 +22,7 @@ import (
|
|||||||
|
|
||||||
func TestKVs(t *testing.T) {
|
func TestKVs(t *testing.T) {
|
||||||
key := "my-key"
|
key := "my-key"
|
||||||
kvs := NewKVs(KV{
|
kvs := NewKVs(&SimpleKV{
|
||||||
Key: key,
|
Key: key,
|
||||||
Value: 12,
|
Value: 12,
|
||||||
})
|
})
|
||||||
|
@ -1,21 +0,0 @@
|
|||||||
// Copyright 2020 beego-dev
|
|
||||||
//
|
|
||||||
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
||||||
// you may not use this file except in compliance with the License.
|
|
||||||
// You may obtain a copy of the License at
|
|
||||||
//
|
|
||||||
// http://www.apache.org/licenses/LICENSE-2.0
|
|
||||||
//
|
|
||||||
// Unless required by applicable law or agreed to in writing, software
|
|
||||||
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
||||||
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
||||||
// See the License for the specific language governing permissions and
|
|
||||||
// limitations under the License.
|
|
||||||
|
|
||||||
package orm
|
|
||||||
|
|
||||||
const (
|
|
||||||
MaxIdleConnsKey = "MaxIdleConns"
|
|
||||||
MaxOpenConnsKey = "MaxOpenConns"
|
|
||||||
ConnMaxLifetimeKey = "ConnMaxLifetime"
|
|
||||||
)
|
|
@ -109,8 +109,9 @@ func (ac *_dbCache) getDefault() (al *alias) {
|
|||||||
|
|
||||||
type DB struct {
|
type DB struct {
|
||||||
*sync.RWMutex
|
*sync.RWMutex
|
||||||
DB *sql.DB
|
DB *sql.DB
|
||||||
stmtDecorators *lru.Cache
|
stmtDecorators *lru.Cache
|
||||||
|
stmtDecoratorsLimit int
|
||||||
}
|
}
|
||||||
|
|
||||||
var _ dbQuerier = new(DB)
|
var _ dbQuerier = new(DB)
|
||||||
@ -165,16 +166,14 @@ func (d *DB) PrepareContext(ctx context.Context, query string) (*sql.Stmt, error
|
|||||||
}
|
}
|
||||||
|
|
||||||
func (d *DB) Exec(query string, args ...interface{}) (sql.Result, error) {
|
func (d *DB) Exec(query string, args ...interface{}) (sql.Result, error) {
|
||||||
sd, err := d.getStmtDecorator(query)
|
return d.ExecContext(context.Background(), query, args...)
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
stmt := sd.getStmt()
|
|
||||||
defer sd.release()
|
|
||||||
return stmt.Exec(args...)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func (d *DB) ExecContext(ctx context.Context, query string, args ...interface{}) (sql.Result, error) {
|
func (d *DB) ExecContext(ctx context.Context, query string, args ...interface{}) (sql.Result, error) {
|
||||||
|
if d.stmtDecorators == nil {
|
||||||
|
return d.DB.ExecContext(ctx, query, args...)
|
||||||
|
}
|
||||||
|
|
||||||
sd, err := d.getStmtDecorator(query)
|
sd, err := d.getStmtDecorator(query)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
@ -185,16 +184,14 @@ func (d *DB) ExecContext(ctx context.Context, query string, args ...interface{})
|
|||||||
}
|
}
|
||||||
|
|
||||||
func (d *DB) Query(query string, args ...interface{}) (*sql.Rows, error) {
|
func (d *DB) Query(query string, args ...interface{}) (*sql.Rows, error) {
|
||||||
sd, err := d.getStmtDecorator(query)
|
return d.QueryContext(context.Background(), query, args...)
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
stmt := sd.getStmt()
|
|
||||||
defer sd.release()
|
|
||||||
return stmt.Query(args...)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func (d *DB) QueryContext(ctx context.Context, query string, args ...interface{}) (*sql.Rows, error) {
|
func (d *DB) QueryContext(ctx context.Context, query string, args ...interface{}) (*sql.Rows, error) {
|
||||||
|
if d.stmtDecorators == nil {
|
||||||
|
return d.DB.QueryContext(ctx, query, args...)
|
||||||
|
}
|
||||||
|
|
||||||
sd, err := d.getStmtDecorator(query)
|
sd, err := d.getStmtDecorator(query)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
@ -205,24 +202,21 @@ func (d *DB) QueryContext(ctx context.Context, query string, args ...interface{}
|
|||||||
}
|
}
|
||||||
|
|
||||||
func (d *DB) QueryRow(query string, args ...interface{}) *sql.Row {
|
func (d *DB) QueryRow(query string, args ...interface{}) *sql.Row {
|
||||||
sd, err := d.getStmtDecorator(query)
|
return d.QueryRowContext(context.Background(), query, args...)
|
||||||
if err != nil {
|
|
||||||
panic(err)
|
|
||||||
}
|
|
||||||
stmt := sd.getStmt()
|
|
||||||
defer sd.release()
|
|
||||||
return stmt.QueryRow(args...)
|
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func (d *DB) QueryRowContext(ctx context.Context, query string, args ...interface{}) *sql.Row {
|
func (d *DB) QueryRowContext(ctx context.Context, query string, args ...interface{}) *sql.Row {
|
||||||
|
if d.stmtDecorators == nil {
|
||||||
|
return d.DB.QueryRowContext(ctx, query, args...)
|
||||||
|
}
|
||||||
|
|
||||||
sd, err := d.getStmtDecorator(query)
|
sd, err := d.getStmtDecorator(query)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
panic(err)
|
panic(err)
|
||||||
}
|
}
|
||||||
stmt := sd.getStmt()
|
stmt := sd.getStmt()
|
||||||
defer sd.release()
|
defer sd.release()
|
||||||
return stmt.QueryRowContext(ctx, args)
|
return stmt.QueryRowContext(ctx, args...)
|
||||||
}
|
}
|
||||||
|
|
||||||
type TxDB struct {
|
type TxDB struct {
|
||||||
@ -345,14 +339,49 @@ func detectTZ(al *alias) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func addAliasWthDB(aliasName, driverName string, db *sql.DB) (*alias, error) {
|
func addAliasWthDB(aliasName, driverName string, db *sql.DB, params ...common.KV) (*alias, error) {
|
||||||
|
existErr := fmt.Errorf("DataBase alias name `%s` already registered, cannot reuse", aliasName)
|
||||||
|
if _, ok := dataBaseCache.get(aliasName); ok {
|
||||||
|
return nil, existErr
|
||||||
|
}
|
||||||
|
|
||||||
|
al, err := newAliasWithDb(aliasName, driverName, db, params...)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
if !dataBaseCache.add(aliasName, al) {
|
||||||
|
return nil, existErr
|
||||||
|
}
|
||||||
|
|
||||||
|
return al, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func newAliasWithDb(aliasName, driverName string, db *sql.DB, params ...common.KV)(*alias, error){
|
||||||
|
kvs := common.NewKVs(params...)
|
||||||
|
|
||||||
|
var stmtCache *lru.Cache
|
||||||
|
var stmtCacheSize int
|
||||||
|
|
||||||
|
maxStmtCacheSize := kvs.GetValueOr(maxStmtCacheSizeKey, 0).(int)
|
||||||
|
if maxStmtCacheSize > 0 {
|
||||||
|
_stmtCache, errC := newStmtDecoratorLruWithEvict(maxStmtCacheSize)
|
||||||
|
if errC != nil {
|
||||||
|
return nil, errC
|
||||||
|
} else {
|
||||||
|
stmtCache = _stmtCache
|
||||||
|
stmtCacheSize = maxStmtCacheSize
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
al := new(alias)
|
al := new(alias)
|
||||||
al.Name = aliasName
|
al.Name = aliasName
|
||||||
al.DriverName = driverName
|
al.DriverName = driverName
|
||||||
al.DB = &DB{
|
al.DB = &DB{
|
||||||
RWMutex: new(sync.RWMutex),
|
RWMutex: new(sync.RWMutex),
|
||||||
DB: db,
|
DB: db,
|
||||||
stmtDecorators: newStmtDecoratorLruWithEvict(),
|
stmtDecorators: stmtCache,
|
||||||
|
stmtDecoratorsLimit: stmtCacheSize,
|
||||||
}
|
}
|
||||||
|
|
||||||
if dr, ok := drivers[driverName]; ok {
|
if dr, ok := drivers[driverName]; ok {
|
||||||
@ -367,28 +396,39 @@ func addAliasWthDB(aliasName, driverName string, db *sql.DB) (*alias, error) {
|
|||||||
return nil, fmt.Errorf("register db Ping `%s`, %s", aliasName, err.Error())
|
return nil, fmt.Errorf("register db Ping `%s`, %s", aliasName, err.Error())
|
||||||
}
|
}
|
||||||
|
|
||||||
if !dataBaseCache.add(aliasName, al) {
|
detectTZ(al)
|
||||||
return nil, fmt.Errorf("DataBase alias name `%s` already registered, cannot reuse", aliasName)
|
|
||||||
}
|
kvs.IfContains(maxIdleConnectionsKey, func(value interface{}) {
|
||||||
|
if m, ok := value.(int); ok {
|
||||||
|
SetMaxIdleConns(al, m)
|
||||||
|
}
|
||||||
|
}).IfContains(maxOpenConnectionsKey, func(value interface{}) {
|
||||||
|
if m, ok := value.(int); ok {
|
||||||
|
SetMaxOpenConns(al, m)
|
||||||
|
}
|
||||||
|
}).IfContains(connMaxLifetimeKey, func(value interface{}) {
|
||||||
|
if m, ok := value.(time.Duration); ok {
|
||||||
|
SetConnMaxLifetime(al, m)
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
return al, nil
|
return al, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// AddAliasWthDB add a aliasName for the drivename
|
// AddAliasWthDB add a aliasName for the drivename
|
||||||
func AddAliasWthDB(aliasName, driverName string, db *sql.DB) error {
|
func AddAliasWthDB(aliasName, driverName string, db *sql.DB, params ...common.KV) error {
|
||||||
_, err := addAliasWthDB(aliasName, driverName, db)
|
_, err := addAliasWthDB(aliasName, driverName, db, params...)
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
// RegisterDataBase Setting the database connect params. Use the database driver self dataSource args.
|
// RegisterDataBase Setting the database connect params. Use the database driver self dataSource args.
|
||||||
func RegisterDataBase(aliasName, driverName, dataSource string, params ...common.KV) error {
|
func RegisterDataBase(aliasName, driverName, dataSource string, hints ...common.KV) error {
|
||||||
var (
|
var (
|
||||||
err error
|
err error
|
||||||
db *sql.DB
|
db *sql.DB
|
||||||
al *alias
|
al *alias
|
||||||
)
|
)
|
||||||
|
|
||||||
kvs := common.NewKVs(params...)
|
|
||||||
|
|
||||||
db, err = sql.Open(driverName, dataSource)
|
db, err = sql.Open(driverName, dataSource)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
@ -396,23 +436,13 @@ func RegisterDataBase(aliasName, driverName, dataSource string, params ...common
|
|||||||
goto end
|
goto end
|
||||||
}
|
}
|
||||||
|
|
||||||
al, err = addAliasWthDB(aliasName, driverName, db)
|
al, err = addAliasWthDB(aliasName, driverName, db, hints...)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
goto end
|
goto end
|
||||||
}
|
}
|
||||||
|
|
||||||
al.DataSource = dataSource
|
al.DataSource = dataSource
|
||||||
|
|
||||||
detectTZ(al)
|
|
||||||
|
|
||||||
kvs.IfContains(MaxIdleConnsKey, func(value interface{}) {
|
|
||||||
SetMaxIdleConns(al.Name, value.(int))
|
|
||||||
}).IfContains(MaxOpenConnsKey, func(value interface{}) {
|
|
||||||
SetMaxOpenConns(al.Name, value.(int))
|
|
||||||
}).IfContains(ConnMaxLifetimeKey, func(value interface{}) {
|
|
||||||
SetConnMaxLifetime(al.Name, value.(time.Duration))
|
|
||||||
})
|
|
||||||
|
|
||||||
end:
|
end:
|
||||||
if err != nil {
|
if err != nil {
|
||||||
if db != nil {
|
if db != nil {
|
||||||
@ -447,21 +477,18 @@ func SetDataBaseTZ(aliasName string, tz *time.Location) error {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// SetMaxIdleConns Change the max idle conns for *sql.DB, use specify database alias name
|
// SetMaxIdleConns Change the max idle conns for *sql.DB, use specify database alias name
|
||||||
func SetMaxIdleConns(aliasName string, maxIdleConns int) {
|
func SetMaxIdleConns(al *alias, maxIdleConns int) {
|
||||||
al := getDbAlias(aliasName)
|
|
||||||
al.MaxIdleConns = maxIdleConns
|
al.MaxIdleConns = maxIdleConns
|
||||||
al.DB.DB.SetMaxIdleConns(maxIdleConns)
|
al.DB.DB.SetMaxIdleConns(maxIdleConns)
|
||||||
}
|
}
|
||||||
|
|
||||||
// SetMaxOpenConns Change the max open conns for *sql.DB, use specify database alias name
|
// SetMaxOpenConns Change the max open conns for *sql.DB, use specify database alias name
|
||||||
func SetMaxOpenConns(aliasName string, maxOpenConns int) {
|
func SetMaxOpenConns(al *alias, maxOpenConns int) {
|
||||||
al := getDbAlias(aliasName)
|
|
||||||
al.MaxOpenConns = maxOpenConns
|
al.MaxOpenConns = maxOpenConns
|
||||||
al.DB.DB.SetMaxOpenConns(maxOpenConns)
|
al.DB.DB.SetMaxOpenConns(maxOpenConns)
|
||||||
}
|
}
|
||||||
|
|
||||||
func SetConnMaxLifetime(aliasName string, lifeTime time.Duration) {
|
func SetConnMaxLifetime(al *alias, lifeTime time.Duration) {
|
||||||
al := getDbAlias(aliasName)
|
|
||||||
al.ConnMaxLifetime = lifeTime
|
al.ConnMaxLifetime = lifeTime
|
||||||
al.DB.DB.SetConnMaxLifetime(lifeTime)
|
al.DB.DB.SetConnMaxLifetime(lifeTime)
|
||||||
}
|
}
|
||||||
@ -517,9 +544,12 @@ func newStmtDecorator(sqlStmt *sql.Stmt) *stmtDecorator {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func newStmtDecoratorLruWithEvict() *lru.Cache {
|
func newStmtDecoratorLruWithEvict(cacheSize int) (*lru.Cache, error) {
|
||||||
cache, _ := lru.NewWithEvict(1000, func(key interface{}, value interface{}) {
|
cache, err := lru.NewWithEvict(cacheSize, func(key interface{}, value interface{}) {
|
||||||
value.(*stmtDecorator).destroy()
|
value.(*stmtDecorator).destroy()
|
||||||
})
|
})
|
||||||
return cache
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
return cache, nil
|
||||||
}
|
}
|
||||||
|
@ -19,21 +19,13 @@ import (
|
|||||||
"time"
|
"time"
|
||||||
|
|
||||||
"github.com/stretchr/testify/assert"
|
"github.com/stretchr/testify/assert"
|
||||||
|
|
||||||
"github.com/astaxie/beego/pkg/common"
|
|
||||||
)
|
)
|
||||||
|
|
||||||
func TestRegisterDataBase(t *testing.T) {
|
func TestRegisterDataBase(t *testing.T) {
|
||||||
err := RegisterDataBase("test-params", DBARGS.Driver, DBARGS.Source, common.KV{
|
err := RegisterDataBase("test-params", DBARGS.Driver, DBARGS.Source,
|
||||||
Key: MaxIdleConnsKey,
|
MaxIdleConnections(20),
|
||||||
Value: 20,
|
MaxOpenConnections(300),
|
||||||
}, common.KV{
|
ConnMaxLifetime(time.Minute))
|
||||||
Key: MaxOpenConnsKey,
|
|
||||||
Value: 300,
|
|
||||||
}, common.KV{
|
|
||||||
Key: ConnMaxLifetimeKey,
|
|
||||||
Value: time.Minute,
|
|
||||||
})
|
|
||||||
assert.Nil(t, err)
|
assert.Nil(t, err)
|
||||||
|
|
||||||
al := getDbAlias("test-params")
|
al := getDbAlias("test-params")
|
||||||
@ -42,3 +34,44 @@ func TestRegisterDataBase(t *testing.T) {
|
|||||||
assert.Equal(t, al.MaxOpenConns, 300)
|
assert.Equal(t, al.MaxOpenConns, 300)
|
||||||
assert.Equal(t, al.ConnMaxLifetime, time.Minute)
|
assert.Equal(t, al.ConnMaxLifetime, time.Minute)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func TestRegisterDataBase_MaxStmtCacheSizeNegative1(t *testing.T) {
|
||||||
|
aliasName := "TestRegisterDataBase_MaxStmtCacheSizeNegative1"
|
||||||
|
err := RegisterDataBase(aliasName, DBARGS.Driver, DBARGS.Source, MaxStmtCacheSize(-1))
|
||||||
|
assert.Nil(t, err)
|
||||||
|
|
||||||
|
al := getDbAlias(aliasName)
|
||||||
|
assert.NotNil(t, al)
|
||||||
|
assert.Equal(t, al.DB.stmtDecoratorsLimit, 0)
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestRegisterDataBase_MaxStmtCacheSize0(t *testing.T) {
|
||||||
|
aliasName := "TestRegisterDataBase_MaxStmtCacheSize0"
|
||||||
|
err := RegisterDataBase(aliasName, DBARGS.Driver, DBARGS.Source, MaxStmtCacheSize(0))
|
||||||
|
assert.Nil(t, err)
|
||||||
|
|
||||||
|
al := getDbAlias(aliasName)
|
||||||
|
assert.NotNil(t, al)
|
||||||
|
assert.Equal(t, al.DB.stmtDecoratorsLimit, 0)
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestRegisterDataBase_MaxStmtCacheSize1(t *testing.T) {
|
||||||
|
aliasName := "TestRegisterDataBase_MaxStmtCacheSize1"
|
||||||
|
err := RegisterDataBase(aliasName, DBARGS.Driver, DBARGS.Source, MaxStmtCacheSize(1))
|
||||||
|
assert.Nil(t, err)
|
||||||
|
|
||||||
|
al := getDbAlias(aliasName)
|
||||||
|
assert.NotNil(t, al)
|
||||||
|
assert.Equal(t, al.DB.stmtDecoratorsLimit, 1)
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestRegisterDataBase_MaxStmtCacheSize841(t *testing.T) {
|
||||||
|
aliasName := "TestRegisterDataBase_MaxStmtCacheSize841"
|
||||||
|
err := RegisterDataBase(aliasName, DBARGS.Driver, DBARGS.Source, MaxStmtCacheSize(841))
|
||||||
|
assert.Nil(t, err)
|
||||||
|
|
||||||
|
al := getDbAlias(aliasName)
|
||||||
|
assert.NotNil(t, al)
|
||||||
|
assert.Equal(t, al.DB.stmtDecoratorsLimit, 841)
|
||||||
|
}
|
||||||
|
|
||||||
|
74
pkg/orm/db_hints.go
Normal file
74
pkg/orm/db_hints.go
Normal file
@ -0,0 +1,74 @@
|
|||||||
|
// Copyright 2020 beego-dev
|
||||||
|
//
|
||||||
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
// you may not use this file except in compliance with the License.
|
||||||
|
// You may obtain a copy of the License at
|
||||||
|
//
|
||||||
|
// http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
//
|
||||||
|
// Unless required by applicable law or agreed to in writing, software
|
||||||
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
// See the License for the specific language governing permissions and
|
||||||
|
// limitations under the License.
|
||||||
|
|
||||||
|
package orm
|
||||||
|
|
||||||
|
import (
|
||||||
|
"github.com/astaxie/beego/pkg/common"
|
||||||
|
"time"
|
||||||
|
)
|
||||||
|
|
||||||
|
type Hint struct {
|
||||||
|
key interface{}
|
||||||
|
value interface{}
|
||||||
|
}
|
||||||
|
|
||||||
|
var _ common.KV = new(Hint)
|
||||||
|
|
||||||
|
// GetKey return key
|
||||||
|
func (s *Hint) GetKey() interface{} {
|
||||||
|
return s.key
|
||||||
|
}
|
||||||
|
|
||||||
|
// GetValue return value
|
||||||
|
func (s *Hint) GetValue() interface{} {
|
||||||
|
return s.value
|
||||||
|
}
|
||||||
|
|
||||||
|
const (
|
||||||
|
maxIdleConnectionsKey = "MaxIdleConnections"
|
||||||
|
maxOpenConnectionsKey = "MaxOpenConnections"
|
||||||
|
connMaxLifetimeKey = "ConnMaxLifetime"
|
||||||
|
maxStmtCacheSizeKey = "MaxStmtCacheSize"
|
||||||
|
)
|
||||||
|
|
||||||
|
var _ common.KV = new(Hint)
|
||||||
|
|
||||||
|
// MaxIdleConnections return a hint about MaxIdleConnections
|
||||||
|
func MaxIdleConnections(v int) *Hint {
|
||||||
|
return NewHint(maxIdleConnectionsKey, v)
|
||||||
|
}
|
||||||
|
|
||||||
|
// MaxOpenConnections return a hint about MaxOpenConnections
|
||||||
|
func MaxOpenConnections(v int) *Hint {
|
||||||
|
return NewHint(maxOpenConnectionsKey, v)
|
||||||
|
}
|
||||||
|
|
||||||
|
// ConnMaxLifetime return a hint about ConnMaxLifetime
|
||||||
|
func ConnMaxLifetime(v time.Duration) *Hint {
|
||||||
|
return NewHint(connMaxLifetimeKey, v)
|
||||||
|
}
|
||||||
|
|
||||||
|
// MaxStmtCacheSize return a hint about MaxStmtCacheSize
|
||||||
|
func MaxStmtCacheSize(v int) *Hint {
|
||||||
|
return NewHint(maxStmtCacheSizeKey, v)
|
||||||
|
}
|
||||||
|
|
||||||
|
// NewHint return a hint
|
||||||
|
func NewHint(key interface{}, value interface{}) *Hint {
|
||||||
|
return &Hint{
|
||||||
|
key: key,
|
||||||
|
value: value,
|
||||||
|
}
|
||||||
|
}
|
76
pkg/orm/db_hints_test.go
Normal file
76
pkg/orm/db_hints_test.go
Normal file
@ -0,0 +1,76 @@
|
|||||||
|
// Copyright 2020 beego-dev
|
||||||
|
//
|
||||||
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
// you may not use this file except in compliance with the License.
|
||||||
|
// You may obtain a copy of the License at
|
||||||
|
//
|
||||||
|
// http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
//
|
||||||
|
// Unless required by applicable law or agreed to in writing, software
|
||||||
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
// See the License for the specific language governing permissions and
|
||||||
|
// limitations under the License.
|
||||||
|
|
||||||
|
package orm
|
||||||
|
|
||||||
|
import (
|
||||||
|
"github.com/stretchr/testify/assert"
|
||||||
|
"testing"
|
||||||
|
"time"
|
||||||
|
)
|
||||||
|
|
||||||
|
func TestNewHint_time(t *testing.T) {
|
||||||
|
key := "qweqwe"
|
||||||
|
value := time.Second
|
||||||
|
hint := NewHint(key, value)
|
||||||
|
|
||||||
|
assert.Equal(t, hint.GetKey(), key)
|
||||||
|
assert.Equal(t, hint.GetValue(), value)
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestNewHint_int(t *testing.T) {
|
||||||
|
key := "qweqwe"
|
||||||
|
value := 281230
|
||||||
|
hint := NewHint(key, value)
|
||||||
|
|
||||||
|
assert.Equal(t, hint.GetKey(), key)
|
||||||
|
assert.Equal(t, hint.GetValue(), value)
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestNewHint_float(t *testing.T) {
|
||||||
|
key := "qweqwe"
|
||||||
|
value := 21.2459753
|
||||||
|
hint := NewHint(key, value)
|
||||||
|
|
||||||
|
assert.Equal(t, hint.GetKey(), key)
|
||||||
|
assert.Equal(t, hint.GetValue(), value)
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestMaxOpenConnections(t *testing.T) {
|
||||||
|
i := 887423
|
||||||
|
hint := MaxOpenConnections(i)
|
||||||
|
assert.Equal(t, hint.GetValue(), i)
|
||||||
|
assert.Equal(t, hint.GetKey(), maxOpenConnectionsKey)
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestConnMaxLifetime(t *testing.T) {
|
||||||
|
i := time.Hour
|
||||||
|
hint := ConnMaxLifetime(i)
|
||||||
|
assert.Equal(t, hint.GetValue(), i)
|
||||||
|
assert.Equal(t, hint.GetKey(), connMaxLifetimeKey)
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestMaxIdleConnections(t *testing.T) {
|
||||||
|
i := 42316
|
||||||
|
hint := MaxIdleConnections(i)
|
||||||
|
assert.Equal(t, hint.GetValue(), i)
|
||||||
|
assert.Equal(t, hint.GetKey(), maxIdleConnectionsKey)
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestMaxStmtCacheSize(t *testing.T) {
|
||||||
|
i := 94157
|
||||||
|
hint := MaxStmtCacheSize(i)
|
||||||
|
assert.Equal(t, hint.GetValue(), i)
|
||||||
|
assert.Equal(t, hint.GetKey(), maxStmtCacheSizeKey)
|
||||||
|
}
|
@ -28,7 +28,6 @@ import (
|
|||||||
// As tidb can't use go get, so disable the tidb testing now
|
// As tidb can't use go get, so disable the tidb testing now
|
||||||
// _ "github.com/pingcap/tidb"
|
// _ "github.com/pingcap/tidb"
|
||||||
|
|
||||||
"github.com/astaxie/beego/pkg/common"
|
|
||||||
)
|
)
|
||||||
|
|
||||||
// A slice string field.
|
// A slice string field.
|
||||||
@ -489,10 +488,7 @@ func init() {
|
|||||||
os.Exit(2)
|
os.Exit(2)
|
||||||
}
|
}
|
||||||
|
|
||||||
err := RegisterDataBase("default", DBARGS.Driver, DBARGS.Source, common.KV{
|
err := RegisterDataBase("default", DBARGS.Driver, DBARGS.Source, MaxIdleConnections(20))
|
||||||
Key: MaxIdleConnsKey,
|
|
||||||
Value: 20,
|
|
||||||
})
|
|
||||||
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
panic(fmt.Sprintf("can not register database: %v", err))
|
panic(fmt.Sprintf("can not register database: %v", err))
|
||||||
|
@ -58,9 +58,9 @@ import (
|
|||||||
"database/sql"
|
"database/sql"
|
||||||
"errors"
|
"errors"
|
||||||
"fmt"
|
"fmt"
|
||||||
|
"github.com/astaxie/beego/pkg/common"
|
||||||
"os"
|
"os"
|
||||||
"reflect"
|
"reflect"
|
||||||
"sync"
|
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
"github.com/astaxie/beego/logs"
|
"github.com/astaxie/beego/logs"
|
||||||
@ -609,27 +609,12 @@ func NewOrm() Ormer {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// NewOrmWithDB create a new ormer object with specify *sql.DB for query
|
// NewOrmWithDB create a new ormer object with specify *sql.DB for query
|
||||||
func NewOrmWithDB(driverName, aliasName string, db *sql.DB) (Ormer, error) {
|
func NewOrmWithDB(driverName, aliasName string, db *sql.DB, params ...common.KV) (Ormer, error) {
|
||||||
var al *alias
|
al, err := newAliasWithDb(aliasName, driverName, db, params...)
|
||||||
|
if err != nil {
|
||||||
if dr, ok := drivers[driverName]; ok {
|
return nil, err
|
||||||
al = new(alias)
|
|
||||||
al.DbBaser = dbBasers[dr]
|
|
||||||
al.Driver = dr
|
|
||||||
} else {
|
|
||||||
return nil, fmt.Errorf("driver name `%s` have not registered", driverName)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
al.Name = aliasName
|
|
||||||
al.DriverName = driverName
|
|
||||||
al.DB = &DB{
|
|
||||||
RWMutex: new(sync.RWMutex),
|
|
||||||
DB: db,
|
|
||||||
stmtDecorators: newStmtDecoratorLruWithEvict(),
|
|
||||||
}
|
|
||||||
|
|
||||||
detectTZ(al)
|
|
||||||
|
|
||||||
o := new(orm)
|
o := new(orm)
|
||||||
o.alias = al
|
o.alias = al
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user