mirror of
https://github.com/octoleo/restic.git
synced 2025-01-26 00:28:26 +00:00
f4b15fdd96
When the `--delete` option is specified, recursively delete directories that should be replaced with a file.
171 lines
4.3 KiB
Go
171 lines
4.3 KiB
Go
package restorer
|
|
|
|
import (
|
|
"fmt"
|
|
"os"
|
|
"path/filepath"
|
|
"runtime"
|
|
"testing"
|
|
|
|
"github.com/restic/restic/internal/errors"
|
|
rtest "github.com/restic/restic/internal/test"
|
|
)
|
|
|
|
func TestFilesWriterBasic(t *testing.T) {
|
|
dir := rtest.TempDir(t)
|
|
w := newFilesWriter(1, false)
|
|
|
|
f1 := dir + "/f1"
|
|
f2 := dir + "/f2"
|
|
|
|
rtest.OK(t, w.writeToFile(f1, []byte{1}, 0, 2, false))
|
|
rtest.Equals(t, 0, len(w.buckets[0].files))
|
|
|
|
rtest.OK(t, w.writeToFile(f2, []byte{2}, 0, 2, false))
|
|
rtest.Equals(t, 0, len(w.buckets[0].files))
|
|
|
|
rtest.OK(t, w.writeToFile(f1, []byte{1}, 1, -1, false))
|
|
rtest.Equals(t, 0, len(w.buckets[0].files))
|
|
|
|
rtest.OK(t, w.writeToFile(f2, []byte{2}, 1, -1, false))
|
|
rtest.Equals(t, 0, len(w.buckets[0].files))
|
|
|
|
buf, err := os.ReadFile(f1)
|
|
rtest.OK(t, err)
|
|
rtest.Equals(t, []byte{1, 1}, buf)
|
|
|
|
buf, err = os.ReadFile(f2)
|
|
rtest.OK(t, err)
|
|
rtest.Equals(t, []byte{2, 2}, buf)
|
|
}
|
|
|
|
func TestFilesWriterRecursiveOverwrite(t *testing.T) {
|
|
path := filepath.Join(t.TempDir(), "test")
|
|
|
|
// create filled directory
|
|
rtest.OK(t, os.Mkdir(path, 0o700))
|
|
rtest.OK(t, os.WriteFile(filepath.Join(path, "file"), []byte("data"), 0o400))
|
|
|
|
// must error if recursive delete is not allowed
|
|
w := newFilesWriter(1, false)
|
|
err := w.writeToFile(path, []byte{1}, 0, 2, false)
|
|
rtest.Assert(t, errors.Is(err, notEmptyDirError()), "unexepected error got %v", err)
|
|
rtest.Equals(t, 0, len(w.buckets[0].files))
|
|
|
|
// must replace directory
|
|
w = newFilesWriter(1, true)
|
|
rtest.OK(t, w.writeToFile(path, []byte{1, 1}, 0, 2, false))
|
|
rtest.Equals(t, 0, len(w.buckets[0].files))
|
|
|
|
buf, err := os.ReadFile(path)
|
|
rtest.OK(t, err)
|
|
rtest.Equals(t, []byte{1, 1}, buf)
|
|
}
|
|
|
|
func TestCreateFile(t *testing.T) {
|
|
basepath := filepath.Join(t.TempDir(), "test")
|
|
|
|
scenarios := []struct {
|
|
name string
|
|
create func(t testing.TB, path string)
|
|
check func(t testing.TB, path string)
|
|
err error
|
|
}{
|
|
{
|
|
name: "file",
|
|
create: func(t testing.TB, path string) {
|
|
rtest.OK(t, os.WriteFile(path, []byte("test-test-test-data"), 0o400))
|
|
},
|
|
},
|
|
{
|
|
name: "empty dir",
|
|
create: func(t testing.TB, path string) {
|
|
rtest.OK(t, os.Mkdir(path, 0o400))
|
|
},
|
|
},
|
|
{
|
|
name: "symlink",
|
|
create: func(t testing.TB, path string) {
|
|
rtest.OK(t, os.Symlink("./something", path))
|
|
},
|
|
},
|
|
{
|
|
name: "filled dir",
|
|
create: func(t testing.TB, path string) {
|
|
rtest.OK(t, os.Mkdir(path, 0o700))
|
|
rtest.OK(t, os.WriteFile(filepath.Join(path, "file"), []byte("data"), 0o400))
|
|
},
|
|
err: notEmptyDirError(),
|
|
},
|
|
{
|
|
name: "hardlinks",
|
|
create: func(t testing.TB, path string) {
|
|
rtest.OK(t, os.WriteFile(path, []byte("test-test-test-data"), 0o400))
|
|
rtest.OK(t, os.Link(path, path+"h"))
|
|
},
|
|
check: func(t testing.TB, path string) {
|
|
if runtime.GOOS == "windows" {
|
|
// hardlinks are not supported on windows
|
|
return
|
|
}
|
|
|
|
data, err := os.ReadFile(path + "h")
|
|
rtest.OK(t, err)
|
|
rtest.Equals(t, "test-test-test-data", string(data), "unexpected content change")
|
|
},
|
|
},
|
|
}
|
|
|
|
tests := []struct {
|
|
size int64
|
|
isSparse bool
|
|
}{
|
|
{5, false},
|
|
{21, false},
|
|
{100, false},
|
|
{5, true},
|
|
{21, true},
|
|
{100, true},
|
|
}
|
|
|
|
for i, sc := range scenarios {
|
|
t.Run(sc.name, func(t *testing.T) {
|
|
for j, test := range tests {
|
|
path := basepath + fmt.Sprintf("%v%v", i, j)
|
|
sc.create(t, path)
|
|
f, err := createFile(path, test.size, test.isSparse, false)
|
|
if sc.err == nil {
|
|
rtest.OK(t, err)
|
|
fi, err := f.Stat()
|
|
rtest.OK(t, err)
|
|
rtest.Assert(t, fi.Mode().IsRegular(), "wrong filetype %v", fi.Mode())
|
|
rtest.Assert(t, fi.Size() <= test.size, "unexpected file size expected %v, got %v", test.size, fi.Size())
|
|
rtest.OK(t, f.Close())
|
|
if sc.check != nil {
|
|
sc.check(t, path)
|
|
}
|
|
} else {
|
|
rtest.Assert(t, errors.Is(err, sc.err), "unexpected error got %v expected %v", err, sc.err)
|
|
}
|
|
rtest.OK(t, os.RemoveAll(path))
|
|
}
|
|
})
|
|
}
|
|
}
|
|
|
|
func TestCreateFileRecursiveDelete(t *testing.T) {
|
|
path := filepath.Join(t.TempDir(), "test")
|
|
|
|
// create filled directory
|
|
rtest.OK(t, os.Mkdir(path, 0o700))
|
|
rtest.OK(t, os.WriteFile(filepath.Join(path, "file"), []byte("data"), 0o400))
|
|
|
|
// replace it
|
|
f, err := createFile(path, 42, false, true)
|
|
rtest.OK(t, err)
|
|
fi, err := f.Stat()
|
|
rtest.OK(t, err)
|
|
rtest.Assert(t, fi.Mode().IsRegular(), "wrong filetype %v", fi.Mode())
|
|
rtest.OK(t, f.Close())
|
|
}
|