forked from TrueCloudLab/restic
Add more integration tests
This commit is contained in:
parent
246fdb13f9
commit
a176b1b5a6
2 changed files with 217 additions and 77 deletions
|
@ -2,9 +2,13 @@ package main
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"fmt"
|
"fmt"
|
||||||
|
"io/ioutil"
|
||||||
"os"
|
"os"
|
||||||
"path/filepath"
|
"path/filepath"
|
||||||
"syscall"
|
"syscall"
|
||||||
|
"testing"
|
||||||
|
|
||||||
|
. "github.com/restic/restic/test"
|
||||||
)
|
)
|
||||||
|
|
||||||
type dirEntry struct {
|
type dirEntry struct {
|
||||||
|
@ -51,31 +55,33 @@ func walkDir(dir string) <-chan *dirEntry {
|
||||||
|
|
||||||
func (e *dirEntry) equals(other *dirEntry) bool {
|
func (e *dirEntry) equals(other *dirEntry) bool {
|
||||||
if e.path != other.path {
|
if e.path != other.path {
|
||||||
fmt.Printf("path does not match\n")
|
fmt.Fprintf(os.Stderr, "%v: path does not match\n", e.path)
|
||||||
return false
|
return false
|
||||||
}
|
}
|
||||||
|
|
||||||
if e.fi.Mode() != other.fi.Mode() {
|
if e.fi.Mode() != other.fi.Mode() {
|
||||||
fmt.Printf("mode does not match\n")
|
fmt.Fprintf(os.Stderr, "%v: mode does not match\n", e.path)
|
||||||
return false
|
return false
|
||||||
}
|
}
|
||||||
|
|
||||||
// if e.fi.ModTime() != other.fi.ModTime() {
|
if e.fi.ModTime() != other.fi.ModTime() {
|
||||||
// fmt.Printf("%s: ModTime does not match\n", e.path)
|
fmt.Fprintf(os.Stderr, "%v: ModTime does not match\n", e.path)
|
||||||
// // TODO: Fix ModTime for directories, return false
|
return false
|
||||||
// return true
|
}
|
||||||
// }
|
|
||||||
|
|
||||||
stat, _ := e.fi.Sys().(*syscall.Stat_t)
|
stat, _ := e.fi.Sys().(*syscall.Stat_t)
|
||||||
stat2, _ := other.fi.Sys().(*syscall.Stat_t)
|
stat2, _ := other.fi.Sys().(*syscall.Stat_t)
|
||||||
|
|
||||||
if stat.Uid != stat2.Uid || stat2.Gid != stat2.Gid {
|
if stat.Uid != stat2.Uid || stat2.Gid != stat2.Gid {
|
||||||
|
fmt.Fprintf(os.Stderr, "%v: UID or GID do not match\n", e.path)
|
||||||
return false
|
return false
|
||||||
}
|
}
|
||||||
|
|
||||||
return true
|
return true
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// directoriesEqualContents checks if both directories contain exactly the same
|
||||||
|
// contents.
|
||||||
func directoriesEqualContents(dir1, dir2 string) bool {
|
func directoriesEqualContents(dir1, dir2 string) bool {
|
||||||
ch1 := walkDir(dir1)
|
ch1 := walkDir(dir1)
|
||||||
ch2 := walkDir(dir2)
|
ch2 := walkDir(dir2)
|
||||||
|
@ -136,3 +142,83 @@ func directoriesEqualContents(dir1, dir2 string) bool {
|
||||||
|
|
||||||
return true
|
return true
|
||||||
}
|
}
|
||||||
|
|
||||||
|
type dirStat struct {
|
||||||
|
files, dirs, other uint
|
||||||
|
size uint64
|
||||||
|
}
|
||||||
|
|
||||||
|
func isFile(fi os.FileInfo) bool {
|
||||||
|
return fi.Mode()&(os.ModeType|os.ModeCharDevice) == 0
|
||||||
|
}
|
||||||
|
|
||||||
|
// dirStats walks dir and collects stats.
|
||||||
|
func dirStats(dir string) (stat dirStat) {
|
||||||
|
for entry := range walkDir(dir) {
|
||||||
|
if isFile(entry.fi) {
|
||||||
|
stat.files++
|
||||||
|
stat.size += uint64(entry.fi.Size())
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
if entry.fi.IsDir() {
|
||||||
|
stat.dirs++
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
stat.other++
|
||||||
|
}
|
||||||
|
|
||||||
|
return stat
|
||||||
|
}
|
||||||
|
|
||||||
|
type testEnvironment struct {
|
||||||
|
base, cache, repo, testdata string
|
||||||
|
}
|
||||||
|
|
||||||
|
func configureRestic(t testing.TB, cache, repo string) {
|
||||||
|
opts.CacheDir = cache
|
||||||
|
opts.Repo = repo
|
||||||
|
opts.Quiet = true
|
||||||
|
|
||||||
|
opts.password = TestPassword
|
||||||
|
}
|
||||||
|
|
||||||
|
func cleanupTempdir(t testing.TB, tempdir string) {
|
||||||
|
if !TestCleanup {
|
||||||
|
t.Logf("leaving temporary directory %v used for test", tempdir)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
OK(t, os.RemoveAll(tempdir))
|
||||||
|
}
|
||||||
|
|
||||||
|
// withTestEnvironment creates a test environment and calls f with it. After f has
|
||||||
|
// returned, the temporary directory is removed.
|
||||||
|
func withTestEnvironment(t testing.TB, f func(*testEnvironment)) {
|
||||||
|
if !RunIntegrationTest {
|
||||||
|
t.Skip("integration tests disabled")
|
||||||
|
}
|
||||||
|
|
||||||
|
tempdir, err := ioutil.TempDir(TestTempDir, "restic-test-")
|
||||||
|
OK(t, err)
|
||||||
|
|
||||||
|
env := testEnvironment{
|
||||||
|
base: tempdir,
|
||||||
|
cache: filepath.Join(tempdir, "cache"),
|
||||||
|
repo: filepath.Join(tempdir, "repo"),
|
||||||
|
testdata: filepath.Join(tempdir, "testdata"),
|
||||||
|
}
|
||||||
|
|
||||||
|
configureRestic(t, env.cache, env.repo)
|
||||||
|
OK(t, os.MkdirAll(env.testdata, 0700))
|
||||||
|
|
||||||
|
f(&env)
|
||||||
|
|
||||||
|
if !TestCleanup {
|
||||||
|
t.Logf("leaving temporary directory %v used for test", tempdir)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
OK(t, os.RemoveAll(tempdir))
|
||||||
|
}
|
||||||
|
|
|
@ -2,8 +2,9 @@ package main
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"bufio"
|
"bufio"
|
||||||
|
"crypto/rand"
|
||||||
|
"fmt"
|
||||||
"io"
|
"io"
|
||||||
"io/ioutil"
|
|
||||||
"os"
|
"os"
|
||||||
"os/exec"
|
"os/exec"
|
||||||
"path/filepath"
|
"path/filepath"
|
||||||
|
@ -13,30 +14,6 @@ import (
|
||||||
. "github.com/restic/restic/test"
|
. "github.com/restic/restic/test"
|
||||||
)
|
)
|
||||||
|
|
||||||
func setupTempdir(t testing.TB) (tempdir string) {
|
|
||||||
tempdir, err := ioutil.TempDir(TestTempDir, "restic-test-")
|
|
||||||
OK(t, err)
|
|
||||||
|
|
||||||
return tempdir
|
|
||||||
}
|
|
||||||
|
|
||||||
func configureRestic(t testing.TB, tempdir string) {
|
|
||||||
opts.CacheDir = filepath.Join(tempdir, "cache")
|
|
||||||
opts.Repo = filepath.Join(tempdir, "repo")
|
|
||||||
opts.Quiet = true
|
|
||||||
|
|
||||||
opts.password = TestPassword
|
|
||||||
}
|
|
||||||
|
|
||||||
func cleanupTempdir(t testing.TB, tempdir string) {
|
|
||||||
if !TestCleanup {
|
|
||||||
t.Logf("leaving temporary directory %v used for test", tempdir)
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
OK(t, os.RemoveAll(tempdir))
|
|
||||||
}
|
|
||||||
|
|
||||||
func setupTarTestFixture(t testing.TB, outputDir, tarFile string) {
|
func setupTarTestFixture(t testing.TB, outputDir, tarFile string) {
|
||||||
err := system("sh", "-c", `mkdir "$1" && (cd "$1" && tar xz) < "$2"`,
|
err := system("sh", "-c", `mkdir "$1" && (cd "$1" && tar xz) < "$2"`,
|
||||||
"sh", outputDir, tarFile)
|
"sh", outputDir, tarFile)
|
||||||
|
@ -85,7 +62,6 @@ func cmdBackup(t testing.TB, target []string, parentID backend.ID) {
|
||||||
}
|
}
|
||||||
|
|
||||||
func cmdList(t testing.TB, tpe string) []backend.ID {
|
func cmdList(t testing.TB, tpe string) []backend.ID {
|
||||||
|
|
||||||
rd, wr := io.Pipe()
|
rd, wr := io.Pipe()
|
||||||
|
|
||||||
cmd := &CmdList{w: wr}
|
cmd := &CmdList{w: wr}
|
||||||
|
@ -97,8 +73,6 @@ func cmdList(t testing.TB, tpe string) []backend.ID {
|
||||||
|
|
||||||
IDs := parseIDsFromReader(t, rd)
|
IDs := parseIDsFromReader(t, rd)
|
||||||
|
|
||||||
t.Logf("Listing %v: %v", tpe, IDs)
|
|
||||||
|
|
||||||
return IDs
|
return IDs
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -113,56 +87,136 @@ func cmdFsck(t testing.TB) {
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestBackup(t *testing.T) {
|
func TestBackup(t *testing.T) {
|
||||||
if !RunIntegrationTest {
|
withTestEnvironment(t, func(env *testEnvironment) {
|
||||||
t.Skip("integration tests disabled")
|
datafile := filepath.Join("testdata", "backup-data.tar.gz")
|
||||||
}
|
fd, err := os.Open(datafile)
|
||||||
|
if os.IsNotExist(err) {
|
||||||
|
t.Skipf("unable to find data file %q, skipping TestBackup", datafile)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
OK(t, err)
|
||||||
|
OK(t, fd.Close())
|
||||||
|
|
||||||
datafile := filepath.Join("testdata", "backup-data.tar.gz")
|
cmdInit(t)
|
||||||
fd, err := os.Open(datafile)
|
|
||||||
if os.IsNotExist(err) {
|
|
||||||
t.Skipf("unable to find data file %q, skipping TestBackup", datafile)
|
|
||||||
return
|
|
||||||
}
|
|
||||||
OK(t, err)
|
|
||||||
OK(t, fd.Close())
|
|
||||||
|
|
||||||
tempdir := setupTempdir(t)
|
datadir := filepath.Join(env.base, "testdata")
|
||||||
defer cleanupTempdir(t, tempdir)
|
setupTarTestFixture(t, datadir, datafile)
|
||||||
|
|
||||||
configureRestic(t, tempdir)
|
// first backup
|
||||||
|
cmdBackup(t, []string{datadir}, nil)
|
||||||
|
snapshotIDs := cmdList(t, "snapshots")
|
||||||
|
Assert(t, len(snapshotIDs) == 1,
|
||||||
|
"more than one snapshot ID in repo")
|
||||||
|
|
||||||
cmdInit(t)
|
cmdFsck(t)
|
||||||
|
stat1 := dirStats(env.repo)
|
||||||
|
|
||||||
datadir := filepath.Join(tempdir, "testdata")
|
// second backup, implicit incremental
|
||||||
|
cmdBackup(t, []string{datadir}, nil)
|
||||||
|
snapshotIDs = cmdList(t, "snapshots")
|
||||||
|
Assert(t, len(snapshotIDs) == 2,
|
||||||
|
"more than one snapshot ID in repo")
|
||||||
|
|
||||||
setupTarTestFixture(t, datadir, datafile)
|
stat2 := dirStats(env.repo)
|
||||||
|
if stat2.size > stat1.size+stat1.size/10 {
|
||||||
|
t.Error("repository size has grown by more than 10 percent")
|
||||||
|
}
|
||||||
|
t.Logf("repository grown by %d bytes", stat2.size-stat1.size)
|
||||||
|
|
||||||
// first backup
|
cmdFsck(t)
|
||||||
cmdBackup(t, []string{datadir}, nil)
|
// third backup, explicit incremental
|
||||||
snapshotIDs := cmdList(t, "snapshots")
|
cmdBackup(t, []string{datadir}, snapshotIDs[0])
|
||||||
Assert(t, len(snapshotIDs) == 1,
|
snapshotIDs = cmdList(t, "snapshots")
|
||||||
"more than one snapshot ID in repo")
|
Assert(t, len(snapshotIDs) == 3,
|
||||||
|
"more than two snapshot IDs in repo")
|
||||||
|
|
||||||
// second backup, implicit incremental
|
stat3 := dirStats(env.repo)
|
||||||
cmdBackup(t, []string{datadir}, nil)
|
if stat3.size > stat1.size+stat1.size/10 {
|
||||||
snapshotIDs = cmdList(t, "snapshots")
|
t.Error("repository size has grown by more than 10 percent")
|
||||||
Assert(t, len(snapshotIDs) == 2,
|
}
|
||||||
"more than one snapshot ID in repo")
|
t.Logf("repository grown by %d bytes", stat3.size-stat2.size)
|
||||||
|
|
||||||
// third backup, explicit incremental
|
// restore all backups and compare
|
||||||
cmdBackup(t, []string{datadir}, snapshotIDs[0])
|
for i, snapshotID := range snapshotIDs {
|
||||||
snapshotIDs = cmdList(t, "snapshots")
|
restoredir := filepath.Join(env.base, fmt.Sprintf("restore%d", i))
|
||||||
Assert(t, len(snapshotIDs) == 3,
|
t.Logf("restoring snapshot %v to %v", snapshotID.Str(), restoredir)
|
||||||
"more than one snapshot ID in repo")
|
cmdRestore(t, restoredir, snapshotIDs[0])
|
||||||
|
Assert(t, directoriesEqualContents(datadir, filepath.Join(restoredir, "testdata")),
|
||||||
|
"directories are not equal")
|
||||||
|
}
|
||||||
|
|
||||||
// restore all backups and compare
|
cmdFsck(t)
|
||||||
for _, snapshotID := range snapshotIDs {
|
})
|
||||||
restoredir := filepath.Join(tempdir, "restore", snapshotID.String())
|
}
|
||||||
t.Logf("restoring snapshot %v to %v", snapshotID.Str(), restoredir)
|
|
||||||
cmdRestore(t, restoredir, snapshotIDs[0])
|
const (
|
||||||
Assert(t, directoriesEqualContents(datadir, filepath.Join(restoredir, "testdata")),
|
incrementalFirstWrite = 20 * 1042 * 1024
|
||||||
"directories are not equal")
|
incrementalSecondWrite = 12 * 1042 * 1024
|
||||||
}
|
incrementalThirdWrite = 4 * 1042 * 1024
|
||||||
|
)
|
||||||
cmdFsck(t)
|
|
||||||
|
func appendRandomData(filename string, bytes uint) error {
|
||||||
|
f, err := os.OpenFile(filename, os.O_WRONLY|os.O_CREATE, 0666)
|
||||||
|
if err != nil {
|
||||||
|
fmt.Fprint(os.Stderr, err)
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
_, err = f.Seek(0, 2)
|
||||||
|
if err != nil {
|
||||||
|
fmt.Fprint(os.Stderr, err)
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
_, err = io.Copy(f, io.LimitReader(rand.Reader, int64(bytes)))
|
||||||
|
if err != nil {
|
||||||
|
fmt.Fprint(os.Stderr, err)
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
return f.Close()
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestIncrementalBackup(t *testing.T) {
|
||||||
|
withTestEnvironment(t, func(env *testEnvironment) {
|
||||||
|
datafile := filepath.Join("testdata", "backup-data.tar.gz")
|
||||||
|
fd, err := os.Open(datafile)
|
||||||
|
if os.IsNotExist(err) {
|
||||||
|
t.Skipf("unable to find data file %q, skipping TestBackup", datafile)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
OK(t, err)
|
||||||
|
OK(t, fd.Close())
|
||||||
|
|
||||||
|
cmdInit(t)
|
||||||
|
|
||||||
|
datadir := filepath.Join(env.base, "testdata")
|
||||||
|
testfile := filepath.Join(datadir, "testfile")
|
||||||
|
|
||||||
|
OK(t, appendRandomData(testfile, incrementalFirstWrite))
|
||||||
|
|
||||||
|
cmdBackup(t, []string{datadir}, nil)
|
||||||
|
cmdFsck(t)
|
||||||
|
stat1 := dirStats(env.repo)
|
||||||
|
|
||||||
|
OK(t, appendRandomData(testfile, incrementalSecondWrite))
|
||||||
|
|
||||||
|
cmdBackup(t, []string{datadir}, nil)
|
||||||
|
cmdFsck(t)
|
||||||
|
stat2 := dirStats(env.repo)
|
||||||
|
if stat2.size-stat1.size > incrementalFirstWrite {
|
||||||
|
t.Errorf("repository size has grown by more than %d bytes", incrementalFirstWrite)
|
||||||
|
}
|
||||||
|
t.Logf("repository grown by %d bytes", stat2.size-stat1.size)
|
||||||
|
|
||||||
|
OK(t, appendRandomData(testfile, incrementalThirdWrite))
|
||||||
|
|
||||||
|
cmdBackup(t, []string{datadir}, nil)
|
||||||
|
cmdFsck(t)
|
||||||
|
stat3 := dirStats(env.repo)
|
||||||
|
if stat3.size-stat2.size > incrementalFirstWrite {
|
||||||
|
t.Errorf("repository size has grown by more than %d bytes", incrementalFirstWrite)
|
||||||
|
}
|
||||||
|
t.Logf("repository grown by %d bytes", stat3.size-stat2.size)
|
||||||
|
})
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in a new issue