mirror of
https://github.com/restic/restic.git
synced 2025-12-08 06:09:56 +00:00
197 lines
6.3 KiB
Go
197 lines
6.3 KiB
Go
package main
|
|
|
|
import (
|
|
"context"
|
|
"fmt"
|
|
"path/filepath"
|
|
"testing"
|
|
|
|
"github.com/restic/restic/internal/global"
|
|
"github.com/restic/restic/internal/restic"
|
|
rtest "github.com/restic/restic/internal/test"
|
|
"github.com/restic/restic/internal/ui"
|
|
)
|
|
|
|
func testRunCopy(t testing.TB, srcGopts global.Options, dstGopts global.Options) {
|
|
gopts := srcGopts
|
|
gopts.Repo = dstGopts.Repo
|
|
gopts.Password = dstGopts.Password
|
|
gopts.InsecureNoPassword = dstGopts.InsecureNoPassword
|
|
copyOpts := CopyOptions{
|
|
SecondaryRepoOptions: global.SecondaryRepoOptions{
|
|
Repo: srcGopts.Repo,
|
|
Password: srcGopts.Password,
|
|
InsecureNoPassword: srcGopts.InsecureNoPassword,
|
|
},
|
|
}
|
|
|
|
rtest.OK(t, withTermStatus(t, gopts, func(ctx context.Context, gopts global.Options) error {
|
|
return runCopy(context.TODO(), copyOpts, gopts, nil, gopts.Term)
|
|
}))
|
|
}
|
|
|
|
func TestCopy(t *testing.T) {
|
|
env, cleanup := withTestEnvironment(t)
|
|
defer cleanup()
|
|
env2, cleanup2 := withTestEnvironment(t)
|
|
defer cleanup2()
|
|
|
|
testSetupBackupData(t, env)
|
|
opts := BackupOptions{}
|
|
testRunBackup(t, "", []string{filepath.Join(env.testdata, "0", "0", "9")}, opts, env.gopts)
|
|
testRunBackup(t, "", []string{filepath.Join(env.testdata, "0", "0", "9", "2")}, opts, env.gopts)
|
|
testRunBackup(t, "", []string{filepath.Join(env.testdata, "0", "0", "9", "3")}, opts, env.gopts)
|
|
testRunCheck(t, env.gopts)
|
|
|
|
testRunInit(t, env2.gopts)
|
|
testRunCopy(t, env.gopts, env2.gopts)
|
|
|
|
snapshotIDs := testListSnapshots(t, env.gopts, 3)
|
|
copiedSnapshotIDs := testListSnapshots(t, env2.gopts, 3)
|
|
|
|
// Check that the copies size seems reasonable
|
|
stat := dirStats(t, env.repo)
|
|
stat2 := dirStats(t, env2.repo)
|
|
sizeDiff := int64(stat.size) - int64(stat2.size)
|
|
if sizeDiff < 0 {
|
|
sizeDiff = -sizeDiff
|
|
}
|
|
rtest.Assert(t, sizeDiff < int64(stat.size)/50, "expected less than 2%% size difference: %v vs. %v",
|
|
stat.size, stat2.size)
|
|
|
|
// Check integrity of the copy
|
|
testRunCheck(t, env2.gopts)
|
|
|
|
// Check that the copied snapshots have the same tree contents as the old ones (= identical tree hash)
|
|
origRestores := make(map[string]struct{})
|
|
for i, snapshotID := range snapshotIDs {
|
|
restoredir := filepath.Join(env.base, fmt.Sprintf("restore%d", i))
|
|
origRestores[restoredir] = struct{}{}
|
|
testRunRestore(t, env.gopts, restoredir, snapshotID.String())
|
|
}
|
|
for i, snapshotID := range copiedSnapshotIDs {
|
|
restoredir := filepath.Join(env2.base, fmt.Sprintf("restore%d", i))
|
|
testRunRestore(t, env2.gopts, restoredir, snapshotID.String())
|
|
foundMatch := false
|
|
for cmpdir := range origRestores {
|
|
diff := directoriesContentsDiff(t, restoredir, cmpdir)
|
|
if diff == "" {
|
|
delete(origRestores, cmpdir)
|
|
foundMatch = true
|
|
}
|
|
}
|
|
|
|
rtest.Assert(t, foundMatch, "found no counterpart for snapshot %v", snapshotID)
|
|
}
|
|
|
|
rtest.Assert(t, len(origRestores) == 0, "found not copied snapshots")
|
|
|
|
// check that snapshots were properly batched while copying
|
|
_, _, countBlobs := testPackAndBlobCounts(t, env.gopts)
|
|
countTreePacksDst, countDataPacksDst, countBlobsDst := testPackAndBlobCounts(t, env2.gopts)
|
|
|
|
rtest.Equals(t, countBlobs, countBlobsDst, "expected blob count in boths repos to be equal")
|
|
rtest.Equals(t, countTreePacksDst, 1, "expected 1 tree packfile")
|
|
rtest.Equals(t, countDataPacksDst, 1, "expected 1 data packfile")
|
|
}
|
|
|
|
func testPackAndBlobCounts(t testing.TB, gopts global.Options) (countTreePacks int, countDataPacks int, countBlobs int) {
|
|
rtest.OK(t, withTermStatus(t, gopts, func(ctx context.Context, gopts global.Options) error {
|
|
printer := ui.NewProgressPrinter(gopts.JSON, gopts.Verbosity, gopts.Term)
|
|
_, repo, unlock, err := openWithReadLock(ctx, gopts, false, printer)
|
|
rtest.OK(t, err)
|
|
defer unlock()
|
|
|
|
rtest.OK(t, repo.List(context.TODO(), restic.PackFile, func(id restic.ID, size int64) error {
|
|
blobs, _, err := repo.ListPack(context.TODO(), id, size)
|
|
rtest.OK(t, err)
|
|
rtest.Assert(t, len(blobs) > 0, "a packfile should contain at least one blob")
|
|
|
|
switch blobs[0].Type {
|
|
case restic.TreeBlob:
|
|
countTreePacks++
|
|
case restic.DataBlob:
|
|
countDataPacks++
|
|
}
|
|
countBlobs += len(blobs)
|
|
return nil
|
|
}))
|
|
return nil
|
|
}))
|
|
|
|
return countTreePacks, countDataPacks, countBlobs
|
|
}
|
|
|
|
func TestCopyIncremental(t *testing.T) {
|
|
env, cleanup := withTestEnvironment(t)
|
|
defer cleanup()
|
|
env2, cleanup2 := withTestEnvironment(t)
|
|
defer cleanup2()
|
|
|
|
testSetupBackupData(t, env)
|
|
opts := BackupOptions{}
|
|
testRunBackup(t, "", []string{filepath.Join(env.testdata, "0", "0", "9")}, opts, env.gopts)
|
|
testRunBackup(t, "", []string{filepath.Join(env.testdata, "0", "0", "9", "2")}, opts, env.gopts)
|
|
testRunCheck(t, env.gopts)
|
|
|
|
testRunInit(t, env2.gopts)
|
|
testRunCopy(t, env.gopts, env2.gopts)
|
|
|
|
testListSnapshots(t, env.gopts, 2)
|
|
testListSnapshots(t, env2.gopts, 2)
|
|
|
|
// Check that the copies size seems reasonable
|
|
testRunCheck(t, env2.gopts)
|
|
|
|
// check that no snapshots are copied, as there are no new ones
|
|
testRunCopy(t, env.gopts, env2.gopts)
|
|
testRunCheck(t, env2.gopts)
|
|
testListSnapshots(t, env2.gopts, 2)
|
|
|
|
// check that only new snapshots are copied
|
|
testRunBackup(t, "", []string{filepath.Join(env.testdata, "0", "0", "9", "3")}, opts, env.gopts)
|
|
testRunCopy(t, env.gopts, env2.gopts)
|
|
testRunCheck(t, env2.gopts)
|
|
testListSnapshots(t, env.gopts, 3)
|
|
testListSnapshots(t, env2.gopts, 3)
|
|
|
|
// also test the reverse direction
|
|
testRunCopy(t, env2.gopts, env.gopts)
|
|
testRunCheck(t, env.gopts)
|
|
testListSnapshots(t, env.gopts, 3)
|
|
}
|
|
|
|
func TestCopyUnstableJSON(t *testing.T) {
|
|
env, cleanup := withTestEnvironment(t)
|
|
defer cleanup()
|
|
env2, cleanup2 := withTestEnvironment(t)
|
|
defer cleanup2()
|
|
|
|
// contains a symlink created using `ln -s '../i/'$'\355\246\361''d/samba' broken-symlink`
|
|
datafile := filepath.Join("testdata", "copy-unstable-json.tar.gz")
|
|
rtest.SetupTarTestFixture(t, env.base, datafile)
|
|
|
|
testRunInit(t, env2.gopts)
|
|
testRunCopy(t, env.gopts, env2.gopts)
|
|
testRunCheck(t, env2.gopts)
|
|
testListSnapshots(t, env2.gopts, 1)
|
|
}
|
|
|
|
func TestCopyToEmptyPassword(t *testing.T) {
|
|
env, cleanup := withTestEnvironment(t)
|
|
defer cleanup()
|
|
env2, cleanup2 := withTestEnvironment(t)
|
|
defer cleanup2()
|
|
env2.gopts.Password = ""
|
|
env2.gopts.InsecureNoPassword = true
|
|
|
|
testSetupBackupData(t, env)
|
|
testRunBackup(t, "", []string{filepath.Join(env.testdata, "0", "0", "9")}, BackupOptions{}, env.gopts)
|
|
|
|
testRunInit(t, env2.gopts)
|
|
testRunCopy(t, env.gopts, env2.gopts)
|
|
|
|
testListSnapshots(t, env.gopts, 1)
|
|
testListSnapshots(t, env2.gopts, 1)
|
|
testRunCheck(t, env2.gopts)
|
|
}
|