2017-02-12 13:13:54 +02:00
|
|
|
// go-mysqlbinlog: a simple binlog tool to sync remote MySQL binlog.
|
|
|
|
// go-mysqlbinlog supports semi-sync mode like facebook mysqlbinlog.
|
|
|
|
// see http://yoshinorimatsunobu.blogspot.com/2014/04/semi-synchronous-replication-at-facebook.html
|
|
|
|
package main
|
|
|
|
|
|
|
|
import (
|
2019-01-01 10:57:46 +02:00
|
|
|
"context"
|
2017-02-12 13:13:54 +02:00
|
|
|
"flag"
|
|
|
|
"fmt"
|
|
|
|
"os"
|
|
|
|
|
|
|
|
"github.com/juju/errors"
|
|
|
|
"github.com/siddontang/go-mysql/mysql"
|
|
|
|
"github.com/siddontang/go-mysql/replication"
|
|
|
|
)
|
|
|
|
|
|
|
|
var host = flag.String("host", "127.0.0.1", "MySQL host")
|
|
|
|
var port = flag.Int("port", 3306, "MySQL port")
|
|
|
|
var user = flag.String("user", "root", "MySQL user, must have replication privilege")
|
|
|
|
var password = flag.String("password", "", "MySQL password")
|
|
|
|
|
|
|
|
var flavor = flag.String("flavor", "mysql", "Flavor: mysql or mariadb")
|
|
|
|
|
|
|
|
var file = flag.String("file", "", "Binlog filename")
|
|
|
|
var pos = flag.Int("pos", 4, "Binlog position")
|
|
|
|
|
|
|
|
var semiSync = flag.Bool("semisync", false, "Support semi sync")
|
|
|
|
var backupPath = flag.String("backup_path", "", "backup path to store binlog files")
|
|
|
|
|
|
|
|
var rawMode = flag.Bool("raw", false, "Use raw mode")
|
|
|
|
|
|
|
|
func main() {
|
|
|
|
flag.Parse()
|
|
|
|
|
|
|
|
cfg := replication.BinlogSyncerConfig{
|
|
|
|
ServerID: 101,
|
|
|
|
Flavor: *flavor,
|
|
|
|
|
|
|
|
Host: *host,
|
|
|
|
Port: uint16(*port),
|
|
|
|
User: *user,
|
|
|
|
Password: *password,
|
2018-04-11 09:11:27 +03:00
|
|
|
RawModeEnabled: *rawMode,
|
2017-02-12 13:13:54 +02:00
|
|
|
SemiSyncEnabled: *semiSync,
|
2019-01-01 10:57:46 +02:00
|
|
|
UseDecimal: true,
|
2017-02-12 13:13:54 +02:00
|
|
|
}
|
|
|
|
|
2018-01-11 14:51:14 +02:00
|
|
|
b := replication.NewBinlogSyncer(cfg)
|
2017-02-12 13:13:54 +02:00
|
|
|
|
2019-01-01 10:57:46 +02:00
|
|
|
pos := mysql.Position{Name: *file, Pos: uint32(*pos)}
|
2017-02-12 13:13:54 +02:00
|
|
|
if len(*backupPath) > 0 {
|
|
|
|
// Backup will always use RawMode.
|
|
|
|
err := b.StartBackup(*backupPath, pos, 0)
|
|
|
|
if err != nil {
|
|
|
|
fmt.Printf("Start backup error: %v\n", errors.ErrorStack(err))
|
|
|
|
return
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
s, err := b.StartSync(pos)
|
|
|
|
if err != nil {
|
|
|
|
fmt.Printf("Start sync error: %v\n", errors.ErrorStack(err))
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
for {
|
|
|
|
e, err := s.GetEvent(context.Background())
|
|
|
|
if err != nil {
|
2019-01-01 10:57:46 +02:00
|
|
|
// Try to output all left events
|
|
|
|
events := s.DumpEvents()
|
|
|
|
for _, e := range events {
|
|
|
|
e.Dump(os.Stdout)
|
|
|
|
}
|
2017-02-12 13:13:54 +02:00
|
|
|
fmt.Printf("Get event error: %v\n", errors.ErrorStack(err))
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
e.Dump(os.Stdout)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|