From 95d0410baacba5cb2c6fa36433c184c55dcbe162 Mon Sep 17 00:00:00 2001 From: Manish Gupta Date: Thu, 12 Nov 2020 17:02:55 +0530 Subject: [PATCH] local: continue listing files/folders when a circular symlink is detected Before this change a circular symlink would cause rclone to error out from the listings. After this change rclone will skip a circular symlink and carry on the listing, producing an error at the end. Fixes #4743 --- backend/local/local.go | 4 ++-- backend/local/symlink.go | 22 ++++++++++++++++++++++ backend/local/symlink_other.go | 17 +++++++++++++++++ 3 files changed, 41 insertions(+), 2 deletions(-) create mode 100644 backend/local/symlink.go create mode 100644 backend/local/symlink_other.go diff --git a/backend/local/local.go b/backend/local/local.go index 7cb11ad45..9b3aa7b76 100644 --- a/backend/local/local.go +++ b/backend/local/local.go @@ -456,8 +456,8 @@ func (f *Fs) List(ctx context.Context, dir string) (entries fs.DirEntries, err e if f.opt.FollowSymlinks && (mode&os.ModeSymlink) != 0 { localPath := filepath.Join(fsDirPath, name) fi, err = os.Stat(localPath) - if os.IsNotExist(err) { - // Skip bad symlinks + if os.IsNotExist(err) || isCircularSymlinkError(err) { + // Skip bad symlinks and circular symlinks err = fserrors.NoRetryError(errors.Wrap(err, "symlink")) fs.Errorf(newRemote, "Listing error: %v", err) err = accounting.Stats(ctx).Error(err) diff --git a/backend/local/symlink.go b/backend/local/symlink.go new file mode 100644 index 000000000..cf49c7282 --- /dev/null +++ b/backend/local/symlink.go @@ -0,0 +1,22 @@ +// +build !windows,!plan9,!js + +package local + +import ( + "os" + "syscall" +) + +// isCircularSymlinkError checks if the current error code is because of a circular symlink +func isCircularSymlinkError(err error) bool { + if err != nil { + if newerr, ok := err.(*os.PathError); ok { + if errcode, ok := newerr.Err.(syscall.Errno); ok { + if errcode == syscall.ELOOP { + return true + } + } + } + } + return false +} diff --git a/backend/local/symlink_other.go b/backend/local/symlink_other.go new file mode 100644 index 000000000..36e012768 --- /dev/null +++ b/backend/local/symlink_other.go @@ -0,0 +1,17 @@ +// +build windows plan9 js + +package local + +import ( + "strings" +) + +// isCircularSymlinkError checks if the current error code is because of a circular symlink +func isCircularSymlinkError(err error) bool { + if err != nil { + if strings.Contains(err.Error(), "The name of the file cannot be resolved by the system") { + return true + } + } + return false +}