2022-09-06 20:30:45 +00:00
|
|
|
package walker
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
"path"
|
|
|
|
|
|
|
|
"github.com/restic/restic/internal/debug"
|
|
|
|
"github.com/restic/restic/internal/restic"
|
|
|
|
)
|
|
|
|
|
2022-09-09 20:33:12 +00:00
|
|
|
// SelectByNameFunc returns true for all items that should be included (files and
|
|
|
|
// dirs). If false is returned, files are ignored and dirs are not even walked.
|
|
|
|
type SelectByNameFunc func(item string) bool
|
2022-09-06 20:30:45 +00:00
|
|
|
|
|
|
|
type TreeFilterVisitor struct {
|
2022-09-09 20:33:12 +00:00
|
|
|
SelectByName SelectByNameFunc
|
2022-09-06 20:30:45 +00:00
|
|
|
PrintExclude func(string)
|
|
|
|
}
|
|
|
|
|
2022-10-14 21:26:13 +00:00
|
|
|
type BlobLoadSaver interface {
|
|
|
|
restic.BlobSaver
|
|
|
|
restic.BlobLoader
|
|
|
|
}
|
|
|
|
|
|
|
|
func FilterTree(ctx context.Context, repo BlobLoadSaver, nodepath string, nodeID restic.ID, visitor *TreeFilterVisitor) (newNodeID restic.ID, err error) {
|
2022-09-06 20:30:45 +00:00
|
|
|
curTree, err := restic.LoadTree(ctx, repo, nodeID)
|
|
|
|
if err != nil {
|
|
|
|
return restic.ID{}, err
|
|
|
|
}
|
|
|
|
|
|
|
|
debug.Log("filterTree: %s, nodeId: %s\n", nodepath, nodeID.Str())
|
|
|
|
|
|
|
|
changed := false
|
|
|
|
tb := restic.NewTreeJSONBuilder()
|
|
|
|
for _, node := range curTree.Nodes {
|
|
|
|
path := path.Join(nodepath, node.Name)
|
2022-09-09 20:33:12 +00:00
|
|
|
if !visitor.SelectByName(path) {
|
2022-09-06 20:30:45 +00:00
|
|
|
if visitor.PrintExclude != nil {
|
|
|
|
visitor.PrintExclude(path)
|
|
|
|
}
|
|
|
|
changed = true
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
if node.Subtree == nil {
|
|
|
|
err = tb.AddNode(node)
|
|
|
|
if err != nil {
|
|
|
|
return restic.ID{}, err
|
|
|
|
}
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
newID, err := FilterTree(ctx, repo, path, *node.Subtree, visitor)
|
|
|
|
if err != nil {
|
|
|
|
return restic.ID{}, err
|
|
|
|
}
|
|
|
|
if !node.Subtree.Equal(newID) {
|
|
|
|
changed = true
|
|
|
|
}
|
|
|
|
node.Subtree = &newID
|
|
|
|
err = tb.AddNode(node)
|
|
|
|
if err != nil {
|
|
|
|
return restic.ID{}, err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if changed {
|
|
|
|
tree, err := tb.Finalize()
|
|
|
|
if err != nil {
|
|
|
|
return restic.ID{}, err
|
|
|
|
}
|
|
|
|
|
|
|
|
// Save new tree
|
|
|
|
newTreeID, _, _, err := repo.SaveBlob(ctx, restic.TreeBlob, tree, restic.ID{}, false)
|
|
|
|
debug.Log("filterTree: save new tree for %s as %v\n", nodepath, newTreeID)
|
|
|
|
return newTreeID, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return nodeID, nil
|
|
|
|
}
|