bisync
is in beta and is considered an advanced command, so use with care.
Make sure you have read and understood the entire manual (especially the Limitations section) before using, or data loss can result. Questions can be asked in the Rclone Forum.
~/.cache/rclone/bisync
on Linux, /Users/yourusername/Library/Caches/rclone/bisync
on Mac,
or C:\Users\MyLogin\AppData\Local\rclone\bisync
on Windows.
Make sure that this location is writable.--resync
flag, specifying the paths
to the local and remote sync directory roots.--resync
flag. (Important!)For example, your first command might look like this:
rclone bisync remote1:path1 remote2:path2 --create-empty-src-dirs --compare size,modtime,checksum --slow-hash-sync-only --resilient -MvP --drive-skip-gdocs --fix-case --resync --dry-run
If all looks good, run it again without --dry-run
. After that, remove --resync
as well.
Here is a typical run log (with timestamps removed for clarity):
rclone bisync /testdir/path1/ /testdir/path2/ --verbose
INFO : Synching Path1 "/testdir/path1/" with Path2 "/testdir/path2/"
INFO : Path1 checking for diffs
INFO : - Path1 File is new - file11.txt
INFO : - Path1 File is newer - file2.txt
INFO : - Path1 File is newer - file5.txt
INFO : - Path1 File is newer - file7.txt
INFO : - Path1 File was deleted - file4.txt
INFO : - Path1 File was deleted - file6.txt
INFO : - Path1 File was deleted - file8.txt
INFO : Path1: 7 changes: 1 new, 3 newer, 0 older, 3 deleted
INFO : Path2 checking for diffs
INFO : - Path2 File is new - file10.txt
INFO : - Path2 File is newer - file1.txt
INFO : - Path2 File is newer - file5.txt
INFO : - Path2 File is newer - file6.txt
INFO : - Path2 File was deleted - file3.txt
INFO : - Path2 File was deleted - file7.txt
INFO : - Path2 File was deleted - file8.txt
INFO : Path2: 7 changes: 1 new, 3 newer, 0 older, 3 deleted
INFO : Applying changes
INFO : - Path1 Queue copy to Path2 - /testdir/path2/file11.txt
INFO : - Path1 Queue copy to Path2 - /testdir/path2/file2.txt
INFO : - Path2 Queue delete - /testdir/path2/file4.txt
NOTICE: - WARNING New or changed in both paths - file5.txt
NOTICE: - Path1 Renaming Path1 copy - /testdir/path1/file5.txt..path1
NOTICE: - Path1 Queue copy to Path2 - /testdir/path2/file5.txt..path1
NOTICE: - Path2 Renaming Path2 copy - /testdir/path2/file5.txt..path2
NOTICE: - Path2 Queue copy to Path1 - /testdir/path1/file5.txt..path2
INFO : - Path2 Queue copy to Path1 - /testdir/path1/file6.txt
INFO : - Path1 Queue copy to Path2 - /testdir/path2/file7.txt
INFO : - Path2 Queue copy to Path1 - /testdir/path1/file1.txt
INFO : - Path2 Queue copy to Path1 - /testdir/path1/file10.txt
INFO : - Path1 Queue delete - /testdir/path1/file3.txt
INFO : - Path2 Do queued copies to - Path1
INFO : - Path1 Do queued copies to - Path2
INFO : - Do queued deletes on - Path1
INFO : - Do queued deletes on - Path2
INFO : Updating listings
INFO : Validating listings for Path1 "/testdir/path1/" vs Path2 "/testdir/path2/"
INFO : Bisync successful
$ rclone bisync --help
Usage:
rclone bisync remote1:path1 remote2:path2 [flags]
Positional arguments:
Path1, Path2 Local path, or remote storage with ':' plus optional path.
Type 'rclone listremotes' for list of configured remotes.
Optional Flags:
--backup-dir1 string --backup-dir for Path1. Must be a non-overlapping path on the same remote.
--backup-dir2 string --backup-dir for Path2. Must be a non-overlapping path on the same remote.
--check-access Ensure expected RCLONE_TEST files are found on both Path1 and Path2 filesystems, else abort.
--check-filename string Filename for --check-access (default: RCLONE_TEST)
--check-sync string Controls comparison of final listings: true|false|only (default: true) (default "true")
--compare string Comma-separated list of bisync-specific compare options ex. 'size,modtime,checksum' (default: 'size,modtime')
--conflict-loser ConflictLoserAction Action to take on the loser of a sync conflict (when there is a winner) or on both files (when there is no winner): , num, pathname, delete (default: num)
--conflict-resolve string Automatically resolve conflicts by preferring the version that is: none, path1, path2, newer, older, larger, smaller (default: none) (default "none")
--conflict-suffix string Suffix to use when renaming a --conflict-loser. Can be either one string or two comma-separated strings to assign different suffixes to Path1/Path2. (default: 'conflict')
--create-empty-src-dirs Sync creation and deletion of empty directories. (Not compatible with --remove-empty-dirs)
--download-hash Compute hash by downloading when otherwise unavailable. (warning: may be slow and use lots of data!)
--filters-file string Read filtering patterns from a file
--force Bypass --max-delete safety check and run the sync. Consider using with --verbose
-h, --help help for bisync
--ignore-listing-checksum Do not use checksums for listings (add --ignore-checksum to additionally skip post-copy checksum checks)
--max-lock Duration Consider lock files older than this to be expired (default: 0 (never expire)) (minimum: 2m) (default 0s)
--no-cleanup Retain working files (useful for troubleshooting and testing).
--no-slow-hash Ignore listing checksums only on backends where they are slow
--recover Automatically recover from interruptions without requiring --resync.
--remove-empty-dirs Remove ALL empty directories at the final cleanup step.
--resilient Allow future runs to retry after certain less-serious errors, instead of requiring --resync. Use at your own risk!
-1, --resync Performs the resync run. Equivalent to --resync-mode path1. Consider using --verbose or --dry-run first.
--resync-mode string During resync, prefer the version that is: path1, path2, newer, older, larger, smaller (default: path1 if --resync, otherwise none for no resync.) (default "none")
--retries int Retry operations this many times if they fail (requires --resilient). (default 3)
--retries-sleep Duration Interval between retrying operations if they fail, e.g. 500ms, 60s, 5m (0 to disable) (default 0s)
--slow-hash-sync-only Ignore slow checksums for listings and deltas, but still consider them during sync calls.
--workdir string Use custom working dir - useful for testing. (default: {WORKDIR})
--max-delete PERCENT Safety check on maximum percentage of deleted files allowed. If exceeded, the bisync run will abort. (default: 50%)
-n, --dry-run Go through the motions - No files are copied/deleted.
-v, --verbose Increases logging verbosity. May be specified more than once for more details.
Arbitrary rclone flags may be specified on the
bisync command line, for example
rclone bisync ./testdir/path1/ gdrive:testdir/path2/ --drive-skip-gdocs -v -v --timeout 10s
Note that interactions of various rclone flags with bisync process flow
has not been fully tested yet.
Path1 and Path2 arguments may be references to any mix of local directory
paths (absolute or relative), UNC paths (//server/share/path
),
Windows drive paths (with a drive letter and :
) or configured
remotes with optional subdirectory paths.
Cloud references are distinguished by having a :
in the argument
(see Windows support below).
Path1 and Path2 are treated equally, in that neither has priority for
file changes (except during --resync
), and access efficiency does not change whether a remote
is on Path1 or Path2.
The listings in bisync working directory (default: ~/.cache/rclone/bisync
)
are named based on the Path1 and Path2 arguments so that separate syncs
to individual directories within the tree may be set up, e.g.:
path_to_local_tree..dropbox_subdir.lst
.
Any empty directories after the sync on both the Path1 and Path2
filesystems are not deleted by default, unless --create-empty-src-dirs
is specified.
If the --remove-empty-dirs
flag is specified, then both paths will have ALL empty directories purged
as the last step in the process.
This will effectively make both Path1 and Path2 filesystems contain a matching superset of all files. By default, Path2 files that do not exist in Path1 will be copied to Path1, and the process will then copy the Path1 tree to Path2.
The --resync
sequence is roughly equivalent to the following (but see --resync-mode
for other options):
rclone copy Path2 Path1 --ignore-existing [--create-empty-src-dirs]
rclone copy Path1 Path2 [--create-empty-src-dirs]
The base directories on both Path1 and Path2 filesystems must exist or bisync will fail. This is required for safety - that bisync can verify that both paths are valid.
When using --resync
, a newer version of a file on the Path2 filesystem
will (by default) be overwritten by the Path1 filesystem version.
(Note that this is NOT entirely symmetrical, and more symmetrical options can be specified with the --resync-mode
flag.)
Carefully evaluate deltas using --dry-run.
For a resync run, one of the paths may be empty (no files in the path tree). The resync run should result in files on both paths, else a normal non-resync run will fail.
For a non-resync run, either path being empty (no files in the tree) fails with
Empty current PathN listing. Cannot sync to an empty directory: X.pathN.lst
This is a safety check that an unexpected empty path does not result in
deleting everything in the other path.
Note that --resync
implies --resync-mode path1
unless a different
--resync-mode
is explicitly specified.
It is not necessary to use both the --resync
and --resync-mode
flags --
either one is sufficient without the other.
Note: --resync
(including --resync-mode
) should only be used under three specific (rare) circumstances:
--filters-file
)--resync
to recoverThe rest of the time, you should omit --resync
. The reason is because --resync
will only copy (not sync) each side to the other.
Therefore, if you included --resync
for every bisync run, it would never be possible to delete a file --
the deleted file would always keep reappearing at the end of every run (because it's being copied from the other side where it still exists).
Similarly, renaming a file would always result in a duplicate copy (both old and new name) on both sides.
If you find that frequent interruptions from #3 are an issue, rather than
automatically running --resync
, the recommended alternative is to use the
--resilient
, --recover
, and
--conflict-resolve
flags, (along with Graceful
Shutdown mode, when needed) for a very robust
"set-it-and-forget-it" bisync setup that can automatically bounce back from
almost any interruption it might encounter. Consider adding something like the
following:
--resilient --recover --max-lock 2m --conflict-resolve newer
In the event that a file differs on both sides during a --resync
,
--resync-mode
controls which version will overwrite the other. The supported
options are similar to --conflict-resolve
. For all of
the following options, the version that is kept is referred to as the "winner",
and the version that is overwritten (deleted) is referred to as the "loser".
The options are named after the "winner":
path1
- (the default) - the version from Path1 is unconditionally
considered the winner (regardless of modtime
and size
, if any). This can be
useful if one side is more trusted or up-to-date than the other, at the time of
the --resync
.path2
- same as path1
, except the path2 version is considered the winner.newer
- the newer file (by modtime
) is considered the winner, regardless
of which side it came from. This may result in having a mix of some winners
from Path1, and some winners from Path2. (The implementation is analogous to
running rclone copy --update
in both directions.)older
- same as newer
, except the older file is considered the winner,
and the newer file is considered the loser.larger
- the larger file (by size
) is considered the winner (regardless
of modtime
, if any). This can be a useful option for remotes without
modtime
support, or with the kinds of files (such as logs) that tend to grow
but not shrink, over time.smaller
- the smaller file (by size
) is considered the winner (regardless
of modtime
, if any).For all of the above options, note the following:
path1
. (For example, if
--resync-mode newer
is set, but one of the paths uses a remote that doesn't
support modtime
.)--compare
methods in effect.
(For example, if --resync-mode newer
is set, but the Path1 and Path2 modtimes
are identical, bisync will compare the sizes.) If bisync concludes that they
differ, preference is given to whichever is the "source" at that moment. (In
practice, this gives a slight advantage to Path2, as the 2to1 copy comes before
the 1to2 copy.) If the files do not differ, nothing is copied (as both sides
are already correct).--resync
(this is one of the main
differences between resync and non-resync runs.).--conflict-resolve
, --conflict-loser
, and --conflict-suffix
do not
apply during --resync
, and unlike these flags, nothing is renamed during
--resync
. When a file differs on both sides during --resync
, one version
always overwrites the other (much like in rclone copy
.) (Consider using
--backup-dir
to retain a backup of the losing
version.)--conflict-resolve
, --resync-mode none
is not a valid option
(or rather, it will be interpreted as "no resync", unless --resync
has also
been specified, in which case it will be ignored.)path1
and path2
options typically produce a similar result.)--resync
flag implies
--resync-mode path1
unless a different --resync-mode
is explicitly
specified. Similarly, all --resync-mode
options (except none
) imply
--resync
, so it is not necessary to use both the --resync
and
--resync-mode
flags simultaneously -- either one is sufficient without the
other.Access check files are an additional safety measure against data loss.
bisync will ensure it can find matching RCLONE_TEST
files in the same places
in the Path1 and Path2 filesystems.
RCLONE_TEST
files are not generated automatically.
For --check-access
to succeed, you must first either:
A) Place one or more RCLONE_TEST
files in both systems, or
B) Set --check-filename
to a filename already in use in various locations
throughout your sync'd fileset. Recommended methods for A) include:
rclone touch Path1/RCLONE_TEST
(create a new file)rclone copyto Path1/RCLONE_TEST Path2/RCLONE_TEST
(copy an existing file)rclone copy Path1/RCLONE_TEST Path2/RCLONE_TEST --include "RCLONE_TEST"
(copy multiple files at once, recursively)bisync
once without --check-access
to set matching files on both filesystems
will also work, but is not preferred, due to potential for user error
(you are temporarily disabling the safety feature).Note that --check-access
is still enforced on --resync
, so bisync --resync --check-access
will not work as a method of initially setting the files (this is to ensure that bisync can't
inadvertently circumvent its own safety switch.)
Time stamps and file contents for RCLONE_TEST
files are not important, just the names and locations.
If you have symbolic links in your sync tree it is recommended to place
RCLONE_TEST
files in the linked-to directory tree to protect against
bisync assuming a bunch of deleted files if the linked-to tree should not be
accessible.
See also the --check-filename flag.
Name of the file(s) used in access health validation.
The default --check-filename
is RCLONE_TEST
.
One or more files having this filename must exist, synchronized between your
source and destination filesets, in order for --check-access
to succeed.
See --check-access for additional details.
As of v1.66
, bisync fully supports comparing based on any combination of
size, modtime, and checksum (lifting the prior restriction on backends without
modtime support.)
By default (without the --compare
flag), bisync inherits the same comparison
options as sync
(that is: size
and modtime
by default, unless modified with flags such as
--checksum
or --size-only
.)
If the --compare
flag is set, it will override these defaults. This can be
useful if you wish to compare based on combinations not currently supported in
sync
, such as comparing all three of size
AND modtime
AND checksum
simultaneously (or just modtime
AND checksum
).
--compare
takes a comma-separated list, with the currently supported values
being size
, modtime
, and checksum
. For example, if you want to compare
size and checksum, but not modtime, you would do:
--compare size,checksum
Or if you want to compare all three:
--compare size,modtime,checksum
--compare
overrides any conflicting flags. For example, if you set the
conflicting flags --compare checksum --size-only
, --size-only
will be
ignored, and bisync will compare checksum and not size. To avoid confusion, it
is recommended to use either --compare
or the normal sync
flags, but not
both.
If --compare
includes checksum
and both remotes support checksums but have
no hash types in common with each other, checksums will be considered only
for comparisons within the same side (to determine what has changed since the
prior sync), but not for comparisons against the opposite side. If one side
supports checksums and the other does not, checksums will only be considered on
the side that supports them.
When comparing with checksum
and/or size
without modtime
, bisync cannot
determine whether a file is newer
or older
-- only whether it is changed
or unchanged
. (If it is changed
on both sides, bisync still does the
standard equality-check to avoid declaring a sync conflict unless it absolutely
has to.)
It is recommended to do a --resync
when changing --compare
settings, as
otherwise your prior listing files may not contain the attributes you wish to
compare (for example, they will not have stored checksums if you were not
previously comparing checksums.)
When --checksum
or --compare checksum
is set, bisync will retrieve (or
generate) checksums (for backends that support them) when creating the listings
for both paths, and store the checksums in the listing files.
--ignore-listing-checksum
will disable this behavior, which may speed things
up considerably, especially on backends (such as local) where hashes
must be computed on the fly instead of retrieved. Please note the following:
v1.66
, --ignore-listing-checksum
is now automatically set when
neither --checksum
nor --compare checksum
are in use (as the checksums
would not be used for anything.)--ignore-listing-checksum
is NOT the same as
--ignore-checksum
,
and you may wish to use one or the other, or both. In a nutshell:
--ignore-listing-checksum
controls whether checksums are considered when
scanning for diffs,
while --ignore-checksum
controls whether checksums are considered during the
copy/sync operations that follow,
if there ARE diffs.--ignore-listing-checksum
is passed, bisync currently computes
hashes for one path
even when there's no common hash with the other path
(for example, a crypt remote.)
This can still be beneficial, as the hashes will still be used to detect
changes within the same side
(if --checksum
or --compare checksum
is set), even if they can't be used to
compare against the opposite side.--no-slow-hash
(or
--slow-hash-sync-only
) instead of
--ignore-listing-checksum
.--ignore-listing-checksum
is used simultaneously with --compare checksum
(or --checksum
), checksums will be ignored for bisync deltas,
but still considered during the sync operations that follow (if deltas are
detected based on modtime and/or size.)On some remotes (notably local
), checksums can dramatically slow down a
bisync run, because hashes cannot be stored and need to be computed in
real-time when they are requested. On other remotes (such as drive
), they add
practically no time at all. The --no-slow-hash
flag will automatically skip
checksums on remotes where they are slow, while still comparing them on others
(assuming --compare
includes checksum
.) This can be useful when one of your
bisync paths is slow but you still want to check checksums on the other, for a more
robust sync.
Same as --no-slow-hash
, except slow hashes are still
considered during sync calls. They are still NOT considered for determining
deltas, nor or they included in listings. They are also skipped during
--resync
. The main use case for this flag is when you have a large number of
files, but relatively few of them change from run to run -- so you don't want
to check your entire tree every time (it would take too long), but you still
want to consider checksums for the smaller group of files for which a modtime
or size
change was detected. Keep in mind that this speed savings comes with
a safety trade-off: if a file's content were to change without a change to its
modtime
or size
, bisync would not detect it, and it would not be synced.
--slow-hash-sync-only
is only useful if both remotes share a common hash
type (if they don't, bisync will automatically fall back to --no-slow-hash
.)
Both --no-slow-hash
and --slow-hash-sync-only
have no effect without
--compare checksum
(or --checksum
).
If --download-hash
is set, bisync will use best efforts to obtain an MD5
checksum by downloading and computing on-the-fly, when checksums are not
otherwise available (for example, a remote that doesn't support them.) Note
that since rclone has to download the entire file, this may dramatically slow
down your bisync runs, and is also likely to use a lot of data, so it is
probably not practical for bisync paths with a large total file size. However,
it can be a good option for syncing small-but-important files with maximum
accuracy (for example, a source code repo on a crypt
remote.) An additional
advantage over methods like cryptcheck
is
that the original file is not required for comparison (for example,
--download-hash
can be used to bisync two different crypt remotes with
different passwords.)
When --download-hash
is set, bisync still looks for more efficient checksums
first, and falls back to downloading only when none are found. It takes
priority over conflicting flags such as --no-slow-hash
. --download-hash
is
not suitable for Google Docs and other files of unknown size, as
their checksums would change from run to run (due to small variances in the
internals of the generated export file.) Therefore, bisync automatically skips
--download-hash
for files with a size less than 0.
See also: Hasher
backend,
cryptcheck
command, rclone check --download
option,
md5sum
command
As a safety check, if greater than the --max-delete
percent of files were
deleted on either the Path1 or Path2 filesystem, then bisync will abort with
a warning message, without making any changes.
The default --max-delete
is 50%
.
One way to trigger this limit is to rename a directory that contains more
than half of your files. This will appear to bisync as a bunch of deleted
files and a bunch of new files.
This safety check is intended to block bisync from deleting all of the
files on both filesystems due to a temporary network access issue, or if
the user had inadvertently deleted the files on one side or the other.
To force the sync, either set a different delete percentage limit,
e.g. --max-delete 75
(allows up to 75% deletion), or use --force
to bypass the check.
Also see the all files changed check.
By using rclone filter features you can exclude file types or directory sub-trees from the sync. See the bisync filters section and generic --filter-from documentation. An example filters file contains filters for non-allowed files for syncing with Dropbox.
If you make changes to your filters file then bisync requires a run
with --resync
. This is a safety feature, which prevents existing files
on the Path1 and/or Path2 side from seeming to disappear from view
(since they are excluded in the new listings), which would fool bisync
into seeing them as deleted (as compared to the prior run listings),
and then bisync would proceed to delete them for real.
To block this from happening, bisync calculates an MD5 hash of the filters file
and stores the hash in a .md5
file in the same place as your filters file.
On the next run with --filters-file
set, bisync re-calculates the MD5 hash
of the current filters file and compares it to the hash stored in the .md5
file.
If they don't match, the run aborts with a critical error and thus forces you
to do a --resync
, likely avoiding a disaster.
In bisync, a "conflict" is a file that is new or changed on both sides
(relative to the prior run) AND is not currently identical on both sides.
--conflict-resolve
controls how bisync handles such a scenario. The currently
supported options are:
none
- (the default) - do not attempt to pick a winner, keep and rename
both files according to --conflict-loser
and
--conflict-suffix
settings. For example, with the default
settings, file.txt
on Path1 is renamed file.txt.conflict1
and file.txt
on
Path2 is renamed file.txt.conflict2
. Both are copied to the opposite path
during the run, so both sides end up with a copy of both files. (As none
is
the default, it is not necessary to specify --conflict-resolve none
-- you
can just omit the flag.)newer
- the newer file (by modtime
) is considered the winner and is
copied without renaming. The older file (the "loser") is handled according to
--conflict-loser
and --conflict-suffix
settings (either renamed or
deleted.) For example, if file.txt
on Path1 is newer than file.txt
on
Path2, the result on both sides (with other default settings) will be file.txt
(winner from Path1) and file.txt.conflict1
(loser from Path2).older
- same as newer
, except the older file is considered the winner,
and the newer file is considered the loser.larger
- the larger file (by size
) is considered the winner (regardless
of modtime
, if any).smaller
- the smaller file (by size
) is considered the winner (regardless
of modtime
, if any).path1
- the version from Path1 is unconditionally considered the winner
(regardless of modtime
and size
, if any). This can be useful if one side is
usually more trusted or up-to-date than the other.path2
- same as path1
, except the path2 version is considered the
winner.For all of the above options, note the following:
none
. (For example, if --conflict-resolve newer
is set, but one of the paths uses a remote that doesn't support
modtime
.)none
. (For example, if
--conflict-resolve newer
is set, but the Path1 and Path2 modtimes are
identical, even if the sizes may differ.)--resync
, as there is
no "prior run" to speak of (but see --resync-mode
for similar
options.)--conflict-loser
determines what happens to the "loser" of a sync conflict
(when --conflict-resolve
determines a winner) or to both
files (when there is no winner.) The currently supported options are:
num
- (the default) - auto-number the conflicts by automatically appending
the next available number to the --conflict-suffix
, in chronological order.
For example, with the default settings, the first conflict for file.txt
will
be renamed file.txt.conflict1
. If file.txt.conflict1
already exists,
file.txt.conflict2
will be used instead (etc., up to a maximum of
9223372036854775807 conflicts.)pathname
- rename the conflicts according to which side they came from,
which was the default behavior prior to v1.66
. For example, with
--conflict-suffix path
, file.txt
from Path1 will be renamed
file.txt.path1
, and file.txt
from Path2 will be renamed file.txt.path2
.
If two non-identical suffixes are provided (ex. --conflict-suffix cloud,local
), the trailing digit is omitted. Importantly, note that with
pathname
, there is no auto-numbering beyond 2
, so if file.txt.path2
somehow already exists, it will be overwritten. Using a dynamic date variable
in your --conflict-suffix
(see below) is one possible way to avoid this. Note
also that conflicts-of-conflicts are possible, if the original conflict is not
manually resolved -- for example, if for some reason you edited
file.txt.path1
on both sides, and those edits were different, the result
would be file.txt.path1.path1
and file.txt.path1.path2
(in addition to
file.txt.path2
.)delete
- keep the winner only and delete the loser, instead of renaming it.
If a winner cannot be determined (see --conflict-resolve
for details on how
this could happen), delete
is ignored and the default num
is used instead
(i.e. both versions are kept and renamed, and neither is deleted.) delete
is
inherently the most destructive option, so use it only with care.For all of the above options, note that if a winner cannot be determined (see
--conflict-resolve
for details on how this could happen), or if
--conflict-resolve
is not in use, both files will be renamed.
--conflict-suffix
controls the suffix that is appended when bisync renames a
--conflict-loser
(default: conflict
).
--conflict-suffix
will accept either one string or two comma-separated
strings to assign different suffixes to Path1 vs. Path2. This may be helpful
later in identifying the source of the conflict. (For example,
--conflict-suffix dropboxconflict,laptopconflict
)
With --conflict-loser num
, a number is always appended to the suffix. With
--conflict-loser pathname
, a number is appended only when one suffix is
specified (or when two identical suffixes are specified.) i.e. with
--conflict-loser pathname
, all of the following would produce exactly the
same result:
--conflict-suffix path
--conflict-suffix path,path
--conflict-suffix path1,path2
Suffixes may be as short as 1 character. By default, the suffix is appended
after any other extensions (ex. file.jpg.conflict1
), however, this can be
changed with the --suffix-keep-extension
flag
(i.e. to instead result in file.conflict1.jpg
).
--conflict-suffix
supports several dynamic date variables when enclosed in
curly braces as globs. This can be helpful to track the date and/or time that
each conflict was handled by bisync. For example:
--conflict-suffix {DateOnly}-conflict
// result: myfile.txt.2006-01-02-conflict1
All of the formats described here and
here are supported, but take
care to ensure that your chosen format does not use any characters that are
illegal on your remotes (for example, macOS does not allow colons in
filenames, and slashes are also best avoided as they are often interpreted as
directory separators.) To address this particular issue, an additional
{MacFriendlyTime}
(or just {mac}
) option is supported, which results in
2006-01-02 0304PM
.
Note that --conflict-suffix
is entirely separate from rclone's main
--sufix
flag. This is intentional, as users may wish
to use both flags simultaneously, if also using
--backup-dir
.
Finally, note that the default in bisync prior to v1.66
was to rename
conflicts with ..path1
and ..path2
(with two periods, and path
instead of
conflict
.) Bisync now defaults to a single dot instead of a double dot, but
additional dots can be added by including them in the specified suffix string.
For example, for behavior equivalent to the previous default, use:
[--conflict-resolve none] --conflict-loser pathname --conflict-suffix .path
Enabled by default, the check-sync function checks that all of the same files exist in both the Path1 and Path2 history listings. This check-sync integrity check is performed at the end of the sync run by default. Any untrapped failing copy/deletes between the two paths might result in differences between the two listings and in the untracked file content differences between the two paths. A resync run would correct the error.
Note that the default-enabled integrity check locally executes a load of both
the final Path1 and Path2 listings, and thus adds to the run time of a sync.
Using --check-sync=false
will disable it and may significantly reduce the
sync run times for very large numbers of files.
The check may be run manually with --check-sync=only
. It runs only the
integrity check and terminates without actually syncing.
Note that currently, --check-sync
only checks listing snapshots and NOT the
actual files on the remotes. Note also that the listing snapshots will not
know about any changes that happened during or after the latest bisync run, as
those will be discovered on the next run. Therefore, while listings should
always match each other at the end of a bisync run, it is expected that
they will not match the underlying remotes, nor will the remotes match each
other, if there were changes during or after the run. This is normal, and any
differences will be detected and synced on the next run.
For a robust integrity check of the current state of the remotes (as opposed to just their listing snapshots), consider using check
(or cryptcheck
, if at least one path is a crypt
remote) instead of --check-sync
,
keeping in mind that differences are expected if files changed during or after your last bisync run.
For example, a possible sequence could look like this:
rclone bisync Path1 Path2 -MPc --check-access --max-delete 10 --filters-file /path/to/filters.txt -v --no-cleanup --ignore-listing-checksum --disable ListR --checkers=16 --drive-pacer-min-sleep=10ms --create-empty-src-dirs --resilient
rclone check -MvPc Path1 Path2 --filter-from /path/to/filters.txt
rclone sync Path1 Path2 --filter-from /path/to/filters.txt --create-empty-src-dirs -MPc -v
(or switch Path1 and Path2 to make Path2 the source-of-truth)
Or, if neither side is totally up-to-date, you could run a --resync
to bring them back into agreement
(but remember that this could cause deleted files to re-appear.)
*Note also that rclone check
does not currently include empty directories,
so if you want to know if any empty directories are out of sync,
consider alternatively running the above rclone sync
command with --dry-run
added.
See also: Concurrent modifications, --resilient
Caution: this is an experimental feature. Use at your own risk!
By default, most errors or interruptions will cause bisync to abort and
require --resync
to recover. This is a safety feature, to prevent
bisync from running again until a user checks things out. However, in some
cases, bisync can go too far and enforce a lockout when one isn't actually
necessary, like for certain less-serious errors that might resolve themselves
on the next run. When --resilient
is specified, bisync tries its best to
recover and self-correct, and only requires --resync
as a last resort when a
human's involvement is absolutely necessary. The intended use case is for
running bisync as a background process (such as via scheduled cron).
When using --resilient
mode, bisync will still report the error and abort,
however it will not lock out future runs -- allowing the possibility of
retrying at the next normally scheduled time, without requiring a --resync
first. Examples of such retryable errors include access test failures, missing
listing files, and filter change detections. These safety features will still
prevent the current run from proceeding -- the difference is that if
conditions have improved by the time of the next run, that next run will be
allowed to proceed. Certain more serious errors will still enforce a
--resync
lockout, even in --resilient
mode, to prevent data loss.
Behavior of --resilient
may change in a future version. (See also:
--recover
, --max-lock
, Graceful
Shutdown)
If --recover
is set, in the event of a sudden interruption or other
un-graceful shutdown, bisync will attempt to automatically recover on the next
run, instead of requiring --resync
. Bisync is able to recover robustly by
keeping one "backup" listing at all times, representing the state of both paths
after the last known successful sync. Bisync can then compare the current state
with this snapshot to determine which changes it needs to retry. Changes that
were synced after this snapshot (during the run that was later interrupted)
will appear to bisync as if they are "new or changed on both sides", but in
most cases this is not a problem, as bisync will simply do its usual "equality
check" and learn that no action needs to be taken on these files, since they
are already identical on both sides.
In the rare event that a file is synced successfully during a run that later
aborts, and then that same file changes AGAIN before the next run, bisync will
think it is a sync conflict, and handle it accordingly. (From bisync's
perspective, the file has changed on both sides since the last trusted sync,
and the files on either side are not currently identical.) Therefore,
--recover
carries with it a slightly increased chance of having conflicts --
though in practice this is pretty rare, as the conditions required to cause it
are quite specific. This risk can be reduced by using bisync's "Graceful
Shutdown" mode (triggered by sending SIGINT
or
Ctrl+C
), when you have the choice, instead of forcing a sudden termination.
--recover
and --resilient
are similar, but distinct -- the main difference
is that --resilient
is about retrying, while --recover
is about
recovering. Most users will probably want both. --resilient
allows retrying
when bisync has chosen to abort itself due to safety features such as failing
--check-access
or detecting a filter change. --resilient
does not cover
external interruptions such as a user shutting down their computer in the
middle of a sync -- that is what --recover
is for.
Bisync uses lock files as a safety feature to prevent
interference from other bisync runs while it is running. Bisync normally
removes these lock files at the end of a run, but if bisync is abruptly
interrupted, these files will be left behind. By default, they will lock out
all future runs, until the user has a chance to manually check things out and
remove the lock. As an alternative, --max-lock
can be used to make them
automatically expire after a certain period of time, so that future runs are
not locked out forever, and auto-recovery is possible. --max-lock
can be any
duration 2m
or greater (or 0
to disable). If set, lock files older than
this will be considered "expired", and future runs will be allowed to disregard
them and proceed. (Note that the --max-lock
duration must be set by the
process that left the lock file -- not the later one interpreting it.)
If set, bisync will also "renew" these lock files every --max-lock minus one minute
throughout a run, for extra safety. (For example, with --max-lock 5m
,
bisync would renew the lock file (for another 5 minutes) every 4 minutes until
the run has completed.) In other words, it should not be possible for a lock
file to pass its expiration time while the process that created it is still
running -- and you can therefore be reasonably sure that any expired lock
file you may find was left there by an interrupted run, not one that is still
running and just taking awhile.
If --max-lock
is 0
or not set, the default is that lock files will never
expire, and will block future runs (of these same two bisync paths)
indefinitely.
For maximum resilience from disruptions, consider setting a relatively short
duration like --max-lock 2m
along with --resilient
and
--recover
, and a relatively frequent cron schedule. The
result will be a very robust "set-it-and-forget-it" bisync run that can
automatically bounce back from almost any interruption it might encounter,
without requiring the user to get involved and run a --resync
. (See also:
Graceful Shutdown mode)
As of v1.66
, --backup-dir
is supported in bisync.
Because --backup-dir
must be a non-overlapping path on the same remote,
Bisync has introduced new --backup-dir1
and --backup-dir2
flags to support
separate backup-dirs for Path1
and Path2
(bisyncing between different
remotes with --backup-dir
would not otherwise be possible.) --backup-dir1
and --backup-dir2
can use different remotes from each other, but
--backup-dir1
must use the same remote as Path1
, and --backup-dir2
must
use the same remote as Path2
. Each backup directory must not overlap its
respective bisync Path without being excluded by a filter rule.
The standard --backup-dir
will also work, if both paths use the same remote
(but note that deleted files from both paths would be mixed together in the
same dir). If either --backup-dir1
and --backup-dir2
are set, they will
override --backup-dir
.
Example:
rclone bisync /Users/someuser/some/local/path/Bisync gdrive:Bisync --backup-dir1 /Users/someuser/some/local/path/BackupDir --backup-dir2 gdrive:BackupDir --suffix -2023-08-26 --suffix-keep-extension --check-access --max-delete 10 --filters-file /Users/someuser/some/local/path/bisync_filters.txt --no-cleanup --ignore-listing-checksum --checkers=16 --drive-pacer-min-sleep=10ms --create-empty-src-dirs --resilient -MvP --drive-skip-gdocs --fix-case
In this example, if the user deletes a file in
/Users/someuser/some/local/path/Bisync
, bisync will propagate the delete to
the other side by moving the corresponding file from gdrive:Bisync
to
gdrive:BackupDir
. If the user deletes a file from gdrive:Bisync
, bisync
moves it from /Users/someuser/some/local/path/Bisync
to
/Users/someuser/some/local/path/BackupDir
.
In the event of a rename due to a sync conflict, the rename is not considered a delete, unless a previous conflict with the same name already exists and would get overwritten.
See also: --suffix
,
--suffix-keep-extension
bisync retains the listings of the Path1
and Path2
filesystems
from the prior run.
On each successive run it will:
path1
and path2
, and check for changes on each side.
Changes include New
, Newer
, Older
, and Deleted
files.path1
to path2
, and vice-versa..conflict1
, .conflict2
, etc. file versions, according to
--conflict-resolve
, --conflict-loser
, and --conflict-suffix
settings.RCLONE_TEST
files
(see the --check-access
flag).--max-delete
and --force
flags.Type | Description | Result | Implementation |
---|---|---|---|
Path2 new | File is new on Path2, does not exist on Path1 | Path2 version survives | rclone copy Path2 to Path1 |
Path2 newer | File is newer on Path2, unchanged on Path1 | Path2 version survives | rclone copy Path2 to Path1 |
Path2 deleted | File is deleted on Path2, unchanged on Path1 | File is deleted | rclone delete Path1 |
Path1 new | File is new on Path1, does not exist on Path2 | Path1 version survives | rclone copy Path1 to Path2 |
Path1 newer | File is newer on Path1, unchanged on Path2 | Path1 version survives | rclone copy Path1 to Path2 |
Path1 older | File is older on Path1, unchanged on Path2 | Path1 version survives | rclone copy Path1 to Path2 |
Path2 older | File is older on Path2, unchanged on Path1 | Path2 version survives | rclone copy Path2 to Path1 |
Path1 deleted | File no longer exists on Path1 | File is deleted | rclone delete Path2 |
Type | Description | Result | Implementation |
---|---|---|---|
Path1 new/changed AND Path2 new/changed AND Path1 == Path2 | File is new/changed on Path1 AND new/changed on Path2 AND Path1 version is currently identical to Path2 | No change | None |
Path1 new AND Path2 new | File is new on Path1 AND new on Path2 (and Path1 version is NOT identical to Path2) | Conflicts handled according to --conflict-resolve & --conflict-loser settings |
default: rclone copy renamed Path2.conflict2 file to Path1, rclone copy renamed Path1.conflict1 file to Path2 |
Path2 newer AND Path1 changed | File is newer on Path2 AND also changed (newer/older/size) on Path1 (and Path1 version is NOT identical to Path2) | Conflicts handled according to --conflict-resolve & --conflict-loser settings |
default: rclone copy renamed Path2.conflict2 file to Path1, rclone copy renamed Path1.conflict1 file to Path2 |
Path2 newer AND Path1 deleted | File is newer on Path2 AND also deleted on Path1 | Path2 version survives | rclone copy Path2 to Path1 |
Path2 deleted AND Path1 changed | File is deleted on Path2 AND changed (newer/older/size) on Path1 | Path1 version survives | rclone copy Path1 to Path2 |
Path1 deleted AND Path2 changed | File is deleted on Path1 AND changed (newer/older/size) on Path2 | Path2 version survives | rclone copy Path2 to Path1 |
As of rclone v1.64
, bisync is now better at detecting false positive sync conflicts,
which would previously have resulted in unnecessary renames and duplicates.
Now, when bisync comes to a file that it wants to rename (because it is new/changed on both sides),
it first checks whether the Path1 and Path2 versions are currently identical
(using the same underlying function as check
.)
If bisync concludes that the files are identical, it will skip them and move on.
Otherwise, it will create renamed duplicates, as before.
This behavior also improves the experience of renaming directories,
as a --resync
is no longer required, so long as the same change has been made on both sides.
If all prior existing files on either of the filesystems have changed
(e.g. timestamps have changed due to changing the system's timezone)
then bisync will abort without making any changes.
Any new files are not considered for this check. You could use --force
to force the sync (whichever side has the changed timestamp files wins).
Alternately, a --resync
may be used (Path1 versions will be pushed
to Path2). Consider the situation carefully and perhaps use --dry-run
before you commit to the changes.
By default, bisync compares files by modification time and size.
If you or your application should change the content of a file
without changing the modification time and size, then bisync will not
notice the change, and thus will not copy it to the other side.
As an alternative, consider comparing by checksum (if your remotes support it).
See --compare
for details.
Certain bisync critical errors, such as file copy/move failing, will result in
a bisync lockout of following runs. The lockout is asserted because the sync
status and history of the Path1 and Path2 filesystems cannot be trusted,
so it is safer to block any further changes until someone checks things out.
The recovery is to do a --resync
again.
It is recommended to use --resync --dry-run --verbose
initially and
carefully review what changes will be made before running the --resync
without --dry-run
.
Most of these events come up due to an error status from an internal call.
On such a critical error the {...}.path1.lst
and {...}.path2.lst
listing files are renamed to extension .lst-err
, which blocks any future
bisync runs (since the normal .lst
files are not found).
Bisync keeps them under bisync
subdirectory of the rclone cache directory,
typically at ${HOME}/.cache/rclone/bisync/
on Linux.
Some errors are considered temporary and re-running the bisync is not blocked. The critical return blocks further bisync runs.
See also: --resilient
, --recover
,
--max-lock
, Graceful Shutdown
When bisync is running, a lock file is created in the bisync working directory,
typically at ~/.cache/rclone/bisync/PATH1..PATH2.lck
on Linux.
If bisync should crash or hang, the lock file will remain in place and block
any further runs of bisync for the same paths.
Delete the lock file as part of debugging the situation.
The lock file effectively blocks follow-on (e.g., scheduled by cron) runs
when the prior invocation is taking a long time.
The lock file contains PID of the blocking process, which may help in debug.
Lock files can be set to automatically expire after a certain amount of time,
using the --max-lock
flag.
Note that while concurrent bisync runs are allowed, be very cautious that there is no overlap in the trees being synched between concurrent runs, lest there be replicated files, deleted files and general mayhem.
rclone bisync
returns the following codes to calling program:
0
on a successful run,1
for a non-critical failing run (a rerun may be successful),2
on syntax or usage error,7
for a critically aborted run (requires a --resync
to recover).See also the section about exit codes in main docs.
Bisync has a "Graceful Shutdown" mode which is activated by sending SIGINT
or
pressing Ctrl+C
during a run. Once triggered, bisync will use best efforts to
exit cleanly before the timer runs out. If bisync is in the middle of
transferring files, it will attempt to cleanly empty its queue by finishing
what it has started but not taking more. If it cannot do so within 30 seconds,
it will cancel the in-progress transfers at that point and then give itself a
maximum of 60 seconds to wrap up, save its state for next time, and exit. With
the -vP
flags you will see constant status updates and a final confirmation
of whether or not the graceful shutdown was successful.
At any point during the "Graceful Shutdown" sequence, a second SIGINT
or
Ctrl+C
will trigger an immediate, un-graceful exit, which will leave things
in a messier state. Usually a robust recovery will still be possible if using
--recover
mode, otherwise you will need to do a --resync
.
If you plan to use Graceful Shutdown mode, it is recommended to use
--resilient
and --recover
, and it is important to
NOT use --inplace
, otherwise you risk leaving
partially-written files on one side, which may be confused for real files on
the next run. Note also that in the event of an abrupt interruption, a lock
file will be left behind to block concurrent runs. You will need
to delete it before you can proceed with the next run (or wait for it to
expire on its own, if using --max-lock
.)
Bisync is considered BETA and has been tested with the following backends:
It has not been fully tested with other services yet. If it works, or sorta works, please let us know and we'll update the list. Run the test suite to check for proper operation as described below.
The first release of rclone bisync
required both underlying backends to support
modification times, and refused to run otherwise.
This limitation has been lifted as of v1.66
, as bisync now supports comparing
checksum and/or size instead of (or in addition to) modtime.
See --compare
for details.
When using Local, FTP or SFTP remotes with --inplace
, rclone does not create temporary
files at the destination when copying, and thus if the connection is lost
the created file may be corrupt, which will likely propagate back to the
original path on the next sync, resulting in data loss.
It is therefore recommended to omit --inplace
.
Files that change during a bisync run may result in data loss.
Prior to rclone v1.66
, this was commonly seen in highly dynamic environments, where the filesystem
was getting hammered by running processes during the sync.
As of rclone v1.66
, bisync was redesigned to use a "snapshot" model,
greatly reducing the risks from changes during a sync.
Changes that are not detected during the current sync will now be detected during the following sync,
and will no longer cause the entire run to throw a critical error.
There is additionally a mechanism to mark files as needing to be internally rechecked next time, for added safety.
It should therefore no longer be necessary to sync only at quiet times --
however, note that an error can still occur if a file happens to change at the exact moment it's
being read/written by bisync (same as would happen in rclone sync
.)
(See also: --ignore-checksum
,
--local-no-check-updated
)
By default, new/deleted empty directories on one path are not propagated to the other side.
This is because bisync (and rclone) natively works on files, not directories.
However, this can be changed with the --create-empty-src-dirs
flag, which works in
much the same way as in sync
and copy
.
When used, empty directories created or deleted on one side will also be created or deleted on the other side.
The following should be noted:
--create-empty-src-dirs
is not compatible with --remove-empty-dirs
. Use only one or the other (or neither).--create-empty-src-dirs
and the default (no --create-empty-src-dirs
) without running --resync
.
This is because it may appear as though all directories (not just the empty ones) were created/deleted,
when actually you've just toggled between making them visible/invisible to bisync.
It looks scarier than it is, but it's still probably best to stick to one or the other,
and use --resync
when you need to switch.By default, renaming a folder on the Path1 side results in deleting all files on the Path2 side and then copying all files again from Path1 to Path2. Bisync sees this as all files in the old directory name as deleted and all files in the new directory name as new.
A recommended solution is to use --track-renames
,
which is now supported in bisync as of rclone v1.66
.
Note that --track-renames
is not available during --resync
,
as --resync
does not delete anything (--track-renames
only supports sync
, not copy
.)
Otherwise, the most effective and efficient method of renaming a directory
is to rename it to the same name on both sides. (As of rclone v1.64
,
a --resync
is no longer required after doing so, as bisync will automatically
detect that Path1 and Path2 are in agreement.)
--fast-list
used by defaultUnlike most other rclone commands, bisync uses --fast-list
by default,
for backends that support it. In many cases this is desirable, however,
there are some scenarios in which bisync could be faster without --fast-list
,
and there is also a known issue concerning Google Drive users with many empty directories.
For now, the recommended way to avoid using --fast-list
is to add --disable ListR
to all bisync commands. The default behavior may change in a future version.
As of v1.66
, case and unicode form differences no longer cause critical errors,
and normalization (when comparing between filesystems) is handled according to the same flags and defaults as rclone sync
.
See the following options (all of which are supported by bisync) to control this behavior more granularly:
--fix-case
--ignore-case-sync
--no-unicode-normalization
--local-unicode-normalization
and
--local-case-sensitive
(caution: these are normally not what you want.)Note that in the (probably rare) event that --fix-case
is used AND a file is new/changed on both sides
AND the checksums match AND the filename case does not match, the Path1 filename is considered the winner,
for the purposes of --fix-case
(Path2 will be renamed to match it).
Bisync has been tested on Windows 8.1, Windows 10 Pro 64-bit and on Windows GitHub runners.
Drive letters are allowed, including drive letters mapped to network drives
(rclone bisync J:\localsync GDrive:
).
If a drive letter is omitted, the shell current drive is the default.
Drive letters are a single character follows by :
, so cloud names
must be more than one character long.
Absolute paths (with or without a drive letter), and relative paths (with or without a drive letter) are supported.
Working directory is created at C:\Users\MyLogin\AppData\Local\rclone\bisync
.
Note that bisync output may show a mix of forward /
and back \
slashes.
Be careful of case independent directory and file naming on Windows vs. case dependent Linux
See filtering documentation for how filter rules are written and interpreted.
Bisync's --filters-file
flag slightly extends the rclone's
--filter-from
filtering mechanism.
For a given bisync run you may provide only one --filters-file
.
The --include*
, --exclude*
, and --filter
flags are also supported.
Filtering portions of the directory tree is a critical feature for syncing.
Examples of directory trees (always beneath the Path1/Path2 root level) you may want to exclude from your sync:
C:\Users\MyLogin\AppData\
tree.On the other hand, there may be only select directories that you actually want to sync, and exclude all others. See the Example include-style filters for Windows user directories below.
- /AppData/
**
on the end is not necessary. Once a given directory level
is excluded then everything beneath it won't be looked at by rclone.+ /Documents/PersonalFiles/**
to select which
directories to include in the sync.**
on the end specifies to include the full depth of the specified tree.+ /*
.- **
at the end
of the filters file.-/Desktop/tempfiles/
, or - /testdir/
.
Again, a **
on the end is not necessary.- **
in the file. Without this line, everything
will be included that has not been explicitly excluded.A few rules for the syntax of a filter file expanding on filtering documentation:
#
then the line is a comment
and will be ignored.+
or -
.+/-
and the path term./
) are used in path terms, even on Windows.This Windows include-style example is based on the sync root (Path1)
set to C:\Users\MyLogin
. The strategy is to select specific directories
to be synched with a network drive (Path2).
- /AppData/
excludes an entire tree of Windows stored stuff
that need not be synched.
In my case, AppData has >11 GB of stuff I don't care about, and there are
some subdirectories beneath AppData that are not accessible to my
user login, resulting in bisync critical aborts.NTUSER
at C:\Users\MyLogin
. These files may be dynamic,
locked, and are generally don't care.+ /<path>
. By selecting only the directory trees I
want to avoid the dozen plus directories that various apps make
at C:\Users\MyLogin\Documents
.C:\Users\MyLogin
,
by adding the + /*
line.- **
which excludes everything not explicitly included.- /AppData/
- NTUSER*
- ntuser*
+ /Documents/Family/**
+ /Documents/Sketchup/**
+ /Documents/Microcapture_Photo/**
+ /Documents/Microcapture_Video/**
+ /Desktop/**
+ /Pictures/**
+ /*
- **
Note also that Windows implements several "library" links such as
C:\Users\MyLogin\My Documents\My Music
pointing to C:\Users\MyLogin\Music
.
rclone sees these as links, so you must add --links
to the
bisync command line if you which to follow these links. I find that I get
permission errors in trying to follow the links, so I don't include the
rclone --links
flag, but then you get lots of Can't follow symlink…
noise from rclone about not following the links. This noise can be
quashed by adding --quiet
to the bisync command line.
- <filename>
filters exclude these files where ever they may occur
in the sync tree. Consider adding similar exclusions for file types
you don't need to sync, such as core dump and software build files./testdir/
at the top level of the sync tree,
and usually deletes the tree after the test. If a normal sync should run
while the /testdir/
tree exists the --check-access
phase may fail
due to unbalanced RCLONE_TEST files.
The - /testdir/
filter blocks this tree from being synched.
You don't need this exclusion if you are not doing bisync development testing.# Filter file for use with bisync
# See https://rclone.org/filtering/ for filtering rules
# NOTICE: If you make changes to this file you MUST do a --resync run.
# Run with --dry-run to see what changes will be made.
# Dropbox won't sync some files so filter them away here.
# See https://help.dropbox.com/installs-integrations/sync-uploads/files-not-syncing
- .dropbox.attr
- ~*.tmp
- ~$*
- .~*
- desktop.ini
- .dropbox
# Used for bisync testing, so excluded from normal runs
- /testdir/
# Other example filters
#- /TiBU/
#- /Photos/
At the start of a bisync run, listings are gathered for Path1 and Path2
while using the user's --filters-file
. During the check access phase,
bisync scans these listings for RCLONE_TEST
files.
Any RCLONE_TEST
files hidden by the --filters-file
are not in the
listings and thus not checked during the check access phase.
Here are two normal runs. The first one has a newer file on the remote. The second has no deltas between local and remote.
2021/05/16 00:24:38 INFO : Synching Path1 "/path/to/local/tree/" with Path2 "dropbox:/"
2021/05/16 00:24:38 INFO : Path1 checking for diffs
2021/05/16 00:24:38 INFO : - Path1 File is new - file.txt
2021/05/16 00:24:38 INFO : Path1: 1 changes: 1 new, 0 newer, 0 older, 0 deleted
2021/05/16 00:24:38 INFO : Path2 checking for diffs
2021/05/16 00:24:38 INFO : Applying changes
2021/05/16 00:24:38 INFO : - Path1 Queue copy to Path2 - dropbox:/file.txt
2021/05/16 00:24:38 INFO : - Path1 Do queued copies to - Path2
2021/05/16 00:24:38 INFO : Updating listings
2021/05/16 00:24:38 INFO : Validating listings for Path1 "/path/to/local/tree/" vs Path2 "dropbox:/"
2021/05/16 00:24:38 INFO : Bisync successful
2021/05/16 00:36:52 INFO : Synching Path1 "/path/to/local/tree/" with Path2 "dropbox:/"
2021/05/16 00:36:52 INFO : Path1 checking for diffs
2021/05/16 00:36:52 INFO : Path2 checking for diffs
2021/05/16 00:36:52 INFO : No changes found
2021/05/16 00:36:52 INFO : Updating listings
2021/05/16 00:36:52 INFO : Validating listings for Path1 "/path/to/local/tree/" vs Path2 "dropbox:/"
2021/05/16 00:36:52 INFO : Bisync successful
The --dry-run
messages may indicate that it would try to delete some files.
For example, if a file is new on Path2 and does not exist on Path1 then
it would normally be copied to Path1, but with --dry-run
enabled those
copies don't happen, which leads to the attempted delete on Path2,
blocked again by --dry-run: ... Not deleting as --dry-run
.
This whole confusing situation is an artifact of the --dry-run
flag.
Scrutinize the proposed deletes carefully, and if the files would have been
copied to Path1 then the threatened deletes on Path2 may be disregarded.
Rclone has built-in retries. If you run with --verbose
you'll see
error and retry messages such as shown below. This is usually not a bug.
If at the end of the run, you see Bisync successful
and not
Bisync critical error
or Bisync aborted
then the run was successful,
and you can ignore the error messages.
The following run shows an intermittent fail. Lines 5 and _6- are low-level messages. Line 6 is a bubbled-up warning message, conveying the error. Rclone normally retries failing commands, so there may be numerous such messages in the log.
Since there are no final error/warning messages on line 7, rclone has recovered from failure after a retry, and the overall sync was successful.
1: 2021/05/14 00:44:12 INFO : Synching Path1 "/path/to/local/tree" with Path2 "dropbox:"
2: 2021/05/14 00:44:12 INFO : Path1 checking for diffs
3: 2021/05/14 00:44:12 INFO : Path2 checking for diffs
4: 2021/05/14 00:44:12 INFO : Path2: 113 changes: 22 new, 0 newer, 0 older, 91 deleted
5: 2021/05/14 00:44:12 ERROR : /path/to/local/tree/objects/af: error listing: unexpected end of JSON input
6: 2021/05/14 00:44:12 NOTICE: WARNING listing try 1 failed. - dropbox:
7: 2021/05/14 00:44:12 INFO : Bisync successful
This log shows a Critical failure which requires a --resync
to recover from.
See the Runtime Error Handling section.
2021/05/12 00:49:40 INFO : Google drive root '': Waiting for checks to finish
2021/05/12 00:49:40 INFO : Google drive root '': Waiting for transfers to finish
2021/05/12 00:49:40 INFO : Google drive root '': not deleting files as there were IO errors
2021/05/12 00:49:40 ERROR : Attempt 3/3 failed with 3 errors and: not deleting files as there were IO errors
2021/05/12 00:49:40 ERROR : Failed to sync: not deleting files as there were IO errors
2021/05/12 00:49:40 NOTICE: WARNING rclone sync try 3 failed. - /path/to/local/tree/
2021/05/12 00:49:40 ERROR : Bisync aborted. Must run --resync to recover.
Google Drive has a filter for certain file types (.exe
, .apk
, et cetera)
that by default cannot be copied from Google Drive to the local filesystem.
If you are having problems, run with --verbose
to see specifically which
files are generating complaints. If the error is
This file has been identified as malware or spam and cannot be downloaded
,
consider using the flag
--drive-acknowledge-abuse.
As of v1.66
, Google Docs
(including Google Sheets, Slides, etc.) are now supported in bisync, subject to
the same options, defaults, and limitations as in rclone sync
. When bisyncing
drive with non-drive backends, the drive -> non-drive direction is controlled
by --drive-export-formats
(default
"docx,xlsx,pptx,svg"
) and the non-drive -> drive direction is controlled by
--drive-import-formats
(default none.)
For example, with the default export/import formats, a Google Sheet on the
drive side will be synced to an .xlsx
file on the non-drive side. In the
reverse direction, .xlsx
files with filenames that match an existing Google
Sheet will be synced to that Google Sheet, while .xlsx
files that do NOT
match an existing Google Sheet will be copied to drive as normal .xlsx
files
(without conversion to Sheets, although the Google Drive web browser UI may
still give you the option to open it as one.)
If --drive-import-formats
is set (it's not, by default), then all of the
specified formats will be converted to Google Docs, if there is no existing
Google Doc with a matching name. Caution: such conversion can be quite lossy,
and in most cases it's probably not what you want!
To bisync Google Docs as URL shortcut links (in a manner similar to "Drive for
Desktop"), use: --drive-export-formats url
(or
alternatives.)
Note that these link files cannot be edited on the non-drive side -- you will
get errors if you try to sync an edited link file back to drive. They CAN be
deleted (it will result in deleting the corresponding Google Doc.) If you
create a .url
file on the non-drive side that does not match an existing
Google Doc, bisyncing it will just result in copying the literal .url
file
over to drive (no Google Doc will be created.) So, as a general rule of thumb,
think of them as read-only placeholders on the non-drive side, and make all
your changes on the drive side.
Likewise, even with other export-formats, it is best to only move/rename Google
Docs on the drive side. This is because otherwise, bisync will interpret this
as a file deleted and another created, and accordingly, it will delete the
Google Doc and create a new file at the new path. (Whether or not that new file
is a Google Doc depends on --drive-import-formats
.)
Lastly, take note that all Google Docs on the drive side have a size of -1
and no checksum. Therefore, they cannot be reliably synced with the
--checksum
or --size-only
flags. (To be exact: they will still get
created/deleted, and bisync's delta engine will notice changes and queue them
for syncing, but the underlying sync function will consider them identical and
skip them.) To work around this, use the default (modtime and size) instead of
--checksum
or --size-only
.
To ignore Google Docs entirely, use
--drive-skip-gdocs
.
Rclone does not yet have a built-in capability to monitor the local file system for changes and must be blindly run periodically. On Windows this can be done using a Task Scheduler, on Linux you can use Cron which is described below.
The 1st example runs a sync every 5 minutes between a local directory and an OwnCloud server, with output logged to a runlog file:
# Minute (0-59)
# Hour (0-23)
# Day of Month (1-31)
# Month (1-12 or Jan-Dec)
# Day of Week (0-6 or Sun-Sat)
# Command
*/5 * * * * /path/to/rclone bisync /local/files MyCloud: --check-access --filters-file /path/to/bysync-filters.txt --log-file /path/to//bisync.log
See crontab syntax for the details of crontab time interval expressions.
If you run rclone bisync
as a cron job, redirect stdout/stderr to a file.
The 2nd example runs a sync to Dropbox every hour and logs all stdout (via the >>
)
and stderr (via 2>&1
) to a log file.
0 * * * * /path/to/rclone bisync /path/to/local/dropbox Dropbox: --check-access --filters-file /home/user/filters.txt >> /path/to/logs/dropbox-run.log 2>&1
bisync can keep a local folder in sync with a cloud service, but what if you have some highly sensitive files to be synched?
Usage of a cloud service is for exchanging both routine and sensitive personal files between one's home network, one's personal notebook when on the road, and with one's work computer. The routine data is not sensitive. For the sensitive data, configure an rclone crypt remote to point to a subdirectory within the local disk tree that is bisync'd to Dropbox, and then set up an bisync for this local crypt directory to a directory outside of the main sync tree.
/path/to/DBoxroot
is the root of my local sync tree.
There are numerous subdirectories./path/to/DBoxroot/crypt
is the root subdirectory for files
that are encrypted. This local directory target is setup as an
rclone crypt remote named Dropcrypt:
.
See rclone.conf snippet below./path/to/my/unencrypted/files
is the root of my sensitive
files - not encrypted, not within the tree synched to Dropbox.bisync /path/to/my/unencrypted/files DropCrypt:
.
This step could be bundled into a script to run before and after
the full Dropbox tree sync in the last step,
thus actively keeping the sensitive files in sync.bisync /path/to/DBoxroot Dropbox:
runs periodically via cron,
keeping my full local sync tree in sync with Dropbox.C:\Users\MyLogin\Dropbox
always in sync with Dropbox. I could have used rclone bisync
instead.C:\Users\MyLogin\Documents\DropLocal
hosts the tree of unencrypted files/folders.rclone bisync C:\Users\MyLogin\Documents\DropLocal Dropcrypt:
.[Dropbox]
type = dropbox
...
[Dropcrypt]
type = crypt
remote = /path/to/DBoxroot/crypt # on the Linux server
remote = C:\Users\MyLogin\Dropbox\crypt # on the Windows notebook
filename_encryption = standard
directory_name_encryption = true
password = ...
...
You should read this section only if you are developing for rclone. You need to have rclone source code locally to work with bisync tests.
Bisync has a dedicated test framework implemented in the bisync_test.go
file located in the rclone source tree. The test suite is based on the
go test
command. Series of tests are stored in subdirectories below the
cmd/bisync/testdata
directory. Individual tests can be invoked by their
directory name, e.g.
go test . -case basic -remote local -remote2 gdrive: -v
Tests will make a temporary folder on remote and purge it afterwards. If during test run there are intermittent errors and rclone retries, these errors will be captured and flagged as invalid MISCOMPAREs. Rerunning the test will let it pass. Consider such failures as noise.
usage: go test ./cmd/bisync [options...]
Options:
-case NAME Name(s) of the test case(s) to run. Multiple names should
be separated by commas. You can remove the `test_` prefix
and replace `_` by `-` in test name for convenience.
If not `all`, the name(s) should map to a directory under
`./cmd/bisync/testdata`.
Use `all` to run all tests (default: all)
-remote PATH1 `local` or name of cloud service with `:` (default: local)
-remote2 PATH2 `local` or name of cloud service with `:` (default: local)
-no-compare Disable comparing test results with the golden directory
(default: compare)
-no-cleanup Disable cleanup of Path1 and Path2 testdirs.
Useful for troubleshooting. (default: cleanup)
-golden Store results in the golden directory (default: false)
This flag can be used with multiple tests.
-debug Print debug messages
-stop-at NUM Stop test after given step number. (default: run to the end)
Implies `-no-compare` and `-no-cleanup`, if the test really
ends prematurely. Only meaningful for a single test case.
-refresh-times Force refreshing the target modtime, useful for Dropbox
(default: false)
-verbose Run tests verbosely
Note: unlike rclone flags which must be prefixed by double dash (--
), the
test command flags can be equally prefixed by a single -
or double dash.
go test . -case basic -remote local -remote2 local
runs the test_basic
test case using only the local filesystem,
syncing one local directory with another local directory.
Test script output is to the console, while commands within scenario.txt
have their output sent to the .../workdir/test.log
file,
which is finally compared to the golden copy.go test
should be a relative name of the
directory containing bisync source code. If you run tests right from there,
the argument will be .
(current directory) as in most examples below.
If you run bisync tests from the rclone source directory, the command
should be go test ./cmd/bisync ...
../cmd/bisync/testdata
. The test -case
argument should match the full name of a subdirectory under that
directory. Every test subdirectory name on disk must start with test_
,
this prefix can be omitted on command line for brevity. Also, underscores
in the name can be replaced by dashes for convenience.go test . -remote local -remote2 local -case all
runs all tests.local
or may be names of configured cloud services.
go test . -remote gdrive: -remote2 dropbox: -case basic
will run the test between these two services, without transferring
any files to the local filesystem.go test . -remote gdrive: -remote2 local -case all > runlog.txt 2>&1
initial
directory of the testcase is applied
on the Path1 and Path2 filesystems (via rclone copy the initial directory
to Path1, then rclone sync Path1 to Path2).test.log
file in the test working directory.
Typically, the first actual command in the scenario.txt
file is
to do a --resync
, which establishes the baseline
{...}.path1.lst
and {...}.path2.lst
files in the test working
directory (.../workdir/
relative to the temporary test directory).
Various commands and listing snapshots are done within the test../cmd/bisync/testdata
.
A command line reference to a test is understood to reference a directory
beneath testdata
. For example,
go test ./cmd/bisync -case dry-run -remote gdrive: -remote2 local
refers to the test case in ./cmd/bisync/testdata/test_dry_run
..../workdir
relative to a
temporary test directory, usually under /tmp
on Linux.bisync.XXX
under system temporary directory.<remote:>/bisync.XXX/
.path1
and/or path2
subdirectories are created in a temporary
directory under the respective local or cloud test remote.-no-cleanup
flag disables purging these
directories when validating and debugging a given test.
These directories will be flushed before running another test,
independent of the -no-cleanup
usage.- /testdir/
to your normal
bisync --filters-file
so that normal syncs do not attempt to sync
the test temporary directories, which may have RCLONE_TEST
miscompares
in some testcases which would otherwise trip the --check-access
system.
The --check-access
mechanism is hard-coded to ignore RCLONE_TEST
files beneath bisync/testdata
, so the test cases may reside on the
synched tree even if there are check file mismatches in the test tree.src and dst identical but can't set mod time without deleting and re-uploading
This is expected and happens due to the way Dropbox handles modification times.
You should use the -refresh-times
test flag to make up for this.too_many_requests/...: Too many requests or write operations.
then follow the
Dropbox App ID instructions.Sometimes even a slight change in the bisync source can cause little changes spread around many log files. Updating them manually would be a nightmare.
The -golden
flag will store the test.log
and *.lst
listings from each
test case into respective golden directories. Golden results will
automatically contain generic strings instead of local or cloud paths which
means that they should match when run with a different cloud service.
Your normal workflow might be as follows:
go test ./cmd/bisync -remote local
go test ./cmd/bisync -remote local -case basic
go test ./cmd/bisync -remote local -golden
git diff --word-diff ./cmd/bisync/testdata/
.
Please note that normal line-level diff is generally useless here.git commit
) only if you are sure.
If unsure, save your code changes then wipe the log diffs from git:
git reset [--hard]
.<testname>/initial/
contains a tree of files that will be set
as the initial condition on both Path1 and Path2 testdirs.<testname>/modfiles/
contains files that will be used to
modify the Path1 and/or Path2 filesystems.<testname>/golden/
contains the expected content of the test
working directory (workdir
) at the completion of the testcase.<testname>/scenario.txt
contains the body of the test, in the form of
various commands to modify files, run bisync, and snapshot listings.
Output from these commands is captured to .../workdir/test.log
for comparison to the golden files.test <some message>
Print the line to the console and to the test.log
:
test sync is working correctly with options x, y, z
copy-listings <prefix>
Save a copy of all .lst
listings in the test working directory
with the specified prefix:
save-listings exclude-pass-run
move-listings <prefix>
Similar to copy-listings
but removes the sourcepurge-children <dir>
This will delete all child files and purge all child subdirs under given
directory but keep the parent intact. This behavior is important for tests
with Google Drive because removing and re-creating the parent would change
its ID.delete-file <file>
Delete a single file.delete-glob <dir> <pattern>
Delete a group of files located one level deep in the given directory
with names matching a given glob pattern.touch-glob YYYY-MM-DD <dir> <pattern>
Change modification time on a group of files.touch-copy YYYY-MM-DD <source-file> <dest-dir>
Change file modification time then copy it to destination.copy-file <source-file> <dest-dir>
Copy a single file to given directory.copy-as <source-file> <dest-file>
Similar to above but destination must include both directory
and the new file name at destination.copy-dir <src> <dst>
and sync-dir <src> <dst>
Copy/sync a directory. Equivalent of rclone copy
and rclone sync
.list-dirs <dir>
Equivalent to rclone lsf -R --dirs-only <dir>
bisync [options]
Runs bisync against -remote
and -remote2
.{testdir/}
- the root dir of the testcase{datadir/}
- the modfiles
dir under the testcase root{workdir/}
- the temporary test working directory{path1/}
- the root of the Path1 test directory tree{path2/}
- the root of the Path2 test directory tree{session}
- base name of the test listings{/}
- OS-specific path separator{spc}
, {tab}
, {eol}
- whitespace{chr:HH}
- raw byte with given hexadecimal codeSubstitution results of the terms named like {dir/}
will end with
/
(or backslash on Windows), so it is not necessary to include
slash in the usage, for example delete-file {path1/}file1.txt
.
This section is work in progress.
Here are a few data points for scale, execution times, and memory usage.
The first set of data was taken between a local disk to Dropbox. The speedtest.net download speed was ~170 Mbps, and upload speed was ~10 Mbps. 500 files (~9.5 MB each) had been already synched. 50 files were added in a new directory, each ~9.5 MB, ~475 MB total.
Change | Operations and times | Overall run time |
---|---|---|
500 files synched (nothing to move) | 1x listings for Path1 & Path2 | 1.5 sec |
500 files synched with --check-access | 1x listings for Path1 & Path2 | 1.5 sec |
50 new files on remote | Queued 50 copies down: 27 sec | 29 sec |
Moved local dir | Queued 50 copies up: 410 sec, 50 deletes up: 9 sec | 421 sec |
Moved remote dir | Queued 50 copies down: 31 sec, 50 deletes down: <1 sec | 33 sec |
Delete local dir | Queued 50 deletes up: 9 sec | 13 sec |
This next data is from a user's application. They had ~400GB of data over 1.96 million files being sync'ed between a Windows local disk and some remote cloud. The file full path length was on average 35 characters (which factors into load time and RAM required).
rclone's bisync implementation was derived from the rclonesync-V2 project, including documentation and test mechanisms, with @cjnaz's full support and encouragement.
rclone bisync
is similar in nature to a range of other projects:
Bisync adopts the differential synchronization technique, which is based on keeping history of changes performed by both synchronizing sides. See the Dual Shadow Method section in Neil Fraser's article.
Also note a number of academic publications by Benjamin Pierce about Unison and synchronization in general.
v1.68
v1.67
v1.66
--track-renames
and --backup-dir
are now supportedlocal
/ftp
/sftp
has been resolved (unless using --inplace
)--color
(AUTO
|NEVER
|ALWAYS
)check
and sync
,
for performance improvements and less risk of error.--fix-case
, --ignore-case-sync
, --no-unicode-normalization
--resync
is now much more efficient (especially for users of --create-empty-src-dirs
)sync
)cryptcheck
(when possible) or --download
,
instead of of --size-only
, when check
is not available.--resync
.--recover
flag allows robust recovery in the event of interruptions, without requiring --resync
.--max-lock
setting allows lock files to automatically renew and expire, for better automatic recovery when a run is interrupted.--conflict-resolve
, --conflict-loser
, and --conflict-suffix
flags.--resync-mode
flag allows more control over which version of a file gets kept during a --resync
.--retries
and --retries-sleep
(when --resilient
is set.)v1.64
--resync
to erroneously delete empty folders and duplicate files unique to Path2--check-access
is now enforced during --resync
, preventing data loss in certain user error scenarios--create-empty-src-dirs
--resilient
mode to allow recovery from self-correctable errors--ignore-listing-checksum
flag
to distinguish from --ignore-checksum