Skip to content
This repository has been archived by the owner on Nov 24, 2023. It is now read-only.

feat: add handle-error command #850

Merged
merged 46 commits into from
Aug 13, 2020
Merged
Show file tree
Hide file tree
Changes from 10 commits
Commits
Show all changes
46 commits
Select commit Hold shift + click to select a range
16e24e2
remove handle-sql
GMHDBJD Jul 30, 2020
33a9be6
add grpc method
GMHDBJD Jul 31, 2020
db6983b
Merge branch 'master' into errorMechanism
lance6716 Jul 31, 2020
fab914f
return error
GMHDBJD Jul 31, 2020
d184682
revert remove errors
GMHDBJD Jul 31, 2020
f66c734
fix fmt
GMHDBJD Aug 1, 2020
5dae4b9
record error location
GMHDBJD Aug 1, 2020
540020d
fix test
GMHDBJD Aug 2, 2020
80a2dd4
Merge branch 'master' into errorMechanism
GMHDBJD Aug 3, 2020
cdf10bf
support handle-error
GMHDBJD Aug 3, 2020
e3251ba
Merge remote-tracking branch 'upstream/master' into errorMechanism
GMHDBJD Aug 4, 2020
3e04891
address comment
GMHDBJD Aug 4, 2020
3454208
fix bugs
GMHDBJD Aug 5, 2020
694055b
add integration test
GMHDBJD Aug 5, 2020
4a8323c
Merge remote-tracking branch 'upstream/master' into errorMechanism
GMHDBJD Aug 5, 2020
562b80a
minor update
GMHDBJD Aug 5, 2020
9fffe3a
Merge remote-tracking branch 'upstream/master' into errorMechanism
GMHDBJD Aug 5, 2020
229f825
fix ut
GMHDBJD Aug 5, 2020
84561a4
format comment
GMHDBJD Aug 5, 2020
1de3af6
minor fix
GMHDBJD Aug 5, 2020
4e710d4
Merge remote-tracking branch 'upstream/master' into errorMechanism
GMHDBJD Aug 6, 2020
72ab1d0
add handle-error revert
GMHDBJD Aug 6, 2020
145a366
Merge branch 'master' into errorMechanism
GMHDBJD Aug 6, 2020
b1c7d8c
remove outdated operator
GMHDBJD Aug 6, 2020
d31fb71
add operator test
GMHDBJD Aug 6, 2020
934faab
fix test
GMHDBJD Aug 6, 2020
fce824b
Merge branch 'master' into errorMechanism
GMHDBJD Aug 6, 2020
5290ea2
address comment
GMHDBJD Aug 7, 2020
7673386
fix test
GMHDBJD Aug 7, 2020
9998b83
Merge branch 'master' into errorMechanism
GMHDBJD Aug 7, 2020
81d6e61
Merge branch 'master' into errorMechanism
GMHDBJD Aug 10, 2020
817354a
address comment
GMHDBJD Aug 11, 2020
d6e3adb
Merge remote-tracking branch 'upstream/master' into errorMechanism
GMHDBJD Aug 11, 2020
283cb32
address comment and fix test
GMHDBJD Aug 11, 2020
e045098
Merge remote-tracking branch 'upstream/master' into errorMechanism
GMHDBJD Aug 11, 2020
4c6d0ee
address some comment
GMHDBJD Aug 11, 2020
1d07b9c
remove msg
GMHDBJD Aug 11, 2020
9ce4c9b
address comment
GMHDBJD Aug 11, 2020
92d933c
address comment
GMHDBJD Aug 12, 2020
57bb76d
Merge branch 'master' into errorMechanism
GMHDBJD Aug 12, 2020
0018b5d
address comment
GMHDBJD Aug 12, 2020
b8403ec
update ut
GMHDBJD Aug 12, 2020
d40872a
Merge branch 'master' into errorMechanism
GMHDBJD Aug 12, 2020
567393f
Merge remote-tracking branch 'upstream/master' into errorMechanism
GMHDBJD Aug 13, 2020
eaef36c
address comment
GMHDBJD Aug 13, 2020
ccb3d91
better variable and func name
GMHDBJD Aug 13, 2020
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
2 changes: 2 additions & 0 deletions _utils/terror_gen/errors_release.txt
Original file line number Diff line number Diff line change
Expand Up @@ -118,6 +118,7 @@ ErrShardDDLOptimismTrySyncFail,[code=11111:class=functional:scope=internal:level
ErrConnInvalidTLSConfig,[code=11112:class=functional:scope=internal:level=medium], "Message: invalid TLS config, Workaround: Please check the `ssl-ca`, `ssl-cert` and `ssl-key` config."
ErrConnRegistryTLSConfig,[code=11113:class=functional:scope=internal:level=medium], "Message: fail to registry TLS config"
ErrUpgradeVersionEtcdFail,[code=11114:class=functional:scope=internal:level=high], "Message: fail to operate DM cluster version in etcd, Workaround: Please use `list-member --master` to confirm whether the DM-master cluster is healthy"
ErrVerifyHandleErrorArgs,[code=11115:class=functional:scope=internal:level=low], "Workaround: Please make sure the args are correct."
ErrConfigCheckItemNotSupport,[code=20001:class=config:scope=internal:level=medium], "Message: checking item %s is not supported\n%s, Workaround: Please check `ignore-checking-items` config in task configuration file, which can be set including `all`/`dump_privilege`/`replication_privilege`/`version`/`binlog_enable`/`binlog_format`/`binlog_row_image`/`table_schema`/`schema_of_shard_tables`/`auto_increment_ID`."
ErrConfigTomlTransform,[code=20002:class=config:scope=internal:level=medium], "Message: %s, Workaround: Please check the configuration file has correct TOML format."
ErrConfigYamlTransform,[code=20003:class=config:scope=internal:level=medium], "Message: %s, Workaround: Please check the configuration file has correct YAML format."
Expand Down Expand Up @@ -299,6 +300,7 @@ ErrSyncerUnitGenBAList,[code=36060:class=sync-unit:scope=internal:level=high], "
ErrSyncerUnitHandleDDLFailed,[code=36061:class=sync-unit:scope=internal:level=high], "Message: fail to handle ddl job for %s"
ErrSyncerShardDDLConflict,[code=36062:class=sync-unit:scope=internal:level=high], "Message: fail to handle shard ddl %v in optimistic mode, because schema conflict detected, Workaround: Please use show-ddl-locks command for more details."
ErrSyncerFailpoint,[code=36063:class=sync-unit:scope=internal:level=low], "Message: failpoint specified error"
ErrSyncerReplaceEvent,[code=36064:class=sync-unit:scope=internal:level=high]
ErrMasterSQLOpNilRequest,[code=38001:class=dm-master:scope=internal:level=medium], "Message: nil request not valid"
ErrMasterSQLOpNotSupport,[code=38002:class=dm-master:scope=internal:level=medium], "Message: op %s not supported"
ErrMasterSQLOpWithoutSharding,[code=38003:class=dm-master:scope=internal:level=medium], "Message: operate request without --sharding specified not valid"
Expand Down
52 changes: 7 additions & 45 deletions dm/command/argument.go
Original file line number Diff line number Diff line change
Expand Up @@ -14,9 +14,6 @@
package command

import (
"regexp"
"strings"

"github.com/siddontang/go-mysql/mysql"

"github.com/pingcap/dm/pkg/binlog"
Expand All @@ -32,47 +29,12 @@ func TrimQuoteMark(s string) string {
return s
}

// VerifySQLOperateArgs verify args for sql operation, including sql-skip, sql-replace.
// NOTE: only part of args verified in here and others need to be verified from outer.
func VerifySQLOperateArgs(binlogPosStr, sqlPattern string, sharding bool) (*mysql.Position, *regexp.Regexp, error) {
binlogPosStr = TrimQuoteMark(binlogPosStr)
sqlPattern = TrimQuoteMark(sqlPattern)

var (
pos *mysql.Position
reg *regexp.Regexp
)

if len(binlogPosStr) > 0 && len(sqlPattern) > 0 {
return nil, nil, terror.ErrVerifySQLOperateArgs.New("cannot specify both --binlog-pos and --sql-pattern in sql operation")
// VerifyBinlogPos verify binlog pos string
func VerifyBinlogPos(pos string) (*mysql.Position, error) {
binlogPosStr := TrimQuoteMark(pos)
pos2, err := binlog.PositionFromStr(binlogPosStr)
if err != nil {
return nil, terror.ErrVerifyHandleErrorArgs.Generatef("invalid --binlog-pos %s in handle-error operation: %s", binlogPosStr, terror.Message(err))
}

if len(binlogPosStr) > 0 {
pos2, err := binlog.PositionFromStr(binlogPosStr)
if err != nil {
return nil, nil, terror.ErrVerifySQLOperateArgs.Generatef("invalid --binlog-pos %s in sql operation: %s", binlogPosStr, terror.Message(err))
}
pos = &pos2
} else if len(sqlPattern) > 0 {
var pattern string
if strings.HasPrefix(sqlPattern, "~") {
pattern = sqlPattern[1:]
} else {
pattern = "^" + regexp.QuoteMeta(sqlPattern) + "$"
}

var err error
reg, err = regexp.Compile(pattern)
if err != nil {
return nil, nil, terror.ErrVerifySQLOperateArgs.AnnotateDelegate(err, "invalid --sql-pattern %s in sql operation", sqlPattern)
}
} else {
return nil, nil, terror.ErrVerifySQLOperateArgs.New("must specify one of --binlog-pos and --sql-pattern in sql operation")
}

if sharding && len(binlogPosStr) > 0 {
return nil, nil, terror.ErrVerifySQLOperateArgs.New("cannot specify --binlog-pos with --sharding in sql operation")
}

return pos, reg, nil
return &pos2, nil
}
131 changes: 0 additions & 131 deletions dm/command/argument_test.go

This file was deleted.

2 changes: 2 additions & 0 deletions dm/ctl/common/util.go
Original file line number Diff line number Diff line change
Expand Up @@ -20,6 +20,7 @@ import (
"strings"
"time"

"github.com/pingcap/dm/dm/command"
"github.com/pingcap/dm/dm/config"
"github.com/pingcap/dm/dm/pb"
parserpkg "github.com/pingcap/dm/pkg/parser"
Expand Down Expand Up @@ -189,6 +190,7 @@ func ExtractSQLsFromArgs(args []string) ([]string, error) {
}

concat := strings.TrimSpace(strings.Join(args, " "))
concat = command.TrimQuoteMark(concat)

parser := parser.New()
nodes, err := parserpkg.Parse(parser, concat, "", "")
Expand Down
4 changes: 1 addition & 3 deletions dm/ctl/ctl.go
Original file line number Diff line number Diff line change
Expand Up @@ -63,9 +63,6 @@ func NewRootCmd() *cobra.Command {
master.NewUpdateTaskCmd(),
master.NewQueryStatusCmd(),
master.NewQueryErrorCmd(),
master.NewSQLReplaceCmd(),
master.NewSQLSkipCmd(),
master.NewSQLInjectCmd(),
master.NewShowDDLLocksCmd(),
master.NewUnlockDDLLockCmd(),
master.NewSwitchRelayMasterCmd(),
Expand All @@ -81,6 +78,7 @@ func NewRootCmd() *cobra.Command {
master.NewListMemberCmd(),
master.NewOperateSchemaCmd(),
master.NewGetTaskCfgCmd(),
master.NewHandleErrorCmd(),
)
return cmd
}
Expand Down
119 changes: 119 additions & 0 deletions dm/ctl/master/handle_error.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,119 @@
// Copyright 2020 PingCAP, Inc.
//
// 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,
// See the License for the specific language governing permissions and
// limitations under the License.

package master

import (
"context"
"errors"
"os"

"github.com/spf13/cobra"

"github.com/pingcap/dm/dm/command"
"github.com/pingcap/dm/dm/ctl/common"
"github.com/pingcap/dm/dm/pb"
)

// NewHandleErrorCmd creates a HandleError command
func NewHandleErrorCmd() *cobra.Command {
cmd := &cobra.Command{
Use: "handle-error <task-name> [-s source ...] [-b binlog-pos] <skip/replace> [replace-sql1;replace-sql2;]",
Short: "skip/replace the current error event or a specific binlog position (binlog-pos) event",
RunE: handleErrorFunc,
}
cmd.Flags().StringP("binlog-pos", "b", "", "position used to match binlog event if matched the handler-error operation will be applied. The format like \"mysql-bin|000001.000003:3270\"")
return cmd
}

func convertOp(t string) pb.ErrorOp {
switch t {
case "skip":
return pb.ErrorOp_Skip
case "replace":
return pb.ErrorOp_Replace
default:
return pb.ErrorOp_InvalidErrorOp
}
}

// handleErrorFunc does handle error request
func handleErrorFunc(cmd *cobra.Command, _ []string) (err error) {
if len(cmd.Flags().Args()) < 2 {
cmd.SetOut(os.Stdout)
cmd.Usage()
err = errors.New("please check output to see error")
return
}

taskName := cmd.Flags().Arg(0)
operation := cmd.Flags().Arg(1)
var sqls []string

op := convertOp(operation)
switch op {
case pb.ErrorOp_Skip:
if len(cmd.Flags().Args()) > 2 {
common.PrintLines("replace-sqls will be ignored for 'skip' operation")
}
case pb.ErrorOp_Replace:
if len(cmd.Flags().Args()) <= 2 {
common.PrintLines("must specify the replace-sqls for replace operation")
err = errors.New("please check output to see error")
return
}

sqls, err = common.ExtractSQLsFromArgs(cmd.Flags().Args()[2:])
if err != nil {
return
}
default:
common.PrintLines("invalid operation '%s', please use `skip` or `relpace`", operation)
err = errors.New("please check output to see error")
return
}

binlogPos, err := cmd.Flags().GetString("binlog-pos")
if err != nil {
return
}
if len(binlogPos) != 0 {
_, err = command.VerifyBinlogPos(binlogPos)
if err != nil {
return
}
}

sources, err := common.GetSourceArgs(cmd)
if err != nil {
return
}

ctx, cancel := context.WithCancel(context.Background())
defer cancel()
cli := common.MasterClient()

resp, err := cli.HandleError(ctx, &pb.HandleErrorRequest{
Op: op,
Task: taskName,
BinlogPos: binlogPos,
Sqls: sqls,
Sources: sources,
})
if err != nil {
return
}

common.PrettyPrintResponse(resp)
return
}
Loading