delay stat calls until required for creating the object

Signed-off-by: Anagh Kumar Baranwal <6824881+darthShadow@users.noreply.github.com>
This commit is contained in:
Anagh Kumar Baranwal 2023-08-04 13:18:46 +05:30
parent 050c7159de
commit 0b35ff3893
No known key found for this signature in database

View File

@ -33,9 +33,8 @@ import (
) )
// Constants // Constants
const devUnset = 0xdeadbeefcafebabe // a device id meaning it is unset const devUnset = 0xdeadbeefcafebabe // a device id meaning it is unset
const linkSuffix = ".rclonelink" // The suffix added to a translated symbolic link const linkSuffix = ".rclonelink" // The suffix added to a translated symbolic link
const useReadDir = (runtime.GOOS == "windows" || runtime.GOOS == "plan9") // these OSes read FileInfos directly
// Register with Fs // Register with Fs
func init() { func init() {
@ -486,91 +485,68 @@ func (f *Fs) List(ctx context.Context, dir string) (entries fs.DirEntries, err e
}() }()
for { for {
var fis []os.FileInfo var des []os.DirEntry
if useReadDir { des, err = fd.ReadDir(1024)
// Windows and Plan9 read the directory entries with the stat information in which if err == io.EOF && len(des) == 0 {
// shouldn't fail because of unreadable entries. break
fis, err = fd.Readdir(1024)
if err == io.EOF && len(fis) == 0 {
break
}
} else {
// For other OSes we read the names only (which shouldn't fail) then stat the
// individual ourselves, so we can log errors but not fail the directory read.
var names []string
names, err = fd.Readdirnames(1024)
if err == io.EOF && len(names) == 0 {
break
}
fis = make([]os.FileInfo, len(names))
g, gCtx := errgroup.WithContext(ctx)
g.SetLimit(fs.GetConfig(ctx).Checkers)
if err == nil {
for i, name := range names {
i, name := i, name // https://golang.org/doc/faq#closures_and_goroutines
g.Go(func() error {
// No point in continuing if context has been cancelled
if gCtx.Err() != nil {
return nil
}
var err error
namepath := filepath.Join(fsDirPath, name)
fi, fierr := os.Lstat(namepath)
if os.IsNotExist(fierr) {
// skip entry removed by a concurrent goroutine
return nil
}
if fierr != nil {
if useFilter {
newRemote := f.cleanRemote(dir, name)
if !filter.IncludeRemote(newRemote) {
return nil
}
}
err = fmt.Errorf("failed to get info about directory entry %q: %w", namepath, fierr)
fs.Errorf(dir, "%v", err)
_ = accounting.Stats(gCtx).Error(fserrors.NoRetryError(err)) // fail the sync
return nil
}
fis[i] = fi
return nil
})
}
}
err = g.Wait()
} }
if err != nil { if err != nil {
return nil, fmt.Errorf("failed to read directory entry: %w", err) return nil, fmt.Errorf("failed to read directory entry: %w", err)
} }
loopEntries := make(fs.DirEntries, len(fis)) loopEntries := make(fs.DirEntries, len(des))
g, gCtx := errgroup.WithContext(ctx) g, gCtx := errgroup.WithContext(ctx)
g.SetLimit(fs.GetConfig(ctx).Checkers) g.SetLimit(fs.GetConfig(ctx).Checkers)
for i, fi := range fis { for i, de := range des {
if fi == nil { i, de := i, de // https://golang.org/doc/faq#closures_and_goroutines
continue
}
i, fi := i, fi // https://golang.org/doc/faq#closures_and_goroutines
g.Go(func() error { g.Go(func() error {
// No point in continuing if context has been cancelled // No point in continuing if context has been cancelled
if gCtx.Err() != nil { if gCtx.Err() != nil {
return nil return nil
} }
var err error var (
name := fi.Name() err error
mode := fi.Mode() fi os.FileInfo
)
name := de.Name()
mode := de.Type()
namepath := filepath.Join(fsDirPath, name)
newRemote := f.cleanRemote(dir, name) newRemote := f.cleanRemote(dir, name)
if de.IsDir() {
// Ignore directories which are symlinks. These are junction points under windows which
// are kind of a souped up symlink. Unix doesn't have directories which are symlinks.
if (mode & os.ModeSymlink) != 0 {
return nil
}
}
fi, err = de.Info()
if err != nil {
if os.IsNotExist(err) {
// skip entry removed by a concurrent goroutine
return nil
}
if useFilter && !filter.IncludeRemote(newRemote) {
return nil
}
err = fmt.Errorf("failed to get info about directory entry %q: %w", namepath, err)
fs.Errorf(dir, "%v", err)
_ = accounting.Stats(gCtx).Error(fserrors.NoRetryError(err)) // fail the sync
return nil
}
name = fi.Name()
mode = fi.Mode()
namepath = filepath.Join(fsDirPath, name)
newRemote = f.cleanRemote(dir, name)
// Follow symlinks if required // Follow symlinks if required
if f.opt.FollowSymlinks && (mode&os.ModeSymlink) != 0 { if f.opt.FollowSymlinks && (mode&os.ModeSymlink) != 0 {
localPath := filepath.Join(fsDirPath, name) fi, err = os.Stat(namepath)
fi, err = os.Stat(localPath)
if err != nil { if err != nil {
// Quietly skip errors on excluded files and directories // Quietly skip errors on excluded files and directories
if useFilter && !filter.IncludeRemote(newRemote) { if useFilter && !filter.IncludeRemote(newRemote) {
@ -594,16 +570,14 @@ func (f *Fs) List(ctx context.Context, dir string) (entries fs.DirEntries, err e
} }
if fi.IsDir() { if fi.IsDir() {
// Ignore directories which are symlinks. These are junction points under windows which if f.dev == readDevice(fi, f.opt.OneFileSystem) {
// are kind of a souped up symlink. Unix doesn't have directories which are symlinks.
if (mode&os.ModeSymlink) == 0 && f.dev == readDevice(fi, f.opt.OneFileSystem) {
d := fs.NewDir(newRemote, fi.ModTime()) d := fs.NewDir(newRemote, fi.ModTime())
loopEntries[i] = d loopEntries[i] = d
return nil return nil
} }
} else { } else {
// Check whether this link should be translated // Check whether this link should be translated
if f.opt.TranslateSymlinks && fi.Mode()&os.ModeSymlink != 0 { if f.opt.TranslateSymlinks && mode&os.ModeSymlink != 0 {
newRemote += linkSuffix newRemote += linkSuffix
} }
// Don't include non directory if not included // Don't include non directory if not included