Skip to content

Commit

Permalink
fix possible create backup failures during UNFREEZE not exists tables…
Browse files Browse the repository at this point in the history
…, affected 2.2.7+ version, fix #704, also improve integration_test.go which cover this feature.
  • Loading branch information
Slach committed Jul 25, 2023
1 parent b387f9a commit a0c1439
Show file tree
Hide file tree
Showing 3 changed files with 43 additions and 12 deletions.
7 changes: 7 additions & 0 deletions ChangeLog.md
Original file line number Diff line number Diff line change
@@ -1,3 +1,10 @@
# v2.4.0 (not released yet)
IMPROVEMENTS
- first implementation for properly backup S3/GCS/Azure disks, support server-side copy to back up bucket during `clickhouse-backup` create and during `clickhouse-backup restore`, requires add `object_disk_path` to `s3`,`gcs`,`azblob` section, fix [447](https://github.com/Altinity/clickhouse-backup/issues/447)

BUG FIXES
- fix possible create backup failures during UNFREEZE not exists tables, affected 2.2.7+ version, fix [704](https://github.com/Altinity/clickhouse-backup/issues/704)

# v2.3.2
BUG FIXES
- fix error when `backups_to_keep_local: -1`, fix [698](https://github.com/Altinity/clickhouse-backup/issues/698)
Expand Down
7 changes: 6 additions & 1 deletion pkg/backup/create.go
Original file line number Diff line number Diff line change
Expand Up @@ -541,7 +541,12 @@ func (b *Backuper) AddTableToBackup(ctx context.Context, backupName, shadowBacku
// Unfreeze to unlock data on S3 disks, https://github.com/Altinity/clickhouse-backup/issues/423
if version > 21004000 {
if err := b.ch.QueryContext(ctx, fmt.Sprintf("ALTER TABLE `%s`.`%s` UNFREEZE WITH NAME '%s'", table.Database, table.Name, shadowBackupUUID)); err != nil {
return disksToPartsMap, realSize, err
if (strings.Contains(err.Error(), "code: 60") || strings.Contains(err.Error(), "code: 81")) && b.cfg.ClickHouse.IgnoreNotExistsErrorDuringFreeze {
b.ch.Log.Warnf("can't unfreeze table: %v", err)
} else {
return disksToPartsMap, realSize, err
}

}
}
if b.dst != nil {
Expand Down
41 changes: 30 additions & 11 deletions test/integration/integration_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -1383,7 +1383,7 @@ func TestTablePatterns(t *testing.T) {
}

func TestSkipNotExistsTable(t *testing.T) {
t.Skip("TestSkipNotExistsTable is flaky now, need more precise algorithm for pause calculation")
//t.Skip("TestSkipNotExistsTable is flaky now, need more precise algorithm for pause calculation")
ch := &TestClickHouse{}
r := require.New(t)
ch.connectWithWait(r, 0*time.Second, 1*time.Second)
Expand All @@ -1395,10 +1395,10 @@ func TestSkipNotExistsTable(t *testing.T) {
chVersion, err := ch.chbackend.GetVersion(context.Background())
r.NoError(err)

errorCaught := false
freezeErrorHandled := false
pauseChannel := make(chan int64)
resumeChannel := make(chan int64)

ch.chbackend.Config.LogSQLQueries = true
wg := sync.WaitGroup{}
wg.Add(1)
go func() {
Expand All @@ -1407,7 +1407,7 @@ func TestSkipNotExistsTable(t *testing.T) {
wg.Done()
}()
pause := int64(0)
pausePercent := int64(93)
// pausePercent := int64(90)
for i := 0; i < 100; i++ {
testBackupName := fmt.Sprintf("not_exists_%d", i)
err = ch.chbackend.Query(ifNotExistsCreateSQL)
Expand All @@ -1417,18 +1417,36 @@ func TestSkipNotExistsTable(t *testing.T) {
pauseChannel <- pause
startTime := time.Now()
out, err := dockerExecOut("clickhouse", "bash", "-ce", "LOG_LEVEL=debug clickhouse-backup create --table default.if_not_exists "+testBackupName)
pause = time.Since(startTime).Nanoseconds() * pausePercent / 100
log.Info(out)
if (err != nil && strings.Contains(out, "can't freeze")) || (err == nil && !strings.Contains(out, "can't freeze")) {
parseTime := func(line string) time.Time {
parsedTime, err := time.Parse("2006/01/02 15:04:05.999999", line[:26])
if err != nil {
r.Failf("%s, Error parsing time: %v", line, err)
}
return parsedTime
}
lines := strings.Split(out, "\n")
firstTime := parseTime(lines[0])
var freezeTime time.Time
for _, line := range lines {
if strings.Contains(line, "create_table_query") {
freezeTime = parseTime(line)
break
}
}
pause = (firstTime.Sub(startTime) + freezeTime.Sub(firstTime) + (10 * time.Microsecond)).Nanoseconds()
}
if err != nil {
if !strings.Contains(out, "no tables for backup") {
assert.NoError(t, err)
} else {
} /* else {
pausePercent += 1
}
} */
}

if strings.Contains(out, "warn") && strings.Contains(out, "can't freeze") && strings.Contains(out, "code: 60") && err == nil {
errorCaught = true
if strings.Contains(out, "code: 60") && err == nil {
freezeErrorHandled = true
<-resumeChannel
r.NoError(dockerExec("clickhouse", "clickhouse-backup", "delete", "local", testBackupName))
break
Expand All @@ -1448,16 +1466,17 @@ func TestSkipNotExistsTable(t *testing.T) {
}()
for pause := range pauseChannel {
if pause > 0 {
pauseStart := time.Now()
time.Sleep(time.Duration(pause) * time.Nanosecond)
log.Infof("pause=%s", time.Duration(pause).String())
log.Infof("pause=%s pauseStart=%s", time.Duration(pause).String(), pauseStart.String())
err = ch.chbackend.DropTable(clickhouse.Table{Database: "default", Name: "if_not_exists"}, ifNotExistsCreateSQL, "", false, chVersion)
r.NoError(err)
}
resumeChannel <- 1
}
}()
wg.Wait()
r.True(errorCaught)
r.True(freezeErrorHandled)
}

func TestProjections(t *testing.T) {
Expand Down

0 comments on commit a0c1439

Please sign in to comment.