diff --git a/archiver.go b/archiver.go index dfe91b79b..69d7d8053 100644 --- a/archiver.go +++ b/archiver.go @@ -538,32 +538,6 @@ func (arch *Archiver) dirWorker(wg *sync.WaitGroup, p *Progress, done <-chan str } } -func compareWithOldTree(newCh <-chan interface{}, oldCh <-chan WalkTreeJob, outCh chan<- interface{}) { - debug.Log("Archiver.compareWithOldTree", "start") - defer func() { - debug.Log("Archiver.compareWithOldTree", "done") - }() - for { - debug.Log("Archiver.compareWithOldTree", "waiting for new job") - newJob, ok := <-newCh - if !ok { - // channel is closed - return - } - - debug.Log("Archiver.compareWithOldTree", "received new job %v", newJob) - oldJob, ok := <-oldCh - if !ok { - // channel is closed - return - } - - debug.Log("Archiver.compareWithOldTree", "received old job %v", oldJob) - - outCh <- newJob - } -} - type ArchivePipe struct { Old <-chan WalkTreeJob New <-chan pipe.Job diff --git a/tree.go b/tree.go index ce25414a9..7be386758 100644 --- a/tree.go +++ b/tree.go @@ -3,9 +3,7 @@ package restic import ( "errors" "fmt" - "path/filepath" "sort" - "sync" "github.com/restic/restic/backend" "github.com/restic/restic/debug" @@ -42,116 +40,6 @@ func LoadTree(s Server, id backend.ID) (*Tree, error) { return tree, nil } -// LoadTreeRecursive loads the tree and all subtrees via s. -func LoadTreeRecursive(path string, s Server, blob Blob) (*Tree, error) { - // TODO: load subtrees in parallel - tree, err := LoadTree(s, blob.Storage) - if err != nil { - return nil, err - } - - for _, n := range tree.Nodes { - n.path = filepath.Join(path, n.Name) - if n.Type == "dir" && n.Subtree != nil { - subtreeBlob, err := tree.Map.FindID(n.Subtree) - if err != nil { - return nil, err - } - - t, err := LoadTreeRecursive(n.path, s, subtreeBlob) - if err != nil { - return nil, err - } - - n.tree = t - } - } - - return tree, nil -} - -// CopyFrom recursively copies all content from other to t. -func (t Tree) CopyFrom(other *Tree, s *Server) error { - debug.Log("Tree.CopyFrom", "CopyFrom(%v)\n", other) - for _, node := range t.Nodes { - // only process files and dirs - if node.Type != "file" && node.Type != "dir" { - continue - } - - // find entry in other tree - oldNode, err := other.Find(node.Name) - - // if the node could not be found or the type has changed, proceed to the next - if err == ErrNodeNotFound || node.Type != oldNode.Type { - debug.Log("Tree.CopyFrom", " node %v is new\n", node) - continue - } - - if node.Type == "file" { - // compare content - if node.SameContent(oldNode) { - debug.Log("Tree.CopyFrom", " file node %v has same content\n", node) - - // check if all content is still available in the repository - for _, id := range oldNode.Content { - blob, err := other.Map.FindID(id) - if err != nil { - continue - } - - if ok, err := s.Test(backend.Data, blob.Storage); !ok || err != nil { - continue - } - } - - // copy Content - node.Content = oldNode.Content - - // copy storage IDs - for _, id := range node.Content { - blob, err := other.Map.FindID(id) - if err != nil { - return err - } - - debug.Log("Tree.CopyFrom", " insert blob %v\n", blob) - t.Map.Insert(blob) - } - } - } else if node.Type == "dir" { - // fill in all subtrees from old subtree - err := node.tree.CopyFrom(oldNode.tree, s) - if err != nil { - return err - } - - // check if tree has changed - if node.tree.Equals(*oldNode.tree) { - debug.Log("Tree.CopyFrom", " tree node %v has same content\n", node) - - // if nothing has changed, copy subtree ID - node.Subtree = oldNode.Subtree - - // and store blob in bloblist - blob, err := other.Map.FindID(oldNode.Subtree) - if err != nil { - return err - } - - debug.Log("Tree.CopyFrom", " insert blob %v\n", blob) - t.Map.Insert(blob) - } else { - debug.Log("Tree.CopyFrom", " trees are not equal: %v\n", node) - debug.Log("Tree.CopyFrom", " %#v\n", node.tree) - debug.Log("Tree.CopyFrom", " %#v\n", oldNode.tree) - } - } - } - - return nil -} - // Equals returns true if t and other have exactly the same nodes and map. func (t Tree) Equals(other Tree) bool { if len(t.Nodes) != len(other.Nodes) { @@ -207,120 +95,3 @@ func (t Tree) Find(name string) (*Node, error) { _, node, err := t.find(name) return node, err } - -func (t Tree) Stat() Stat { - s := Stat{} - for _, n := range t.Nodes { - switch n.Type { - case "file": - s.Files++ - s.Bytes += n.Size - case "dir": - s.Dirs++ - if n.tree != nil { - s.Add(n.tree.Stat()) - } - } - } - - return s -} - -func (t Tree) StatTodo() Stat { - s := Stat{} - for _, n := range t.Nodes { - switch n.Type { - case "file": - if n.Content == nil { - s.Files++ - s.Bytes += n.Size - } - case "dir": - if n.Subtree == nil { - s.Dirs++ - if n.tree != nil { - s.Add(n.tree.StatTodo()) - } - } - } - } - - return s -} - -// EachNode calls fn recursively for each node in t and all subtrees -// (depth-first). This in done concurrently in p goroutines. -func (t *Tree) EachNode(p int, fn func(*Node)) { - processNodeWorker := func(wg *sync.WaitGroup, ch <-chan *Node, f func(*Node)) { - for node := range ch { - f(node) - } - wg.Done() - } - - // start workers - var wg sync.WaitGroup - ch := make(chan *Node) - - for i := 0; i < p; i++ { - wg.Add(1) - go processNodeWorker(&wg, ch, fn) - } - - var processTree func(t *Tree, ch chan<- *Node) - processTree = func(t *Tree, ch chan<- *Node) { - for _, n := range t.Nodes { - if n.Type == "dir" && n.Tree() != nil { - processTree(n.Tree(), ch) - } - - ch <- n - } - } - - // run on root - processTree(t, ch) - close(ch) - - // wait for all goroutines to terminate - wg.Wait() -} - -// EachSubtree calls fn recursively for t and all subtrees (depth-first). This -// is done concurrently in p goroutines. -func (t *Tree) EachSubtree(p int, fn func(*Tree)) { - processTreeWorker := func(wg *sync.WaitGroup, ch <-chan *Tree, f func(*Tree)) { - for tree := range ch { - f(tree) - } - wg.Done() - } - - // start workers - var wg sync.WaitGroup - ch := make(chan *Tree) - - for i := 0; i < p; i++ { - wg.Add(1) - go processTreeWorker(&wg, ch, fn) - } - - // extra variable needed for recursion - var processTree func(t *Tree, ch chan<- *Tree) - processTree = func(t *Tree, ch chan<- *Tree) { - for _, n := range t.Nodes { - if n.Type == "dir" && n.Tree() != nil { - processTree(n.Tree(), ch) - } - } - - ch <- t - } - - // run on root - processTree(t, ch) - close(ch) - - // wait for all goroutines to terminate - wg.Wait() -}