restore: track expected filenames in a folder

This commit is contained in:
Michael Eischer 2024-06-29 18:58:17 +02:00
parent d762f4ee64
commit 144e2a451f
2 changed files with 117 additions and 43 deletions

View file

@ -37,6 +37,7 @@ type Options struct {
Sparse bool Sparse bool
Progress *restoreui.Progress Progress *restoreui.Progress
Overwrite OverwriteBehavior Overwrite OverwriteBehavior
Delete bool
} }
type OverwriteBehavior int type OverwriteBehavior int
@ -107,22 +108,61 @@ func NewRestorer(repo restic.Repository, sn *restic.Snapshot, opts Options) *Res
type treeVisitor struct { type treeVisitor struct {
enterDir func(node *restic.Node, target, location string) error enterDir func(node *restic.Node, target, location string) error
visitNode func(node *restic.Node, target, location string) error visitNode func(node *restic.Node, target, location string) error
leaveDir func(node *restic.Node, target, location string) error // 'entries' contains all files the snapshot contains for this node. This also includes files
// ignored by the SelectFilter.
leaveDir func(node *restic.Node, target, location string, entries []string) error
} }
// traverseTree traverses a tree from the repo and calls treeVisitor. // traverseTree traverses a tree from the repo and calls treeVisitor.
// target is the path in the file system, location within the snapshot. // target is the path in the file system, location within the snapshot.
func (res *Restorer) traverseTree(ctx context.Context, target, location string, treeID restic.ID, visitor treeVisitor) (hasRestored bool, err error) { func (res *Restorer) traverseTree(ctx context.Context, target string, treeID restic.ID, visitor treeVisitor) error {
location := string(filepath.Separator)
sanitizeError := func(err error) error {
switch err {
case nil, context.Canceled, context.DeadlineExceeded:
// Context errors are permanent.
return err
default:
return res.Error(location, err)
}
}
if visitor.enterDir != nil {
err := sanitizeError(visitor.enterDir(nil, target, location))
if err != nil {
return err
}
}
childFilenames, hasRestored, err := res.traverseTreeInner(ctx, target, location, treeID, visitor)
if err != nil {
return err
}
if hasRestored && visitor.leaveDir != nil {
err = sanitizeError(visitor.leaveDir(nil, target, location, childFilenames))
}
return err
}
func (res *Restorer) traverseTreeInner(ctx context.Context, target, location string, treeID restic.ID, visitor treeVisitor) (filenames []string, hasRestored bool, err error) {
debug.Log("%v %v %v", target, location, treeID) debug.Log("%v %v %v", target, location, treeID)
tree, err := restic.LoadTree(ctx, res.repo, treeID) tree, err := restic.LoadTree(ctx, res.repo, treeID)
if err != nil { if err != nil {
debug.Log("error loading tree %v: %v", treeID, err) debug.Log("error loading tree %v: %v", treeID, err)
return hasRestored, res.Error(location, err) return nil, hasRestored, res.Error(location, err)
} }
if res.opts.Delete {
filenames = make([]string, 0, len(tree.Nodes))
}
for i, node := range tree.Nodes { for i, node := range tree.Nodes {
// allow GC of tree node // allow GC of tree node
tree.Nodes[i] = nil tree.Nodes[i] = nil
if res.opts.Delete {
// just track all files included in the tree node to simplify the control flow.
// tracking too many files does not matter except for a slightly elevated memory usage
filenames = append(filenames, node.Name)
}
// ensure that the node name does not contain anything that refers to a // ensure that the node name does not contain anything that refers to a
// top-level directory. // top-level directory.
@ -131,8 +171,10 @@ func (res *Restorer) traverseTree(ctx context.Context, target, location string,
debug.Log("node %q has invalid name %q", node.Name, nodeName) debug.Log("node %q has invalid name %q", node.Name, nodeName)
err := res.Error(location, errors.Errorf("invalid child node name %s", node.Name)) err := res.Error(location, errors.Errorf("invalid child node name %s", node.Name))
if err != nil { if err != nil {
return hasRestored, err return nil, hasRestored, err
} }
// force disable deletion to prevent unexpected behavior
res.opts.Delete = false
continue continue
} }
@ -144,8 +186,10 @@ func (res *Restorer) traverseTree(ctx context.Context, target, location string,
debug.Log("node %q has invalid target path %q", node.Name, nodeTarget) debug.Log("node %q has invalid target path %q", node.Name, nodeTarget)
err := res.Error(nodeLocation, errors.New("node has invalid path")) err := res.Error(nodeLocation, errors.New("node has invalid path"))
if err != nil { if err != nil {
return hasRestored, err return nil, hasRestored, err
} }
// force disable deletion to prevent unexpected behavior
res.opts.Delete = false
continue continue
} }
@ -173,25 +217,26 @@ func (res *Restorer) traverseTree(ctx context.Context, target, location string,
if node.Type == "dir" { if node.Type == "dir" {
if node.Subtree == nil { if node.Subtree == nil {
return hasRestored, errors.Errorf("Dir without subtree in tree %v", treeID.Str()) return nil, hasRestored, errors.Errorf("Dir without subtree in tree %v", treeID.Str())
} }
if selectedForRestore && visitor.enterDir != nil { if selectedForRestore && visitor.enterDir != nil {
err = sanitizeError(visitor.enterDir(node, nodeTarget, nodeLocation)) err = sanitizeError(visitor.enterDir(node, nodeTarget, nodeLocation))
if err != nil { if err != nil {
return hasRestored, err return nil, hasRestored, err
} }
} }
// keep track of restored child status // keep track of restored child status
// so metadata of the current directory are restored on leaveDir // so metadata of the current directory are restored on leaveDir
childHasRestored := false childHasRestored := false
var childFilenames []string
if childMayBeSelected { if childMayBeSelected {
childHasRestored, err = res.traverseTree(ctx, nodeTarget, nodeLocation, *node.Subtree, visitor) childFilenames, childHasRestored, err = res.traverseTreeInner(ctx, nodeTarget, nodeLocation, *node.Subtree, visitor)
err = sanitizeError(err) err = sanitizeError(err)
if err != nil { if err != nil {
return hasRestored, err return nil, hasRestored, err
} }
// inform the parent directory to restore parent metadata on leaveDir if needed // inform the parent directory to restore parent metadata on leaveDir if needed
if childHasRestored { if childHasRestored {
@ -202,9 +247,9 @@ func (res *Restorer) traverseTree(ctx context.Context, target, location string,
// metadata need to be restore when leaving the directory in both cases // metadata need to be restore when leaving the directory in both cases
// selected for restore or any child of any subtree have been restored // selected for restore or any child of any subtree have been restored
if (selectedForRestore || childHasRestored) && visitor.leaveDir != nil { if (selectedForRestore || childHasRestored) && visitor.leaveDir != nil {
err = sanitizeError(visitor.leaveDir(node, nodeTarget, nodeLocation)) err = sanitizeError(visitor.leaveDir(node, nodeTarget, nodeLocation, childFilenames))
if err != nil { if err != nil {
return hasRestored, err return nil, hasRestored, err
} }
} }
@ -214,12 +259,12 @@ func (res *Restorer) traverseTree(ctx context.Context, target, location string,
if selectedForRestore { if selectedForRestore {
err = sanitizeError(visitor.visitNode(node, nodeTarget, nodeLocation)) err = sanitizeError(visitor.visitNode(node, nodeTarget, nodeLocation))
if err != nil { if err != nil {
return hasRestored, err return nil, hasRestored, err
} }
} }
} }
return hasRestored, nil return filenames, hasRestored, nil
} }
func (res *Restorer) restoreNodeTo(ctx context.Context, node *restic.Node, target, location string) error { func (res *Restorer) restoreNodeTo(ctx context.Context, node *restic.Node, target, location string) error {
@ -310,10 +355,12 @@ func (res *Restorer) RestoreTo(ctx context.Context, dst string) error {
var buf []byte var buf []byte
// first tree pass: create directories and collect all files to restore // first tree pass: create directories and collect all files to restore
_, err = res.traverseTree(ctx, dst, string(filepath.Separator), *res.sn.Tree, treeVisitor{ err = res.traverseTree(ctx, dst, *res.sn.Tree, treeVisitor{
enterDir: func(_ *restic.Node, target, location string) error { enterDir: func(_ *restic.Node, target, location string) error {
debug.Log("first pass, enterDir: mkdir %q, leaveDir should restore metadata", location) debug.Log("first pass, enterDir: mkdir %q, leaveDir should restore metadata", location)
res.opts.Progress.AddFile(0) if location != "/" {
res.opts.Progress.AddFile(0)
}
return res.ensureDir(target) return res.ensureDir(target)
}, },
@ -373,7 +420,7 @@ func (res *Restorer) RestoreTo(ctx context.Context, dst string) error {
debug.Log("second pass for %q", dst) debug.Log("second pass for %q", dst)
// second tree pass: restore special files and filesystem metadata // second tree pass: restore special files and filesystem metadata
_, err = res.traverseTree(ctx, dst, string(filepath.Separator), *res.sn.Tree, treeVisitor{ err = res.traverseTree(ctx, dst, *res.sn.Tree, treeVisitor{
visitNode: func(node *restic.Node, target, location string) error { visitNode: func(node *restic.Node, target, location string) error {
debug.Log("second pass, visitNode: restore node %q", location) debug.Log("second pass, visitNode: restore node %q", location)
if node.Type != "file" { if node.Type != "file" {
@ -396,7 +443,11 @@ func (res *Restorer) RestoreTo(ctx context.Context, dst string) error {
// don't touch skipped files // don't touch skipped files
return nil return nil
}, },
leaveDir: func(node *restic.Node, target, location string) error { leaveDir: func(node *restic.Node, target, location string, expectedFilenames []string) error {
if node == nil {
return nil
}
err := res.restoreNodeMetadataTo(node, target, location) err := res.restoreNodeMetadataTo(node, target, location)
if err == nil { if err == nil {
res.opts.Progress.AddProgress(location, restoreui.ActionDirRestored, 0, 0) res.opts.Progress.AddProgress(location, restoreui.ActionDirRestored, 0, 0)
@ -493,7 +544,7 @@ func (res *Restorer) VerifyFiles(ctx context.Context, dst string) (int, error) {
g.Go(func() error { g.Go(func() error {
defer close(work) defer close(work)
_, err := res.traverseTree(ctx, dst, string(filepath.Separator), *res.sn.Tree, treeVisitor{ err := res.traverseTree(ctx, dst, *res.sn.Tree, treeVisitor{
visitNode: func(node *restic.Node, target, location string) error { visitNode: func(node *restic.Node, target, location string) error {
if node.Type != "file" { if node.Type != "file" {
return nil return nil

View file

@ -8,6 +8,7 @@ import (
"math" "math"
"os" "os"
"path/filepath" "path/filepath"
"reflect"
"runtime" "runtime"
"strings" "strings"
"syscall" "syscall"
@ -527,16 +528,17 @@ func TestRestorerRelative(t *testing.T) {
type TraverseTreeCheck func(testing.TB) treeVisitor type TraverseTreeCheck func(testing.TB) treeVisitor
type TreeVisit struct { type TreeVisit struct {
funcName string // name of the function funcName string // name of the function
location string // location passed to the function location string // location passed to the function
files []string // file list passed to the function
} }
func checkVisitOrder(list []TreeVisit) TraverseTreeCheck { func checkVisitOrder(list []TreeVisit) TraverseTreeCheck {
var pos int var pos int
return func(t testing.TB) treeVisitor { return func(t testing.TB) treeVisitor {
check := func(funcName string) func(*restic.Node, string, string) error { check := func(funcName string) func(*restic.Node, string, string, []string) error {
return func(node *restic.Node, target, location string) error { return func(node *restic.Node, target, location string, expectedFilenames []string) error {
if pos >= len(list) { if pos >= len(list) {
t.Errorf("step %v, %v(%v): expected no more than %d function calls", pos, funcName, location, len(list)) t.Errorf("step %v, %v(%v): expected no more than %d function calls", pos, funcName, location, len(list))
pos++ pos++
@ -554,14 +556,24 @@ func checkVisitOrder(list []TreeVisit) TraverseTreeCheck {
t.Errorf("step %v: want location %v, got %v", pos, list[pos].location, location) t.Errorf("step %v: want location %v, got %v", pos, list[pos].location, location)
} }
if !reflect.DeepEqual(expectedFilenames, v.files) {
t.Errorf("step %v: want files %v, got %v", pos, list[pos].files, expectedFilenames)
}
pos++ pos++
return nil return nil
} }
} }
checkNoFilename := func(funcName string) func(*restic.Node, string, string) error {
f := check(funcName)
return func(node *restic.Node, target, location string) error {
return f(node, target, location, nil)
}
}
return treeVisitor{ return treeVisitor{
enterDir: check("enterDir"), enterDir: checkNoFilename("enterDir"),
visitNode: check("visitNode"), visitNode: checkNoFilename("visitNode"),
leaveDir: check("leaveDir"), leaveDir: check("leaveDir"),
} }
} }
@ -590,13 +602,15 @@ func TestRestorerTraverseTree(t *testing.T) {
return true, true return true, true
}, },
Visitor: checkVisitOrder([]TreeVisit{ Visitor: checkVisitOrder([]TreeVisit{
{"enterDir", "/dir"}, {"enterDir", "/", nil},
{"visitNode", "/dir/otherfile"}, {"enterDir", "/dir", nil},
{"enterDir", "/dir/subdir"}, {"visitNode", "/dir/otherfile", nil},
{"visitNode", "/dir/subdir/file"}, {"enterDir", "/dir/subdir", nil},
{"leaveDir", "/dir/subdir"}, {"visitNode", "/dir/subdir/file", nil},
{"leaveDir", "/dir"}, {"leaveDir", "/dir/subdir", []string{"file"}},
{"visitNode", "/foo"}, {"leaveDir", "/dir", []string{"otherfile", "subdir"}},
{"visitNode", "/foo", nil},
{"leaveDir", "/", []string{"dir", "foo"}},
}), }),
}, },
@ -620,7 +634,9 @@ func TestRestorerTraverseTree(t *testing.T) {
return false, false return false, false
}, },
Visitor: checkVisitOrder([]TreeVisit{ Visitor: checkVisitOrder([]TreeVisit{
{"visitNode", "/foo"}, {"enterDir", "/", nil},
{"visitNode", "/foo", nil},
{"leaveDir", "/", []string{"dir", "foo"}},
}), }),
}, },
{ {
@ -642,7 +658,9 @@ func TestRestorerTraverseTree(t *testing.T) {
return false, false return false, false
}, },
Visitor: checkVisitOrder([]TreeVisit{ Visitor: checkVisitOrder([]TreeVisit{
{"visitNode", "/aaa"}, {"enterDir", "/", nil},
{"visitNode", "/aaa", nil},
{"leaveDir", "/", []string{"aaa", "dir"}},
}), }),
}, },
@ -666,12 +684,14 @@ func TestRestorerTraverseTree(t *testing.T) {
return false, false return false, false
}, },
Visitor: checkVisitOrder([]TreeVisit{ Visitor: checkVisitOrder([]TreeVisit{
{"enterDir", "/dir"}, {"enterDir", "/", nil},
{"visitNode", "/dir/otherfile"}, {"enterDir", "/dir", nil},
{"enterDir", "/dir/subdir"}, {"visitNode", "/dir/otherfile", nil},
{"visitNode", "/dir/subdir/file"}, {"enterDir", "/dir/subdir", nil},
{"leaveDir", "/dir/subdir"}, {"visitNode", "/dir/subdir/file", nil},
{"leaveDir", "/dir"}, {"leaveDir", "/dir/subdir", []string{"file"}},
{"leaveDir", "/dir", []string{"otherfile", "subdir"}},
{"leaveDir", "/", []string{"dir", "foo"}},
}), }),
}, },
@ -699,8 +719,10 @@ func TestRestorerTraverseTree(t *testing.T) {
} }
}, },
Visitor: checkVisitOrder([]TreeVisit{ Visitor: checkVisitOrder([]TreeVisit{
{"visitNode", "/dir/otherfile"}, {"enterDir", "/", nil},
{"leaveDir", "/dir"}, {"visitNode", "/dir/otherfile", nil},
{"leaveDir", "/dir", []string{"otherfile", "subdir"}},
{"leaveDir", "/", []string{"dir", "foo"}},
}), }),
}, },
} }
@ -710,7 +732,8 @@ func TestRestorerTraverseTree(t *testing.T) {
repo := repository.TestRepository(t) repo := repository.TestRepository(t)
sn, _ := saveSnapshot(t, repo, test.Snapshot, noopGetGenericAttributes) sn, _ := saveSnapshot(t, repo, test.Snapshot, noopGetGenericAttributes)
res := NewRestorer(repo, sn, Options{}) // set Delete option to enable tracking filenames in a directory
res := NewRestorer(repo, sn, Options{Delete: true})
res.SelectFilter = test.Select res.SelectFilter = test.Select
@ -721,7 +744,7 @@ func TestRestorerTraverseTree(t *testing.T) {
// make sure we're creating a new subdir of the tempdir // make sure we're creating a new subdir of the tempdir
target := filepath.Join(tempdir, "target") target := filepath.Join(tempdir, "target")
_, err := res.traverseTree(ctx, target, string(filepath.Separator), *sn.Tree, test.Visitor(t)) err := res.traverseTree(ctx, target, *sn.Tree, test.Visitor(t))
if err != nil { if err != nil {
t.Fatal(err) t.Fatal(err)
} }