This repository has been archived by the owner on Jul 24, 2024. It is now read-only.
-
Notifications
You must be signed in to change notification settings - Fork 101
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
backup,restore: support backing up / restore system databases (#1048)
- Loading branch information
Showing
17 changed files
with
307 additions
and
11 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,185 @@ | ||
// Copyright 2020 PingCAP, Inc. Licensed under Apache-2.0. | ||
|
||
package restore | ||
|
||
import ( | ||
"context" | ||
"fmt" | ||
|
||
"github.com/pingcap/errors" | ||
"github.com/pingcap/log" | ||
"github.com/pingcap/parser/model" | ||
"github.com/pingcap/parser/mysql" | ||
filter "github.com/pingcap/tidb-tools/pkg/table-filter" | ||
"go.uber.org/multierr" | ||
"go.uber.org/zap" | ||
|
||
berrors "github.com/pingcap/br/pkg/errors" | ||
"github.com/pingcap/br/pkg/logutil" | ||
"github.com/pingcap/br/pkg/utils" | ||
) | ||
|
||
var statsTables = map[string]struct{}{ | ||
"stats_buckets": {}, | ||
"stats_extended": {}, | ||
"stats_feedback": {}, | ||
"stats_fm_sketch": {}, | ||
"stats_histograms": {}, | ||
"stats_meta": {}, | ||
"stats_top_n": {}, | ||
} | ||
|
||
func isStatsTable(tableName string) bool { | ||
_, ok := statsTables[tableName] | ||
return ok | ||
} | ||
|
||
// RestoreSystemSchemas restores the system schema(i.e. the `mysql` schema). | ||
// Detail see https://github.com/pingcap/br/issues/679#issuecomment-762592254. | ||
func (rc *Client) RestoreSystemSchemas(ctx context.Context, f filter.Filter) { | ||
sysDB := mysql.SystemDB | ||
|
||
temporaryDB := utils.TemporaryDBName(sysDB) | ||
defer rc.cleanTemporaryDatabase(ctx, sysDB) | ||
|
||
if !f.MatchSchema(temporaryDB.O) { | ||
log.Debug("system database filtered out", zap.String("database", sysDB)) | ||
return | ||
} | ||
originDatabase, ok := rc.databases[temporaryDB.O] | ||
if !ok { | ||
log.Info("system database not backed up, skipping", zap.String("database", sysDB)) | ||
return | ||
} | ||
db, ok := rc.getDatabaseByName(sysDB) | ||
if !ok { | ||
// Or should we create the database here? | ||
log.Warn("target database not exist, aborting", zap.String("database", sysDB)) | ||
return | ||
} | ||
|
||
tablesRestored := make([]string, 0, len(originDatabase.Tables)) | ||
for _, table := range originDatabase.Tables { | ||
tableName := table.Info.Name | ||
if f.MatchTable(sysDB, tableName.O) { | ||
if err := rc.replaceTemporaryTableToSystable(ctx, tableName.L, db); err != nil { | ||
logutil.WarnTerm("error during merging temporary tables into system tables", | ||
logutil.ShortError(err), | ||
zap.Stringer("table", tableName), | ||
) | ||
} | ||
} | ||
tablesRestored = append(tablesRestored, tableName.L) | ||
} | ||
if err := rc.afterSystemTablesReplaced(ctx, tablesRestored); err != nil { | ||
for _, e := range multierr.Errors(err) { | ||
logutil.WarnTerm("error during reconfigurating the system tables", zap.String("database", sysDB), logutil.ShortError(e)) | ||
} | ||
} | ||
} | ||
|
||
// database is a record of a database. | ||
// For fast querying whether a table exists and the temporary database of it. | ||
type database struct { | ||
ExistingTables map[string]*model.TableInfo | ||
Name model.CIStr | ||
TemporaryName model.CIStr | ||
} | ||
|
||
// getDatabaseByName make a record of a database from info schema by its name. | ||
func (rc *Client) getDatabaseByName(name string) (*database, bool) { | ||
infoSchema := rc.dom.InfoSchema() | ||
schema, ok := infoSchema.SchemaByName(model.NewCIStr(name)) | ||
if !ok { | ||
return nil, false | ||
} | ||
db := &database{ | ||
ExistingTables: map[string]*model.TableInfo{}, | ||
Name: model.NewCIStr(name), | ||
TemporaryName: utils.TemporaryDBName(name), | ||
} | ||
for _, t := range schema.Tables { | ||
db.ExistingTables[t.Name.L] = t | ||
} | ||
return db, true | ||
} | ||
|
||
// afterSystemTablesReplaced do some extra work for special system tables. | ||
// e.g. after inserting to the table mysql.user, we must execute `FLUSH PRIVILEGES` to allow it take effect. | ||
func (rc *Client) afterSystemTablesReplaced(ctx context.Context, tables []string) error { | ||
var err error | ||
for _, table := range tables { | ||
switch { | ||
case table == "user": | ||
// We cannot execute `rc.dom.NotifyUpdatePrivilege` here, because there isn't | ||
// sessionctx.Context provided by the glue. | ||
// TODO: update the glue type and allow we retrive a session context from it. | ||
err = multierr.Append(err, errors.Annotatef(berrors.ErrUnsupportedSystemTable, | ||
"restored user info may not take effect, until you should execute `FLUSH PRIVILEGES` manually")) | ||
} | ||
} | ||
return err | ||
} | ||
|
||
// replaceTemporaryTableToSystable replaces the temporary table to real system table. | ||
func (rc *Client) replaceTemporaryTableToSystable(ctx context.Context, tableName string, db *database) error { | ||
execSQL := func(sql string) error { | ||
// SQLs here only contain table name and database name, seems it is no need to redact them. | ||
if err := rc.db.se.Execute(ctx, sql); err != nil { | ||
log.Warn("failed to execute SQL restore system database", | ||
zap.String("table", tableName), | ||
zap.Stringer("database", db.Name), | ||
zap.String("sql", sql), | ||
zap.Error(err), | ||
) | ||
return berrors.ErrUnknown.Wrap(err).GenWithStack("failed to execute %s", sql) | ||
} | ||
log.Info("successfully restore system database", | ||
zap.String("table", tableName), | ||
zap.Stringer("database", db.Name), | ||
zap.String("sql", sql), | ||
) | ||
return nil | ||
} | ||
|
||
// The newly created tables have different table IDs with original tables, | ||
// hence the old statistics are invalid. | ||
// | ||
// TODO: | ||
// 1 ) Rewrite the table IDs via `UPDATE _temporary_mysql.stats_xxx SET table_id = new_table_id WHERE table_id = old_table_id` | ||
// BEFORE replacing into and then execute `rc.statsHandler.Update(rc.dom.InfoSchema())`. | ||
// 1.5 ) (Optional) The UPDATE statement sometimes costs, the whole system tables restore step can be place into the restore pipeline. | ||
// 2 ) Deprecate the origin interface for backing up statistics. | ||
if isStatsTable(tableName) { | ||
return berrors.ErrUnsupportedSystemTable.GenWithStack("restoring stats via `mysql` schema isn't support yet: " + | ||
"the table ID is out-of-date and may corrupt existing statistics") | ||
} | ||
|
||
if db.ExistingTables[tableName] != nil { | ||
log.Info("table existing, using replace into for restore", | ||
zap.String("table", tableName), | ||
zap.Stringer("schema", db.Name)) | ||
replaceIntoSQL := fmt.Sprintf("REPLACE INTO %s SELECT * FROM %s;", | ||
utils.EncloseDBAndTable(db.Name.L, tableName), | ||
utils.EncloseDBAndTable(db.TemporaryName.L, tableName)) | ||
return execSQL(replaceIntoSQL) | ||
} | ||
|
||
renameSQL := fmt.Sprintf("RENAME TABLE %s TO %s;", | ||
utils.EncloseDBAndTable(db.TemporaryName.L, tableName), | ||
utils.EncloseDBAndTable(db.Name.L, tableName), | ||
) | ||
return execSQL(renameSQL) | ||
} | ||
|
||
func (rc *Client) cleanTemporaryDatabase(ctx context.Context, originDB string) { | ||
database := utils.TemporaryDBName(originDB) | ||
log.Debug("dropping temporary database", zap.Stringer("database", database)) | ||
sql := fmt.Sprintf("DROP DATABASE IF EXISTS %s", utils.EncloseName(database.L)) | ||
if err := rc.db.se.Execute(ctx, sql); err != nil { | ||
logutil.WarnTerm("failed to drop temporary database, it should be dropped manually", | ||
zap.Stringer("database", database), | ||
logutil.ShortError(err), | ||
) | ||
} | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Oops, something went wrong.