From 629623a482250ad92ab7198fcc7533972bddaa4a Mon Sep 17 00:00:00 2001 From: Shlomi Noach <2607934+shlomi-noach@users.noreply.github.com> Date: Sun, 31 Mar 2024 09:35:34 +0300 Subject: [PATCH 1/6] Do not require 'MySQL56/' prefix in PITR restore Signed-off-by: Shlomi Noach <2607934+shlomi-noach@users.noreply.github.com> --- go/vt/vttablet/tabletmanager/restore.go | 9 ++++++++- 1 file changed, 8 insertions(+), 1 deletion(-) diff --git a/go/vt/vttablet/tabletmanager/restore.go b/go/vt/vttablet/tabletmanager/restore.go index 6b37edb5244..5f57fc9a593 100644 --- a/go/vt/vttablet/tabletmanager/restore.go +++ b/go/vt/vttablet/tabletmanager/restore.go @@ -36,6 +36,7 @@ import ( "vitess.io/vitess/go/vt/logutil" "vitess.io/vitess/go/vt/mysqlctl" "vitess.io/vitess/go/vt/mysqlctl/backupstats" + "vitess.io/vitess/go/vt/proto/vtrpc" "vitess.io/vitess/go/vt/proto/vttime" "vitess.io/vitess/go/vt/servenv" "vitess.io/vitess/go/vt/topo" @@ -237,10 +238,16 @@ func (tm *TabletManager) restoreDataLocked(ctx context.Context, logger logutil.L return vterrors.Errorf(vtrpcpb.Code_INVALID_ARGUMENT, "--restore-to-pos and --restore-to-timestamp are mutually exclusive") } if request.RestoreToPos != "" { - pos, err := replication.DecodePosition(request.RestoreToPos) + pos, err := replication.DecodePositionDefaultFlavor(request.RestoreToPos, replication.Mysql56FlavorID) if err != nil { return vterrors.Wrapf(err, "restore failed: unable to decode --restore-to-pos: %s", request.RestoreToPos) } + if !pos.MatchesFlavor(replication.Mysql56FlavorID) { + return vterrors.Errorf(vtrpc.Code_FAILED_PRECONDITION, "restore to position only supports MySQL GTID positions. Got: %v", request.RestoreToPos) + } + if _, ok := pos.GTIDSet.(replication.Mysql56GTIDSet); !ok { + return vterrors.Errorf(vtrpc.Code_FAILED_PRECONDITION, "cannot get MySQL GTID value: %v", pos) + } params.RestoreToPos = pos } if !restoreToTimestamp.IsZero() { From d3e143cda7221dde79584de869f5f41074bae615 Mon Sep 17 00:00:00 2001 From: Shlomi Noach <2607934+shlomi-noach@users.noreply.github.com> Date: Sun, 31 Mar 2024 09:36:42 +0300 Subject: [PATCH 2/6] randomly strip 'MySQL56/' prefix in RestoreFromBackup Signed-off-by: Shlomi Noach <2607934+shlomi-noach@users.noreply.github.com> --- go/test/endtoend/backup/vtctlbackup/backup_utils.go | 8 ++++++++ 1 file changed, 8 insertions(+) diff --git a/go/test/endtoend/backup/vtctlbackup/backup_utils.go b/go/test/endtoend/backup/vtctlbackup/backup_utils.go index 7fc872e934c..14063a8daac 100644 --- a/go/test/endtoend/backup/vtctlbackup/backup_utils.go +++ b/go/test/endtoend/backup/vtctlbackup/backup_utils.go @@ -21,6 +21,7 @@ import ( "context" "encoding/json" "fmt" + "math/rand" "os" "os/exec" "path" @@ -1299,6 +1300,13 @@ func TestReplicaRestoreToPos(t *testing.T, replicaIndex int, restoreToPos replic require.False(t, restoreToPos.IsZero()) restoreToPosArg := replication.EncodePosition(restoreToPos) + assert.Contains(t, restoreToPosArg, "MySQL56/") + if rand.Intn(2) == 0 { + // Verify that restore works whether or not the MySQL56/ prefix is present. + restoreToPosArg = strings.Replace(restoreToPosArg, "MySQL56/", "", 1) + assert.NotContains(t, restoreToPosArg, "MySQL56/") + } + output, err := localCluster.VtctldClientProcess.ExecuteCommandWithOutput("RestoreFromBackup", "--restore-to-pos", restoreToPosArg, replica.Alias) if expectError != "" { require.Errorf(t, err, "expected: %v", expectError) From fff521fe60508200e517e5e5e62bd20c741f37e9 Mon Sep 17 00:00:00 2001 From: Shlomi Noach <2607934+shlomi-noach@users.noreply.github.com> Date: Sun, 31 Mar 2024 09:38:29 +0300 Subject: [PATCH 3/6] test incremental backup with/out 'MySQL56/' position prefix Signed-off-by: Shlomi Noach <2607934+shlomi-noach@users.noreply.github.com> --- .../backup/vtctlbackup/pitr_test_framework.go | 12 +++++++++++- 1 file changed, 11 insertions(+), 1 deletion(-) diff --git a/go/test/endtoend/backup/vtctlbackup/pitr_test_framework.go b/go/test/endtoend/backup/vtctlbackup/pitr_test_framework.go index 64e058dfe65..7f611d81ad6 100644 --- a/go/test/endtoend/backup/vtctlbackup/pitr_test_framework.go +++ b/go/test/endtoend/backup/vtctlbackup/pitr_test_framework.go @@ -20,6 +20,7 @@ import ( "context" "fmt" "math/rand/v2" + "strings" "testing" "time" @@ -227,9 +228,17 @@ func ExecTestIncrementalBackupAndRestoreToPos(t *testing.T, tcase *PITRTestCase) if tc.fromFullPosition { incrementalFromPos = replication.EncodePosition(fullBackupPos) } + assert.Contains(t, incrementalFromPos, "MySQL56/") + } + incrementalFromPosArg := incrementalFromPos + if tc.incrementalFrom == incrementalFromPosPosition && tc.fromFullPosition { + // Verify that backup works whether or not the MySQL56/ prefix is present. + // We arbitrarily decide to strip the prefix when "tc.fromFullPosition" is true, and keep it when false. + incrementalFromPosArg = strings.Replace(incrementalFromPosArg, "MySQL56/", "", 1) + assert.NotContains(t, incrementalFromPosArg, "MySQL56/") } // always use same 1st replica - manifest, backupName := TestReplicaIncrementalBackup(t, 0, incrementalFromPos, tc.expectEmpty, tc.expectError) + manifest, backupName := TestReplicaIncrementalBackup(t, 0, incrementalFromPosArg, tc.expectEmpty, tc.expectError) if tc.expectError != "" { return } @@ -271,6 +280,7 @@ func ExecTestIncrementalBackupAndRestoreToPos(t *testing.T, tcase *PITRTestCase) t.Run(testName, func(t *testing.T) { restoreToPos, err := replication.DecodePosition(pos) require.NoError(t, err) + require.False(t, restoreToPos.IsZero()) TestReplicaRestoreToPos(t, 0, restoreToPos, "") msgs := ReadRowsFromReplica(t, 0) count, ok := rowsPerPosition[pos] From 8ee1cf645c7411359700bce47fefc467bf726a45 Mon Sep 17 00:00:00 2001 From: Shlomi Noach <2607934+shlomi-noach@users.noreply.github.com> Date: Sun, 31 Mar 2024 10:11:21 +0300 Subject: [PATCH 4/6] refactor: consolidate logic in new DecodePositionMySQL56() function. Retire (but not purge as it may still be useful) DecodePositionDefaultFlavor Signed-off-by: Shlomi Noach <2607934+shlomi-noach@users.noreply.github.com> --- go/mysql/replication/mysql56_gtid.go | 33 +++++++++++++++++ go/mysql/replication/mysql56_gtid_test.go | 37 +++++++++++++++++++ .../foreignkey/stress/fk_stress_test.go | 2 +- go/vt/mysqlctl/builtinbackupengine.go | 13 ++----- go/vt/vttablet/tabletmanager/restore.go | 9 +---- 5 files changed, 75 insertions(+), 19 deletions(-) diff --git a/go/mysql/replication/mysql56_gtid.go b/go/mysql/replication/mysql56_gtid.go index 4ec861b84e5..48d199e4360 100644 --- a/go/mysql/replication/mysql56_gtid.go +++ b/go/mysql/replication/mysql56_gtid.go @@ -29,6 +29,10 @@ import ( // Mysql56FlavorID is the string identifier for the Mysql56 flavor. const Mysql56FlavorID = "MySQL56" +var ( + ErrExpectMysql56Flavor = vterrors.Errorf(vtrpc.Code_INVALID_ARGUMENT, "Expecting MySQL56 position flavor") +) + // parseMysql56GTID is registered as a GTID parser. func parseMysql56GTID(s string) (GTID, error) { // Split into parts. @@ -128,3 +132,32 @@ func (gtid Mysql56GTID) GTIDSet() GTIDSet { func init() { gtidParsers[Mysql56FlavorID] = parseMysql56GTID } + +// DecodePositionMySQL56 converts a string into a Position value with the MySQL56 flavor. The function returns an error if the given +// string does not translate to a MySQL56 GTID set. +// The prefix "MySQL56/" is optional in the input string. Examples of inputs strings that produce valid result: +// - "MySQL56/16b1039f-22b6-11ed-b765-0a43f95f28a3:1-615" +// - "16b1039f-22b6-11ed-b765-0a43f95f28a3:1-615" +func DecodePositionMySQL56(s string) (rp Position, gtidSet Mysql56GTIDSet, err error) { + if s == "" { + return rp, nil, nil + } + + flav, gtid, ok := strings.Cut(s, "/") + if !ok { + gtid = s + flav = Mysql56FlavorID + } + rp, err = ParsePosition(flav, gtid) + if err != nil { + return rp, nil, err + } + if !rp.MatchesFlavor(Mysql56FlavorID) { + return rp, nil, vterrors.Wrapf(ErrExpectMysql56Flavor, s) + } + gtidSet, ok = rp.GTIDSet.(Mysql56GTIDSet) + if !ok { + return rp, nil, vterrors.Wrapf(ErrExpectMysql56Flavor, s) + } + return rp, gtidSet, nil +} diff --git a/go/mysql/replication/mysql56_gtid_test.go b/go/mysql/replication/mysql56_gtid_test.go index 7a4bc9862a8..6db23cc48bc 100644 --- a/go/mysql/replication/mysql56_gtid_test.go +++ b/go/mysql/replication/mysql56_gtid_test.go @@ -153,3 +153,40 @@ func TestMysql56ParseGTID(t *testing.T) { require.NoError(t, err, "unexpected error: %v", err) assert.Equal(t, want, got, "(&mysql56{}).ParseGTID(%#v) = %#v, want %#v", input, got, want) } + +func TestDecodePositionMySQL56(t *testing.T) { + { + pos, gtidSet, err := DecodePositionMySQL56("") + assert.NoError(t, err) + assert.True(t, pos.IsZero()) + assert.Nil(t, gtidSet) + } + { + pos, gtidSet, err := DecodePositionMySQL56("MySQL56/16b1039f-22b6-11ed-b765-0a43f95f28a3:1-615") + assert.NoError(t, err) + assert.False(t, pos.IsZero()) + assert.NotNil(t, gtidSet) + } + { + pos, gtidSet, err := DecodePositionMySQL56("16b1039f-22b6-11ed-b765-0a43f95f28a3:1-615") + assert.NoError(t, err) + assert.False(t, pos.IsZero()) + assert.NotNil(t, gtidSet) + } + { + _, _, err := DecodePositionMySQL56("q-22b6-11ed-b765-0a43f95f28a3:1-615") + assert.Error(t, err) + } + { + _, _, err := DecodePositionMySQL56("16b1039f-22b6-11ed-b765-0a43f95f28a3") + assert.Error(t, err) + } + { + _, _, err := DecodePositionMySQL56("FilePos/mysql-bin.000001:234") + assert.Error(t, err) + } + { + _, _, err := DecodePositionMySQL56("mysql-bin.000001:234") + assert.Error(t, err) + } +} diff --git a/go/test/endtoend/vtgate/foreignkey/stress/fk_stress_test.go b/go/test/endtoend/vtgate/foreignkey/stress/fk_stress_test.go index 73baad2615b..b9240f46605 100644 --- a/go/test/endtoend/vtgate/foreignkey/stress/fk_stress_test.go +++ b/go/test/endtoend/vtgate/foreignkey/stress/fk_stress_test.go @@ -415,7 +415,7 @@ func getTabletPosition(t *testing.T, tablet *cluster.Vttablet) replication.Posit require.NotNil(t, row) gtidExecuted := row.AsString("gtid_executed", "") require.NotEmpty(t, gtidExecuted) - pos, err := replication.DecodePositionDefaultFlavor(gtidExecuted, replication.Mysql56FlavorID) + pos, _, err := replication.DecodePositionMySQL56(gtidExecuted) assert.NoError(t, err) return pos } diff --git a/go/vt/mysqlctl/builtinbackupengine.go b/go/vt/mysqlctl/builtinbackupengine.go index 5cdf39084da..fbc078c870b 100644 --- a/go/vt/mysqlctl/builtinbackupengine.go +++ b/go/vt/mysqlctl/builtinbackupengine.go @@ -223,18 +223,11 @@ func (be *BuiltinBackupEngine) ExecuteBackup(ctx context.Context, params BackupP // getIncrementalFromPosGTIDSet turns the given string into a valid Mysql56GTIDSet func getIncrementalFromPosGTIDSet(incrementalFromPos string) (replication.Mysql56GTIDSet, error) { - pos, err := replication.DecodePositionDefaultFlavor(incrementalFromPos, replication.Mysql56FlavorID) + _, gtidSet, err := replication.DecodePositionMySQL56(incrementalFromPos) if err != nil { return nil, vterrors.Wrapf(err, "cannot decode position in incremental backup: %v", incrementalFromPos) } - if !pos.MatchesFlavor(replication.Mysql56FlavorID) { - return nil, vterrors.Errorf(vtrpc.Code_FAILED_PRECONDITION, "incremental backup only supports MySQL GTID positions. Got: %v", incrementalFromPos) - } - ifPosGTIDSet, ok := pos.GTIDSet.(replication.Mysql56GTIDSet) - if !ok { - return nil, vterrors.Errorf(vtrpc.Code_FAILED_PRECONDITION, "cannot get MySQL GTID value: %v", pos) - } - return ifPosGTIDSet, nil + return gtidSet, nil } // executeIncrementalBackup runs an incremental backup, based on given 'incremental_from_pos', which can be: @@ -269,7 +262,7 @@ func (be *BuiltinBackupEngine) executeIncrementalBackup(ctx context.Context, par params.Logger.Infof("auto evaluated incremental_from_pos: %s", params.IncrementalFromPos) } - if _, err := replication.DecodePositionDefaultFlavor(params.IncrementalFromPos, replication.Mysql56FlavorID); err != nil { + if _, _, err := replication.DecodePositionMySQL56(params.IncrementalFromPos); err != nil { // This does not seem to be a valid position. Maybe it's a backup name? backupName := params.IncrementalFromPos pos, err := findBackupPosition(ctx, params, backupName) diff --git a/go/vt/vttablet/tabletmanager/restore.go b/go/vt/vttablet/tabletmanager/restore.go index 5f57fc9a593..3e4b4a89555 100644 --- a/go/vt/vttablet/tabletmanager/restore.go +++ b/go/vt/vttablet/tabletmanager/restore.go @@ -36,7 +36,6 @@ import ( "vitess.io/vitess/go/vt/logutil" "vitess.io/vitess/go/vt/mysqlctl" "vitess.io/vitess/go/vt/mysqlctl/backupstats" - "vitess.io/vitess/go/vt/proto/vtrpc" "vitess.io/vitess/go/vt/proto/vttime" "vitess.io/vitess/go/vt/servenv" "vitess.io/vitess/go/vt/topo" @@ -238,16 +237,10 @@ func (tm *TabletManager) restoreDataLocked(ctx context.Context, logger logutil.L return vterrors.Errorf(vtrpcpb.Code_INVALID_ARGUMENT, "--restore-to-pos and --restore-to-timestamp are mutually exclusive") } if request.RestoreToPos != "" { - pos, err := replication.DecodePositionDefaultFlavor(request.RestoreToPos, replication.Mysql56FlavorID) + pos, _, err := replication.DecodePositionMySQL56(request.RestoreToPos) if err != nil { return vterrors.Wrapf(err, "restore failed: unable to decode --restore-to-pos: %s", request.RestoreToPos) } - if !pos.MatchesFlavor(replication.Mysql56FlavorID) { - return vterrors.Errorf(vtrpc.Code_FAILED_PRECONDITION, "restore to position only supports MySQL GTID positions. Got: %v", request.RestoreToPos) - } - if _, ok := pos.GTIDSet.(replication.Mysql56GTIDSet); !ok { - return vterrors.Errorf(vtrpc.Code_FAILED_PRECONDITION, "cannot get MySQL GTID value: %v", pos) - } params.RestoreToPos = pos } if !restoreToTimestamp.IsZero() { From a88dd2063ac464436106dee166afee0a47aa8fdf Mon Sep 17 00:00:00 2001 From: Shlomi Noach <2607934+shlomi-noach@users.noreply.github.com> Date: Tue, 2 Apr 2024 07:45:41 +0300 Subject: [PATCH 5/6] rewording Signed-off-by: Shlomi Noach <2607934+shlomi-noach@users.noreply.github.com> --- go/mysql/replication/mysql56_gtid.go | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/go/mysql/replication/mysql56_gtid.go b/go/mysql/replication/mysql56_gtid.go index 48d199e4360..dd23fb2092b 100644 --- a/go/mysql/replication/mysql56_gtid.go +++ b/go/mysql/replication/mysql56_gtid.go @@ -30,7 +30,7 @@ import ( const Mysql56FlavorID = "MySQL56" var ( - ErrExpectMysql56Flavor = vterrors.Errorf(vtrpc.Code_INVALID_ARGUMENT, "Expecting MySQL56 position flavor") + ErrExpectMysql56Flavor = vterrors.Errorf(vtrpc.Code_INVALID_ARGUMENT, "expected MySQL GTID position but found a different or invalid format.") ) // parseMysql56GTID is registered as a GTID parser. From 043decabf4dda8a568b261f89b4bdaa33b9a45f3 Mon Sep 17 00:00:00 2001 From: Shlomi Noach <2607934+shlomi-noach@users.noreply.github.com> Date: Wed, 3 Apr 2024 09:51:39 +0300 Subject: [PATCH 6/6] Validate GTID value Signed-off-by: Shlomi Noach <2607934+shlomi-noach@users.noreply.github.com> --- go/mysql/replication/mysql56_gtid_test.go | 14 ++++++++++++-- 1 file changed, 12 insertions(+), 2 deletions(-) diff --git a/go/mysql/replication/mysql56_gtid_test.go b/go/mysql/replication/mysql56_gtid_test.go index 6db23cc48bc..a8bffed72b9 100644 --- a/go/mysql/replication/mysql56_gtid_test.go +++ b/go/mysql/replication/mysql56_gtid_test.go @@ -162,16 +162,26 @@ func TestDecodePositionMySQL56(t *testing.T) { assert.Nil(t, gtidSet) } { - pos, gtidSet, err := DecodePositionMySQL56("MySQL56/16b1039f-22b6-11ed-b765-0a43f95f28a3:1-615") + pos, gtidSet, err := DecodePositionMySQL56("MySQL56/00010203-0405-0607-0809-0A0B0C0D0E0F:1-615") assert.NoError(t, err) assert.False(t, pos.IsZero()) assert.NotNil(t, gtidSet) + expectGTID := Mysql56GTIDSet{ + SID{ + 0x0, 0x1, 0x2, 0x3, 0x4, 0x5, 0x6, 0x7, 0x8, 0x9, 0xa, 0xb, 0xc, 0xd, 0xe, 0xf, + }: []interval{{start: 1, end: 615}}} + assert.Equal(t, expectGTID, gtidSet) } { - pos, gtidSet, err := DecodePositionMySQL56("16b1039f-22b6-11ed-b765-0a43f95f28a3:1-615") + pos, gtidSet, err := DecodePositionMySQL56("00010203-0405-0607-0809-0A0B0C0D0E0F:1-615") assert.NoError(t, err) assert.False(t, pos.IsZero()) assert.NotNil(t, gtidSet) + expectGTID := Mysql56GTIDSet{ + SID{ + 0x0, 0x1, 0x2, 0x3, 0x4, 0x5, 0x6, 0x7, 0x8, 0x9, 0xa, 0xb, 0xc, 0xd, 0xe, 0xf, + }: []interval{{start: 1, end: 615}}} + assert.Equal(t, expectGTID, gtidSet) } { _, _, err := DecodePositionMySQL56("q-22b6-11ed-b765-0a43f95f28a3:1-615")