forked from TrueCloudLab/rclone
Implement -I, --ignore-times for unconditional upload - fixes #311
This commit is contained in:
parent
cb22583212
commit
ced3a4bc19
4 changed files with 54 additions and 0 deletions
|
@ -399,6 +399,15 @@ While this isn't a generally recommended option, it can be useful
|
||||||
in cases where your files change due to encryption. However, it cannot
|
in cases where your files change due to encryption. However, it cannot
|
||||||
correct partial transfers in case a transfer was interrupted.
|
correct partial transfers in case a transfer was interrupted.
|
||||||
|
|
||||||
|
### -I, --ignore-times ###
|
||||||
|
|
||||||
|
Using this option will cause rclone to unconditionally upload all
|
||||||
|
files regardless of the state of files on the destination.
|
||||||
|
|
||||||
|
Normally rclone would skip any files that have the same
|
||||||
|
modification time and are the same size (or have the same checksum if
|
||||||
|
using `--checksum`).
|
||||||
|
|
||||||
### --log-file=FILE ###
|
### --log-file=FILE ###
|
||||||
|
|
||||||
Log all of rclone's output to FILE. This is not active by default.
|
Log all of rclone's output to FILE. This is not active by default.
|
||||||
|
|
|
@ -69,6 +69,7 @@ var (
|
||||||
configFile = pflag.StringP("config", "", ConfigPath, "Config file.")
|
configFile = pflag.StringP("config", "", ConfigPath, "Config file.")
|
||||||
checkSum = pflag.BoolP("checksum", "c", false, "Skip based on checksum & size, not mod-time & size")
|
checkSum = pflag.BoolP("checksum", "c", false, "Skip based on checksum & size, not mod-time & size")
|
||||||
sizeOnly = pflag.BoolP("size-only", "", false, "Skip based on size only, not mod-time or checksum")
|
sizeOnly = pflag.BoolP("size-only", "", false, "Skip based on size only, not mod-time or checksum")
|
||||||
|
ignoreTimes = pflag.BoolP("ignore-times", "I", false, "Don't skip files that match size and time - transfer all files")
|
||||||
ignoreExisting = pflag.BoolP("ignore-existing", "", false, "Skip all files that exist on destination")
|
ignoreExisting = pflag.BoolP("ignore-existing", "", false, "Skip all files that exist on destination")
|
||||||
dryRun = pflag.BoolP("dry-run", "n", false, "Do a trial run with no permanent changes")
|
dryRun = pflag.BoolP("dry-run", "n", false, "Do a trial run with no permanent changes")
|
||||||
connectTimeout = pflag.DurationP("contimeout", "", 60*time.Second, "Connect timeout")
|
connectTimeout = pflag.DurationP("contimeout", "", 60*time.Second, "Connect timeout")
|
||||||
|
@ -188,6 +189,7 @@ type ConfigInfo struct {
|
||||||
DryRun bool
|
DryRun bool
|
||||||
CheckSum bool
|
CheckSum bool
|
||||||
SizeOnly bool
|
SizeOnly bool
|
||||||
|
IgnoreTimes bool
|
||||||
IgnoreExisting bool
|
IgnoreExisting bool
|
||||||
ModifyWindow time.Duration
|
ModifyWindow time.Duration
|
||||||
Checkers int
|
Checkers int
|
||||||
|
@ -299,6 +301,7 @@ func LoadConfig() {
|
||||||
Config.ConnectTimeout = *connectTimeout
|
Config.ConnectTimeout = *connectTimeout
|
||||||
Config.CheckSum = *checkSum
|
Config.CheckSum = *checkSum
|
||||||
Config.SizeOnly = *sizeOnly
|
Config.SizeOnly = *sizeOnly
|
||||||
|
Config.IgnoreTimes = *ignoreTimes
|
||||||
Config.IgnoreExisting = *ignoreExisting
|
Config.IgnoreExisting = *ignoreExisting
|
||||||
Config.DumpHeaders = *dumpHeaders
|
Config.DumpHeaders = *dumpHeaders
|
||||||
Config.DumpBodies = *dumpBodies
|
Config.DumpBodies = *dumpBodies
|
||||||
|
|
|
@ -315,6 +315,12 @@ func checkOne(pair ObjectPair, out ObjectPairChan) {
|
||||||
Debug(src, "Destination exists, skipping")
|
Debug(src, "Destination exists, skipping")
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
// If we should upload unconditionally
|
||||||
|
if Config.IgnoreTimes {
|
||||||
|
Debug(src, "Uploading unconditionally as --ignore-times is in use")
|
||||||
|
out <- pair
|
||||||
|
return
|
||||||
|
}
|
||||||
// If UpdateOlder is in effect, skip if dst is newer than src
|
// If UpdateOlder is in effect, skip if dst is newer than src
|
||||||
if Config.UpdateOlder {
|
if Config.UpdateOlder {
|
||||||
srcModTime := src.ModTime()
|
srcModTime := src.ModTime()
|
||||||
|
|
|
@ -432,6 +432,42 @@ func TestSyncSizeOnly(t *testing.T) {
|
||||||
fstest.CheckItems(t, r.fremote, file1)
|
fstest.CheckItems(t, r.fremote, file1)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func TestSyncIgnoreTimes(t *testing.T) {
|
||||||
|
r := NewRun(t)
|
||||||
|
defer r.Finalise()
|
||||||
|
file1 := r.WriteBoth("existing", "potato", t1)
|
||||||
|
|
||||||
|
fs.Stats.ResetCounters()
|
||||||
|
err := fs.Sync(r.fremote, r.flocal)
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("Sync failed: %v", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
// We should have transferred exactly 0 files because the
|
||||||
|
// files were identical.
|
||||||
|
if fs.Stats.GetTransfers() != 0 {
|
||||||
|
t.Fatalf("Sync 1: want 0 transfer, got %d", fs.Stats.GetTransfers())
|
||||||
|
}
|
||||||
|
|
||||||
|
fs.Config.IgnoreTimes = true
|
||||||
|
defer func() { fs.Config.IgnoreTimes = false }()
|
||||||
|
|
||||||
|
fs.Stats.ResetCounters()
|
||||||
|
err = fs.Sync(r.fremote, r.flocal)
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("Sync failed: %v", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
// We should have transferred exactly one file even though the
|
||||||
|
// files were identical.
|
||||||
|
if fs.Stats.GetTransfers() != 1 {
|
||||||
|
t.Fatalf("Sync 2: want 1 transfer, got %d", fs.Stats.GetTransfers())
|
||||||
|
}
|
||||||
|
|
||||||
|
fstest.CheckItems(t, r.flocal, file1)
|
||||||
|
fstest.CheckItems(t, r.fremote, file1)
|
||||||
|
}
|
||||||
|
|
||||||
func TestSyncIgnoreExisting(t *testing.T) {
|
func TestSyncIgnoreExisting(t *testing.T) {
|
||||||
r := NewRun(t)
|
r := NewRun(t)
|
||||||
defer r.Finalise()
|
defer r.Finalise()
|
||||||
|
|
Loading…
Reference in a new issue