236 lines
7.5 KiB
Go
236 lines
7.5 KiB
Go
/*
|
|
Copyright 2016 GitHub Inc.
|
|
See https://github.com/github/gh-ost/blob/master/LICENSE
|
|
*/
|
|
|
|
package logic
|
|
|
|
import (
|
|
gosql "database/sql"
|
|
"fmt"
|
|
"strings"
|
|
"sync"
|
|
"time"
|
|
|
|
"github.com/github/gh-ost/go/base"
|
|
"github.com/github/gh-ost/go/binlog"
|
|
"github.com/github/gh-ost/go/mysql"
|
|
|
|
"github.com/outbrain/golib/log"
|
|
"github.com/outbrain/golib/sqlutils"
|
|
)
|
|
|
|
type BinlogEventListenerFunc func(event *binlog.BinlogDMLEvent, coordinates *mysql.BinlogCoordinates) error
|
|
|
|
type BinlogEventListener struct {
|
|
async bool
|
|
databaseName string
|
|
tableName string
|
|
onDmlEvent BinlogEventListenerFunc
|
|
}
|
|
|
|
const (
|
|
EventsChannelBufferSize = 1
|
|
ReconnectStreamerSleepSeconds = 5
|
|
)
|
|
|
|
// EventsStreamer reads data from binary logs and streams it on. It acts as a publisher,
|
|
// and interested parties may subscribe for per-table events.
|
|
type EventsStreamer struct {
|
|
connectionConfig *mysql.ConnectionConfig
|
|
db *gosql.DB
|
|
migrationContext *base.MigrationContext
|
|
initialBinlogCoordinates *mysql.BinlogCoordinates
|
|
listeners [](*BinlogEventListener)
|
|
listenersMutex *sync.Mutex
|
|
eventsChannel chan *binlog.BinlogEntry
|
|
binlogReader *binlog.GoMySQLReader
|
|
}
|
|
|
|
func NewEventsStreamer() *EventsStreamer {
|
|
return &EventsStreamer{
|
|
connectionConfig: base.GetMigrationContext().InspectorConnectionConfig,
|
|
migrationContext: base.GetMigrationContext(),
|
|
listeners: [](*BinlogEventListener){},
|
|
listenersMutex: &sync.Mutex{},
|
|
eventsChannel: make(chan *binlog.BinlogEntry, EventsChannelBufferSize),
|
|
}
|
|
}
|
|
|
|
// AddListener registers a new listener for binlog events, on a per-table basis
|
|
func (this *EventsStreamer) AddListener(
|
|
async bool, databaseName string, tableName string, onDmlEvent BinlogEventListenerFunc) (err error) {
|
|
|
|
this.listenersMutex.Lock()
|
|
defer this.listenersMutex.Unlock()
|
|
|
|
if databaseName == "" {
|
|
return fmt.Errorf("Empty database name in AddListener")
|
|
}
|
|
if tableName == "" {
|
|
return fmt.Errorf("Empty table name in AddListener")
|
|
}
|
|
listener := &BinlogEventListener{
|
|
async: async,
|
|
databaseName: databaseName,
|
|
tableName: tableName,
|
|
onDmlEvent: onDmlEvent,
|
|
}
|
|
this.listeners = append(this.listeners, listener)
|
|
return nil
|
|
}
|
|
|
|
// notifyListeners will notify relevant listeners with given DML event. Only
|
|
// listeners registered for changes on the table on which the DML operates are notified.
|
|
func (this *EventsStreamer) notifyListeners(binlogEntry *binlog.BinlogEntry) {
|
|
this.listenersMutex.Lock()
|
|
defer this.listenersMutex.Unlock()
|
|
|
|
binlogEvent := binlogEntry.DmlEvent
|
|
for _, listener := range this.listeners {
|
|
listener := listener
|
|
if strings.ToLower(listener.databaseName) != strings.ToLower(binlogEvent.DatabaseName) {
|
|
continue
|
|
}
|
|
if strings.ToLower(listener.tableName) != strings.ToLower(binlogEvent.TableName) {
|
|
continue
|
|
}
|
|
if listener.async {
|
|
go func() {
|
|
listener.onDmlEvent(binlogEvent, &binlogEntry.Coordinates)
|
|
}()
|
|
} else {
|
|
listener.onDmlEvent(binlogEvent, &binlogEntry.Coordinates)
|
|
}
|
|
}
|
|
}
|
|
|
|
func (this *EventsStreamer) InitDBConnections(resurrectedContext *base.MigrationContext) (err error) {
|
|
EventsStreamerUri := this.connectionConfig.GetDBUri(this.migrationContext.DatabaseName)
|
|
if this.db, _, err = sqlutils.GetDB(EventsStreamerUri); err != nil {
|
|
return err
|
|
}
|
|
if err := this.validateConnection(); err != nil {
|
|
return err
|
|
}
|
|
if this.migrationContext.Resurrect {
|
|
// Rewinding to beginning of logfile:
|
|
resurrectedContext.AppliedBinlogCoordinates.LogPos = 4
|
|
log.Infof("Resurrection: initiating streamer at resurrected coordinates %+v", resurrectedContext.AppliedBinlogCoordinates)
|
|
this.initialBinlogCoordinates = &resurrectedContext.AppliedBinlogCoordinates
|
|
} else {
|
|
if err := this.readCurrentBinlogCoordinates(); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
if err := this.initBinlogReader(this.initialBinlogCoordinates); err != nil {
|
|
return err
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
// initBinlogReader creates and connects the reader: we hook up to a MySQL server as a replica
|
|
func (this *EventsStreamer) initBinlogReader(binlogCoordinates *mysql.BinlogCoordinates) error {
|
|
goMySQLReader, err := binlog.NewGoMySQLReader(this.migrationContext.InspectorConnectionConfig)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
if err := goMySQLReader.ConnectBinlogStreamer(*binlogCoordinates); err != nil {
|
|
return err
|
|
}
|
|
this.binlogReader = goMySQLReader
|
|
return nil
|
|
}
|
|
|
|
// validateConnection issues a simple can-connect to MySQL
|
|
func (this *EventsStreamer) validateConnection() error {
|
|
query := `select @@global.port`
|
|
var port int
|
|
if err := this.db.QueryRow(query).Scan(&port); err != nil {
|
|
return err
|
|
}
|
|
if port != this.connectionConfig.Key.Port {
|
|
return fmt.Errorf("Unexpected database port reported: %+v", port)
|
|
}
|
|
log.Infof("connection validated on %+v", this.connectionConfig.Key)
|
|
return nil
|
|
}
|
|
|
|
func (this *EventsStreamer) GetCurrentBinlogCoordinates() *mysql.BinlogCoordinates {
|
|
return this.binlogReader.GetCurrentBinlogCoordinates()
|
|
}
|
|
|
|
func (this *EventsStreamer) GetReconnectBinlogCoordinates() *mysql.BinlogCoordinates {
|
|
return &mysql.BinlogCoordinates{LogFile: this.GetCurrentBinlogCoordinates().LogFile, LogPos: 4}
|
|
}
|
|
|
|
// readCurrentBinlogCoordinates reads master status from hooked server
|
|
func (this *EventsStreamer) readCurrentBinlogCoordinates() error {
|
|
query := `show /* gh-ost readCurrentBinlogCoordinates */ master status`
|
|
foundMasterStatus := false
|
|
err := sqlutils.QueryRowsMap(this.db, query, func(m sqlutils.RowMap) error {
|
|
this.initialBinlogCoordinates = &mysql.BinlogCoordinates{
|
|
LogFile: m.GetString("File"),
|
|
LogPos: m.GetInt64("Position"),
|
|
}
|
|
foundMasterStatus = true
|
|
|
|
return nil
|
|
})
|
|
if err != nil {
|
|
return err
|
|
}
|
|
if !foundMasterStatus {
|
|
return fmt.Errorf("Got no results from SHOW MASTER STATUS. Bailing out")
|
|
}
|
|
log.Debugf("Streamer binlog coordinates: %+v", *this.initialBinlogCoordinates)
|
|
return nil
|
|
}
|
|
|
|
// StreamEvents will begin streaming events. It will be blocking, so should be
|
|
// executed by a goroutine
|
|
func (this *EventsStreamer) StreamEvents(canStopStreaming func() bool) error {
|
|
go func() {
|
|
for binlogEntry := range this.eventsChannel {
|
|
if binlogEntry.DmlEvent != nil {
|
|
this.notifyListeners(binlogEntry)
|
|
}
|
|
}
|
|
}()
|
|
// The next should block and execute forever, unless there's a serious error
|
|
var successiveFailures int64
|
|
var lastAppliedRowsEventHint mysql.BinlogCoordinates
|
|
for {
|
|
if err := this.binlogReader.StreamEvents(canStopStreaming, this.eventsChannel); err != nil {
|
|
log.Infof("StreamEvents encountered unexpected error: %+v", err)
|
|
this.migrationContext.MarkPointOfInterest()
|
|
time.Sleep(ReconnectStreamerSleepSeconds * time.Second)
|
|
|
|
// See if there's retry overflow
|
|
if this.binlogReader.LastAppliedRowsEventHint.Equals(&lastAppliedRowsEventHint) {
|
|
successiveFailures += 1
|
|
} else {
|
|
successiveFailures = 0
|
|
}
|
|
if successiveFailures > this.migrationContext.MaxRetries() {
|
|
return fmt.Errorf("%d successive failures in streamer reconnect at coordinates %+v", successiveFailures, this.GetReconnectBinlogCoordinates())
|
|
}
|
|
|
|
// Reposition at same binlog file.
|
|
lastAppliedRowsEventHint = this.binlogReader.LastAppliedRowsEventHint
|
|
log.Infof("Reconnecting... Will resume at %+v", lastAppliedRowsEventHint)
|
|
if err := this.initBinlogReader(this.GetReconnectBinlogCoordinates()); err != nil {
|
|
return err
|
|
}
|
|
this.binlogReader.LastAppliedRowsEventHint = lastAppliedRowsEventHint
|
|
}
|
|
}
|
|
}
|
|
|
|
func (this *EventsStreamer) Close() (err error) {
|
|
err = this.binlogReader.Close()
|
|
log.Infof("Closed streamer connection. err=%+v", err)
|
|
return err
|
|
}
|