Cache DB connection pools on the migrationContext where applicable
This commit is contained in:
parent
5788ab5347
commit
84bdfdb1ad
@ -14,9 +14,12 @@ import (
|
|||||||
"sync/atomic"
|
"sync/atomic"
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
|
gosql "database/sql"
|
||||||
"github.com/github/gh-ost/go/mysql"
|
"github.com/github/gh-ost/go/mysql"
|
||||||
"github.com/github/gh-ost/go/sql"
|
"github.com/github/gh-ost/go/sql"
|
||||||
|
|
||||||
|
"github.com/outbrain/golib/sqlutils"
|
||||||
|
|
||||||
"gopkg.in/gcfg.v1"
|
"gopkg.in/gcfg.v1"
|
||||||
gcfgscanner "gopkg.in/gcfg.v1/scanner"
|
gcfgscanner "gopkg.in/gcfg.v1/scanner"
|
||||||
)
|
)
|
||||||
@ -197,6 +200,9 @@ type MigrationContext struct {
|
|||||||
recentBinlogCoordinates mysql.BinlogCoordinates
|
recentBinlogCoordinates mysql.BinlogCoordinates
|
||||||
|
|
||||||
CanStopStreaming func() bool
|
CanStopStreaming func() bool
|
||||||
|
|
||||||
|
knownDBs map[string]*gosql.DB
|
||||||
|
knownDBsMutex *sync.Mutex
|
||||||
}
|
}
|
||||||
|
|
||||||
type ContextConfig struct {
|
type ContextConfig struct {
|
||||||
@ -230,6 +236,8 @@ func NewMigrationContext() *MigrationContext {
|
|||||||
pointOfInterestTimeMutex: &sync.Mutex{},
|
pointOfInterestTimeMutex: &sync.Mutex{},
|
||||||
ColumnRenameMap: make(map[string]string),
|
ColumnRenameMap: make(map[string]string),
|
||||||
PanicAbort: make(chan error),
|
PanicAbort: make(chan error),
|
||||||
|
knownDBsMutex: &sync.Mutex{},
|
||||||
|
knownDBs: make(map[string]*gosql.DB),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -242,6 +250,23 @@ func getSafeTableName(baseName string, suffix string) string {
|
|||||||
return fmt.Sprintf("_%s_%s", baseName[0:len(baseName)-extraCharacters], suffix)
|
return fmt.Sprintf("_%s_%s", baseName[0:len(baseName)-extraCharacters], suffix)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// GetDB returns a DB instance based on uri.
|
||||||
|
// bool result indicates whether the DB was returned from cache; err
|
||||||
|
func (this *MigrationContext) GetDB(mysql_uri string) (*gosql.DB, bool, error) {
|
||||||
|
this.knownDBsMutex.Lock()
|
||||||
|
defer this.knownDBsMutex.Unlock()
|
||||||
|
|
||||||
|
var exists bool
|
||||||
|
if _, exists = this.knownDBs[mysql_uri]; !exists {
|
||||||
|
if db, err := sqlutils.GetDB(mysql_uri); err == nil {
|
||||||
|
this.knownDBs[mysql_uri] = db
|
||||||
|
} else {
|
||||||
|
return db, exists, err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return this.knownDBs[mysql_uri], exists, nil
|
||||||
|
}
|
||||||
|
|
||||||
// GetGhostTableName generates the name of ghost table, based on original table name
|
// GetGhostTableName generates the name of ghost table, based on original table name
|
||||||
// or a given table name
|
// or a given table name
|
||||||
func (this *MigrationContext) GetGhostTableName() string {
|
func (this *MigrationContext) GetGhostTableName() string {
|
||||||
|
@ -47,11 +47,11 @@ func NewApplier(migrationContext *base.MigrationContext) *Applier {
|
|||||||
|
|
||||||
func (this *Applier) InitDBConnections() (err error) {
|
func (this *Applier) InitDBConnections() (err error) {
|
||||||
applierUri := this.connectionConfig.GetDBUri(this.migrationContext.DatabaseName)
|
applierUri := this.connectionConfig.GetDBUri(this.migrationContext.DatabaseName)
|
||||||
if this.db, _, err = sqlutils.GetDB(applierUri); err != nil {
|
if this.db, _, err = this.migrationContext.GetDB(applierUri); err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
singletonApplierUri := fmt.Sprintf("%s?timeout=0", applierUri)
|
singletonApplierUri := fmt.Sprintf("%s?timeout=0", applierUri)
|
||||||
if this.singletonDB, _, err = sqlutils.GetDB(singletonApplierUri); err != nil {
|
if this.singletonDB, _, err = this.migrationContext.GetDB(singletonApplierUri); err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
this.singletonDB.SetMaxOpenConns(1)
|
this.singletonDB.SetMaxOpenConns(1)
|
||||||
|
@ -26,9 +26,10 @@ const startSlavePostWaitMilliseconds = 500 * time.Millisecond
|
|||||||
// Inspector reads data from the read-MySQL-server (typically a replica, but can be the master)
|
// Inspector reads data from the read-MySQL-server (typically a replica, but can be the master)
|
||||||
// It is used for gaining initial status and structure, and later also follow up on progress and changelog
|
// It is used for gaining initial status and structure, and later also follow up on progress and changelog
|
||||||
type Inspector struct {
|
type Inspector struct {
|
||||||
connectionConfig *mysql.ConnectionConfig
|
connectionConfig *mysql.ConnectionConfig
|
||||||
db *gosql.DB
|
db *gosql.DB
|
||||||
migrationContext *base.MigrationContext
|
informationSchemaDb *gosql.DB
|
||||||
|
migrationContext *base.MigrationContext
|
||||||
}
|
}
|
||||||
|
|
||||||
func NewInspector(migrationContext *base.MigrationContext) *Inspector {
|
func NewInspector(migrationContext *base.MigrationContext) *Inspector {
|
||||||
@ -40,9 +41,15 @@ func NewInspector(migrationContext *base.MigrationContext) *Inspector {
|
|||||||
|
|
||||||
func (this *Inspector) InitDBConnections() (err error) {
|
func (this *Inspector) InitDBConnections() (err error) {
|
||||||
inspectorUri := this.connectionConfig.GetDBUri(this.migrationContext.DatabaseName)
|
inspectorUri := this.connectionConfig.GetDBUri(this.migrationContext.DatabaseName)
|
||||||
if this.db, _, err = sqlutils.GetDB(inspectorUri); err != nil {
|
if this.db, _, err = this.migrationContext.GetDB(inspectorUri); err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
|
informationSchemaUri := this.connectionConfig.GetDBUri("information_schema")
|
||||||
|
if this.informationSchemaDb, _, err = this.migrationContext.GetDB(informationSchemaUri); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
if err := this.validateConnection(); err != nil {
|
if err := this.validateConnection(); err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
@ -755,6 +762,7 @@ func (this *Inspector) getMasterConnectionConfig() (applierConfig *mysql.Connect
|
|||||||
|
|
||||||
func (this *Inspector) getReplicationLag() (replicationLag time.Duration, err error) {
|
func (this *Inspector) getReplicationLag() (replicationLag time.Duration, err error) {
|
||||||
replicationLag, err = mysql.GetReplicationLag(
|
replicationLag, err = mysql.GetReplicationLag(
|
||||||
|
this.informationSchemaDb,
|
||||||
this.migrationContext.InspectorConnectionConfig,
|
this.migrationContext.InspectorConnectionConfig,
|
||||||
)
|
)
|
||||||
return replicationLag, err
|
return replicationLag, err
|
||||||
@ -762,5 +770,6 @@ func (this *Inspector) getReplicationLag() (replicationLag time.Duration, err er
|
|||||||
|
|
||||||
func (this *Inspector) Teardown() {
|
func (this *Inspector) Teardown() {
|
||||||
this.db.Close()
|
this.db.Close()
|
||||||
|
this.informationSchemaDb.Close()
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
@ -20,7 +20,6 @@ import (
|
|||||||
"github.com/github/gh-ost/go/sql"
|
"github.com/github/gh-ost/go/sql"
|
||||||
|
|
||||||
"github.com/outbrain/golib/log"
|
"github.com/outbrain/golib/log"
|
||||||
"github.com/outbrain/golib/sqlutils"
|
|
||||||
)
|
)
|
||||||
|
|
||||||
type ChangelogState string
|
type ChangelogState string
|
||||||
@ -1248,6 +1247,4 @@ func (this *Migrator) teardown() {
|
|||||||
log.Infof("Tearing down streamer")
|
log.Infof("Tearing down streamer")
|
||||||
this.eventsStreamer.Teardown()
|
this.eventsStreamer.Teardown()
|
||||||
}
|
}
|
||||||
|
|
||||||
sqlutils.ResetDBCache()
|
|
||||||
}
|
}
|
||||||
|
@ -104,7 +104,7 @@ func (this *EventsStreamer) notifyListeners(binlogEvent *binlog.BinlogDMLEvent)
|
|||||||
|
|
||||||
func (this *EventsStreamer) InitDBConnections() (err error) {
|
func (this *EventsStreamer) InitDBConnections() (err error) {
|
||||||
EventsStreamerUri := this.connectionConfig.GetDBUri(this.migrationContext.DatabaseName)
|
EventsStreamerUri := this.connectionConfig.GetDBUri(this.migrationContext.DatabaseName)
|
||||||
if this.db, _, err = sqlutils.GetDB(EventsStreamerUri); err != nil {
|
if this.db, _, err = this.migrationContext.GetDB(EventsStreamerUri); err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
if err := this.validateConnection(); err != nil {
|
if err := this.validateConnection(); err != nil {
|
||||||
|
@ -16,7 +16,6 @@ import (
|
|||||||
"github.com/github/gh-ost/go/mysql"
|
"github.com/github/gh-ost/go/mysql"
|
||||||
"github.com/github/gh-ost/go/sql"
|
"github.com/github/gh-ost/go/sql"
|
||||||
"github.com/outbrain/golib/log"
|
"github.com/outbrain/golib/log"
|
||||||
"github.com/outbrain/golib/sqlutils"
|
|
||||||
)
|
)
|
||||||
|
|
||||||
var (
|
var (
|
||||||
@ -140,7 +139,7 @@ func (this *Throttler) collectReplicationLag(firstThrottlingCollected chan<- boo
|
|||||||
// when running on replica, the heartbeat injection is also done on the replica.
|
// when running on replica, the heartbeat injection is also done on the replica.
|
||||||
// This means we will always get a good heartbeat value.
|
// This means we will always get a good heartbeat value.
|
||||||
// When runnign on replica, we should instead check the `SHOW SLAVE STATUS` output.
|
// When runnign on replica, we should instead check the `SHOW SLAVE STATUS` output.
|
||||||
if lag, err := mysql.GetReplicationLag(this.inspector.connectionConfig); err != nil {
|
if lag, err := mysql.GetReplicationLag(this.inspector.informationSchemaDb, this.inspector.connectionConfig); err != nil {
|
||||||
return log.Errore(err)
|
return log.Errore(err)
|
||||||
} else {
|
} else {
|
||||||
atomic.StoreInt64(&this.migrationContext.CurrentLag, int64(lag))
|
atomic.StoreInt64(&this.migrationContext.CurrentLag, int64(lag))
|
||||||
@ -182,7 +181,7 @@ func (this *Throttler) collectControlReplicasLag() {
|
|||||||
dbUri := connectionConfig.GetDBUri("information_schema")
|
dbUri := connectionConfig.GetDBUri("information_schema")
|
||||||
|
|
||||||
var heartbeatValue string
|
var heartbeatValue string
|
||||||
if db, _, err := sqlutils.GetDB(dbUri); err != nil {
|
if db, _, err := this.migrationContext.GetDB(dbUri); err != nil {
|
||||||
return lag, err
|
return lag, err
|
||||||
} else if err = db.QueryRow(replicationLagQuery).Scan(&heartbeatValue); err != nil {
|
} else if err = db.QueryRow(replicationLagQuery).Scan(&heartbeatValue); err != nil {
|
||||||
return lag, err
|
return lag, err
|
||||||
|
@ -35,15 +35,8 @@ func (this *ReplicationLagResult) HasLag() bool {
|
|||||||
|
|
||||||
// GetReplicationLag returns replication lag for a given connection config; either by explicit query
|
// GetReplicationLag returns replication lag for a given connection config; either by explicit query
|
||||||
// or via SHOW SLAVE STATUS
|
// or via SHOW SLAVE STATUS
|
||||||
func GetReplicationLag(connectionConfig *ConnectionConfig) (replicationLag time.Duration, err error) {
|
func GetReplicationLag(informationSchemaDb *gosql.DB, connectionConfig *ConnectionConfig) (replicationLag time.Duration, err error) {
|
||||||
dbUri := connectionConfig.GetDBUri("information_schema")
|
err = sqlutils.QueryRowsMap(informationSchemaDb, `show slave status`, func(m sqlutils.RowMap) error {
|
||||||
var db *gosql.DB
|
|
||||||
if db, _, err = sqlutils.GetDB(dbUri); err != nil {
|
|
||||||
return replicationLag, err
|
|
||||||
}
|
|
||||||
defer db.Close()
|
|
||||||
|
|
||||||
err = sqlutils.QueryRowsMap(db, `show slave status`, func(m sqlutils.RowMap) error {
|
|
||||||
slaveIORunning := m.GetString("Slave_IO_Running")
|
slaveIORunning := m.GetString("Slave_IO_Running")
|
||||||
slaveSQLRunning := m.GetString("Slave_SQL_Running")
|
slaveSQLRunning := m.GetString("Slave_SQL_Running")
|
||||||
secondsBehindMaster := m.GetNullInt64("Seconds_Behind_Master")
|
secondsBehindMaster := m.GetNullInt64("Seconds_Behind_Master")
|
||||||
@ -59,7 +52,10 @@ func GetReplicationLag(connectionConfig *ConnectionConfig) (replicationLag time.
|
|||||||
|
|
||||||
func GetMasterKeyFromSlaveStatus(connectionConfig *ConnectionConfig) (masterKey *InstanceKey, err error) {
|
func GetMasterKeyFromSlaveStatus(connectionConfig *ConnectionConfig) (masterKey *InstanceKey, err error) {
|
||||||
currentUri := connectionConfig.GetDBUri("information_schema")
|
currentUri := connectionConfig.GetDBUri("information_schema")
|
||||||
db, _, err := sqlutils.GetDB(currentUri)
|
// This function is only called once, okay to not have a cached connection pool
|
||||||
|
db, err := sqlutils.GetDB(currentUri)
|
||||||
|
defer db.Close()
|
||||||
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
|
32
vendor/github.com/outbrain/golib/sqlutils/sqlutils.go
generated
vendored
32
vendor/github.com/outbrain/golib/sqlutils/sqlutils.go
generated
vendored
@ -25,7 +25,6 @@ import (
|
|||||||
"github.com/outbrain/golib/log"
|
"github.com/outbrain/golib/log"
|
||||||
"strconv"
|
"strconv"
|
||||||
"strings"
|
"strings"
|
||||||
"sync"
|
|
||||||
)
|
)
|
||||||
|
|
||||||
// RowMap represents one row in a result set. Its objective is to allow
|
// RowMap represents one row in a result set. Its objective is to allow
|
||||||
@ -121,34 +120,13 @@ func (this *RowMap) GetBool(key string) bool {
|
|||||||
return this.GetInt(key) != 0
|
return this.GetInt(key) != 0
|
||||||
}
|
}
|
||||||
|
|
||||||
// knownDBs is a DB cache by uri
|
|
||||||
var knownDBs map[string]*sql.DB = make(map[string]*sql.DB)
|
|
||||||
var knownDBsMutex = &sync.Mutex{}
|
|
||||||
|
|
||||||
// GetDB returns a DB instance based on uri.
|
// GetDB returns a DB instance based on uri.
|
||||||
// bool result indicates whether the DB was returned from cache; err
|
func GetDB(mysql_uri string) (*sql.DB, error) {
|
||||||
func GetDB(mysql_uri string) (*sql.DB, bool, error) {
|
if db, err := sql.Open("mysql", mysql_uri); err == nil {
|
||||||
knownDBsMutex.Lock()
|
return db, nil
|
||||||
defer knownDBsMutex.Unlock()
|
} else {
|
||||||
|
return db, err
|
||||||
var exists bool
|
|
||||||
if _, exists = knownDBs[mysql_uri]; !exists {
|
|
||||||
if db, err := sql.Open("mysql", mysql_uri); err == nil {
|
|
||||||
knownDBs[mysql_uri] = db
|
|
||||||
} else {
|
|
||||||
return db, exists, err
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
return knownDBs[mysql_uri], exists, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// Resets the knownDBs cache, used when the DB connections have been closed,
|
|
||||||
// and new connections are needed to access the DB
|
|
||||||
func ResetDBCache() {
|
|
||||||
knownDBsMutex.Lock()
|
|
||||||
defer knownDBsMutex.Unlock()
|
|
||||||
|
|
||||||
knownDBs = make(map[string]*sql.DB)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// RowToArray is a convenience function, typically not called directly, which maps a
|
// RowToArray is a convenience function, typically not called directly, which maps a
|
||||||
|
Loading…
Reference in New Issue
Block a user