2018-03-30 22:43:18 +02:00
|
|
|
package archiver
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
"os"
|
|
|
|
"path/filepath"
|
2020-02-17 09:22:32 +01:00
|
|
|
"sort"
|
2018-03-30 22:43:18 +02: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 {
|
2018-07-31 17:25:25 +02:00
|
|
|
FS fs.FS
|
|
|
|
SelectByName SelectByNameFunc
|
|
|
|
Select SelectFunc
|
|
|
|
Error ErrorFunc
|
|
|
|
Result func(item string, s ScanStats)
|
2018-03-30 22:43:18 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// NewScanner initializes a new Scanner.
|
|
|
|
func NewScanner(fs fs.FS) *Scanner {
|
|
|
|
return &Scanner{
|
2018-07-31 17:25:25 +02:00
|
|
|
FS: fs,
|
|
|
|
SelectByName: func(item string) bool { return true },
|
|
|
|
Select: func(item string, fi os.FileInfo) bool { return true },
|
|
|
|
Error: func(item string, fi os.FileInfo, err error) error { return err },
|
|
|
|
Result: func(item string, s ScanStats) {},
|
2018-03-30 22:43:18 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// ScanStats collect statistics.
|
|
|
|
type ScanStats struct {
|
|
|
|
Files, Dirs, Others uint
|
|
|
|
Bytes uint64
|
|
|
|
}
|
|
|
|
|
|
|
|
// 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 {
|
|
|
|
var stats ScanStats
|
|
|
|
for _, target := range targets {
|
|
|
|
abstarget, err := s.FS.Abs(target)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
stats, err = s.scan(ctx, stats, abstarget)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
if ctx.Err() != nil {
|
2018-09-08 18:35:49 +02:00
|
|
|
return nil
|
2018-03-30 22:43:18 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
s.Result("", stats)
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *Scanner) scan(ctx context.Context, stats ScanStats, target string) (ScanStats, error) {
|
|
|
|
if ctx.Err() != nil {
|
2018-09-08 18:35:49 +02:00
|
|
|
return stats, nil
|
2018-03-30 22:43:18 +02:00
|
|
|
}
|
|
|
|
|
2018-07-31 17:25:25 +02: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 22:43:18 +02:00
|
|
|
fi, err := s.FS.Lstat(target)
|
|
|
|
if err != nil {
|
|
|
|
return stats, s.Error(target, fi, err)
|
|
|
|
}
|
|
|
|
|
2018-07-31 17:25:25 +02:00
|
|
|
// run remaining select functions that require file information
|
2018-03-30 22:43:18 +02:00
|
|
|
if !s.Select(target, fi) {
|
|
|
|
return stats, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
switch {
|
|
|
|
case fi.Mode().IsRegular():
|
|
|
|
stats.Files++
|
|
|
|
stats.Bytes += uint64(fi.Size())
|
|
|
|
case fi.Mode().IsDir():
|
2020-02-17 09:22:32 +01:00
|
|
|
names, err := readdirnames(s.FS, target, fs.O_NOFOLLOW)
|
2018-03-30 22:43:18 +02:00
|
|
|
if err != nil {
|
|
|
|
return stats, s.Error(target, fi, err)
|
|
|
|
}
|
2020-02-17 09:22:32 +01:00
|
|
|
sort.Strings(names)
|
2018-03-30 22:43:18 +02:00
|
|
|
|
|
|
|
for _, name := range names {
|
|
|
|
stats, err = s.scan(ctx, stats, filepath.Join(target, name))
|
|
|
|
if err != nil {
|
|
|
|
return stats, err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
stats.Dirs++
|
|
|
|
default:
|
|
|
|
stats.Others++
|
|
|
|
}
|
|
|
|
|
|
|
|
s.Result(target, stats)
|
|
|
|
return stats, nil
|
|
|
|
}
|