1
0
Fork 0
mirror of https://github.com/restic/restic.git synced 2025-01-18 13:31:08 +00:00
restic/internal/archiver/scanner.go

146 lines
3.3 KiB
Go
Raw Normal View History

2018-03-30 20:43:18 +00:00
package archiver
import (
"context"
"os"
"sort"
2018-03-30 20:43:18 +00:00
"github.com/restic/restic/internal/debug"
2018-03-30 20:43:18 +00:00
"github.com/restic/restic/internal/fs"
)
// Scanner traverses the targets and calls the function Result with cumulated
// stats concerning the files and folders found. Select is used to decide which
// items should be included. Error is called when an error occurs.
type Scanner struct {
FS fs.FS
SelectByName SelectByNameFunc
Select SelectFunc
Error ErrorFunc
Result func(item string, s ScanStats)
2018-03-30 20:43:18 +00:00
}
// NewScanner initializes a new Scanner.
func NewScanner(filesystem fs.FS) *Scanner {
2018-03-30 20:43:18 +00:00
return &Scanner{
FS: filesystem,
2024-02-10 21:58:10 +00:00
SelectByName: func(_ string) bool { return true },
Select: func(_ string, _ os.FileInfo, _ fs.FS) bool { return true },
2024-02-10 21:58:10 +00:00
Error: func(_ string, err error) error { return err },
Result: func(_ string, _ ScanStats) {},
2018-03-30 20:43:18 +00:00
}
}
// ScanStats collect statistics.
type ScanStats struct {
Files, Dirs, Others uint
Bytes uint64
}
2024-08-27 09:26:52 +00:00
func (s *Scanner) scanTree(ctx context.Context, stats ScanStats, tree tree) (ScanStats, error) {
// traverse the path in the file system for all leaf nodes
if tree.Leaf() {
abstarget, err := s.FS.Abs(tree.Path)
2018-03-30 20:43:18 +00:00
if err != nil {
return ScanStats{}, err
2018-03-30 20:43:18 +00:00
}
stats, err = s.scan(ctx, stats, abstarget)
if err != nil {
return ScanStats{}, err
}
return stats, nil
}
// otherwise recurse into the nodes in a deterministic order
for _, name := range tree.NodeNames() {
var err error
stats, err = s.scanTree(ctx, stats, tree.Nodes[name])
if err != nil {
return ScanStats{}, err
2018-03-30 20:43:18 +00:00
}
if ctx.Err() != nil {
return stats, nil
2018-03-30 20:43:18 +00:00
}
}
return stats, nil
}
// Scan traverses the targets. The function Result is called for each new item
// found, the complete result is also returned by Scan.
func (s *Scanner) Scan(ctx context.Context, targets []string) error {
debug.Log("start scan for %v", targets)
cleanTargets, err := resolveRelativeTargets(s.FS, targets)
if err != nil {
return err
}
debug.Log("clean targets %v", cleanTargets)
// we're using the same tree representation as the archiver does
2024-08-27 09:26:52 +00:00
tree, err := newTree(s.FS, cleanTargets)
if err != nil {
return err
}
stats, err := s.scanTree(ctx, ScanStats{}, *tree)
if err != nil {
return err
}
2018-03-30 20:43:18 +00:00
s.Result("", stats)
debug.Log("result: %+v", stats)
2018-03-30 20:43:18 +00:00
return nil
}
func (s *Scanner) scan(ctx context.Context, stats ScanStats, target string) (ScanStats, error) {
if ctx.Err() != nil {
return stats, nil
2018-03-30 20:43:18 +00:00
}
// exclude files by path before running stat to reduce number of lstat calls
if !s.SelectByName(target) {
return stats, nil
}
// get file information
2018-03-30 20:43:18 +00:00
fi, err := s.FS.Lstat(target)
if err != nil {
return stats, s.Error(target, err)
2018-03-30 20:43:18 +00:00
}
// run remaining select functions that require file information
if !s.Select(target, fi, s.FS) {
2018-03-30 20:43:18 +00:00
return stats, nil
}
switch {
case fi.Mode().IsRegular():
stats.Files++
stats.Bytes += uint64(fi.Size())
case fi.Mode().IsDir():
names, err := fs.Readdirnames(s.FS, target, fs.O_NOFOLLOW)
2018-03-30 20:43:18 +00:00
if err != nil {
return stats, s.Error(target, err)
2018-03-30 20:43:18 +00:00
}
sort.Strings(names)
2018-03-30 20:43:18 +00:00
for _, name := range names {
2024-08-27 12:25:35 +00:00
stats, err = s.scan(ctx, stats, s.FS.Join(target, name))
2018-03-30 20:43:18 +00:00
if err != nil {
return stats, err
}
}
stats.Dirs++
default:
stats.Others++
}
s.Result(target, stats)
return stats, nil
}