sync: use operations.DirMove instead of sync.MoveDir for --fix-case - #7591
This should be more efficient for the purposes of --fix-case, as operations.DirMove accepts `srcRemote` and `dstRemote` arguments, while sync.MoveDir does not. This also factors the two-step-move logic to operations.DirMoveCaseInsensitive, so that it is reusable by other commands.
This commit is contained in:
parent
dfe76570a1
commit
137f7f62fb
2 changed files with 15 additions and 19 deletions
|
@ -2288,6 +2288,17 @@ func DirMove(ctx context.Context, f fs.Fs, srcRemote, dstRemote string) (err err
|
|||
return nil
|
||||
}
|
||||
|
||||
// DirMoveCaseInsensitive does DirMove in two steps (to temp name, then real name)
|
||||
// which is necessary for some case-insensitive backends
|
||||
func DirMoveCaseInsensitive(ctx context.Context, f fs.Fs, srcRemote, dstRemote string) (err error) {
|
||||
tmpDstRemote := dstRemote + "-rclone-move-" + random.String(8)
|
||||
err = DirMove(ctx, f, srcRemote, tmpDstRemote)
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
return DirMove(ctx, f, tmpDstRemote, dstRemote)
|
||||
}
|
||||
|
||||
// FsInfo provides information about a remote
|
||||
type FsInfo struct {
|
||||
// Name of the remote (as passed into NewFs)
|
||||
|
|
|
@ -6,7 +6,6 @@ import (
|
|||
"errors"
|
||||
"fmt"
|
||||
"path"
|
||||
"path/filepath"
|
||||
"sort"
|
||||
"strings"
|
||||
"sync"
|
||||
|
@ -19,7 +18,6 @@ import (
|
|||
"github.com/rclone/rclone/fs/hash"
|
||||
"github.com/rclone/rclone/fs/march"
|
||||
"github.com/rclone/rclone/fs/operations"
|
||||
"github.com/rclone/rclone/lib/random"
|
||||
)
|
||||
|
||||
// ErrorMaxDurationReached defines error when transfer duration is reached
|
||||
|
@ -1151,27 +1149,14 @@ func (s *syncCopyMove) Match(ctx context.Context, dst, src fs.DirEntry) (recurse
|
|||
if s.ci.FixCase && !s.ci.Immutable && src.Remote() != dst.Remote() {
|
||||
// Fix case for case insensitive filesystems
|
||||
// Fix each dir before recursing into subdirs and files
|
||||
oldDirFs, err := fs.NewFs(s.ctx, filepath.Join(fs.ConfigStringFull(s.fdst), dst.Remote()))
|
||||
s.processError(err)
|
||||
newDirPath := filepath.Join(fs.ConfigStringFull(s.fdst), filepath.Dir(dst.Remote()), filepath.Base(src.Remote()))
|
||||
newDirFs, err := fs.NewFs(s.ctx, newDirPath)
|
||||
s.processError(err)
|
||||
// Create random name to temporarily move dir to
|
||||
tmpDirName := newDirPath + "-rclone-move-" + random.String(8)
|
||||
tmpDirFs, err := fs.NewFs(s.ctx, tmpDirName)
|
||||
s.processError(err)
|
||||
if err = MoveDir(s.ctx, tmpDirFs, oldDirFs, s.deleteEmptySrcDirs, s.copyEmptySrcDirs); err != nil {
|
||||
fs.Errorf(dst, "Error while attempting to move dir to temporary location %s: %v", tmpDirName, err)
|
||||
s.processError(err)
|
||||
} else {
|
||||
if err = MoveDir(s.ctx, newDirFs, tmpDirFs, s.deleteEmptySrcDirs, s.copyEmptySrcDirs); err != nil {
|
||||
err := operations.DirMoveCaseInsensitive(s.ctx, s.fdst, dst.Remote(), src.Remote())
|
||||
if err != nil {
|
||||
fs.Errorf(dst, "Error while attempting to rename to %s: %v", src.Remote(), err)
|
||||
s.processError(err)
|
||||
} else {
|
||||
fs.Infof(dst, "Fixed case by renaming to: %s", src.Remote())
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
return true
|
||||
}
|
||||
|
|
Loading…
Reference in a new issue