mirror of
https://github.com/rclone/rclone.git
synced 2025-01-25 15:49:33 +01:00
sync: fix --track-renames-strategy tests
This commit corrects the logic for --track-renames-strategy which broke the integration tests. It also improves the parsing of the argument and adds a test for that.
This commit is contained in:
parent
410e17a2ec
commit
93f5125f51
110
fs/sync/sync.go
110
fs/sync/sync.go
@ -37,7 +37,7 @@ type syncCopyMove struct {
|
|||||||
deletersWg sync.WaitGroup // for delete before go routine
|
deletersWg sync.WaitGroup // for delete before go routine
|
||||||
deleteFilesCh chan fs.Object // channel to receive deletes if delete before
|
deleteFilesCh chan fs.Object // channel to receive deletes if delete before
|
||||||
trackRenames bool // set if we should do server side renames
|
trackRenames bool // set if we should do server side renames
|
||||||
trackRenamesStrategy []string // stratgies used for tracking renames
|
trackRenamesStrategy trackRenamesStrategy // stratgies used for tracking renames
|
||||||
dstFilesMu sync.Mutex // protect dstFiles
|
dstFilesMu sync.Mutex // protect dstFiles
|
||||||
dstFiles map[string]fs.Object // dst files, always filled
|
dstFiles map[string]fs.Object // dst files, always filled
|
||||||
srcFiles map[string]fs.Object // src files, only used if deleteBefore
|
srcFiles map[string]fs.Object // src files, only used if deleteBefore
|
||||||
@ -68,30 +68,44 @@ type syncCopyMove struct {
|
|||||||
backupDir fs.Fs // place to store overwrites/deletes
|
backupDir fs.Fs // place to store overwrites/deletes
|
||||||
}
|
}
|
||||||
|
|
||||||
|
type trackRenamesStrategy byte
|
||||||
|
|
||||||
|
const (
|
||||||
|
trackRenamesStrategyHash trackRenamesStrategy = 1 << iota
|
||||||
|
trackRenamesStrategyModtime
|
||||||
|
)
|
||||||
|
|
||||||
|
func (strategy trackRenamesStrategy) hash() bool {
|
||||||
|
return (strategy & trackRenamesStrategyHash) != 0
|
||||||
|
}
|
||||||
|
|
||||||
|
func (strategy trackRenamesStrategy) modTime() bool {
|
||||||
|
return (strategy & trackRenamesStrategyModtime) != 0
|
||||||
|
}
|
||||||
|
|
||||||
func newSyncCopyMove(ctx context.Context, fdst, fsrc fs.Fs, deleteMode fs.DeleteMode, DoMove bool, deleteEmptySrcDirs bool, copyEmptySrcDirs bool) (*syncCopyMove, error) {
|
func newSyncCopyMove(ctx context.Context, fdst, fsrc fs.Fs, deleteMode fs.DeleteMode, DoMove bool, deleteEmptySrcDirs bool, copyEmptySrcDirs bool) (*syncCopyMove, error) {
|
||||||
if (deleteMode != fs.DeleteModeOff || DoMove) && operations.Overlapping(fdst, fsrc) {
|
if (deleteMode != fs.DeleteModeOff || DoMove) && operations.Overlapping(fdst, fsrc) {
|
||||||
return nil, fserrors.FatalError(fs.ErrorOverlapping)
|
return nil, fserrors.FatalError(fs.ErrorOverlapping)
|
||||||
}
|
}
|
||||||
s := &syncCopyMove{
|
s := &syncCopyMove{
|
||||||
fdst: fdst,
|
fdst: fdst,
|
||||||
fsrc: fsrc,
|
fsrc: fsrc,
|
||||||
deleteMode: deleteMode,
|
deleteMode: deleteMode,
|
||||||
DoMove: DoMove,
|
DoMove: DoMove,
|
||||||
copyEmptySrcDirs: copyEmptySrcDirs,
|
copyEmptySrcDirs: copyEmptySrcDirs,
|
||||||
deleteEmptySrcDirs: deleteEmptySrcDirs,
|
deleteEmptySrcDirs: deleteEmptySrcDirs,
|
||||||
dir: "",
|
dir: "",
|
||||||
srcFilesChan: make(chan fs.Object, fs.Config.Checkers+fs.Config.Transfers),
|
srcFilesChan: make(chan fs.Object, fs.Config.Checkers+fs.Config.Transfers),
|
||||||
srcFilesResult: make(chan error, 1),
|
srcFilesResult: make(chan error, 1),
|
||||||
dstFilesResult: make(chan error, 1),
|
dstFilesResult: make(chan error, 1),
|
||||||
dstEmptyDirs: make(map[string]fs.DirEntry),
|
dstEmptyDirs: make(map[string]fs.DirEntry),
|
||||||
srcEmptyDirs: make(map[string]fs.DirEntry),
|
srcEmptyDirs: make(map[string]fs.DirEntry),
|
||||||
noTraverse: fs.Config.NoTraverse,
|
noTraverse: fs.Config.NoTraverse,
|
||||||
noCheckDest: fs.Config.NoCheckDest,
|
noCheckDest: fs.Config.NoCheckDest,
|
||||||
deleteFilesCh: make(chan fs.Object, fs.Config.Checkers),
|
deleteFilesCh: make(chan fs.Object, fs.Config.Checkers),
|
||||||
trackRenames: fs.Config.TrackRenames,
|
trackRenames: fs.Config.TrackRenames,
|
||||||
trackRenamesStrategy: strings.Split(fs.Config.TrackRenamesStrategy, ","),
|
commonHash: fsrc.Hashes().Overlap(fdst.Hashes()).GetOne(),
|
||||||
commonHash: fsrc.Hashes().Overlap(fdst.Hashes()).GetOne(),
|
trackRenamesCh: make(chan fs.Object, fs.Config.Checkers),
|
||||||
trackRenamesCh: make(chan fs.Object, fs.Config.Checkers),
|
|
||||||
}
|
}
|
||||||
var err error
|
var err error
|
||||||
s.toBeChecked, err = newPipe(fs.Config.OrderBy, accounting.Stats(ctx).SetCheckQueue, fs.Config.MaxBacklog)
|
s.toBeChecked, err = newPipe(fs.Config.OrderBy, accounting.Stats(ctx).SetCheckQueue, fs.Config.MaxBacklog)
|
||||||
@ -118,6 +132,10 @@ func newSyncCopyMove(ctx context.Context, fdst, fsrc fs.Fs, deleteMode fs.Delete
|
|||||||
fs.Errorf(nil, "Ignoring --no-traverse with sync")
|
fs.Errorf(nil, "Ignoring --no-traverse with sync")
|
||||||
s.noTraverse = false
|
s.noTraverse = false
|
||||||
}
|
}
|
||||||
|
s.trackRenamesStrategy, err = parseTrackRenamesStrategy(fs.Config.TrackRenamesStrategy)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
if s.noCheckDest {
|
if s.noCheckDest {
|
||||||
if s.deleteMode != fs.DeleteModeOff {
|
if s.deleteMode != fs.DeleteModeOff {
|
||||||
return nil, errors.New("can't use --no-check-dest with sync: use copy instead")
|
return nil, errors.New("can't use --no-check-dest with sync: use copy instead")
|
||||||
@ -135,12 +153,12 @@ func newSyncCopyMove(ctx context.Context, fdst, fsrc fs.Fs, deleteMode fs.Delete
|
|||||||
fs.Errorf(fdst, "Ignoring --track-renames as the destination does not support server-side move or copy")
|
fs.Errorf(fdst, "Ignoring --track-renames as the destination does not support server-side move or copy")
|
||||||
s.trackRenames = false
|
s.trackRenames = false
|
||||||
}
|
}
|
||||||
if s.commonHash == hash.None && isUsingRenameStrategy("hash", s.trackRenamesStrategy) {
|
if s.trackRenamesStrategy.hash() && s.commonHash == hash.None {
|
||||||
fs.Errorf(fdst, "Ignoring --track-renames as the source and destination do not have a common hash")
|
fs.Errorf(fdst, "Ignoring --track-renames as the source and destination do not have a common hash")
|
||||||
s.trackRenames = false
|
s.trackRenames = false
|
||||||
}
|
}
|
||||||
|
|
||||||
if fs.GetModifyWindow(fsrc, fdst) == fs.ModTimeNotSupported && isUsingRenameStrategy("modtime", s.trackRenamesStrategy) {
|
if s.trackRenamesStrategy.modTime() && fs.GetModifyWindow(fsrc, fdst) == fs.ModTimeNotSupported {
|
||||||
fs.Errorf(fdst, "Ignoring --track-renames as either the source or destination do not support modtime")
|
fs.Errorf(fdst, "Ignoring --track-renames as either the source or destination do not support modtime")
|
||||||
s.trackRenames = false
|
s.trackRenames = false
|
||||||
}
|
}
|
||||||
@ -569,25 +587,35 @@ func (s *syncCopyMove) srcParentDirCheck(entry fs.DirEntry) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// isUsingRenameStrategy checks if a strategy is included in a list of strategies
|
// parseTrackRenamesStrategy turns a config string into a trackRenamesStrategy
|
||||||
func isUsingRenameStrategy(strategy string, strategies []string) bool {
|
func parseTrackRenamesStrategy(strategies string) (strategy trackRenamesStrategy, err error) {
|
||||||
for _, s := range strategies {
|
if len(strategies) == 0 {
|
||||||
if s == strategy {
|
return strategy, nil
|
||||||
return true
|
}
|
||||||
|
for _, s := range strings.Split(strategies, ",") {
|
||||||
|
switch s {
|
||||||
|
case "hash":
|
||||||
|
strategy |= trackRenamesStrategyHash
|
||||||
|
case "modtime":
|
||||||
|
strategy |= trackRenamesStrategyModtime
|
||||||
|
case "size":
|
||||||
|
// ignore
|
||||||
|
default:
|
||||||
|
return strategy, errors.Errorf("unknown track renames strategy %q", s)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return false
|
return strategy, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// renameID makes a string with the size and the other identifiers of the requested rename strategies
|
// renameID makes a string with the size and the other identifiers of the requested rename strategies
|
||||||
//
|
//
|
||||||
// it may return an empty string in which case no hash could be made
|
// it may return an empty string in which case no hash could be made
|
||||||
func (s *syncCopyMove) renameID(obj fs.Object, renameStrategy []string, precision time.Duration) string {
|
func (s *syncCopyMove) renameID(obj fs.Object, renamesStrategy trackRenamesStrategy, precision time.Duration) string {
|
||||||
var builder strings.Builder
|
var builder strings.Builder
|
||||||
|
|
||||||
fmt.Fprintf(&builder, "%d", obj.Size())
|
fmt.Fprintf(&builder, "%d", obj.Size())
|
||||||
|
|
||||||
if isUsingRenameStrategy("hash", renameStrategy) {
|
if renamesStrategy.hash() {
|
||||||
var err error
|
var err error
|
||||||
hash, err := obj.Hash(s.ctx, s.commonHash)
|
hash, err := obj.Hash(s.ctx, s.commonHash)
|
||||||
|
|
||||||
@ -602,16 +630,28 @@ func (s *syncCopyMove) renameID(obj fs.Object, renameStrategy []string, precisio
|
|||||||
fmt.Fprintf(&builder, ",%s", hash)
|
fmt.Fprintf(&builder, ",%s", hash)
|
||||||
}
|
}
|
||||||
|
|
||||||
if isUsingRenameStrategy("modtime", renameStrategy) {
|
if renamesStrategy.modTime() {
|
||||||
|
// We normally check precisions with
|
||||||
|
// dt = a - b
|
||||||
|
// if dt < precision && dt > -precision then times match
|
||||||
|
//
|
||||||
|
// We want to emulate this with a hash.
|
||||||
|
//
|
||||||
|
// hash(x) = floor((x + divisor/2 - 1)/divisor)
|
||||||
|
// for a given hash value, H
|
||||||
|
// x > H*divisor - divisor/2 &&
|
||||||
|
// x < H*divisor + divisor/2
|
||||||
|
//
|
||||||
|
// To match the normal check of +/- precision then we
|
||||||
|
// need to use 2*precision as the divisor
|
||||||
modTime := obj.ModTime(s.ctx)
|
modTime := obj.ModTime(s.ctx)
|
||||||
divisor := precision.Nanoseconds() / 2
|
divisor := precision.Nanoseconds() * 2
|
||||||
rounding := divisor / 2
|
rounding := divisor / 2
|
||||||
|
|
||||||
if rounding > 0 {
|
if rounding > 0 {
|
||||||
rounding--
|
rounding--
|
||||||
}
|
}
|
||||||
|
// NB modTime.Unix()*int64(time.Nanosecond) won't overflow an int64 until the year 2264
|
||||||
timeHash := (modTime.Unix()*time.Nanosecond.Nanoseconds() + int64(modTime.Nanosecond()) + rounding)
|
timeHash := (modTime.Unix()*int64(time.Nanosecond) + modTime.UnixNano() + rounding) / divisor
|
||||||
fmt.Fprintf(&builder, ",%d", timeHash)
|
fmt.Fprintf(&builder, ",%d", timeHash)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1127,19 +1127,51 @@ func TestSyncWithTrackRenames(t *testing.T) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func TestParseRenamesStrategyModtime(t *testing.T) {
|
||||||
|
for _, test := range []struct {
|
||||||
|
in string
|
||||||
|
want trackRenamesStrategy
|
||||||
|
wantErr bool
|
||||||
|
}{
|
||||||
|
{"", 0, false},
|
||||||
|
{"modtime", trackRenamesStrategyModtime, false},
|
||||||
|
{"hash", trackRenamesStrategyHash, false},
|
||||||
|
{"size", 0, false},
|
||||||
|
{"modtime,hash", trackRenamesStrategyModtime | trackRenamesStrategyHash, false},
|
||||||
|
{"hash,modtime,size", trackRenamesStrategyModtime | trackRenamesStrategyHash, false},
|
||||||
|
{"size,boom", 0, true},
|
||||||
|
} {
|
||||||
|
got, err := parseTrackRenamesStrategy(test.in)
|
||||||
|
assert.Equal(t, test.want, got, test.in)
|
||||||
|
assert.Equal(t, test.wantErr, err != nil, test.in)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestRenamesStrategyModtime(t *testing.T) {
|
||||||
|
both := trackRenamesStrategyHash | trackRenamesStrategyModtime
|
||||||
|
hash := trackRenamesStrategyHash
|
||||||
|
modTime := trackRenamesStrategyModtime
|
||||||
|
|
||||||
|
assert.True(t, both.hash())
|
||||||
|
assert.True(t, both.modTime())
|
||||||
|
assert.True(t, hash.hash())
|
||||||
|
assert.False(t, hash.modTime())
|
||||||
|
assert.False(t, modTime.hash())
|
||||||
|
assert.True(t, modTime.modTime())
|
||||||
|
}
|
||||||
|
|
||||||
func TestSyncWithTrackRenamesStrategyModtime(t *testing.T) {
|
func TestSyncWithTrackRenamesStrategyModtime(t *testing.T) {
|
||||||
r := fstest.NewRun(t)
|
r := fstest.NewRun(t)
|
||||||
defer r.Finalise()
|
defer r.Finalise()
|
||||||
|
|
||||||
fs.Config.TrackRenames = true
|
fs.Config.TrackRenames = true
|
||||||
fs.Config.TrackRenamesStrategy = "hash,modtime"
|
fs.Config.TrackRenamesStrategy = "modtime"
|
||||||
defer func() {
|
defer func() {
|
||||||
fs.Config.TrackRenames = false
|
fs.Config.TrackRenames = false
|
||||||
fs.Config.TrackRenamesStrategy = "hash"
|
fs.Config.TrackRenamesStrategy = "hash"
|
||||||
}()
|
}()
|
||||||
|
|
||||||
haveHash := r.Fremote.Hashes().Overlap(r.Flocal.Hashes()).GetOne() != hash.None
|
canTrackRenames := operations.CanServerSideMove(r.Fremote)
|
||||||
canTrackRenames := haveHash && operations.CanServerSideMove(r.Fremote)
|
|
||||||
t.Logf("Can track renames: %v", canTrackRenames)
|
t.Logf("Can track renames: %v", canTrackRenames)
|
||||||
|
|
||||||
f1 := r.WriteFile("potato", "Potato Content", t1)
|
f1 := r.WriteFile("potato", "Potato Content", t1)
|
||||||
|
Loading…
Reference in New Issue
Block a user