mirror of
https://github.com/rclone/rclone.git
synced 2026-01-07 11:03:15 +00:00
Compare commits
89 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
e04acb09ce | ||
|
|
87ed7fc932 | ||
|
|
90744301d3 | ||
|
|
bf4879f57f | ||
|
|
e22b445cff | ||
|
|
5ab7970e18 | ||
|
|
e984eeedc4 | ||
|
|
968b5a0984 | ||
|
|
7af1282375 | ||
|
|
d9fcc32f70 | ||
|
|
870a9fc3b2 | ||
|
|
8e3703abeb | ||
|
|
ba81277bbe | ||
|
|
88293a4b8a | ||
|
|
981104519e | ||
|
|
1d254a3674 | ||
|
|
f88d171afd | ||
|
|
ba2091725e | ||
|
|
7c120b8bc5 | ||
|
|
5cc5429f99 | ||
|
|
09d71239b6 | ||
|
|
c643e4585e | ||
|
|
873db29391 | ||
|
|
81a933ae38 | ||
|
|
ecb3c7bcef | ||
|
|
80000b904c | ||
|
|
c47c9cd440 | ||
|
|
b4a0941d4c | ||
|
|
c03d6a1ec3 | ||
|
|
46d39ebaf7 | ||
|
|
fe68737268 | ||
|
|
2360bf907a | ||
|
|
aa093e991e | ||
|
|
a5974999eb | ||
|
|
24a6ff54c2 | ||
|
|
e89ea3360e | ||
|
|
85f8552c4d | ||
|
|
a287e3ced7 | ||
|
|
8e4d8d13b8 | ||
|
|
cf208ad21b | ||
|
|
0faed16899 | ||
|
|
8d1c0ad07c | ||
|
|
165e89c266 | ||
|
|
b4e19cfd62 | ||
|
|
20ad96f3cd | ||
|
|
d64a37772f | ||
|
|
5fb6f94579 | ||
|
|
20535348db | ||
|
|
3d83a265c5 | ||
|
|
18a8a61cc5 | ||
|
|
1758621a51 | ||
|
|
5710247bf6 | ||
|
|
78b03929b7 | ||
|
|
492362ec7d | ||
|
|
51b24a1dc6 | ||
|
|
cfdb48c864 | ||
|
|
14567952b3 | ||
|
|
2b052671e2 | ||
|
|
439a126af6 | ||
|
|
0fb35f081a | ||
|
|
9ba25c7219 | ||
|
|
af9c447146 | ||
|
|
ee6b39aa6c | ||
|
|
839133c5e1 | ||
|
|
f4eb48e531 | ||
|
|
18439cf2d7 | ||
|
|
d3c16608e4 | ||
|
|
3e27ff1b95 | ||
|
|
ff91698fb5 | ||
|
|
c389616657 | ||
|
|
442578ca25 | ||
|
|
0b51d6221a | ||
|
|
2f9f9afac2 | ||
|
|
9711a5d647 | ||
|
|
cc679aa714 | ||
|
|
457ef2c190 | ||
|
|
17ffb0855f | ||
|
|
125fc8f1f0 | ||
|
|
1660903aa2 | ||
|
|
b013c58537 | ||
|
|
a5b0d88608 | ||
|
|
02d50f8c6e | ||
|
|
e09ef62d5b | ||
|
|
a75bc0703f | ||
|
|
80ecea82e8 | ||
|
|
54cd46372a | ||
|
|
282cba20a0 | ||
|
|
2479ce2c8e | ||
|
|
9aa4b6bd9b |
5
.gitignore
vendored
5
.gitignore
vendored
@@ -4,6 +4,7 @@ rclone
|
|||||||
rclonetest/rclonetest
|
rclonetest/rclonetest
|
||||||
build
|
build
|
||||||
docs/public
|
docs/public
|
||||||
README.html
|
MANUAL.md
|
||||||
README.txt
|
MANUAL.html
|
||||||
|
MANUAL.txt
|
||||||
rclone.1
|
rclone.1
|
||||||
|
|||||||
@@ -3,7 +3,8 @@ language: go
|
|||||||
go:
|
go:
|
||||||
- 1.1.2
|
- 1.1.2
|
||||||
- 1.2.2
|
- 1.2.2
|
||||||
- 1.3
|
- 1.3.3
|
||||||
|
- 1.4
|
||||||
- tip
|
- tip
|
||||||
|
|
||||||
script:
|
script:
|
||||||
|
|||||||
40
Makefile
40
Makefile
@@ -2,39 +2,46 @@ TAG := $(shell git describe --tags)
|
|||||||
LAST_TAG := $(shell git describe --tags --abbrev=0)
|
LAST_TAG := $(shell git describe --tags --abbrev=0)
|
||||||
NEW_TAG := $(shell echo $(LAST_TAG) | perl -lpe 's/v//; $$_ += 0.01; $$_ = sprintf("v%.2f", $$_)')
|
NEW_TAG := $(shell echo $(LAST_TAG) | perl -lpe 's/v//; $$_ += 0.01; $$_ = sprintf("v%.2f", $$_)')
|
||||||
|
|
||||||
rclone: *.go */*.go
|
rclone:
|
||||||
@go version
|
@go version
|
||||||
go build
|
go install -v ./...
|
||||||
|
|
||||||
doc: rclone.1 README.html README.txt
|
test: rclone
|
||||||
|
go test ./...
|
||||||
|
cd fs && ./test_all.sh
|
||||||
|
|
||||||
rclone.1: README.md
|
doc: rclone.1 MANUAL.html MANUAL.txt
|
||||||
pandoc -s --from markdown --to man README.md -o rclone.1
|
|
||||||
|
|
||||||
README.html: README.md
|
rclone.1: MANUAL.md
|
||||||
pandoc -s --from markdown_github --to html README.md -o README.html
|
pandoc -s --from markdown --to man MANUAL.md -o rclone.1
|
||||||
|
|
||||||
README.txt: README.md
|
MANUAL.md: make_manual.py docs/content/*.md
|
||||||
pandoc -s --from markdown_github --to plain README.md -o README.txt
|
./make_manual.py
|
||||||
|
|
||||||
|
MANUAL.html: MANUAL.md
|
||||||
|
pandoc -s --from markdown --to html MANUAL.md -o MANUAL.html
|
||||||
|
|
||||||
|
MANUAL.txt: MANUAL.md
|
||||||
|
pandoc -s --from markdown --to plain MANUAL.md -o MANUAL.txt
|
||||||
|
|
||||||
install: rclone
|
install: rclone
|
||||||
install -d ${DESTDIR}/usr/bin
|
install -d ${DESTDIR}/usr/bin
|
||||||
install -t ${DESTDIR}/usr/bin rclone
|
install -t ${DESTDIR}/usr/bin ${GOPATH}/bin/rclone
|
||||||
|
|
||||||
clean:
|
clean:
|
||||||
go clean ./...
|
go clean ./...
|
||||||
find . -name \*~ | xargs -r rm -f
|
find . -name \*~ | xargs -r rm -f
|
||||||
rm -rf build docs/public
|
rm -rf build docs/public
|
||||||
rm -f rclone rclonetest/rclonetest rclone.1 README.html README.txt
|
rm -f rclone rclonetest/rclonetest rclone.1 MANUAL.md MANUAL.html MANUAL.txt
|
||||||
|
|
||||||
website:
|
website:
|
||||||
cd docs && hugo
|
cd docs && hugo
|
||||||
|
|
||||||
upload_website: website
|
upload_website: website
|
||||||
./rclone -v sync docs/public memstore:www-rclone-org
|
rclone -v sync docs/public memstore:www-rclone-org
|
||||||
|
|
||||||
upload:
|
upload:
|
||||||
./rclone -v copy build/ memstore:downloads-rclone-org
|
rclone -v copy build/ memstore:downloads-rclone-org
|
||||||
|
|
||||||
cross: doc
|
cross: doc
|
||||||
./cross-compile $(TAG)
|
./cross-compile $(TAG)
|
||||||
@@ -48,12 +55,15 @@ tag:
|
|||||||
echo -e "package fs\n const Version = \"$(NEW_TAG)\"\n" | gofmt > fs/version.go
|
echo -e "package fs\n const Version = \"$(NEW_TAG)\"\n" | gofmt > fs/version.go
|
||||||
perl -lpe 's/VERSION/${NEW_TAG}/g; s/DATE/'`date -I`'/g;' docs/content/downloads.md.in > docs/content/downloads.md
|
perl -lpe 's/VERSION/${NEW_TAG}/g; s/DATE/'`date -I`'/g;' docs/content/downloads.md.in > docs/content/downloads.md
|
||||||
git tag $(NEW_TAG)
|
git tag $(NEW_TAG)
|
||||||
@echo "Add this to changelog in README.md"
|
@echo "Add this to changelog in docs/content/changelog.md"
|
||||||
@echo " * $(NEW_TAG) -" `date -I`
|
@echo " * $(NEW_TAG) -" `date -I`
|
||||||
@git log $(LAST_TAG)..$(NEW_TAG) --oneline
|
@git log $(LAST_TAG)..$(NEW_TAG) --oneline
|
||||||
@echo "Then commit the changes"
|
@echo "Then commit the changes"
|
||||||
@echo git commit -m "Version $(NEW_TAG)" -a -v
|
@echo git commit -m \"Version $(NEW_TAG)\" -a -v
|
||||||
@echo "And finally run make retag before make cross etc"
|
@echo "And finally run make retag before make cross etc"
|
||||||
|
|
||||||
retag:
|
retag:
|
||||||
git tag -f $(LAST_TAG)
|
git tag -f $(LAST_TAG)
|
||||||
|
|
||||||
|
gen_tests:
|
||||||
|
cd fstest/fstests && go run gen_tests.go
|
||||||
|
|||||||
305
README.md
305
README.md
@@ -1,12 +1,12 @@
|
|||||||
% rclone(1) User Manual
|
|
||||||
% Nick Craig-Wood
|
|
||||||
% Jul 7, 2014
|
|
||||||
|
|
||||||
Rclone
|
|
||||||
======
|
|
||||||
|
|
||||||
[](http://rclone.org/)
|
[](http://rclone.org/)
|
||||||
|
|
||||||
|
[Website](http://rclone.org) |
|
||||||
|
[Documentation](http://rclone.org/docs/) |
|
||||||
|
[Installation](http://rclone.org/install/) |
|
||||||
|
[G+](https://google.com/+RcloneOrg)
|
||||||
|
|
||||||
|
[](https://travis-ci.org/ncw/rclone) [](https://godoc.org/github.com/ncw/rclone)
|
||||||
|
|
||||||
Rclone is a command line program to sync files and directories to and from
|
Rclone is a command line program to sync files and directories to and from
|
||||||
|
|
||||||
* Google Drive
|
* Google Drive
|
||||||
@@ -26,300 +26,13 @@ Features
|
|||||||
* Check mode to check all MD5SUMs
|
* Check mode to check all MD5SUMs
|
||||||
* Can sync to and from network, eg two different Drive accounts
|
* Can sync to and from network, eg two different Drive accounts
|
||||||
|
|
||||||
See the Home page for more documentation and configuration walkthroughs.
|
See the home page for installation, usage, documentation, changelog
|
||||||
|
and configuration walkthroughs.
|
||||||
|
|
||||||
* http://rclone.org/
|
* http://rclone.org/
|
||||||
|
|
||||||
Install
|
|
||||||
-------
|
|
||||||
|
|
||||||
Rclone is a Go program and comes as a single binary file.
|
|
||||||
|
|
||||||
Download the binary for your OS from
|
|
||||||
|
|
||||||
* http://rclone.org/downloads/
|
|
||||||
|
|
||||||
Or alternatively if you have Go installed use
|
|
||||||
|
|
||||||
go install github.com/ncw/rclone
|
|
||||||
|
|
||||||
and this will build the binary in `$GOPATH/bin`.
|
|
||||||
|
|
||||||
Configure
|
|
||||||
---------
|
|
||||||
|
|
||||||
First you'll need to configure rclone. As the object storage systems
|
|
||||||
have quite complicated authentication these are kept in a config file
|
|
||||||
`.rclone.conf` in your home directory by default. (You can use the
|
|
||||||
`--config` option to choose a different config file.)
|
|
||||||
|
|
||||||
The easiest way to make the config is to run rclone with the config
|
|
||||||
option, Eg
|
|
||||||
|
|
||||||
rclone config
|
|
||||||
|
|
||||||
Usage
|
|
||||||
-----
|
|
||||||
|
|
||||||
Rclone syncs a directory tree from local to remote.
|
|
||||||
|
|
||||||
Its basic syntax is
|
|
||||||
|
|
||||||
Syntax: [options] subcommand <parameters> <parameters...>
|
|
||||||
|
|
||||||
See below for how to specify the source and destination paths.
|
|
||||||
|
|
||||||
Subcommands
|
|
||||||
-----------
|
|
||||||
|
|
||||||
rclone copy source:path dest:path
|
|
||||||
|
|
||||||
Copy the source to the destination. Doesn't transfer
|
|
||||||
unchanged files, testing first by modification time then by
|
|
||||||
MD5SUM. Doesn't delete files from the destination.
|
|
||||||
|
|
||||||
rclone sync source:path dest:path
|
|
||||||
|
|
||||||
Sync the source to the destination. Doesn't transfer
|
|
||||||
unchanged files, testing first by modification time then by
|
|
||||||
MD5SUM. Deletes any files that exist in source that don't
|
|
||||||
exist in destination. Since this can cause data loss, test
|
|
||||||
first with the `--dry-run` flag.
|
|
||||||
|
|
||||||
rclone ls [remote:path]
|
|
||||||
|
|
||||||
List all the objects in the the path with sizes.
|
|
||||||
|
|
||||||
rclone lsl [remote:path]
|
|
||||||
|
|
||||||
List all the objects in the the path with sizes and timestamps.
|
|
||||||
|
|
||||||
rclone lsd [remote:path]
|
|
||||||
|
|
||||||
List all directories/objects/buckets in the the path.
|
|
||||||
|
|
||||||
rclone mkdir remote:path
|
|
||||||
|
|
||||||
Make the path if it doesn't already exist
|
|
||||||
|
|
||||||
rclone rmdir remote:path
|
|
||||||
|
|
||||||
Remove the path. Note that you can't remove a path with
|
|
||||||
objects in it, use purge for that.
|
|
||||||
|
|
||||||
rclone purge remote:path
|
|
||||||
|
|
||||||
Remove the path and all of its contents.
|
|
||||||
|
|
||||||
rclone check source:path dest:path
|
|
||||||
|
|
||||||
Checks the files in the source and destination match. It
|
|
||||||
compares sizes and MD5SUMs and prints a report of files which
|
|
||||||
don't match. It doesn't alter the source or destination.
|
|
||||||
|
|
||||||
rclone md5sum remote:path
|
|
||||||
|
|
||||||
Produces an md5sum file for all the objects in the path. This is in
|
|
||||||
the same format as the standard md5sum tool produces.
|
|
||||||
|
|
||||||
General options:
|
|
||||||
|
|
||||||
```
|
|
||||||
--checkers=8: Number of checkers to run in parallel.
|
|
||||||
--config="~/.rclone.conf": Config file.
|
|
||||||
-n, --dry-run=false: Do a trial run with no permanent changes
|
|
||||||
--modify-window=1ns: Max time diff to be considered the same
|
|
||||||
-q, --quiet=false: Print as little stuff as possible
|
|
||||||
--stats=1m0s: Interval to print stats
|
|
||||||
--transfers=4: Number of file transfers to run in parallel.
|
|
||||||
-v, --verbose=false: Print lots more stuff
|
|
||||||
```
|
|
||||||
|
|
||||||
Developer options:
|
|
||||||
|
|
||||||
```
|
|
||||||
--cpuprofile="": Write cpu profile to file
|
|
||||||
```
|
|
||||||
|
|
||||||
Local Filesystem
|
|
||||||
----------------
|
|
||||||
|
|
||||||
Paths are specified as normal filesystem paths, so
|
|
||||||
|
|
||||||
rclone sync /home/source /tmp/destination
|
|
||||||
|
|
||||||
Will sync `/home/source` to `/tmp/destination`
|
|
||||||
|
|
||||||
Swift / Rackspace cloudfiles / Memset Memstore
|
|
||||||
----------------------------------------------
|
|
||||||
|
|
||||||
Paths are specified as remote:container (or remote: for the `lsd`
|
|
||||||
command.) You may put subdirectories in too, eg
|
|
||||||
`remote:container/path/to/dir`.
|
|
||||||
|
|
||||||
So to copy a local directory to a swift container called backup:
|
|
||||||
|
|
||||||
rclone sync /home/source swift:backup
|
|
||||||
|
|
||||||
The modified time is stored as metadata on the object as
|
|
||||||
`X-Object-Meta-Mtime` as floating point since the epoch.
|
|
||||||
|
|
||||||
This is a defacto standard (used in the official python-swiftclient
|
|
||||||
amongst others) for storing the modification time (as read using
|
|
||||||
os.Stat) for an object.
|
|
||||||
|
|
||||||
Amazon S3
|
|
||||||
---------
|
|
||||||
|
|
||||||
Paths are specified as remote:bucket. You may put subdirectories in
|
|
||||||
too, eg `remote:bucket/path/to/dir`.
|
|
||||||
|
|
||||||
So to copy a local directory to a s3 container called backup
|
|
||||||
|
|
||||||
rclone sync /home/source s3:backup
|
|
||||||
|
|
||||||
The modified time is stored as metadata on the object as
|
|
||||||
`X-Amz-Meta-Mtime` as floating point since the epoch.
|
|
||||||
|
|
||||||
Google drive
|
|
||||||
------------
|
|
||||||
|
|
||||||
Paths are specified as remote:path Drive paths may be as deep as required.
|
|
||||||
|
|
||||||
The initial setup for drive involves getting a token from Google drive
|
|
||||||
which you need to do in your browser. `rclone config` walks you
|
|
||||||
through it.
|
|
||||||
|
|
||||||
To copy a local directory to a drive directory called backup
|
|
||||||
|
|
||||||
rclone copy /home/source remote:backup
|
|
||||||
|
|
||||||
Google drive stores modification times accurate to 1 ms natively.
|
|
||||||
|
|
||||||
Dropbox
|
|
||||||
-------
|
|
||||||
|
|
||||||
Paths are specified as remote:path Dropbox paths may be as deep as required.
|
|
||||||
|
|
||||||
The initial setup for dropbox involves getting a token from Dropbox
|
|
||||||
which you need to do in your browser. `rclone config` walks you
|
|
||||||
through it.
|
|
||||||
|
|
||||||
To copy a local directory to a drive directory called backup
|
|
||||||
|
|
||||||
rclone copy /home/source dropbox:backup
|
|
||||||
|
|
||||||
Md5sums and timestamps in RFC3339 format accurate to 1ns are stored in
|
|
||||||
a Dropbox datastore called "rclone". Dropbox datastores are limited
|
|
||||||
to 100,000 rows so this is the maximum number of files rclone can
|
|
||||||
manage on Dropbox.
|
|
||||||
|
|
||||||
Google Cloud Storage
|
|
||||||
--------------------
|
|
||||||
|
|
||||||
Paths are specified as remote:path Google Cloud Storage paths may be
|
|
||||||
as deep as required.
|
|
||||||
|
|
||||||
The initial setup for Google Cloud Storage involves getting a token
|
|
||||||
from Google which you need to do in your browser. `rclone config`
|
|
||||||
walks you through it.
|
|
||||||
|
|
||||||
To copy a local directory to a google cloud storage directory called backup
|
|
||||||
|
|
||||||
rclone copy /home/source remote:backup
|
|
||||||
|
|
||||||
Google google cloud storage stores md5sums natively and rclone stores
|
|
||||||
modification times as metadata on the object, under the "mtime" key in
|
|
||||||
RFC3339 format accurate to 1ns.
|
|
||||||
|
|
||||||
Single file copies
|
|
||||||
------------------
|
|
||||||
|
|
||||||
Rclone can copy single files
|
|
||||||
|
|
||||||
rclone src:path/to/file dst:path/dir
|
|
||||||
|
|
||||||
Or
|
|
||||||
|
|
||||||
rclone src:path/to/file dst:path/to/file
|
|
||||||
|
|
||||||
Note that you can't rename the file if you are copying from one file to another.
|
|
||||||
|
|
||||||
License
|
License
|
||||||
-------
|
-------
|
||||||
|
|
||||||
This is free software under the terms of MIT the license (check the
|
This is free software under the terms of MIT the license (check the
|
||||||
COPYING file included in this package).
|
COPYING file included in this package).
|
||||||
|
|
||||||
Bugs
|
|
||||||
----
|
|
||||||
|
|
||||||
* Drive: Sometimes get: Failed to copy: Upload failed: googleapi: Error 403: Rate Limit Exceeded
|
|
||||||
* quota is 100.0 requests/second/user
|
|
||||||
* Empty directories left behind with Local and Drive
|
|
||||||
* eg purging a local directory with subdirectories doesn't work
|
|
||||||
|
|
||||||
Changelog
|
|
||||||
---------
|
|
||||||
* v1.02 - 2014-07-19
|
|
||||||
* Implement Dropbox remote
|
|
||||||
* Implement Google Cloud Storage remote
|
|
||||||
* Verify Md5sums and Sizes after copies
|
|
||||||
* Remove times from "ls" command - lists sizes only
|
|
||||||
* Add add "lsl" - lists times and sizes
|
|
||||||
* Add "md5sum" command
|
|
||||||
* v1.01 - 2014-07-04
|
|
||||||
* drive: fix transfer of big files using up lots of memory
|
|
||||||
* v1.00 - 2014-07-03
|
|
||||||
* drive: fix whole second dates
|
|
||||||
* v0.99 - 2014-06-26
|
|
||||||
* Fix --dry-run not working
|
|
||||||
* Make compatible with go 1.1
|
|
||||||
* v0.98 - 2014-05-30
|
|
||||||
* s3: Treat missing Content-Length as 0 for some ceph installations
|
|
||||||
* rclonetest: add file with a space in
|
|
||||||
* v0.97 - 2014-05-05
|
|
||||||
* Implement copying of single files
|
|
||||||
* s3 & swift: support paths inside containers/buckets
|
|
||||||
* v0.96 - 2014-04-24
|
|
||||||
* drive: Fix multiple files of same name being created
|
|
||||||
* drive: Use o.Update and fs.Put to optimise transfers
|
|
||||||
* Add version number, -V and --version
|
|
||||||
* v0.95 - 2014-03-28
|
|
||||||
* rclone.org: website, docs and graphics
|
|
||||||
* drive: fix path parsing
|
|
||||||
* v0.94 - 2014-03-27
|
|
||||||
* Change remote format one last time
|
|
||||||
* GNU style flags
|
|
||||||
* v0.93 - 2014-03-16
|
|
||||||
* drive: store token in config file
|
|
||||||
* cross compile other versions
|
|
||||||
* set strict permissions on config file
|
|
||||||
* v0.92 - 2014-03-15
|
|
||||||
* Config fixes and --config option
|
|
||||||
* v0.91 - 2014-03-15
|
|
||||||
* Make config file
|
|
||||||
* v0.90 - 2013-06-27
|
|
||||||
* Project named rclone
|
|
||||||
* v0.00 - 2012-11-18
|
|
||||||
* Project started
|
|
||||||
|
|
||||||
|
|
||||||
Contact and support
|
|
||||||
-------------------
|
|
||||||
|
|
||||||
The project website is at:
|
|
||||||
|
|
||||||
* https://github.com/ncw/rclone
|
|
||||||
|
|
||||||
There you can file bug reports, ask for help or send pull requests.
|
|
||||||
|
|
||||||
Authors
|
|
||||||
-------
|
|
||||||
|
|
||||||
* Nick Craig-Wood <nick@craig-wood.com>
|
|
||||||
|
|
||||||
Contributors
|
|
||||||
------------
|
|
||||||
|
|
||||||
* Your name goes here!
|
|
||||||
|
|||||||
19
RELEASE.md
Normal file
19
RELEASE.md
Normal file
@@ -0,0 +1,19 @@
|
|||||||
|
Required software for making a release
|
||||||
|
* [github-release](https://github.com/aktau/github-release) for uploading packages
|
||||||
|
* [gox](https://github.com/mitchellh/gox) for cross compiling
|
||||||
|
* Run `gox -build-toolchain`
|
||||||
|
* This assumes you have your own source checkout
|
||||||
|
* pandoc for making the html and man pages
|
||||||
|
|
||||||
|
Making a release
|
||||||
|
* go get -u -f -v ./...
|
||||||
|
* make test
|
||||||
|
* make tag
|
||||||
|
* edit docs/content/changelog.md
|
||||||
|
* git commit -a -v
|
||||||
|
* make retag
|
||||||
|
* # Set the GOPATH for a gox enabled compiler - . ~/bin/go-cross
|
||||||
|
* make cross
|
||||||
|
* make upload
|
||||||
|
* make upload_website
|
||||||
|
* git push --tags origin master
|
||||||
@@ -21,8 +21,8 @@ mv build/rclone-${VERSION}-darwin-386 build/rclone-${VERSION}-osx-386
|
|||||||
cd build
|
cd build
|
||||||
|
|
||||||
for d in `ls`; do
|
for d in `ls`; do
|
||||||
cp -a ../README.txt $d/
|
cp -a ../MANUAL.txt $d/README.txt
|
||||||
cp -a ../README.html $d/
|
cp -a ../MANUAL.html $d/README.html
|
||||||
cp -a ../rclone.1 $d/
|
cp -a ../rclone.1 $d/
|
||||||
zip -r9 $d.zip $d
|
zip -r9 $d.zip $d
|
||||||
rm -rf $d
|
rm -rf $d
|
||||||
|
|||||||
15
docs/content/authors.md
Normal file
15
docs/content/authors.md
Normal file
@@ -0,0 +1,15 @@
|
|||||||
|
---
|
||||||
|
title: "Authors"
|
||||||
|
description: "Rclone Authors and Contributors"
|
||||||
|
date: "2014-06-16"
|
||||||
|
---
|
||||||
|
|
||||||
|
Authors
|
||||||
|
-------
|
||||||
|
|
||||||
|
* Nick Craig-Wood <nick@craig-wood.com>
|
||||||
|
|
||||||
|
Contributors
|
||||||
|
------------
|
||||||
|
|
||||||
|
* Alex Couper <amcouper@gmail.com>
|
||||||
28
docs/content/bugs.md
Normal file
28
docs/content/bugs.md
Normal file
@@ -0,0 +1,28 @@
|
|||||||
|
---
|
||||||
|
title: "Bugs"
|
||||||
|
description: "Rclone Bugs and Limitations"
|
||||||
|
date: "2014-06-16"
|
||||||
|
---
|
||||||
|
|
||||||
|
Bugs and Limitations
|
||||||
|
--------------------
|
||||||
|
|
||||||
|
### Empty directories are left behind / not created ##
|
||||||
|
|
||||||
|
With remotes that have a concept of directory, eg Local and Drive,
|
||||||
|
empty directories may be left behind, or not created when one was
|
||||||
|
expected.
|
||||||
|
|
||||||
|
This is because rclone doesn't have a concept of a directory - it only
|
||||||
|
works on objects. Most of the object storage systems can't actually
|
||||||
|
store a directory so there is nowhere for rclone to store anything
|
||||||
|
about directories.
|
||||||
|
|
||||||
|
You can work round this to some extent with the`purge` command which
|
||||||
|
will delete everything under the path, **inluding** empty directories.
|
||||||
|
|
||||||
|
### Directory timestamps aren't preserved ##
|
||||||
|
|
||||||
|
For the same reason as the above, rclone doesn't have a concept of a
|
||||||
|
directory - it only works on objects, therefore it can't preserve the
|
||||||
|
timestamps of directories.
|
||||||
112
docs/content/changelog.md
Normal file
112
docs/content/changelog.md
Normal file
@@ -0,0 +1,112 @@
|
|||||||
|
---
|
||||||
|
title: "Documentation"
|
||||||
|
description: "Rclone Changelog"
|
||||||
|
date: "2015-06-06"
|
||||||
|
---
|
||||||
|
|
||||||
|
Changelog
|
||||||
|
---------
|
||||||
|
|
||||||
|
* v1.15 - 2015-06-06
|
||||||
|
* Add --checksum flag to only discard transfers by MD5SUM - thanks Alex Couper
|
||||||
|
* Implement --size-only flag to sync on size not checksum & modtime
|
||||||
|
* Expand docs and remove duplicated information
|
||||||
|
* Document rclone's limitations with directories
|
||||||
|
* dropbox: update docs about case insensitivity
|
||||||
|
* v1.14 - 2015-05-21
|
||||||
|
* local: fix encoding of non utf-8 file names - fixes a duplicate file problem
|
||||||
|
* drive: docs about rate limiting
|
||||||
|
* google cloud storage: Fix compile after API change in "google.golang.org/api/storage/v1"
|
||||||
|
* v1.13 - 2015-05-10
|
||||||
|
* Revise documentation (especially sync)
|
||||||
|
* Implement --timeout and --conntimeout
|
||||||
|
* s3: ignore etags from multipart uploads which aren't md5sums
|
||||||
|
* v1.12 - 2015-03-15
|
||||||
|
* drive: Use chunked upload for files above a certain size
|
||||||
|
* drive: add --drive-chunk-size and --drive-upload-cutoff parameters
|
||||||
|
* drive: switch to insert from update when a failed copy deletes the upload
|
||||||
|
* core: Log duplicate files if they are detected
|
||||||
|
* v1.11 - 2015-03-04
|
||||||
|
* swift: add region parameter
|
||||||
|
* drive: fix crash on failed to update remote mtime
|
||||||
|
* In remote paths, change native directory separators to /
|
||||||
|
* Add synchronization to ls/lsl/lsd output to stop corruptions
|
||||||
|
* Ensure all stats/log messages to go stderr
|
||||||
|
* Add --log-file flag to log everything (including panics) to file
|
||||||
|
* Make it possible to disable stats printing with --stats=0
|
||||||
|
* Implement --bwlimit to limit data transfer bandwidth
|
||||||
|
* v1.10 - 2015-02-12
|
||||||
|
* s3: list an unlimited number of items
|
||||||
|
* Fix getting stuck in the configurator
|
||||||
|
* v1.09 - 2015-02-07
|
||||||
|
* windows: Stop drive letters (eg C:) getting mixed up with remotes (eg drive:)
|
||||||
|
* local: Fix directory separators on Windows
|
||||||
|
* drive: fix rate limit exceeded errors
|
||||||
|
* v1.08 - 2015-02-04
|
||||||
|
* drive: fix subdirectory listing to not list entire drive
|
||||||
|
* drive: Fix SetModTime
|
||||||
|
* dropbox: adapt code to recent library changes
|
||||||
|
* v1.07 - 2014-12-23
|
||||||
|
* google cloud storage: fix memory leak
|
||||||
|
* v1.06 - 2014-12-12
|
||||||
|
* Fix "Couldn't find home directory" on OSX
|
||||||
|
* swift: Add tenant parameter
|
||||||
|
* Use new location of Google API packages
|
||||||
|
* v1.05 - 2014-08-09
|
||||||
|
* Improved tests and consequently lots of minor fixes
|
||||||
|
* core: Fix race detected by go race detector
|
||||||
|
* core: Fixes after running errcheck
|
||||||
|
* drive: reset root directory on Rmdir and Purge
|
||||||
|
* fs: Document that Purger returns error on empty directory, test and fix
|
||||||
|
* google cloud storage: fix ListDir on subdirectory
|
||||||
|
* google cloud storage: re-read metadata in SetModTime
|
||||||
|
* s3: make reading metadata more reliable to work around eventual consistency problems
|
||||||
|
* s3: strip trailing / from ListDir()
|
||||||
|
* swift: return directories without / in ListDir
|
||||||
|
* v1.04 - 2014-07-21
|
||||||
|
* google cloud storage: Fix crash on Update
|
||||||
|
* v1.03 - 2014-07-20
|
||||||
|
* swift, s3, dropbox: fix updated files being marked as corrupted
|
||||||
|
* Make compile with go 1.1 again
|
||||||
|
* v1.02 - 2014-07-19
|
||||||
|
* Implement Dropbox remote
|
||||||
|
* Implement Google Cloud Storage remote
|
||||||
|
* Verify Md5sums and Sizes after copies
|
||||||
|
* Remove times from "ls" command - lists sizes only
|
||||||
|
* Add add "lsl" - lists times and sizes
|
||||||
|
* Add "md5sum" command
|
||||||
|
* v1.01 - 2014-07-04
|
||||||
|
* drive: fix transfer of big files using up lots of memory
|
||||||
|
* v1.00 - 2014-07-03
|
||||||
|
* drive: fix whole second dates
|
||||||
|
* v0.99 - 2014-06-26
|
||||||
|
* Fix --dry-run not working
|
||||||
|
* Make compatible with go 1.1
|
||||||
|
* v0.98 - 2014-05-30
|
||||||
|
* s3: Treat missing Content-Length as 0 for some ceph installations
|
||||||
|
* rclonetest: add file with a space in
|
||||||
|
* v0.97 - 2014-05-05
|
||||||
|
* Implement copying of single files
|
||||||
|
* s3 & swift: support paths inside containers/buckets
|
||||||
|
* v0.96 - 2014-04-24
|
||||||
|
* drive: Fix multiple files of same name being created
|
||||||
|
* drive: Use o.Update and fs.Put to optimise transfers
|
||||||
|
* Add version number, -V and --version
|
||||||
|
* v0.95 - 2014-03-28
|
||||||
|
* rclone.org: website, docs and graphics
|
||||||
|
* drive: fix path parsing
|
||||||
|
* v0.94 - 2014-03-27
|
||||||
|
* Change remote format one last time
|
||||||
|
* GNU style flags
|
||||||
|
* v0.93 - 2014-03-16
|
||||||
|
* drive: store token in config file
|
||||||
|
* cross compile other versions
|
||||||
|
* set strict permissions on config file
|
||||||
|
* v0.92 - 2014-03-15
|
||||||
|
* Config fixes and --config option
|
||||||
|
* v0.91 - 2014-03-15
|
||||||
|
* Make config file
|
||||||
|
* v0.90 - 2013-06-27
|
||||||
|
* Project named rclone
|
||||||
|
* v0.00 - 2012-11-18
|
||||||
|
* Project started
|
||||||
@@ -5,8 +5,17 @@ date: "2014-04-26"
|
|||||||
---
|
---
|
||||||
|
|
||||||
Contact the rclone project
|
Contact the rclone project
|
||||||
|
--------------------------
|
||||||
|
|
||||||
|
The project website is at:
|
||||||
|
|
||||||
|
* https://github.com/ncw/rclone
|
||||||
|
|
||||||
|
There you can file bug reports, ask for help or contribute pull
|
||||||
|
requests.
|
||||||
|
|
||||||
|
See also
|
||||||
|
|
||||||
* [Github project page for source, reporting bugs and pull requests](http://github.com/ncw/rclone)
|
|
||||||
* <a href="https://google.com/+RcloneOrg" rel="publisher">Google+ page for general comments</a></li>
|
* <a href="https://google.com/+RcloneOrg" rel="publisher">Google+ page for general comments</a></li>
|
||||||
|
|
||||||
Or email [Nick Craig-Wood](mailto:nick@craig-wood.com)
|
Or email [Nick Craig-Wood](mailto:nick@craig-wood.com)
|
||||||
|
|||||||
@@ -1,22 +1,9 @@
|
|||||||
---
|
---
|
||||||
title: "Documentation"
|
title: "Documentation"
|
||||||
description: "Rclone Documentation"
|
description: "Rclone Usage"
|
||||||
date: "2014-07-17"
|
date: "2015-06-06"
|
||||||
---
|
---
|
||||||
|
|
||||||
Install
|
|
||||||
-------
|
|
||||||
|
|
||||||
Rclone is a Go program and comes as a single binary file.
|
|
||||||
|
|
||||||
[Download](/downloads/) the relevant binary.
|
|
||||||
|
|
||||||
Or alternatively if you have Go installed use
|
|
||||||
|
|
||||||
go get github.com/ncw/rclone
|
|
||||||
|
|
||||||
and this will build the binary in `$GOPATH/bin`.
|
|
||||||
|
|
||||||
Configure
|
Configure
|
||||||
---------
|
---------
|
||||||
|
|
||||||
@@ -30,11 +17,13 @@ option:
|
|||||||
|
|
||||||
rclone config
|
rclone config
|
||||||
|
|
||||||
See below for detailed instructions for
|
See the following for detailed instructions for
|
||||||
|
|
||||||
* [Google drive](/drive/)
|
* [Google drive](/drive/)
|
||||||
* [Amazon S3](/s3/)
|
* [Amazon S3](/s3/)
|
||||||
* [Swift / Rackspace Cloudfiles / Memset Memstore](/swift/)
|
* [Swift / Rackspace Cloudfiles / Memset Memstore](/swift/)
|
||||||
|
* [Dropbox](/dropbox/)
|
||||||
|
* [Google Cloud Storage](/googlcloudstorage/)
|
||||||
* [Local filesystem](/local/)
|
* [Local filesystem](/local/)
|
||||||
|
|
||||||
Usage
|
Usage
|
||||||
@@ -55,104 +44,216 @@ You can define as many storage paths as you like in the config file.
|
|||||||
Subcommands
|
Subcommands
|
||||||
-----------
|
-----------
|
||||||
|
|
||||||
rclone copy source:path dest:path
|
### rclone copy source:path dest:path ###
|
||||||
|
|
||||||
Copy the source to the destination. Doesn't transfer
|
Copy the source to the destination. Doesn't transfer
|
||||||
unchanged files, testing first by modification time then by
|
unchanged files, testing by size and modification time or
|
||||||
MD5SUM. Doesn't delete files from the destination.
|
MD5SUM. Doesn't delete files from the destination.
|
||||||
|
|
||||||
rclone sync source:path dest:path
|
### rclone sync source:path dest:path ###
|
||||||
|
|
||||||
Sync the source to the destination. Doesn't transfer
|
Sync the source to the destination, changing the destination
|
||||||
unchanged files, testing first by modification time then by
|
only. Doesn't transfer unchanged files, testing by size and
|
||||||
MD5SUM. Deletes any files that exist in source that don't
|
modification time or MD5SUM. Destination is updated to match
|
||||||
exist in destination. Since this can cause data loss, test
|
source, including deleting files if necessary. Since this can
|
||||||
first with the -dry-run flag.
|
cause data loss, test first with the `--dry-run` flag.
|
||||||
|
|
||||||
rclone ls [remote:path]
|
### rclone ls [remote:path] ###
|
||||||
|
|
||||||
List all the objects in the the path with sizes.
|
List all the objects in the the path with size and path.
|
||||||
|
|
||||||
rclone lsl [remote:path]
|
### rclone lsd [remote:path] ###
|
||||||
|
|
||||||
List all the objects in the the path with sizes and timestamps.
|
List all directories/containers/buckets in the the path.
|
||||||
|
|
||||||
rclone lsd [remote:path]
|
### rclone lsl [remote:path] ###
|
||||||
|
|
||||||
List all directories/objects/buckets in the the path.
|
List all the objects in the the path with modification time,
|
||||||
|
size and path.
|
||||||
|
|
||||||
rclone mkdir remote:path
|
### rclone md5sum [remote:path] ###
|
||||||
|
|
||||||
|
Produces an md5sum file for all the objects in the path. This
|
||||||
|
is in the same format as the standard md5sum tool produces.
|
||||||
|
|
||||||
|
### rclone mkdir remote:path ###
|
||||||
|
|
||||||
Make the path if it doesn't already exist
|
Make the path if it doesn't already exist
|
||||||
|
|
||||||
rclone rmdir remote:path
|
### rclone rmdir remote:path ###
|
||||||
|
|
||||||
Remove the path. Note that you can't remove a path with
|
Remove the path. Note that you can't remove a path with
|
||||||
objects in it, use purge for that.
|
objects in it, use purge for that.
|
||||||
|
|
||||||
rclone purge remote:path
|
### rclone purge remote:path ###
|
||||||
|
|
||||||
Remove the path and all of its contents.
|
Remove the path and all of its contents.
|
||||||
|
|
||||||
rclone check source:path dest:path
|
### rclone check source:path dest:path ###
|
||||||
|
|
||||||
Checks the files in the source and destination match. It
|
Checks the files in the source and destination match. It
|
||||||
compares sizes and MD5SUMs and prints a report of files which
|
compares sizes and MD5SUMs and prints a report of files which
|
||||||
don't match. It doesn't alter the source or destination.
|
don't match. It doesn't alter the source or destination.
|
||||||
|
|
||||||
rclone md5sum remote:path
|
### rclone config ###
|
||||||
|
|
||||||
Produces an md5sum file for all the objects in the path. This is in
|
Enter an interactive configuration session.
|
||||||
the same format as the standard md5sum tool produces.
|
|
||||||
General options:
|
|
||||||
|
|
||||||
```
|
### rclone help ###
|
||||||
--checkers=8: Number of checkers to run in parallel.
|
|
||||||
--transfers=4: Number of file transfers to run in parallel.
|
|
||||||
--config="~/.rclone.conf": Config file.
|
|
||||||
-n, --dry-run=false: Do a trial run with no permanent changes
|
|
||||||
--modify-window=1ns: Max time diff to be considered the same
|
|
||||||
-q, --quiet=false: Print as little stuff as possible
|
|
||||||
--stats=1m0s: Interval to print stats
|
|
||||||
-v, --verbose=false: Print lots more stuff
|
|
||||||
```
|
|
||||||
|
|
||||||
Developer options:
|
Prints help on rclone commands and options.
|
||||||
|
|
||||||
```
|
Options
|
||||||
--cpuprofile="": Write cpu profile to file
|
|
||||||
```
|
|
||||||
|
|
||||||
License
|
|
||||||
-------
|
-------
|
||||||
|
|
||||||
This is free software under the terms of MIT the license (check the
|
Rclone has a number of options to control its behaviour.
|
||||||
COPYING file included in this package).
|
|
||||||
|
|
||||||
Bugs
|
Options which use TIME use the go time parser. A duration string is a
|
||||||
----
|
possibly signed sequence of decimal numbers, each with optional
|
||||||
|
fraction and a unit suffix, such as "300ms", "-1.5h" or "2h45m". Valid
|
||||||
|
time units are "ns", "us" (or "µs"), "ms", "s", "m", "h".
|
||||||
|
|
||||||
* Doesn't sync individual files yet, only directories.
|
Options which use SIZE use kByte by default. However a suffix of `k`
|
||||||
* Drive: Sometimes get: Failed to copy: Upload failed: googleapi: Error 403: Rate Limit Exceeded
|
for kBytes, `M` for MBytes and `G` for GBytes may be used. These are
|
||||||
* quota is 100.0 requests/second/user
|
the binary units, eg 2**10, 2**20, 2**30 respectively.
|
||||||
* Empty directories left behind with Local and Drive
|
|
||||||
* eg purging a local directory with subdirectories doesn't work
|
|
||||||
|
|
||||||
Contact and support
|
### --bwlimit=SIZE ###
|
||||||
-------------------
|
|
||||||
|
|
||||||
The project website is at:
|
Bandwidth limit in kBytes/s, or use suffix k|M|G. The default is `0`
|
||||||
|
which means to not limit bandwidth.
|
||||||
|
|
||||||
* https://github.com/ncw/rclone
|
For example to limit bandwidth usage to 10 MBytes/s use `--bwlimit 10M`
|
||||||
|
|
||||||
There you can file bug reports, ask for help or contribute patches.
|
This only limits the bandwidth of the data transfer, it doesn't limit
|
||||||
|
the bandwith of the directory listings etc.
|
||||||
|
|
||||||
Authors
|
### --checkers=N ###
|
||||||
-------
|
|
||||||
|
|
||||||
* Nick Craig-Wood <nick@craig-wood.com>
|
The number of checkers to run in parallel. Checkers do the equality
|
||||||
|
checking of files during a sync. For some storage systems (eg s3,
|
||||||
|
swift, dropbox) this can take a significant amount of time so they are
|
||||||
|
run in parallel.
|
||||||
|
|
||||||
Contributors
|
The default is to run 8 checkers in parallel.
|
||||||
------------
|
|
||||||
|
|
||||||
* Your name goes here!
|
### -c, --checksum ###
|
||||||
|
|
||||||
|
Normally rclone will look at modification time and size of files to
|
||||||
|
see if they are equal. If you set this flag then rclone will check
|
||||||
|
MD5SUM and size to determine if files are equal.
|
||||||
|
|
||||||
|
This is very useful when transferring between remotes which store the
|
||||||
|
MD5SUM on the object which include swift, s3, drive, and google cloud
|
||||||
|
storage.
|
||||||
|
|
||||||
|
Eg `rclone --checksum sync s3:/bucket swift:/bucket` would run much
|
||||||
|
quicker than without the `--checksum` flag.
|
||||||
|
|
||||||
|
When using this flag, rclone won't update mtimes of remote files if
|
||||||
|
they are incorrect as it would normally.
|
||||||
|
|
||||||
|
### --config=CONFIG_FILE ###
|
||||||
|
|
||||||
|
Specify the location of the rclone config file. Normally this is in
|
||||||
|
your home directory as a file called `.rclone.conf`. If you run
|
||||||
|
`rclone -h` and look at the help for the `--config` option you will
|
||||||
|
see where the default location is for you. Use this flag to override
|
||||||
|
the config location, eg `rclone --config=".myconfig" .config`.
|
||||||
|
|
||||||
|
### --contimeout=TIME ###
|
||||||
|
|
||||||
|
Set the connection timeout. This should be in go time format which
|
||||||
|
looks like `5s` for 5 seconds, `10m` for 10 minutes, or `3h30m`.
|
||||||
|
|
||||||
|
The connection timeout is the amount of time rclone will wait for a
|
||||||
|
connection to go through to a remote object storage system. It is
|
||||||
|
`1m` by default.
|
||||||
|
|
||||||
|
### -n, --dry-run ###
|
||||||
|
|
||||||
|
Do a trial run with no permanent changes. Use this in combination
|
||||||
|
with the `-v` flag to see what rclone would do without actually doing
|
||||||
|
it. Useful when setting up the `sync` command.
|
||||||
|
|
||||||
|
### --log-file=FILE ###
|
||||||
|
|
||||||
|
Log all of rclone's output to FILE. This is not active by default.
|
||||||
|
This can be useful for tracking down problems with syncs in
|
||||||
|
combination with the `-v` flag.
|
||||||
|
|
||||||
|
### --modify-window=TIME ###
|
||||||
|
|
||||||
|
When checking whether a file has been modified, this is the maximum
|
||||||
|
allowed time difference that a file can have and still be considered
|
||||||
|
equivalent.
|
||||||
|
|
||||||
|
The default is `1ns` unless this is overridden by a remote. For
|
||||||
|
example OS X only stores modification times to the nearest second so
|
||||||
|
if you are reading and writing to an OS X filing system this will be
|
||||||
|
`1s` by default.
|
||||||
|
|
||||||
|
This command line flag allows you to override that computed default.
|
||||||
|
|
||||||
|
### -q, --quiet ###
|
||||||
|
|
||||||
|
Normally rclone outputs stats and a completion message. If you set
|
||||||
|
this flag it will make as little output as possible.
|
||||||
|
|
||||||
|
### --size-only ###
|
||||||
|
|
||||||
|
Normally rclone will look at modification time and size of files to
|
||||||
|
see if they are equal. If you set this flag then rclone will check
|
||||||
|
only the size.
|
||||||
|
|
||||||
|
This can be useful transferring files from dropbox which have been
|
||||||
|
modified by the desktop sync client which doesn't set checksums of
|
||||||
|
modification times in the same way as rclone.
|
||||||
|
|
||||||
|
When using this flag, rclone won't update mtimes of remote files if
|
||||||
|
they are incorrect as it would normally.
|
||||||
|
|
||||||
|
### --stats=TIME ###
|
||||||
|
|
||||||
|
Rclone will print stats at regular intervals to show its progress.
|
||||||
|
|
||||||
|
This sets the interval.
|
||||||
|
|
||||||
|
The default is `1m`. Use 0 to disable.
|
||||||
|
|
||||||
|
### --timeout=TIME ###
|
||||||
|
|
||||||
|
This sets the IO idle timeout. If a transfer has started but then
|
||||||
|
becomes idle for this long it is considered broken and disconnected.
|
||||||
|
|
||||||
|
The default is `5m`. Set to 0 to disable.
|
||||||
|
|
||||||
|
### --transfers=N ###
|
||||||
|
|
||||||
|
The number of file transfers to run in parallel. It can sometimes be
|
||||||
|
useful to set this to a smaller number if the remote is giving a lot
|
||||||
|
of timeouts or bigger if you have lots of bandwidth and a fast remote.
|
||||||
|
|
||||||
|
The default is to run 4 file transfers in parallel.
|
||||||
|
|
||||||
|
### -v, --verbose ###
|
||||||
|
|
||||||
|
If you set this flag, rclone will become very verbose telling you
|
||||||
|
about every file it considers and transfers.
|
||||||
|
|
||||||
|
Very useful for debugging.
|
||||||
|
|
||||||
|
### -V, --version ###
|
||||||
|
|
||||||
|
Prints the version number
|
||||||
|
|
||||||
|
Developer options
|
||||||
|
-----------------
|
||||||
|
|
||||||
|
These options are useful when developing or debugging rclone. There
|
||||||
|
are also some more remote specific options which aren't documented
|
||||||
|
here which are used for testing. These start with remote name eg
|
||||||
|
`--drive-test-option`.
|
||||||
|
|
||||||
|
### --cpuprofile=FILE ###
|
||||||
|
|
||||||
|
Write cpu profile to file. This can be analysed with `go tool pprof`.
|
||||||
|
|||||||
@@ -2,34 +2,34 @@
|
|||||||
title: "Rclone downloads"
|
title: "Rclone downloads"
|
||||||
description: "Download rclone binaries for your OS."
|
description: "Download rclone binaries for your OS."
|
||||||
type: page
|
type: page
|
||||||
date: "2014-07-19"
|
date: "2015-06-06"
|
||||||
---
|
---
|
||||||
|
|
||||||
Rclone Download v1.02
|
Rclone Download v1.15
|
||||||
=====================
|
=====================
|
||||||
|
|
||||||
* Windows
|
* Windows
|
||||||
* [386 - 32 Bit](http://downloads.rclone.org/rclone-v1.02-windows-386.zip)
|
* [386 - 32 Bit](http://downloads.rclone.org/rclone-v1.15-windows-386.zip)
|
||||||
* [AMD64 - 64 Bit](http://downloads.rclone.org/rclone-v1.02-windows-amd64.zip)
|
* [AMD64 - 64 Bit](http://downloads.rclone.org/rclone-v1.15-windows-amd64.zip)
|
||||||
* OSX
|
* OSX
|
||||||
* [386 - 32 Bit](http://downloads.rclone.org/rclone-v1.02-osx-386.zip)
|
* [386 - 32 Bit](http://downloads.rclone.org/rclone-v1.15-osx-386.zip)
|
||||||
* [AMD64 - 64 Bit](http://downloads.rclone.org/rclone-v1.02-osx-amd64.zip)
|
* [AMD64 - 64 Bit](http://downloads.rclone.org/rclone-v1.15-osx-amd64.zip)
|
||||||
* Linux
|
* Linux
|
||||||
* [386 - 32 Bit](http://downloads.rclone.org/rclone-v1.02-linux-386.zip)
|
* [386 - 32 Bit](http://downloads.rclone.org/rclone-v1.15-linux-386.zip)
|
||||||
* [AMD64 - 64 Bit](http://downloads.rclone.org/rclone-v1.02-linux-amd64.zip)
|
* [AMD64 - 64 Bit](http://downloads.rclone.org/rclone-v1.15-linux-amd64.zip)
|
||||||
* [ARM - 32 Bit](http://downloads.rclone.org/rclone-v1.02-linux-arm.zip)
|
* [ARM - 32 Bit](http://downloads.rclone.org/rclone-v1.15-linux-arm.zip)
|
||||||
* FreeBSD
|
* FreeBSD
|
||||||
* [386 - 32 Bit](http://downloads.rclone.org/rclone-v1.02-freebsd-386.zip)
|
* [386 - 32 Bit](http://downloads.rclone.org/rclone-v1.15-freebsd-386.zip)
|
||||||
* [AMD64 - 64 Bit](http://downloads.rclone.org/rclone-v1.02-freebsd-amd64.zip)
|
* [AMD64 - 64 Bit](http://downloads.rclone.org/rclone-v1.15-freebsd-amd64.zip)
|
||||||
* [ARM - 32 Bit](http://downloads.rclone.org/rclone-v1.02-freebsd-arm.zip)
|
* [ARM - 32 Bit](http://downloads.rclone.org/rclone-v1.15-freebsd-arm.zip)
|
||||||
* NetBSD
|
* NetBSD
|
||||||
* [386 - 32 Bit](http://downloads.rclone.org/rclone-v1.02-netbsd-386.zip)
|
* [386 - 32 Bit](http://downloads.rclone.org/rclone-v1.15-netbsd-386.zip)
|
||||||
* [AMD64 - 64 Bit](http://downloads.rclone.org/rclone-v1.02-netbsd-amd64.zip)
|
* [AMD64 - 64 Bit](http://downloads.rclone.org/rclone-v1.15-netbsd-amd64.zip)
|
||||||
* [ARM - 32 Bit](http://downloads.rclone.org/rclone-v1.02-netbsd-arm.zip)
|
* [ARM - 32 Bit](http://downloads.rclone.org/rclone-v1.15-netbsd-arm.zip)
|
||||||
* OpenBSD
|
* OpenBSD
|
||||||
* [386 - 32 Bit](http://downloads.rclone.org/rclone-v1.02-openbsd-386.zip)
|
* [386 - 32 Bit](http://downloads.rclone.org/rclone-v1.15-openbsd-386.zip)
|
||||||
* [AMD64 - 64 Bit](http://downloads.rclone.org/rclone-v1.02-openbsd-amd64.zip)
|
* [AMD64 - 64 Bit](http://downloads.rclone.org/rclone-v1.15-openbsd-amd64.zip)
|
||||||
* Plan 9
|
* Plan 9
|
||||||
* [386 - 32 Bit](http://downloads.rclone.org/rclone-v1.02-plan9-386.zip)
|
* [386 - 32 Bit](http://downloads.rclone.org/rclone-v1.15-plan9-386.zip)
|
||||||
|
|
||||||
Older downloads can be found [here](http://downloads.rclone.org/)
|
Older downloads can be found [here](http://downloads.rclone.org/)
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
---
|
---
|
||||||
title: "Google drive"
|
title: "Google drive"
|
||||||
description: "Rclone docs for Google drive"
|
description: "Rclone docs for Google drive"
|
||||||
date: "2014-04-26"
|
date: "2015-05-10"
|
||||||
---
|
---
|
||||||
|
|
||||||
<i class="fa fa-google"></i> Google Drive
|
<i class="fa fa-google"></i> Google Drive
|
||||||
@@ -69,7 +69,25 @@ To copy a local directory to a drive directory called backup
|
|||||||
|
|
||||||
rclone copy /home/source remote:backup
|
rclone copy /home/source remote:backup
|
||||||
|
|
||||||
Modified time
|
### Modified time ###
|
||||||
-------------
|
|
||||||
|
|
||||||
Google drive stores modification times accurate to 1 ms.
|
Google drive stores modification times accurate to 1 ms.
|
||||||
|
|
||||||
|
### Revisions ###
|
||||||
|
|
||||||
|
Google drive stores revisions of files. When you upload a change to
|
||||||
|
an existing file to google drive using rclone it will create a new
|
||||||
|
revision of that file.
|
||||||
|
|
||||||
|
Revisions follow the standard google policy which at time of writing
|
||||||
|
was
|
||||||
|
|
||||||
|
* They are deleted after 30 days or 100 revisions (whatever comes first).
|
||||||
|
* They do not count towards a user storage quota.
|
||||||
|
|
||||||
|
### Limitations ###
|
||||||
|
|
||||||
|
Drive has quite a lot of rate limiting. This causes rclone to be
|
||||||
|
limited to transferring about 2 files per second only. Individual
|
||||||
|
files may be transferred much faster at 100s of MBytes/s but lots of
|
||||||
|
small files can take a long time.
|
||||||
|
|||||||
@@ -71,10 +71,21 @@ To copy a local directory to a dropbox directory called backup
|
|||||||
|
|
||||||
rclone copy /home/source remote:backup
|
rclone copy /home/source remote:backup
|
||||||
|
|
||||||
Modified time
|
### Modified time ###
|
||||||
-------------
|
|
||||||
|
|
||||||
Md5sums and timestamps in RFC3339 format accurate to 1ns are stored in
|
Md5sums and timestamps in RFC3339 format accurate to 1ns are stored in
|
||||||
a Dropbox datastore called "rclone". Dropbox datastores are limited
|
a Dropbox datastore called "rclone".
|
||||||
to 100,000 rows so this is the maximum number of files rclone can
|
|
||||||
manage on Dropbox.
|
### Limitations ###
|
||||||
|
|
||||||
|
Dropbox datastores are limited to 100,000 rows so this is the maximum
|
||||||
|
number of files rclone can manage on Dropbox.
|
||||||
|
|
||||||
|
Dropbox is case sensitive which can sometimes cause duplicated files.
|
||||||
|
|
||||||
|
If you use the desktop sync tool and rclone on the same files then the
|
||||||
|
md5sums and modification times may get out of sync as far as rclone is
|
||||||
|
concerned. This will cause `Corrupted on transfer: md5sums differ`
|
||||||
|
error message when fetching files. You can work around this by using
|
||||||
|
the `--size-only` flag to ignore the md5sums and modification times
|
||||||
|
for these files.
|
||||||
|
|||||||
@@ -109,8 +109,7 @@ files in the bucket.
|
|||||||
|
|
||||||
rclone sync /home/local/directory remote:bucket
|
rclone sync /home/local/directory remote:bucket
|
||||||
|
|
||||||
Modified time
|
### Modified time ###
|
||||||
-------------
|
|
||||||
|
|
||||||
Google google cloud storage stores md5sums natively and rclone stores
|
Google google cloud storage stores md5sums natively and rclone stores
|
||||||
modification times as metadata on the object, under the "mtime" key in
|
modification times as metadata on the object, under the "mtime" key in
|
||||||
|
|||||||
21
docs/content/install.md
Normal file
21
docs/content/install.md
Normal file
@@ -0,0 +1,21 @@
|
|||||||
|
---
|
||||||
|
title: "Install"
|
||||||
|
description: "Rclone Installation"
|
||||||
|
date: "2014-07-17"
|
||||||
|
---
|
||||||
|
|
||||||
|
Install
|
||||||
|
-------
|
||||||
|
|
||||||
|
Rclone is a Go program and comes as a single binary file.
|
||||||
|
|
||||||
|
[Download](/downloads/) the relevant binary.
|
||||||
|
|
||||||
|
Or alternatively if you have Go installed use
|
||||||
|
|
||||||
|
go get github.com/ncw/rclone
|
||||||
|
|
||||||
|
and this will build the binary in `$GOPATH/bin`.
|
||||||
|
|
||||||
|
See the [Usage section](/usage/) of the docs for how to use rclone, or
|
||||||
|
run `rclone -h`.
|
||||||
34
docs/content/licence.md
Normal file
34
docs/content/licence.md
Normal file
@@ -0,0 +1,34 @@
|
|||||||
|
---
|
||||||
|
title: "Licence"
|
||||||
|
description: "Rclone Licence"
|
||||||
|
date: "2015-06-06"
|
||||||
|
---
|
||||||
|
|
||||||
|
License
|
||||||
|
-------
|
||||||
|
|
||||||
|
This is free software under the terms of MIT the license (check the
|
||||||
|
COPYING file included with the source code).
|
||||||
|
|
||||||
|
```
|
||||||
|
Copyright (C) 2012 by Nick Craig-Wood http://www.craig-wood.com/nick/
|
||||||
|
|
||||||
|
Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
|
of this software and associated documentation files (the "Software"), to deal
|
||||||
|
in the Software without restriction, including without limitation the rights
|
||||||
|
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
||||||
|
copies of the Software, and to permit persons to whom the Software is
|
||||||
|
furnished to do so, subject to the following conditions:
|
||||||
|
|
||||||
|
The above copyright notice and this permission notice shall be included in
|
||||||
|
all copies or substantial portions of the Software.
|
||||||
|
|
||||||
|
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||||
|
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||||
|
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||||||
|
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||||||
|
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
||||||
|
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
||||||
|
THE SOFTWARE.
|
||||||
|
```
|
||||||
|
|
||||||
@@ -16,10 +16,23 @@ Will sync `/home/source` to `/tmp/destination`
|
|||||||
These can be configured into the config file for consistencies sake,
|
These can be configured into the config file for consistencies sake,
|
||||||
but it is probably easier not to.
|
but it is probably easier not to.
|
||||||
|
|
||||||
Modified time
|
### Modified time ###
|
||||||
-------------
|
|
||||||
|
|
||||||
Rclone reads and writes the modified time using an accuracy determined by
|
Rclone reads and writes the modified time using an accuracy determined by
|
||||||
the OS. Typically this is 1ns on Linux, 10 ns on Windows and 1 Second
|
the OS. Typically this is 1ns on Linux, 10 ns on Windows and 1 Second
|
||||||
on OS X.
|
on OS X.
|
||||||
|
|
||||||
|
### Filenames ###
|
||||||
|
|
||||||
|
Filenames are expected to be encoded in UTF-8 on disk. This is the
|
||||||
|
normal case for Windows and OS X. There is a bit more uncertainty in
|
||||||
|
the Linux world, but new distributions will have UTF-8 encoded files
|
||||||
|
names.
|
||||||
|
|
||||||
|
If an invalid (non-UTF8) filename is read, the invalid caracters will
|
||||||
|
be replaced with the unicode replacement character, '<27>'. `rclone`
|
||||||
|
will emit a debug message in this case (use `-v` to see), eg
|
||||||
|
|
||||||
|
```
|
||||||
|
Local file system at .: Replacing invalid UTF-8 characters in "gro\xdf"
|
||||||
|
```
|
||||||
|
|||||||
@@ -100,8 +100,7 @@ files in the bucket.
|
|||||||
|
|
||||||
rclone sync /home/local/directory remote:bucket
|
rclone sync /home/local/directory remote:bucket
|
||||||
|
|
||||||
Modified time
|
### Modified time ###
|
||||||
-------------
|
|
||||||
|
|
||||||
The modified time is stored as metadata on the object as
|
The modified time is stored as metadata on the object as
|
||||||
`X-Amz-Meta-Mtime` as floating point since the epoch accurate to 1 ns.
|
`X-Amz-Meta-Mtime` as floating point since the epoch accurate to 1 ns.
|
||||||
|
|||||||
@@ -52,12 +52,15 @@ Choose a number from below, or type in your own value
|
|||||||
* Memset Memstore UK v2
|
* Memset Memstore UK v2
|
||||||
5) https://auth.storage.memset.com/v2.0
|
5) https://auth.storage.memset.com/v2.0
|
||||||
auth> 1
|
auth> 1
|
||||||
|
Tenant name - optional
|
||||||
|
tenant>
|
||||||
Remote config
|
Remote config
|
||||||
--------------------
|
--------------------
|
||||||
[remote]
|
[remote]
|
||||||
user = user_name
|
user = user_name
|
||||||
key = password_or_api_key
|
key = password_or_api_key
|
||||||
auth = https://auth.api.rackspacecloud.com/v1.0
|
auth = https://auth.api.rackspacecloud.com/v1.0
|
||||||
|
tenant =
|
||||||
--------------------
|
--------------------
|
||||||
y) Yes this is OK
|
y) Yes this is OK
|
||||||
e) Edit this remote
|
e) Edit this remote
|
||||||
@@ -84,8 +87,7 @@ excess files in the container.
|
|||||||
|
|
||||||
rclone sync /home/local/directory remote:container
|
rclone sync /home/local/directory remote:container
|
||||||
|
|
||||||
Modified time
|
### Modified time ###
|
||||||
-------------
|
|
||||||
|
|
||||||
The modified time is stored as metadata on the object as
|
The modified time is stored as metadata on the object as
|
||||||
`X-Object-Meta-Mtime` as floating point since the epoch accurate to 1
|
`X-Object-Meta-Mtime` as floating point since the epoch accurate to 1
|
||||||
|
|||||||
@@ -12,8 +12,17 @@
|
|||||||
<div class="collapse navbar-collapse navbar-ex1-collapse">
|
<div class="collapse navbar-collapse navbar-ex1-collapse">
|
||||||
<ul class="nav navbar-nav">
|
<ul class="nav navbar-nav">
|
||||||
<li><a href="/downloads/"><i class="fa fa-cloud-download"></i> Downloads</a></li>
|
<li><a href="/downloads/"><i class="fa fa-cloud-download"></i> Downloads</a></li>
|
||||||
<li><a href="/docs/"><i class="fa fa-book"></i> Docs</a></li>
|
<li class="dropdown">
|
||||||
<li><a href="/contact/"><i class="fa fa-envelope"></i> Contact</a></li>
|
<a href="#" class="dropdown-toggle" data-toggle="dropdown"><b class="caret"></b> Docs</a>
|
||||||
|
<ul class="dropdown-menu">
|
||||||
|
<li><a href="/install/"><i class="fa fa-book"></i> Installation</a></li>
|
||||||
|
<li><a href="/docs/"><i class="fa fa-book"></i> Usage</a></li>
|
||||||
|
<li><a href="/changelog/"><i class="fa fa-book"></i> Changelog</a></li>
|
||||||
|
<li><a href="/bugs/"><i class="fa fa-book"></i> Bugs</a></li>
|
||||||
|
<li><a href="/licence/"><i class="fa fa-book"></i> Licence</a></li>
|
||||||
|
<li><a href="/authors/"><i class="fa fa-book"></i> Authors</a></li>
|
||||||
|
</ul>
|
||||||
|
</li>
|
||||||
<li class="dropdown">
|
<li class="dropdown">
|
||||||
<a href="#" class="dropdown-toggle" data-toggle="dropdown"><b class="caret"></b> Storage Systems</a>
|
<a href="#" class="dropdown-toggle" data-toggle="dropdown"><b class="caret"></b> Storage Systems</a>
|
||||||
<ul class="dropdown-menu">
|
<ul class="dropdown-menu">
|
||||||
@@ -25,6 +34,7 @@
|
|||||||
<li><a href="/local/"><i class="fa fa-file"></i> Local</a></li>
|
<li><a href="/local/"><i class="fa fa-file"></i> Local</a></li>
|
||||||
</ul>
|
</ul>
|
||||||
</li>
|
</li>
|
||||||
|
<li><a href="/contact/"><i class="fa fa-envelope"></i> Contact</a></li>
|
||||||
</ul>
|
</ul>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|||||||
349
drive/drive.go
349
drive/drive.go
@@ -1,14 +1,6 @@
|
|||||||
// Drive interface
|
// Drive interface
|
||||||
package drive
|
package drive
|
||||||
|
|
||||||
// Gets this quite often
|
|
||||||
// Failed to set mtime: googleapi: Error 403: Rate Limit Exceeded
|
|
||||||
|
|
||||||
// FIXME list containers equivalent should list directories?
|
|
||||||
|
|
||||||
// FIXME list directory should list to channel for concurrency not
|
|
||||||
// append to array
|
|
||||||
|
|
||||||
// FIXME need to deal with some corner cases
|
// FIXME need to deal with some corner cases
|
||||||
// * multiple files with the same name
|
// * multiple files with the same name
|
||||||
// * files can be in multiple directories
|
// * files can be in multiple directories
|
||||||
@@ -18,16 +10,14 @@ package drive
|
|||||||
import (
|
import (
|
||||||
"fmt"
|
"fmt"
|
||||||
"io"
|
"io"
|
||||||
"log"
|
|
||||||
"mime"
|
|
||||||
"net/http"
|
"net/http"
|
||||||
"os"
|
|
||||||
"path"
|
|
||||||
"strings"
|
"strings"
|
||||||
"sync"
|
"sync"
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
"code.google.com/p/google-api-go-client/drive/v2"
|
"google.golang.org/api/drive/v2"
|
||||||
|
"google.golang.org/api/googleapi"
|
||||||
|
|
||||||
"github.com/ncw/rclone/fs"
|
"github.com/ncw/rclone/fs"
|
||||||
"github.com/ncw/rclone/googleauth"
|
"github.com/ncw/rclone/googleauth"
|
||||||
"github.com/ogier/pflag"
|
"github.com/ogier/pflag"
|
||||||
@@ -38,14 +28,21 @@ const (
|
|||||||
rcloneClientId = "202264815644.apps.googleusercontent.com"
|
rcloneClientId = "202264815644.apps.googleusercontent.com"
|
||||||
rcloneClientSecret = "X4Z3ca8xfWDb1Voo-F9a7ZxJ"
|
rcloneClientSecret = "X4Z3ca8xfWDb1Voo-F9a7ZxJ"
|
||||||
driveFolderType = "application/vnd.google-apps.folder"
|
driveFolderType = "application/vnd.google-apps.folder"
|
||||||
RFC3339In = time.RFC3339
|
timeFormatIn = time.RFC3339
|
||||||
RFC3339Out = "2006-01-02T15:04:05.000000000Z07:00"
|
timeFormatOut = "2006-01-02T15:04:05.000000000Z07:00"
|
||||||
|
minSleep = 10 * time.Millisecond
|
||||||
|
maxSleep = 2 * time.Second
|
||||||
|
decayConstant = 2 // bigger for slower decay, exponential
|
||||||
)
|
)
|
||||||
|
|
||||||
// Globals
|
// Globals
|
||||||
var (
|
var (
|
||||||
// Flags
|
// Flags
|
||||||
driveFullList = pflag.BoolP("drive-full-list", "", true, "Use a full listing for directory list. More data but usually quicker.")
|
driveFullList = pflag.BoolP("drive-full-list", "", true, "Use a full listing for directory list. More data but usually quicker.")
|
||||||
|
// chunkSize is the size of the chunks created during a resumable upload and should be a power of two.
|
||||||
|
// 1<<18 is the minimum size supported by the Google uploader, and there is no maximum.
|
||||||
|
chunkSize = fs.SizeSuffix(256 * 1024)
|
||||||
|
driveUploadCutoff = chunkSize
|
||||||
// Description of how to auth for this app
|
// Description of how to auth for this app
|
||||||
driveAuth = &googleauth.Auth{
|
driveAuth = &googleauth.Auth{
|
||||||
Scope: "https://www.googleapis.com/auth/drive",
|
Scope: "https://www.googleapis.com/auth/drive",
|
||||||
@@ -70,6 +67,8 @@ func init() {
|
|||||||
Help: "Google Application Client Secret - leave blank to use rclone's.",
|
Help: "Google Application Client Secret - leave blank to use rclone's.",
|
||||||
}},
|
}},
|
||||||
})
|
})
|
||||||
|
pflag.VarP(&driveUploadCutoff, "drive-upload-cutoff", "", "Cutoff for switching to chunked upload")
|
||||||
|
pflag.VarP(&chunkSize, "drive-chunk-size", "", "Upload chunk size. Must a power of 2 >= 256k.")
|
||||||
}
|
}
|
||||||
|
|
||||||
// FsDrive represents a remote drive server
|
// FsDrive represents a remote drive server
|
||||||
@@ -81,8 +80,10 @@ type FsDrive struct {
|
|||||||
rootId string // Id of the root directory
|
rootId string // Id of the root directory
|
||||||
foundRoot bool // Whether we have found the root or not
|
foundRoot bool // Whether we have found the root or not
|
||||||
findRootLock sync.Mutex // Protect findRoot from concurrent use
|
findRootLock sync.Mutex // Protect findRoot from concurrent use
|
||||||
dirCache dirCache // Map of directory path to directory id
|
dirCache *dirCache // Map of directory path to directory id
|
||||||
findDirLock sync.Mutex // Protect findDir from concurrent use
|
findDirLock sync.Mutex // Protect findDir from concurrent use
|
||||||
|
pacer chan struct{} // To pace the operations
|
||||||
|
sleepTime time.Duration // Time to sleep for each transaction
|
||||||
}
|
}
|
||||||
|
|
||||||
// FsObjectDrive describes a drive object
|
// FsObjectDrive describes a drive object
|
||||||
@@ -104,8 +105,8 @@ type dirCache struct {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Make a new locked map
|
// Make a new locked map
|
||||||
func newDirCache() dirCache {
|
func newDirCache() *dirCache {
|
||||||
d := dirCache{}
|
d := &dirCache{}
|
||||||
d.Flush()
|
d.Flush()
|
||||||
return d
|
return d
|
||||||
}
|
}
|
||||||
@@ -149,6 +150,97 @@ func (f *FsDrive) String() string {
|
|||||||
return fmt.Sprintf("Google drive root '%s'", f.root)
|
return fmt.Sprintf("Google drive root '%s'", f.root)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Start a call to the drive API
|
||||||
|
//
|
||||||
|
// This must be called as a pair with endCall
|
||||||
|
//
|
||||||
|
// This waits for the pacer token
|
||||||
|
func (f *FsDrive) beginCall() {
|
||||||
|
// pacer starts with a token in and whenever we take one out
|
||||||
|
// XXX ms later we put another in. We could do this with a
|
||||||
|
// Ticker more accurately, but then we'd have to work out how
|
||||||
|
// not to run it when it wasn't needed
|
||||||
|
<-f.pacer
|
||||||
|
|
||||||
|
// Restart the timer
|
||||||
|
go func(t time.Duration) {
|
||||||
|
// fs.Debug(f, "New sleep for %v at %v", t, time.Now())
|
||||||
|
time.Sleep(t)
|
||||||
|
f.pacer <- struct{}{}
|
||||||
|
}(f.sleepTime)
|
||||||
|
}
|
||||||
|
|
||||||
|
// End a call to the drive API
|
||||||
|
//
|
||||||
|
// Refresh the pace given an error that was returned. It returns a
|
||||||
|
// boolean as to whether the operation should be retried.
|
||||||
|
//
|
||||||
|
// See https://developers.google.com/drive/web/handle-errors
|
||||||
|
// http://stackoverflow.com/questions/18529524/403-rate-limit-after-only-1-insert-per-second
|
||||||
|
func (f *FsDrive) endCall(err error) bool {
|
||||||
|
again := false
|
||||||
|
oldSleepTime := f.sleepTime
|
||||||
|
if err == nil {
|
||||||
|
f.sleepTime = (f.sleepTime<<decayConstant - f.sleepTime) >> decayConstant
|
||||||
|
if f.sleepTime < minSleep {
|
||||||
|
f.sleepTime = minSleep
|
||||||
|
}
|
||||||
|
if f.sleepTime != oldSleepTime {
|
||||||
|
fs.Debug(f, "Reducing sleep to %v", f.sleepTime)
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
fs.Debug(f, "Error recived: %T %#v", err, err)
|
||||||
|
// Check for net error Timeout()
|
||||||
|
if x, ok := err.(interface {
|
||||||
|
Timeout() bool
|
||||||
|
}); ok && x.Timeout() {
|
||||||
|
again = true
|
||||||
|
}
|
||||||
|
// Check for net error Temporary()
|
||||||
|
if x, ok := err.(interface {
|
||||||
|
Temporary() bool
|
||||||
|
}); ok && x.Temporary() {
|
||||||
|
again = true
|
||||||
|
}
|
||||||
|
switch gerr := err.(type) {
|
||||||
|
case *googleapi.Error:
|
||||||
|
if gerr.Code >= 500 && gerr.Code < 600 {
|
||||||
|
// All 5xx errors should be retried
|
||||||
|
again = true
|
||||||
|
} else if len(gerr.Errors) > 0 {
|
||||||
|
reason := gerr.Errors[0].Reason
|
||||||
|
if reason == "rateLimitExceeded" || reason == "userRateLimitExceeded" {
|
||||||
|
again = true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if again {
|
||||||
|
f.sleepTime *= 2
|
||||||
|
if f.sleepTime > maxSleep {
|
||||||
|
f.sleepTime = maxSleep
|
||||||
|
}
|
||||||
|
if f.sleepTime != oldSleepTime {
|
||||||
|
fs.Debug(f, "Rate limited, increasing sleep to %v", f.sleepTime)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return again
|
||||||
|
}
|
||||||
|
|
||||||
|
// Pace the remote operations to not exceed Google's limits and retry
|
||||||
|
// on 403 rate limit exceeded
|
||||||
|
//
|
||||||
|
// This calls fn, expecting it to place its error in perr
|
||||||
|
func (f *FsDrive) call(perr *error, fn func()) {
|
||||||
|
for {
|
||||||
|
f.beginCall()
|
||||||
|
fn()
|
||||||
|
if !f.endCall(*perr) {
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
// parseParse parses a drive 'url'
|
// parseParse parses a drive 'url'
|
||||||
func parseDrivePath(path string) (root string, err error) {
|
func parseDrivePath(path string) (root string, err error) {
|
||||||
root = strings.Trim(path, "/")
|
root = strings.Trim(path, "/")
|
||||||
@@ -186,7 +278,10 @@ func (f *FsDrive) listAll(dirId string, title string, directoriesOnly bool, file
|
|||||||
list := f.svc.Files.List().Q(query).MaxResults(1000)
|
list := f.svc.Files.List().Q(query).MaxResults(1000)
|
||||||
OUTER:
|
OUTER:
|
||||||
for {
|
for {
|
||||||
files, err := list.Do()
|
var files *drive.FileList
|
||||||
|
f.call(&err, func() {
|
||||||
|
files, err = list.Do()
|
||||||
|
})
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return false, fmt.Errorf("Couldn't list directory: %s", err)
|
return false, fmt.Errorf("Couldn't list directory: %s", err)
|
||||||
}
|
}
|
||||||
@@ -204,8 +299,27 @@ OUTER:
|
|||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Returns true of x is a power of 2 or zero
|
||||||
|
func isPowerOfTwo(x int64) bool {
|
||||||
|
switch {
|
||||||
|
case x == 0:
|
||||||
|
return true
|
||||||
|
case x < 0:
|
||||||
|
return false
|
||||||
|
default:
|
||||||
|
return (x & (x - 1)) == 0
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
// NewFs contstructs an FsDrive from the path, container:path
|
// NewFs contstructs an FsDrive from the path, container:path
|
||||||
func NewFs(name, path string) (fs.Fs, error) {
|
func NewFs(name, path string) (fs.Fs, error) {
|
||||||
|
if !isPowerOfTwo(int64(chunkSize)) {
|
||||||
|
return nil, fmt.Errorf("drive: chunk size %v isn't a power of two", chunkSize)
|
||||||
|
}
|
||||||
|
if chunkSize < 256*1024 {
|
||||||
|
return nil, fmt.Errorf("drive: chunk size can't be less than 256k - was %v", chunkSize)
|
||||||
|
}
|
||||||
|
|
||||||
t, err := driveAuth.NewTransport(name)
|
t, err := driveAuth.NewTransport(name)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
@@ -217,10 +331,15 @@ func NewFs(name, path string) (fs.Fs, error) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
f := &FsDrive{
|
f := &FsDrive{
|
||||||
root: root,
|
root: root,
|
||||||
dirCache: newDirCache(),
|
dirCache: newDirCache(),
|
||||||
|
pacer: make(chan struct{}, 1),
|
||||||
|
sleepTime: minSleep,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Put the first pacing token in
|
||||||
|
f.pacer <- struct{}{}
|
||||||
|
|
||||||
// Create a new authorized Drive client.
|
// Create a new authorized Drive client.
|
||||||
f.client = t.Client()
|
f.client = t.Client()
|
||||||
f.svc, err = drive.New(f.client)
|
f.svc, err = drive.New(f.client)
|
||||||
@@ -229,15 +348,15 @@ func NewFs(name, path string) (fs.Fs, error) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Read About so we know the root path
|
// Read About so we know the root path
|
||||||
f.about, err = f.svc.About.Get().Do()
|
f.call(&err, func() {
|
||||||
|
f.about, err = f.svc.About.Get().Do()
|
||||||
|
})
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, fmt.Errorf("Couldn't read info about Drive: %s", err)
|
return nil, fmt.Errorf("Couldn't read info about Drive: %s", err)
|
||||||
}
|
}
|
||||||
|
|
||||||
// Find the Id of the root directory and the Id of its parent
|
// Find the Id of the true root and clear everything
|
||||||
f.rootId = f.about.RootFolderId
|
f.resetRoot()
|
||||||
// Put the root directory in
|
|
||||||
f.dirCache.Put("", f.rootId)
|
|
||||||
// Find the current root
|
// Find the current root
|
||||||
err = f.findRoot(false)
|
err = f.findRoot(false)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
@@ -251,7 +370,7 @@ func NewFs(name, path string) (fs.Fs, error) {
|
|||||||
// No root so return old f
|
// No root so return old f
|
||||||
return f, nil
|
return f, nil
|
||||||
}
|
}
|
||||||
obj, err := newF.newFsObjectWithInfo(remote, nil)
|
obj, err := newF.newFsObjectWithInfoErr(remote, nil)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
// File doesn't exist so return old f
|
// File doesn't exist so return old f
|
||||||
return f, nil
|
return f, nil
|
||||||
@@ -264,7 +383,7 @@ func NewFs(name, path string) (fs.Fs, error) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Return an FsObject from a path
|
// Return an FsObject from a path
|
||||||
func (f *FsDrive) newFsObjectWithInfo(remote string, info *drive.File) (fs.Object, error) {
|
func (f *FsDrive) newFsObjectWithInfoErr(remote string, info *drive.File) (fs.Object, error) {
|
||||||
fs := &FsObjectDrive{
|
fs := &FsObjectDrive{
|
||||||
drive: f,
|
drive: f,
|
||||||
remote: remote,
|
remote: remote,
|
||||||
@@ -284,8 +403,8 @@ func (f *FsDrive) newFsObjectWithInfo(remote string, info *drive.File) (fs.Objec
|
|||||||
// Return an FsObject from a path
|
// Return an FsObject from a path
|
||||||
//
|
//
|
||||||
// May return nil if an error occurred
|
// May return nil if an error occurred
|
||||||
func (f *FsDrive) NewFsObjectWithInfo(remote string, info *drive.File) fs.Object {
|
func (f *FsDrive) newFsObjectWithInfo(remote string, info *drive.File) fs.Object {
|
||||||
fs, _ := f.newFsObjectWithInfo(remote, info)
|
fs, _ := f.newFsObjectWithInfoErr(remote, info)
|
||||||
// Errors have already been logged
|
// Errors have already been logged
|
||||||
return fs
|
return fs
|
||||||
}
|
}
|
||||||
@@ -294,7 +413,7 @@ func (f *FsDrive) NewFsObjectWithInfo(remote string, info *drive.File) fs.Object
|
|||||||
//
|
//
|
||||||
// May return nil if an error occurred
|
// May return nil if an error occurred
|
||||||
func (f *FsDrive) NewFsObject(remote string) fs.Object {
|
func (f *FsDrive) NewFsObject(remote string) fs.Object {
|
||||||
return f.NewFsObjectWithInfo(remote, nil)
|
return f.newFsObjectWithInfo(remote, nil)
|
||||||
}
|
}
|
||||||
|
|
||||||
// Path should be directory path either "" or "path/"
|
// Path should be directory path either "" or "path/"
|
||||||
@@ -318,7 +437,7 @@ func (f *FsDrive) listDirRecursive(dirId string, path string, out fs.ObjectsChan
|
|||||||
} else {
|
} else {
|
||||||
// If item has no MD5 sum it isn't stored on drive, so ignore it
|
// If item has no MD5 sum it isn't stored on drive, so ignore it
|
||||||
if item.Md5Checksum != "" {
|
if item.Md5Checksum != "" {
|
||||||
if fs := f.NewFsObjectWithInfo(path+item.Title, item); fs != nil {
|
if fs := f.newFsObjectWithInfo(path+item.Title, item); fs != nil {
|
||||||
out <- fs
|
out <- fs
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -368,7 +487,7 @@ func (f *FsDrive) listDirFull(dirId string, path string, out fs.ObjectsChan) err
|
|||||||
// fmt.Printf("file %s %s %s\n", path, item.Title, item.Id)
|
// fmt.Printf("file %s %s %s\n", path, item.Title, item.Id)
|
||||||
// If item has no MD5 sum it isn't stored on drive, so ignore it
|
// If item has no MD5 sum it isn't stored on drive, so ignore it
|
||||||
if item.Md5Checksum != "" {
|
if item.Md5Checksum != "" {
|
||||||
if fs := f.NewFsObjectWithInfo(path, item); fs != nil {
|
if fs := f.newFsObjectWithInfo(path, item); fs != nil {
|
||||||
out <- fs
|
out <- fs
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -491,15 +610,18 @@ func (f *FsDrive) _findDir(path string, create bool) (pathId string, err error)
|
|||||||
if create {
|
if create {
|
||||||
// fmt.Println("Making", path)
|
// fmt.Println("Making", path)
|
||||||
// Define the metadata for the directory we are going to create.
|
// Define the metadata for the directory we are going to create.
|
||||||
info := &drive.File{
|
createInfo := &drive.File{
|
||||||
Title: leaf,
|
Title: leaf,
|
||||||
Description: leaf,
|
Description: leaf,
|
||||||
MimeType: driveFolderType,
|
MimeType: driveFolderType,
|
||||||
Parents: []*drive.ParentReference{{Id: pathId}},
|
Parents: []*drive.ParentReference{{Id: pathId}},
|
||||||
}
|
}
|
||||||
info, err := f.svc.Files.Insert(info).Do()
|
var info *drive.File
|
||||||
|
f.call(&err, func() {
|
||||||
|
info, err = f.svc.Files.Insert(createInfo).Do()
|
||||||
|
})
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return pathId, fmt.Errorf("Failed to make directory")
|
return pathId, fmt.Errorf("Failed to make directory: %v", err)
|
||||||
}
|
}
|
||||||
pathId = info.Id
|
pathId = info.Id
|
||||||
} else {
|
} else {
|
||||||
@@ -537,6 +659,20 @@ func (f *FsDrive) findRoot(create bool) error {
|
|||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Resets the root directory to the absolute root and clears the dirCache
|
||||||
|
func (f *FsDrive) resetRoot() {
|
||||||
|
f.findRootLock.Lock()
|
||||||
|
defer f.findRootLock.Unlock()
|
||||||
|
f.foundRoot = false
|
||||||
|
f.dirCache.Flush()
|
||||||
|
|
||||||
|
// Put the true root in
|
||||||
|
f.rootId = f.about.RootFolderId
|
||||||
|
|
||||||
|
// Put the root directory in
|
||||||
|
f.dirCache.Put("", f.rootId)
|
||||||
|
}
|
||||||
|
|
||||||
// Walk the path returning a channel of FsObjects
|
// Walk the path returning a channel of FsObjects
|
||||||
func (f *FsDrive) List() fs.ObjectsChan {
|
func (f *FsDrive) List() fs.ObjectsChan {
|
||||||
out := make(fs.ObjectsChan, fs.Config.Checkers)
|
out := make(fs.ObjectsChan, fs.Config.Checkers)
|
||||||
@@ -545,16 +681,16 @@ func (f *FsDrive) List() fs.ObjectsChan {
|
|||||||
err := f.findRoot(false)
|
err := f.findRoot(false)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
fs.Stats.Error()
|
fs.Stats.Error()
|
||||||
log.Printf("Couldn't find root: %s", err)
|
fs.Log(f, "Couldn't find root: %s", err)
|
||||||
} else {
|
} else {
|
||||||
if *driveFullList {
|
if f.root == "" && *driveFullList {
|
||||||
err = f.listDirFull(f.rootId, "", out)
|
err = f.listDirFull(f.rootId, "", out)
|
||||||
} else {
|
} else {
|
||||||
err = f.listDirRecursive(f.rootId, "", out)
|
err = f.listDirRecursive(f.rootId, "", out)
|
||||||
}
|
}
|
||||||
if err != nil {
|
if err != nil {
|
||||||
fs.Stats.Error()
|
fs.Stats.Error()
|
||||||
log.Printf("List failed: %s", err)
|
fs.Log(f, "List failed: %s", err)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}()
|
}()
|
||||||
@@ -569,7 +705,7 @@ func (f *FsDrive) ListDir() fs.DirChan {
|
|||||||
err := f.findRoot(false)
|
err := f.findRoot(false)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
fs.Stats.Error()
|
fs.Stats.Error()
|
||||||
log.Printf("Couldn't find root: %s", err)
|
fs.Log(f, "Couldn't find root: %s", err)
|
||||||
} else {
|
} else {
|
||||||
_, err := f.listAll(f.rootId, "", true, false, func(item *drive.File) bool {
|
_, err := f.listAll(f.rootId, "", true, false, func(item *drive.File) bool {
|
||||||
dir := &fs.Dir{
|
dir := &fs.Dir{
|
||||||
@@ -577,43 +713,19 @@ func (f *FsDrive) ListDir() fs.DirChan {
|
|||||||
Bytes: -1,
|
Bytes: -1,
|
||||||
Count: -1,
|
Count: -1,
|
||||||
}
|
}
|
||||||
dir.When, _ = time.Parse(RFC3339In, item.ModifiedDate)
|
dir.When, _ = time.Parse(timeFormatIn, item.ModifiedDate)
|
||||||
out <- dir
|
out <- dir
|
||||||
return false
|
return false
|
||||||
})
|
})
|
||||||
if err != nil {
|
if err != nil {
|
||||||
fs.Stats.Error()
|
fs.Stats.Error()
|
||||||
log.Printf("ListDir failed: %s", err)
|
fs.Log(f, "ListDir failed: %s", err)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}()
|
}()
|
||||||
return out
|
return out
|
||||||
}
|
}
|
||||||
|
|
||||||
// seekWrapper wraps an io.Reader with a basic Seek for
|
|
||||||
// code.google.com/p/google-api-go-client/googleapi
|
|
||||||
// to detect the length (see getReaderSize function)
|
|
||||||
type seekWrapper struct {
|
|
||||||
in io.Reader
|
|
||||||
size int64
|
|
||||||
}
|
|
||||||
|
|
||||||
// Read bytes from the object - see io.Reader
|
|
||||||
func (file *seekWrapper) Read(p []byte) (n int, err error) {
|
|
||||||
return file.in.Read(p)
|
|
||||||
}
|
|
||||||
|
|
||||||
// Seek - minimal implementation for Google Drive's length detection
|
|
||||||
func (file *seekWrapper) Seek(offset int64, whence int) (int64, error) {
|
|
||||||
switch whence {
|
|
||||||
case os.SEEK_CUR:
|
|
||||||
return 0, nil
|
|
||||||
case os.SEEK_END:
|
|
||||||
return file.size, nil
|
|
||||||
}
|
|
||||||
return 0, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// Put the object
|
// Put the object
|
||||||
//
|
//
|
||||||
// This assumes that the object doesn't not already exists - if you
|
// This assumes that the object doesn't not already exists - if you
|
||||||
@@ -633,27 +745,33 @@ func (f *FsDrive) Put(in io.Reader, remote string, modTime time.Time, size int64
|
|||||||
return o, fmt.Errorf("Couldn't find or make directory: %s", err)
|
return o, fmt.Errorf("Couldn't find or make directory: %s", err)
|
||||||
}
|
}
|
||||||
|
|
||||||
// Guess the mime type
|
|
||||||
mimeType := mime.TypeByExtension(path.Ext(o.remote))
|
|
||||||
if mimeType == "" {
|
|
||||||
mimeType = "application/octet-stream"
|
|
||||||
}
|
|
||||||
modifiedDate := modTime.Format(RFC3339Out)
|
|
||||||
|
|
||||||
// Define the metadata for the file we are going to create.
|
// Define the metadata for the file we are going to create.
|
||||||
info := &drive.File{
|
createInfo := &drive.File{
|
||||||
Title: leaf,
|
Title: leaf,
|
||||||
Description: leaf,
|
Description: leaf,
|
||||||
Parents: []*drive.ParentReference{{Id: directoryId}},
|
Parents: []*drive.ParentReference{{Id: directoryId}},
|
||||||
MimeType: mimeType,
|
MimeType: fs.MimeType(o),
|
||||||
ModifiedDate: modifiedDate,
|
ModifiedDate: modTime.Format(timeFormatOut),
|
||||||
}
|
}
|
||||||
|
|
||||||
// Make the API request to upload metadata and file data.
|
var info *drive.File
|
||||||
in = &seekWrapper{in: in, size: size}
|
if size == 0 || size < int64(driveUploadCutoff) {
|
||||||
info, err = f.svc.Files.Insert(info).Media(in).Do()
|
// Make the API request to upload metadata and file data.
|
||||||
if err != nil {
|
// Don't retry, return a retry error instead
|
||||||
return o, fmt.Errorf("Upload failed: %s", err)
|
f.beginCall()
|
||||||
|
info, err = f.svc.Files.Insert(createInfo).Media(in).Do()
|
||||||
|
if f.endCall(err) {
|
||||||
|
return o, fs.RetryErrorf("Upload failed - retry: %s", err)
|
||||||
|
}
|
||||||
|
if err != nil {
|
||||||
|
return o, fmt.Errorf("Upload failed: %s", err)
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
// Upload the file in chunks
|
||||||
|
info, err = f.Upload(in, size, createInfo.MimeType, createInfo, remote)
|
||||||
|
if err != nil {
|
||||||
|
return o, err
|
||||||
|
}
|
||||||
}
|
}
|
||||||
o.setMetaData(info)
|
o.setMetaData(info)
|
||||||
return o, nil
|
return o, nil
|
||||||
@@ -672,7 +790,10 @@ func (f *FsDrive) Rmdir() error {
|
|||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
children, err := f.svc.Children.List(f.rootId).MaxResults(10).Do()
|
var children *drive.ChildList
|
||||||
|
f.call(&err, func() {
|
||||||
|
children, err = f.svc.Children.List(f.rootId).MaxResults(10).Do()
|
||||||
|
})
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
@@ -681,11 +802,14 @@ func (f *FsDrive) Rmdir() error {
|
|||||||
}
|
}
|
||||||
// Delete the directory if it isn't the root
|
// Delete the directory if it isn't the root
|
||||||
if f.root != "" {
|
if f.root != "" {
|
||||||
err = f.svc.Files.Delete(f.rootId).Do()
|
f.call(&err, func() {
|
||||||
|
err = f.svc.Files.Delete(f.rootId).Do()
|
||||||
|
})
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
f.resetRoot()
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -707,8 +831,10 @@ func (f *FsDrive) Purge() error {
|
|||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
err = f.svc.Files.Delete(f.rootId).Do()
|
f.call(&err, func() {
|
||||||
f.dirCache.Flush()
|
err = f.svc.Files.Delete(f.rootId).Do()
|
||||||
|
})
|
||||||
|
f.resetRoot()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
@@ -795,7 +921,7 @@ func (o *FsObjectDrive) ModTime() time.Time {
|
|||||||
fs.Log(o, "Failed to read metadata: %s", err)
|
fs.Log(o, "Failed to read metadata: %s", err)
|
||||||
return time.Now()
|
return time.Now()
|
||||||
}
|
}
|
||||||
modTime, err := time.Parse(RFC3339In, o.modifiedDate)
|
modTime, err := time.Parse(timeFormatIn, o.modifiedDate)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
fs.Log(o, "Failed to read mtime from object: %s", err)
|
fs.Log(o, "Failed to read mtime from object: %s", err)
|
||||||
return time.Now()
|
return time.Now()
|
||||||
@@ -812,15 +938,21 @@ func (o *FsObjectDrive) SetModTime(modTime time.Time) {
|
|||||||
return
|
return
|
||||||
}
|
}
|
||||||
// New metadata
|
// New metadata
|
||||||
info := &drive.File{
|
updateInfo := &drive.File{
|
||||||
ModifiedDate: modTime.Format(RFC3339Out),
|
ModifiedDate: modTime.Format(timeFormatOut),
|
||||||
}
|
}
|
||||||
// Set modified date
|
// Set modified date
|
||||||
_, err = o.drive.svc.Files.Update(o.id, info).SetModifiedDate(true).Do()
|
var info *drive.File
|
||||||
|
o.drive.call(&err, func() {
|
||||||
|
info, err = o.drive.svc.Files.Update(o.id, updateInfo).SetModifiedDate(true).Do()
|
||||||
|
})
|
||||||
if err != nil {
|
if err != nil {
|
||||||
fs.Stats.Error()
|
fs.Stats.Error()
|
||||||
fs.Log(o, "Failed to update remote mtime: %s", err)
|
fs.Log(o, "Failed to update remote mtime: %s", err)
|
||||||
|
return
|
||||||
}
|
}
|
||||||
|
// Update info from read data
|
||||||
|
o.setMetaData(info)
|
||||||
}
|
}
|
||||||
|
|
||||||
// Is this object storable
|
// Is this object storable
|
||||||
@@ -835,12 +967,15 @@ func (o *FsObjectDrive) Open() (in io.ReadCloser, err error) {
|
|||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
req.Header.Set("User-Agent", fs.UserAgent)
|
req.Header.Set("User-Agent", fs.UserAgent)
|
||||||
res, err := o.drive.client.Do(req)
|
var res *http.Response
|
||||||
|
o.drive.call(&err, func() {
|
||||||
|
res, err = o.drive.client.Do(req)
|
||||||
|
})
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
if res.StatusCode != 200 {
|
if res.StatusCode != 200 {
|
||||||
res.Body.Close()
|
_ = res.Body.Close() // ignore error
|
||||||
return nil, fmt.Errorf("Bad response: %d: %s", res.StatusCode, res.Status)
|
return nil, fmt.Errorf("Bad response: %d: %s", res.StatusCode, res.Status)
|
||||||
}
|
}
|
||||||
return res.Body, nil
|
return res.Body, nil
|
||||||
@@ -852,16 +987,30 @@ func (o *FsObjectDrive) Open() (in io.ReadCloser, err error) {
|
|||||||
//
|
//
|
||||||
// The new object may have been created if an error is returned
|
// The new object may have been created if an error is returned
|
||||||
func (o *FsObjectDrive) Update(in io.Reader, modTime time.Time, size int64) error {
|
func (o *FsObjectDrive) Update(in io.Reader, modTime time.Time, size int64) error {
|
||||||
info := &drive.File{
|
updateInfo := &drive.File{
|
||||||
Id: o.id,
|
Id: o.id,
|
||||||
ModifiedDate: modTime.Format(RFC3339Out),
|
ModifiedDate: modTime.Format(timeFormatOut),
|
||||||
}
|
}
|
||||||
|
|
||||||
// Make the API request to upload metadata and file data.
|
// Make the API request to upload metadata and file data.
|
||||||
in = &seekWrapper{in: in, size: size}
|
var err error
|
||||||
info, err := o.drive.svc.Files.Update(info.Id, info).SetModifiedDate(true).Media(in).Do()
|
var info *drive.File
|
||||||
if err != nil {
|
if size == 0 || size < int64(driveUploadCutoff) {
|
||||||
return fmt.Errorf("Update failed: %s", err)
|
// Don't retry, return a retry error instead
|
||||||
|
o.drive.beginCall()
|
||||||
|
info, err = o.drive.svc.Files.Update(updateInfo.Id, updateInfo).SetModifiedDate(true).Media(in).Do()
|
||||||
|
if o.drive.endCall(err) {
|
||||||
|
return fs.RetryErrorf("Update failed - retry: %s", err)
|
||||||
|
}
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("Update failed: %s", err)
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
// Upload the file in chunks
|
||||||
|
info, err = o.drive.Upload(in, size, fs.MimeType(o), updateInfo, o.remote)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
}
|
}
|
||||||
o.setMetaData(info)
|
o.setMetaData(info)
|
||||||
return nil
|
return nil
|
||||||
@@ -869,7 +1018,11 @@ func (o *FsObjectDrive) Update(in io.Reader, modTime time.Time, size int64) erro
|
|||||||
|
|
||||||
// Remove an object
|
// Remove an object
|
||||||
func (o *FsObjectDrive) Remove() error {
|
func (o *FsObjectDrive) Remove() error {
|
||||||
return o.drive.svc.Files.Delete(o.id).Do()
|
var err error
|
||||||
|
o.drive.call(&err, func() {
|
||||||
|
err = o.drive.svc.Files.Delete(o.id).Do()
|
||||||
|
})
|
||||||
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
// Check the interfaces are satisfied
|
// Check the interfaces are satisfied
|
||||||
|
|||||||
53
drive/drive_test.go
Normal file
53
drive/drive_test.go
Normal file
@@ -0,0 +1,53 @@
|
|||||||
|
// Test Drive filesystem interface
|
||||||
|
//
|
||||||
|
// Automatically generated - DO NOT EDIT
|
||||||
|
// Regenerate with: go run gen_tests.go or make gen_tests
|
||||||
|
package drive_test
|
||||||
|
|
||||||
|
import (
|
||||||
|
"testing"
|
||||||
|
|
||||||
|
"github.com/ncw/rclone/drive"
|
||||||
|
"github.com/ncw/rclone/fs"
|
||||||
|
"github.com/ncw/rclone/fstest/fstests"
|
||||||
|
)
|
||||||
|
|
||||||
|
func init() {
|
||||||
|
fstests.NilObject = fs.Object((*drive.FsObjectDrive)(nil))
|
||||||
|
fstests.RemoteName = "TestDrive:"
|
||||||
|
}
|
||||||
|
|
||||||
|
// Generic tests for the Fs
|
||||||
|
func TestInit(t *testing.T) { fstests.TestInit(t) }
|
||||||
|
func TestFsString(t *testing.T) { fstests.TestFsString(t) }
|
||||||
|
func TestFsRmdirEmpty(t *testing.T) { fstests.TestFsRmdirEmpty(t) }
|
||||||
|
func TestFsRmdirNotFound(t *testing.T) { fstests.TestFsRmdirNotFound(t) }
|
||||||
|
func TestFsMkdir(t *testing.T) { fstests.TestFsMkdir(t) }
|
||||||
|
func TestFsListEmpty(t *testing.T) { fstests.TestFsListEmpty(t) }
|
||||||
|
func TestFsListDirEmpty(t *testing.T) { fstests.TestFsListDirEmpty(t) }
|
||||||
|
func TestFsNewFsObjectNotFound(t *testing.T) { fstests.TestFsNewFsObjectNotFound(t) }
|
||||||
|
func TestFsPutFile1(t *testing.T) { fstests.TestFsPutFile1(t) }
|
||||||
|
func TestFsPutFile2(t *testing.T) { fstests.TestFsPutFile2(t) }
|
||||||
|
func TestFsListDirFile2(t *testing.T) { fstests.TestFsListDirFile2(t) }
|
||||||
|
func TestFsListDirRoot(t *testing.T) { fstests.TestFsListDirRoot(t) }
|
||||||
|
func TestFsListRoot(t *testing.T) { fstests.TestFsListRoot(t) }
|
||||||
|
func TestFsListFile1(t *testing.T) { fstests.TestFsListFile1(t) }
|
||||||
|
func TestFsNewFsObject(t *testing.T) { fstests.TestFsNewFsObject(t) }
|
||||||
|
func TestFsListFile1and2(t *testing.T) { fstests.TestFsListFile1and2(t) }
|
||||||
|
func TestFsRmdirFull(t *testing.T) { fstests.TestFsRmdirFull(t) }
|
||||||
|
func TestFsPrecision(t *testing.T) { fstests.TestFsPrecision(t) }
|
||||||
|
func TestObjectString(t *testing.T) { fstests.TestObjectString(t) }
|
||||||
|
func TestObjectFs(t *testing.T) { fstests.TestObjectFs(t) }
|
||||||
|
func TestObjectRemote(t *testing.T) { fstests.TestObjectRemote(t) }
|
||||||
|
func TestObjectMd5sum(t *testing.T) { fstests.TestObjectMd5sum(t) }
|
||||||
|
func TestObjectModTime(t *testing.T) { fstests.TestObjectModTime(t) }
|
||||||
|
func TestObjectSetModTime(t *testing.T) { fstests.TestObjectSetModTime(t) }
|
||||||
|
func TestObjectSize(t *testing.T) { fstests.TestObjectSize(t) }
|
||||||
|
func TestObjectOpen(t *testing.T) { fstests.TestObjectOpen(t) }
|
||||||
|
func TestObjectUpdate(t *testing.T) { fstests.TestObjectUpdate(t) }
|
||||||
|
func TestObjectStorable(t *testing.T) { fstests.TestObjectStorable(t) }
|
||||||
|
func TestLimitedFs(t *testing.T) { fstests.TestLimitedFs(t) }
|
||||||
|
func TestLimitedFsNotFound(t *testing.T) { fstests.TestLimitedFsNotFound(t) }
|
||||||
|
func TestObjectRemove(t *testing.T) { fstests.TestObjectRemove(t) }
|
||||||
|
func TestObjectPurge(t *testing.T) { fstests.TestObjectPurge(t) }
|
||||||
|
func TestFinalise(t *testing.T) { fstests.TestFinalise(t) }
|
||||||
246
drive/upload.go
Normal file
246
drive/upload.go
Normal file
@@ -0,0 +1,246 @@
|
|||||||
|
// Upload for drive
|
||||||
|
//
|
||||||
|
// Docs
|
||||||
|
// Resumable upload: https://developers.google.com/drive/web/manage-uploads#resumable
|
||||||
|
// Best practices: https://developers.google.com/drive/web/manage-uploads#best-practices
|
||||||
|
// Files insert: https://developers.google.com/drive/v2/reference/files/insert
|
||||||
|
// Files update: https://developers.google.com/drive/v2/reference/files/update
|
||||||
|
//
|
||||||
|
// This contains code adapted from google.golang.org/api (C) the GO AUTHORS
|
||||||
|
|
||||||
|
package drive
|
||||||
|
|
||||||
|
import (
|
||||||
|
"bytes"
|
||||||
|
"encoding/json"
|
||||||
|
"fmt"
|
||||||
|
"io"
|
||||||
|
"net/http"
|
||||||
|
"net/url"
|
||||||
|
"regexp"
|
||||||
|
"strconv"
|
||||||
|
|
||||||
|
"github.com/ncw/rclone/fs"
|
||||||
|
"google.golang.org/api/drive/v2"
|
||||||
|
"google.golang.org/api/googleapi"
|
||||||
|
)
|
||||||
|
|
||||||
|
const (
|
||||||
|
// statusResumeIncomplete is the code returned by the Google uploader when the transfer is not yet complete.
|
||||||
|
statusResumeIncomplete = 308
|
||||||
|
|
||||||
|
// Number of times to try each chunk
|
||||||
|
maxTries = 10
|
||||||
|
)
|
||||||
|
|
||||||
|
// resumableUpload is used by the generated APIs to provide resumable uploads.
|
||||||
|
// It is not used by developers directly.
|
||||||
|
type resumableUpload struct {
|
||||||
|
f *FsDrive
|
||||||
|
remote string
|
||||||
|
// URI is the resumable resource destination provided by the server after specifying "&uploadType=resumable".
|
||||||
|
URI string
|
||||||
|
// Media is the object being uploaded.
|
||||||
|
Media io.Reader
|
||||||
|
// MediaType defines the media type, e.g. "image/jpeg".
|
||||||
|
MediaType string
|
||||||
|
// ContentLength is the full size of the object being uploaded.
|
||||||
|
ContentLength int64
|
||||||
|
// Return value
|
||||||
|
ret *drive.File
|
||||||
|
}
|
||||||
|
|
||||||
|
// Upload the io.Reader in of size bytes with contentType and info
|
||||||
|
func (f *FsDrive) Upload(in io.Reader, size int64, contentType string, info *drive.File, remote string) (*drive.File, error) {
|
||||||
|
fileId := info.Id
|
||||||
|
var body io.Reader = nil
|
||||||
|
body, err := googleapi.WithoutDataWrapper.JSONReader(info)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
params := make(url.Values)
|
||||||
|
params.Set("alt", "json")
|
||||||
|
params.Set("uploadType", "resumable")
|
||||||
|
urls := "https://www.googleapis.com/upload/drive/v2/files"
|
||||||
|
method := "POST"
|
||||||
|
if fileId != "" {
|
||||||
|
params.Set("setModifiedDate", "true")
|
||||||
|
urls += "/{fileId}"
|
||||||
|
method = "PUT"
|
||||||
|
}
|
||||||
|
urls += "?" + params.Encode()
|
||||||
|
req, _ := http.NewRequest(method, urls, body)
|
||||||
|
googleapi.Expand(req.URL, map[string]string{
|
||||||
|
"fileId": fileId,
|
||||||
|
})
|
||||||
|
req.Header.Set("Content-Type", "application/json; charset=UTF-8")
|
||||||
|
req.Header.Set("X-Upload-Content-Type", contentType)
|
||||||
|
req.Header.Set("X-Upload-Content-Length", fmt.Sprintf("%v", size))
|
||||||
|
req.Header.Set("User-Agent", fs.UserAgent)
|
||||||
|
var res *http.Response
|
||||||
|
f.call(&err, func() {
|
||||||
|
res, err = f.client.Do(req)
|
||||||
|
if err == nil {
|
||||||
|
defer googleapi.CloseBody(res)
|
||||||
|
err = googleapi.CheckResponse(res)
|
||||||
|
}
|
||||||
|
})
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
loc := res.Header.Get("Location")
|
||||||
|
rx := &resumableUpload{
|
||||||
|
f: f,
|
||||||
|
remote: remote,
|
||||||
|
URI: loc,
|
||||||
|
Media: in,
|
||||||
|
MediaType: contentType,
|
||||||
|
ContentLength: size,
|
||||||
|
}
|
||||||
|
return rx.Upload()
|
||||||
|
}
|
||||||
|
|
||||||
|
// Make an http.Request for the range passed in
|
||||||
|
func (rx *resumableUpload) makeRequest(start int64, body []byte) *http.Request {
|
||||||
|
reqSize := int64(len(body))
|
||||||
|
req, _ := http.NewRequest("POST", rx.URI, bytes.NewBuffer(body))
|
||||||
|
req.ContentLength = reqSize
|
||||||
|
if reqSize != 0 {
|
||||||
|
req.Header.Set("Content-Range", fmt.Sprintf("bytes %v-%v/%v", start, start+reqSize-1, rx.ContentLength))
|
||||||
|
} else {
|
||||||
|
req.Header.Set("Content-Range", fmt.Sprintf("bytes */%v", rx.ContentLength))
|
||||||
|
}
|
||||||
|
req.Header.Set("Content-Type", rx.MediaType)
|
||||||
|
req.Header.Set("User-Agent", fs.UserAgent)
|
||||||
|
return req
|
||||||
|
}
|
||||||
|
|
||||||
|
// rangeRE matches the transfer status response from the server. $1 is
|
||||||
|
// the last byte index uploaded.
|
||||||
|
var rangeRE = regexp.MustCompile(`^0\-(\d+)$`)
|
||||||
|
|
||||||
|
// Query drive for the amount transferred so far
|
||||||
|
//
|
||||||
|
// If error is nil, then start should be valid
|
||||||
|
func (rx *resumableUpload) transferStatus() (start int64, err error) {
|
||||||
|
req := rx.makeRequest(0, nil)
|
||||||
|
res, err := rx.f.client.Do(req)
|
||||||
|
if err != nil {
|
||||||
|
return 0, err
|
||||||
|
}
|
||||||
|
defer googleapi.CloseBody(res)
|
||||||
|
if res.StatusCode == http.StatusCreated || res.StatusCode == http.StatusOK {
|
||||||
|
return rx.ContentLength, nil
|
||||||
|
}
|
||||||
|
if res.StatusCode != statusResumeIncomplete {
|
||||||
|
err = googleapi.CheckResponse(res)
|
||||||
|
if err != nil {
|
||||||
|
return 0, err
|
||||||
|
}
|
||||||
|
return 0, fmt.Errorf("unexpected http return code %v", res.StatusCode)
|
||||||
|
}
|
||||||
|
Range := res.Header.Get("Range")
|
||||||
|
if m := rangeRE.FindStringSubmatch(Range); len(m) == 2 {
|
||||||
|
start, err = strconv.ParseInt(m[1], 10, 64)
|
||||||
|
if err == nil {
|
||||||
|
return start, nil
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return 0, fmt.Errorf("unable to parse range %q", Range)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Transfer a chunk - caller must call googleapi.CloseBody(res) if err == nil || res != nil
|
||||||
|
func (rx *resumableUpload) transferChunk(start int64, body []byte) (int, error) {
|
||||||
|
req := rx.makeRequest(start, body)
|
||||||
|
res, err := rx.f.client.Do(req)
|
||||||
|
if err != nil {
|
||||||
|
return 599, err
|
||||||
|
}
|
||||||
|
defer googleapi.CloseBody(res)
|
||||||
|
if res.StatusCode == statusResumeIncomplete {
|
||||||
|
return res.StatusCode, nil
|
||||||
|
}
|
||||||
|
err = googleapi.CheckResponse(res)
|
||||||
|
if err != nil {
|
||||||
|
return res.StatusCode, err
|
||||||
|
}
|
||||||
|
|
||||||
|
// When the entire file upload is complete, the server
|
||||||
|
// responds with an HTTP 201 Created along with any metadata
|
||||||
|
// associated with this resource. If this request had been
|
||||||
|
// updating an existing entity rather than creating a new one,
|
||||||
|
// the HTTP response code for a completed upload would have
|
||||||
|
// been 200 OK.
|
||||||
|
//
|
||||||
|
// So parse the response out of the body. We aren't expecting
|
||||||
|
// any other 2xx codes, so we parse it unconditionaly on
|
||||||
|
// StatusCode
|
||||||
|
if err = json.NewDecoder(res.Body).Decode(&rx.ret); err != nil {
|
||||||
|
return 598, err
|
||||||
|
}
|
||||||
|
|
||||||
|
return res.StatusCode, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// Upload uploads the chunks from the input
|
||||||
|
// It retries each chunk maxTries times (with a pause of uploadPause between attempts).
|
||||||
|
func (rx *resumableUpload) Upload() (*drive.File, error) {
|
||||||
|
start := int64(0)
|
||||||
|
buf := make([]byte, chunkSize)
|
||||||
|
var StatusCode int
|
||||||
|
for start < rx.ContentLength {
|
||||||
|
reqSize := rx.ContentLength - start
|
||||||
|
if reqSize >= int64(chunkSize) {
|
||||||
|
reqSize = int64(chunkSize)
|
||||||
|
} else {
|
||||||
|
buf = buf[:reqSize]
|
||||||
|
}
|
||||||
|
|
||||||
|
// Read the chunk
|
||||||
|
_, err := io.ReadFull(rx.Media, buf)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
// Transfer the chunk
|
||||||
|
for try := 1; try <= maxTries; try++ {
|
||||||
|
fs.Debug(rx.remote, "Sending chunk %d length %d, %d/%d", start, reqSize, try, maxTries)
|
||||||
|
rx.f.beginCall()
|
||||||
|
StatusCode, err = rx.transferChunk(start, buf)
|
||||||
|
rx.f.endCall(err)
|
||||||
|
if StatusCode == statusResumeIncomplete || StatusCode == http.StatusCreated || StatusCode == http.StatusOK {
|
||||||
|
goto success
|
||||||
|
}
|
||||||
|
fs.Debug(rx.remote, "Retrying chunk %d/%d, code=%d, err=%v", try, maxTries, StatusCode, err)
|
||||||
|
}
|
||||||
|
fs.Debug(rx.remote, "Failed to send chunk")
|
||||||
|
return nil, fs.RetryErrorf("Chunk upload failed - retry: code=%d, err=%v", StatusCode, err)
|
||||||
|
success:
|
||||||
|
|
||||||
|
start += reqSize
|
||||||
|
}
|
||||||
|
// Resume or retry uploads that fail due to connection interruptions or
|
||||||
|
// any 5xx errors, including:
|
||||||
|
//
|
||||||
|
// 500 Internal Server Error
|
||||||
|
// 502 Bad Gateway
|
||||||
|
// 503 Service Unavailable
|
||||||
|
// 504 Gateway Timeout
|
||||||
|
//
|
||||||
|
// Use an exponential backoff strategy if any 5xx server error is
|
||||||
|
// returned when resuming or retrying upload requests. These errors can
|
||||||
|
// occur if a server is getting overloaded. Exponential backoff can help
|
||||||
|
// alleviate these kinds of problems during periods of high volume of
|
||||||
|
// requests or heavy network traffic. Other kinds of requests should not
|
||||||
|
// be handled by exponential backoff but you can still retry a number of
|
||||||
|
// them. When retrying these requests, limit the number of times you
|
||||||
|
// retry them. For example your code could limit to ten retries or less
|
||||||
|
// before reporting an error.
|
||||||
|
//
|
||||||
|
// Handle 404 Not Found errors when doing resumable uploads by starting
|
||||||
|
// the entire upload over from the beginning.
|
||||||
|
if rx.ret == nil {
|
||||||
|
return nil, fs.RetryErrorf("Incomplete upload - retry, last error %d", StatusCode)
|
||||||
|
}
|
||||||
|
return rx.ret, nil
|
||||||
|
}
|
||||||
@@ -17,6 +17,20 @@ This is a JSON decode error - from Update / UploadByChunk
|
|||||||
- Caused by 500 error from dropbox
|
- Caused by 500 error from dropbox
|
||||||
- See https://github.com/stacktic/dropbox/issues/1
|
- See https://github.com/stacktic/dropbox/issues/1
|
||||||
- Possibly confusing dropbox with excess concurrency?
|
- Possibly confusing dropbox with excess concurrency?
|
||||||
|
|
||||||
|
FIXME implement timeouts - need to get "github.com/stacktic/dropbox"
|
||||||
|
and hence "golang.org/x/oauth2" which uses DefaultTransport unless it
|
||||||
|
is set in the context passed into .Client()
|
||||||
|
|
||||||
|
func (db *Dropbox) client() *http.Client {
|
||||||
|
return db.config.Client(oauth2.NoContext, db.token)
|
||||||
|
}
|
||||||
|
|
||||||
|
// HTTPClient is the context key to use with golang.org/x/net/context's
|
||||||
|
// WithValue function to associate an *http.Client value with a context.
|
||||||
|
var HTTPClient ContextKey
|
||||||
|
|
||||||
|
So pass in a context with HTTPClient set...
|
||||||
*/
|
*/
|
||||||
|
|
||||||
import (
|
import (
|
||||||
@@ -45,8 +59,8 @@ const (
|
|||||||
md5sumField = "md5sum"
|
md5sumField = "md5sum"
|
||||||
mtimeField = "mtime"
|
mtimeField = "mtime"
|
||||||
maxCommitRetries = 5
|
maxCommitRetries = 5
|
||||||
RFC3339In = time.RFC3339
|
timeFormatIn = time.RFC3339
|
||||||
RFC3339Out = "2006-01-02T15:04:05.000000000Z07:00"
|
timeFormatOut = "2006-01-02T15:04:05.000000000Z07:00"
|
||||||
)
|
)
|
||||||
|
|
||||||
// Register with Fs
|
// Register with Fs
|
||||||
@@ -54,7 +68,7 @@ func init() {
|
|||||||
fs.Register(&fs.FsInfo{
|
fs.Register(&fs.FsInfo{
|
||||||
Name: "dropbox",
|
Name: "dropbox",
|
||||||
NewFs: NewFs,
|
NewFs: NewFs,
|
||||||
Config: Config,
|
Config: configHelper,
|
||||||
Options: []fs.Option{{
|
Options: []fs.Option{{
|
||||||
Name: "app_key",
|
Name: "app_key",
|
||||||
Help: "Dropbox App Key - leave blank to use rclone's.",
|
Help: "Dropbox App Key - leave blank to use rclone's.",
|
||||||
@@ -66,7 +80,7 @@ func init() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Configuration helper - called after the user has put in the defaults
|
// Configuration helper - called after the user has put in the defaults
|
||||||
func Config(name string) {
|
func configHelper(name string) {
|
||||||
// See if already have a token
|
// See if already have a token
|
||||||
token := fs.ConfigFile.MustValue(name, "token")
|
token := fs.ConfigFile.MustValue(name, "token")
|
||||||
if token != "" {
|
if token != "" {
|
||||||
@@ -214,7 +228,9 @@ func (f *FsDropbox) openDataStore() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Return an FsObject from a path
|
// Return an FsObject from a path
|
||||||
func (f *FsDropbox) newFsObjectWithInfo(remote string, info *dropbox.Entry) (fs.Object, error) {
|
//
|
||||||
|
// May return nil if an error occurred
|
||||||
|
func (f *FsDropbox) newFsObjectWithInfo(remote string, info *dropbox.Entry) fs.Object {
|
||||||
o := &FsObjectDropbox{
|
o := &FsObjectDropbox{
|
||||||
dropbox: f,
|
dropbox: f,
|
||||||
remote: remote,
|
remote: remote,
|
||||||
@@ -225,26 +241,17 @@ func (f *FsDropbox) newFsObjectWithInfo(remote string, info *dropbox.Entry) (fs.
|
|||||||
err := o.readEntryAndSetMetadata()
|
err := o.readEntryAndSetMetadata()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
// logged already fs.Debug("Failed to read info: %s", err)
|
// logged already fs.Debug("Failed to read info: %s", err)
|
||||||
return nil, err
|
return nil
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return o, nil
|
return o
|
||||||
}
|
|
||||||
|
|
||||||
// Return an FsObject from a path
|
|
||||||
//
|
|
||||||
// May return nil if an error occurred
|
|
||||||
func (f *FsDropbox) NewFsObjectWithInfo(remote string, info *dropbox.Entry) fs.Object {
|
|
||||||
fs, _ := f.newFsObjectWithInfo(remote, info)
|
|
||||||
// Errors have already been logged
|
|
||||||
return fs
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Return an FsObject from a path
|
// Return an FsObject from a path
|
||||||
//
|
//
|
||||||
// May return nil if an error occurred
|
// May return nil if an error occurred
|
||||||
func (f *FsDropbox) NewFsObject(remote string) fs.Object {
|
func (f *FsDropbox) NewFsObject(remote string) fs.Object {
|
||||||
return f.NewFsObjectWithInfo(remote, nil)
|
return f.newFsObjectWithInfo(remote, nil)
|
||||||
}
|
}
|
||||||
|
|
||||||
// Strips the root off entry and returns it
|
// Strips the root off entry and returns it
|
||||||
@@ -290,14 +297,14 @@ func (f *FsDropbox) list(out fs.ObjectsChan) {
|
|||||||
// ignore directories
|
// ignore directories
|
||||||
} else {
|
} else {
|
||||||
path := f.stripRoot(entry)
|
path := f.stripRoot(entry)
|
||||||
out <- f.NewFsObjectWithInfo(path, entry)
|
out <- f.newFsObjectWithInfo(path, entry)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if !deltaPage.HasMore {
|
if !deltaPage.HasMore {
|
||||||
break
|
break
|
||||||
}
|
}
|
||||||
cursor = deltaPage.Cursor
|
cursor = deltaPage.Cursor.Cursor
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -573,7 +580,9 @@ func (o *FsObjectDropbox) remotePath() string {
|
|||||||
func metadataKey(path string) string {
|
func metadataKey(path string) string {
|
||||||
// NB File system is case insensitive
|
// NB File system is case insensitive
|
||||||
path = strings.ToLower(path)
|
path = strings.ToLower(path)
|
||||||
return fmt.Sprintf("%x", md5.Sum([]byte(path)))
|
hash := md5.New()
|
||||||
|
_, _ = hash.Write([]byte(path))
|
||||||
|
return fmt.Sprintf("%x", hash.Sum(nil))
|
||||||
}
|
}
|
||||||
|
|
||||||
// Returns the key for the metadata database
|
// Returns the key for the metadata database
|
||||||
@@ -623,7 +632,7 @@ func (o *FsObjectDropbox) readMetaData() (err error) {
|
|||||||
if !ok {
|
if !ok {
|
||||||
fs.Debug(o, "mtime not a string")
|
fs.Debug(o, "mtime not a string")
|
||||||
} else {
|
} else {
|
||||||
modTime, err := time.Parse(RFC3339In, mtime)
|
modTime, err := time.Parse(timeFormatIn, mtime)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
@@ -633,8 +642,7 @@ func (o *FsObjectDropbox) readMetaData() (err error) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Last resort
|
// Last resort
|
||||||
o.readEntryAndSetMetadata()
|
return o.readEntryAndSetMetadata()
|
||||||
return nil
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// ModTime returns the modification time of the object
|
// ModTime returns the modification time of the object
|
||||||
@@ -666,14 +674,16 @@ func (o *FsObjectDropbox) setModTimeAndMd5sum(modTime time.Time, md5sum string)
|
|||||||
if err != nil {
|
if err != nil {
|
||||||
return fmt.Errorf("Couldn't set md5sum record: %s", err)
|
return fmt.Errorf("Couldn't set md5sum record: %s", err)
|
||||||
}
|
}
|
||||||
|
o.md5sum = md5sum
|
||||||
}
|
}
|
||||||
|
|
||||||
if !modTime.IsZero() {
|
if !modTime.IsZero() {
|
||||||
mtime := modTime.Format(RFC3339Out)
|
mtime := modTime.Format(timeFormatOut)
|
||||||
err := record.Set(mtimeField, mtime)
|
err := record.Set(mtimeField, mtime)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return fmt.Errorf("Couldn't set mtime record: %s", err)
|
return fmt.Errorf("Couldn't set mtime record: %s", err)
|
||||||
}
|
}
|
||||||
|
o.modTime = modTime
|
||||||
}
|
}
|
||||||
|
|
||||||
return nil
|
return nil
|
||||||
|
|||||||
53
dropbox/dropbox_test.go
Normal file
53
dropbox/dropbox_test.go
Normal file
@@ -0,0 +1,53 @@
|
|||||||
|
// Test Dropbox filesystem interface
|
||||||
|
//
|
||||||
|
// Automatically generated - DO NOT EDIT
|
||||||
|
// Regenerate with: go run gen_tests.go or make gen_tests
|
||||||
|
package dropbox_test
|
||||||
|
|
||||||
|
import (
|
||||||
|
"testing"
|
||||||
|
|
||||||
|
"github.com/ncw/rclone/dropbox"
|
||||||
|
"github.com/ncw/rclone/fs"
|
||||||
|
"github.com/ncw/rclone/fstest/fstests"
|
||||||
|
)
|
||||||
|
|
||||||
|
func init() {
|
||||||
|
fstests.NilObject = fs.Object((*dropbox.FsObjectDropbox)(nil))
|
||||||
|
fstests.RemoteName = "TestDropbox:"
|
||||||
|
}
|
||||||
|
|
||||||
|
// Generic tests for the Fs
|
||||||
|
func TestInit(t *testing.T) { fstests.TestInit(t) }
|
||||||
|
func TestFsString(t *testing.T) { fstests.TestFsString(t) }
|
||||||
|
func TestFsRmdirEmpty(t *testing.T) { fstests.TestFsRmdirEmpty(t) }
|
||||||
|
func TestFsRmdirNotFound(t *testing.T) { fstests.TestFsRmdirNotFound(t) }
|
||||||
|
func TestFsMkdir(t *testing.T) { fstests.TestFsMkdir(t) }
|
||||||
|
func TestFsListEmpty(t *testing.T) { fstests.TestFsListEmpty(t) }
|
||||||
|
func TestFsListDirEmpty(t *testing.T) { fstests.TestFsListDirEmpty(t) }
|
||||||
|
func TestFsNewFsObjectNotFound(t *testing.T) { fstests.TestFsNewFsObjectNotFound(t) }
|
||||||
|
func TestFsPutFile1(t *testing.T) { fstests.TestFsPutFile1(t) }
|
||||||
|
func TestFsPutFile2(t *testing.T) { fstests.TestFsPutFile2(t) }
|
||||||
|
func TestFsListDirFile2(t *testing.T) { fstests.TestFsListDirFile2(t) }
|
||||||
|
func TestFsListDirRoot(t *testing.T) { fstests.TestFsListDirRoot(t) }
|
||||||
|
func TestFsListRoot(t *testing.T) { fstests.TestFsListRoot(t) }
|
||||||
|
func TestFsListFile1(t *testing.T) { fstests.TestFsListFile1(t) }
|
||||||
|
func TestFsNewFsObject(t *testing.T) { fstests.TestFsNewFsObject(t) }
|
||||||
|
func TestFsListFile1and2(t *testing.T) { fstests.TestFsListFile1and2(t) }
|
||||||
|
func TestFsRmdirFull(t *testing.T) { fstests.TestFsRmdirFull(t) }
|
||||||
|
func TestFsPrecision(t *testing.T) { fstests.TestFsPrecision(t) }
|
||||||
|
func TestObjectString(t *testing.T) { fstests.TestObjectString(t) }
|
||||||
|
func TestObjectFs(t *testing.T) { fstests.TestObjectFs(t) }
|
||||||
|
func TestObjectRemote(t *testing.T) { fstests.TestObjectRemote(t) }
|
||||||
|
func TestObjectMd5sum(t *testing.T) { fstests.TestObjectMd5sum(t) }
|
||||||
|
func TestObjectModTime(t *testing.T) { fstests.TestObjectModTime(t) }
|
||||||
|
func TestObjectSetModTime(t *testing.T) { fstests.TestObjectSetModTime(t) }
|
||||||
|
func TestObjectSize(t *testing.T) { fstests.TestObjectSize(t) }
|
||||||
|
func TestObjectOpen(t *testing.T) { fstests.TestObjectOpen(t) }
|
||||||
|
func TestObjectUpdate(t *testing.T) { fstests.TestObjectUpdate(t) }
|
||||||
|
func TestObjectStorable(t *testing.T) { fstests.TestObjectStorable(t) }
|
||||||
|
func TestLimitedFs(t *testing.T) { fstests.TestLimitedFs(t) }
|
||||||
|
func TestLimitedFsNotFound(t *testing.T) { fstests.TestLimitedFsNotFound(t) }
|
||||||
|
func TestObjectRemove(t *testing.T) { fstests.TestObjectRemove(t) }
|
||||||
|
func TestObjectPurge(t *testing.T) { fstests.TestObjectPurge(t) }
|
||||||
|
func TestFinalise(t *testing.T) { fstests.TestFinalise(t) }
|
||||||
@@ -10,13 +10,24 @@ import (
|
|||||||
"strings"
|
"strings"
|
||||||
"sync"
|
"sync"
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
|
"github.com/tsenart/tb"
|
||||||
)
|
)
|
||||||
|
|
||||||
// Globals
|
// Globals
|
||||||
var (
|
var (
|
||||||
Stats = NewStats()
|
Stats = NewStats()
|
||||||
|
tokenBucket *tb.Bucket
|
||||||
)
|
)
|
||||||
|
|
||||||
|
// Start the token bucket if necessary
|
||||||
|
func startTokenBucket() {
|
||||||
|
if bwLimit > 0 {
|
||||||
|
tokenBucket = tb.NewBucket(int64(bwLimit), 100*time.Millisecond)
|
||||||
|
Log(nil, "Starting bandwidth limiter at %vBytes/s", &bwLimit)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
// Stringset holds some strings
|
// Stringset holds some strings
|
||||||
type StringSet map[string]bool
|
type StringSet map[string]bool
|
||||||
|
|
||||||
@@ -113,6 +124,16 @@ func (s *StatsInfo) GetErrors() int64 {
|
|||||||
return s.errors
|
return s.errors
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// ResetCounters sets the counters (bytes, checks, errors, transfers) to 0
|
||||||
|
func (s *StatsInfo) ResetCounters() {
|
||||||
|
s.lock.RLock()
|
||||||
|
defer s.lock.RUnlock()
|
||||||
|
s.bytes = 0
|
||||||
|
s.errors = 0
|
||||||
|
s.checks = 0
|
||||||
|
s.transfers = 0
|
||||||
|
}
|
||||||
|
|
||||||
// Errored returns whether there have been any errors
|
// Errored returns whether there have been any errors
|
||||||
func (s *StatsInfo) Errored() bool {
|
func (s *StatsInfo) Errored() bool {
|
||||||
s.lock.RLock()
|
s.lock.RLock()
|
||||||
@@ -142,6 +163,13 @@ func (s *StatsInfo) DoneChecking(o Object) {
|
|||||||
s.checks += 1
|
s.checks += 1
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// GetTransfers reads the number of transfers
|
||||||
|
func (s *StatsInfo) GetTransfers() int64 {
|
||||||
|
s.lock.RLock()
|
||||||
|
defer s.lock.RUnlock()
|
||||||
|
return s.transfers
|
||||||
|
}
|
||||||
|
|
||||||
// Transferring adds a transfer into the stats
|
// Transferring adds a transfer into the stats
|
||||||
func (s *StatsInfo) Transferring(o Object) {
|
func (s *StatsInfo) Transferring(o Object) {
|
||||||
s.lock.Lock()
|
s.lock.Lock()
|
||||||
@@ -178,6 +206,10 @@ func (file *Account) Read(p []byte) (n int, err error) {
|
|||||||
if err == io.EOF {
|
if err == io.EOF {
|
||||||
// FIXME Do something?
|
// FIXME Do something?
|
||||||
}
|
}
|
||||||
|
// Limit the transfer speed if required
|
||||||
|
if tokenBucket != nil {
|
||||||
|
tokenBucket.Wait(int64(n))
|
||||||
|
}
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
180
fs/config.go
180
fs/config.go
@@ -6,6 +6,8 @@ import (
|
|||||||
"bufio"
|
"bufio"
|
||||||
"fmt"
|
"fmt"
|
||||||
"log"
|
"log"
|
||||||
|
"math"
|
||||||
|
"net/http"
|
||||||
"os"
|
"os"
|
||||||
"os/user"
|
"os/user"
|
||||||
"path"
|
"path"
|
||||||
@@ -15,6 +17,7 @@ import (
|
|||||||
"time"
|
"time"
|
||||||
|
|
||||||
"github.com/Unknwon/goconfig"
|
"github.com/Unknwon/goconfig"
|
||||||
|
"github.com/mreiferson/go-httpclient"
|
||||||
"github.com/ogier/pflag"
|
"github.com/ogier/pflag"
|
||||||
)
|
)
|
||||||
|
|
||||||
@@ -22,6 +25,8 @@ const (
|
|||||||
configFileName = ".rclone.conf"
|
configFileName = ".rclone.conf"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
type SizeSuffix int64
|
||||||
|
|
||||||
// Global
|
// Global
|
||||||
var (
|
var (
|
||||||
// Config file
|
// Config file
|
||||||
@@ -33,34 +38,151 @@ var (
|
|||||||
// Global config
|
// Global config
|
||||||
Config = &ConfigInfo{}
|
Config = &ConfigInfo{}
|
||||||
// Flags
|
// Flags
|
||||||
verbose = pflag.BoolP("verbose", "v", false, "Print lots more stuff")
|
verbose = pflag.BoolP("verbose", "v", false, "Print lots more stuff")
|
||||||
quiet = pflag.BoolP("quiet", "q", false, "Print as little stuff as possible")
|
quiet = pflag.BoolP("quiet", "q", false, "Print as little stuff as possible")
|
||||||
modifyWindow = pflag.DurationP("modify-window", "", time.Nanosecond, "Max time diff to be considered the same")
|
modifyWindow = pflag.DurationP("modify-window", "", time.Nanosecond, "Max time diff to be considered the same")
|
||||||
checkers = pflag.IntP("checkers", "", 8, "Number of checkers to run in parallel.")
|
checkers = pflag.IntP("checkers", "", 8, "Number of checkers to run in parallel.")
|
||||||
transfers = pflag.IntP("transfers", "", 4, "Number of file transfers to run in parallel.")
|
transfers = pflag.IntP("transfers", "", 4, "Number of file transfers to run in parallel.")
|
||||||
configFile = pflag.StringP("config", "", ConfigPath, "Config file.")
|
configFile = pflag.StringP("config", "", ConfigPath, "Config file.")
|
||||||
dryRun = pflag.BoolP("dry-run", "n", false, "Do a trial run with no permanent changes")
|
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")
|
||||||
|
dryRun = pflag.BoolP("dry-run", "n", false, "Do a trial run with no permanent changes")
|
||||||
|
connectTimeout = pflag.DurationP("contimeout", "", 60*time.Second, "Connect timeout")
|
||||||
|
timeout = pflag.DurationP("timeout", "", 5*60*time.Second, "IO idle timeout")
|
||||||
|
bwLimit SizeSuffix
|
||||||
)
|
)
|
||||||
|
|
||||||
|
func init() {
|
||||||
|
pflag.VarP(&bwLimit, "bwlimit", "", "Bandwidth limit in kBytes/s, or use suffix k|M|G")
|
||||||
|
}
|
||||||
|
|
||||||
|
// Turn SizeSuffix into a string
|
||||||
|
func (x SizeSuffix) String() string {
|
||||||
|
scaled := float64(0)
|
||||||
|
suffix := ""
|
||||||
|
switch {
|
||||||
|
case x == 0:
|
||||||
|
return "0"
|
||||||
|
case x < 1024*1024:
|
||||||
|
scaled = float64(x) / 1024
|
||||||
|
suffix = "k"
|
||||||
|
case x < 1024*1024*1024:
|
||||||
|
scaled = float64(x) / 1024 / 1024
|
||||||
|
suffix = "M"
|
||||||
|
default:
|
||||||
|
scaled = float64(x) / 1024 / 1024 / 1024
|
||||||
|
suffix = "G"
|
||||||
|
}
|
||||||
|
if math.Floor(scaled) == scaled {
|
||||||
|
return fmt.Sprintf("%.0f%s", scaled, suffix)
|
||||||
|
}
|
||||||
|
return fmt.Sprintf("%.3f%s", scaled, suffix)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Set a SizeSuffix
|
||||||
|
func (x *SizeSuffix) Set(s string) error {
|
||||||
|
if len(s) == 0 {
|
||||||
|
return fmt.Errorf("Empty string")
|
||||||
|
}
|
||||||
|
suffix := s[len(s)-1]
|
||||||
|
suffixLen := 1
|
||||||
|
var multiplier float64
|
||||||
|
switch suffix {
|
||||||
|
case '0', '1', '2', '3', '4', '5', '6', '7', '8', '9', '.':
|
||||||
|
suffixLen = 0
|
||||||
|
multiplier = 1 << 10
|
||||||
|
case 'k', 'K':
|
||||||
|
multiplier = 1 << 10
|
||||||
|
case 'm', 'M':
|
||||||
|
multiplier = 1 << 20
|
||||||
|
case 'g', 'G':
|
||||||
|
multiplier = 1 << 30
|
||||||
|
default:
|
||||||
|
return fmt.Errorf("Bad suffix %q", suffix)
|
||||||
|
}
|
||||||
|
s = s[:len(s)-suffixLen]
|
||||||
|
value, err := strconv.ParseFloat(s, 64)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
if value < 0 {
|
||||||
|
return fmt.Errorf("Size can't be negative %q", s)
|
||||||
|
}
|
||||||
|
value *= multiplier
|
||||||
|
*x = SizeSuffix(value)
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// Check it satisfies the interface
|
||||||
|
var _ pflag.Value = (*SizeSuffix)(nil)
|
||||||
|
|
||||||
// Filesystem config options
|
// Filesystem config options
|
||||||
type ConfigInfo struct {
|
type ConfigInfo struct {
|
||||||
Verbose bool
|
Verbose bool
|
||||||
Quiet bool
|
Quiet bool
|
||||||
DryRun bool
|
DryRun bool
|
||||||
ModifyWindow time.Duration
|
CheckSum bool
|
||||||
Checkers int
|
SizeOnly bool
|
||||||
Transfers int
|
ModifyWindow time.Duration
|
||||||
|
Checkers int
|
||||||
|
Transfers int
|
||||||
|
ConnectTimeout time.Duration // Connect timeout
|
||||||
|
Timeout time.Duration // Data channel timeout
|
||||||
|
}
|
||||||
|
|
||||||
|
// Transport returns an http.RoundTripper with the correct timeouts
|
||||||
|
func (ci *ConfigInfo) Transport() http.RoundTripper {
|
||||||
|
return &httpclient.Transport{
|
||||||
|
Proxy: http.ProxyFromEnvironment,
|
||||||
|
MaxIdleConnsPerHost: ci.Checkers + ci.Transfers + 1,
|
||||||
|
|
||||||
|
// ConnectTimeout, if non-zero, is the maximum amount of time a dial will wait for
|
||||||
|
// a connect to complete.
|
||||||
|
ConnectTimeout: ci.ConnectTimeout,
|
||||||
|
|
||||||
|
// ResponseHeaderTimeout, if non-zero, specifies the amount of
|
||||||
|
// time to wait for a server's response headers after fully
|
||||||
|
// writing the request (including its body, if any). This
|
||||||
|
// time does not include the time to read the response body.
|
||||||
|
ResponseHeaderTimeout: ci.Timeout,
|
||||||
|
|
||||||
|
// RequestTimeout, if non-zero, specifies the amount of time for the entire
|
||||||
|
// request to complete (including all of the above timeouts + entire response body).
|
||||||
|
// This should never be less than the sum total of the above two timeouts.
|
||||||
|
//RequestTimeout: NOT SET,
|
||||||
|
|
||||||
|
// ReadWriteTimeout, if non-zero, will set a deadline for every Read and
|
||||||
|
// Write operation on the request connection.
|
||||||
|
ReadWriteTimeout: ci.Timeout,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Transport returns an http.Client with the correct timeouts
|
||||||
|
func (ci *ConfigInfo) Client() *http.Client {
|
||||||
|
return &http.Client{
|
||||||
|
Transport: ci.Transport(),
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Find the config directory
|
// Find the config directory
|
||||||
func configHome() string {
|
func configHome() string {
|
||||||
// Find users home directory
|
// Find users home directory
|
||||||
usr, err := user.Current()
|
usr, err := user.Current()
|
||||||
if err != nil {
|
if err == nil {
|
||||||
log.Printf("Couldn't find home directory: %v", err)
|
return usr.HomeDir
|
||||||
return ""
|
|
||||||
}
|
}
|
||||||
return usr.HomeDir
|
// Fall back to reading $HOME - work around user.Current() not
|
||||||
|
// working for cross compiled binaries on OSX.
|
||||||
|
// https://github.com/golang/go/issues/6376
|
||||||
|
home := os.Getenv("HOME")
|
||||||
|
if home != "" {
|
||||||
|
return home
|
||||||
|
}
|
||||||
|
log.Printf("Couldn't find home directory or read HOME environment variable.")
|
||||||
|
log.Printf("Defaulting to storing config in current directory.")
|
||||||
|
log.Printf("Use -config flag to workaround.")
|
||||||
|
log.Printf("Error was: %v", err)
|
||||||
|
return ""
|
||||||
}
|
}
|
||||||
|
|
||||||
// Loads the config file
|
// Loads the config file
|
||||||
@@ -74,6 +196,10 @@ func LoadConfig() {
|
|||||||
Config.Checkers = *checkers
|
Config.Checkers = *checkers
|
||||||
Config.Transfers = *transfers
|
Config.Transfers = *transfers
|
||||||
Config.DryRun = *dryRun
|
Config.DryRun = *dryRun
|
||||||
|
Config.Timeout = *timeout
|
||||||
|
Config.ConnectTimeout = *connectTimeout
|
||||||
|
Config.CheckSum = *checkSum
|
||||||
|
Config.SizeOnly = *sizeOnly
|
||||||
|
|
||||||
ConfigPath = *configFile
|
ConfigPath = *configFile
|
||||||
|
|
||||||
@@ -87,6 +213,9 @@ func LoadConfig() {
|
|||||||
log.Fatalf("Failed to read null config file: %v", err)
|
log.Fatalf("Failed to read null config file: %v", err)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Start the token bucket limiter
|
||||||
|
startTokenBucket()
|
||||||
}
|
}
|
||||||
|
|
||||||
// Save configuration file.
|
// Save configuration file.
|
||||||
@@ -320,9 +449,20 @@ func EditConfig() {
|
|||||||
name := ChooseRemote()
|
name := ChooseRemote()
|
||||||
EditRemote(name)
|
EditRemote(name)
|
||||||
case 'n':
|
case 'n':
|
||||||
fmt.Printf("name> ")
|
nameLoop:
|
||||||
name := ReadLine()
|
for {
|
||||||
NewRemote(name)
|
fmt.Printf("name> ")
|
||||||
|
name := ReadLine()
|
||||||
|
switch {
|
||||||
|
case name == "":
|
||||||
|
fmt.Printf("Can't use empty name\n")
|
||||||
|
case isDriveLetter(name):
|
||||||
|
fmt.Printf("Can't use %q as it can be confused a drive letter\n", name)
|
||||||
|
default:
|
||||||
|
NewRemote(name)
|
||||||
|
break nameLoop
|
||||||
|
}
|
||||||
|
}
|
||||||
case 'd':
|
case 'd':
|
||||||
name := ChooseRemote()
|
name := ChooseRemote()
|
||||||
DeleteRemote(name)
|
DeleteRemote(name)
|
||||||
|
|||||||
57
fs/config_test.go
Normal file
57
fs/config_test.go
Normal file
@@ -0,0 +1,57 @@
|
|||||||
|
package fs
|
||||||
|
|
||||||
|
import "testing"
|
||||||
|
|
||||||
|
func TestSizeSuffixString(t *testing.T) {
|
||||||
|
for _, test := range []struct {
|
||||||
|
in float64
|
||||||
|
want string
|
||||||
|
}{
|
||||||
|
{0, "0"},
|
||||||
|
{102, "0.100k"},
|
||||||
|
{1024, "1k"},
|
||||||
|
{1024 * 1024, "1M"},
|
||||||
|
{1024 * 1024 * 1024, "1G"},
|
||||||
|
{10 * 1024 * 1024 * 1024, "10G"},
|
||||||
|
{10.1 * 1024 * 1024 * 1024, "10.100G"},
|
||||||
|
} {
|
||||||
|
ss := SizeSuffix(test.in)
|
||||||
|
got := ss.String()
|
||||||
|
if test.want != got {
|
||||||
|
t.Errorf("Want %v got %v", test.want, got)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestSizeSuffixSet(t *testing.T) {
|
||||||
|
for i, test := range []struct {
|
||||||
|
in string
|
||||||
|
want int64
|
||||||
|
err bool
|
||||||
|
}{
|
||||||
|
{"0", 0, false},
|
||||||
|
{"0.1k", 102, false},
|
||||||
|
{"0.1", 102, false},
|
||||||
|
{"1K", 1024, false},
|
||||||
|
{"1", 1024, false},
|
||||||
|
{"2.5", 1024 * 2.5, false},
|
||||||
|
{"1M", 1024 * 1024, false},
|
||||||
|
{"1.g", 1024 * 1024 * 1024, false},
|
||||||
|
{"10G", 10 * 1024 * 1024 * 1024, false},
|
||||||
|
{"", 0, true},
|
||||||
|
{"1p", 0, true},
|
||||||
|
{"1.p", 0, true},
|
||||||
|
{"1p", 0, true},
|
||||||
|
{"-1K", 0, true},
|
||||||
|
} {
|
||||||
|
ss := SizeSuffix(0)
|
||||||
|
err := ss.Set(test.in)
|
||||||
|
if (err != nil) != test.err {
|
||||||
|
t.Errorf("%d: Expecting error %v but got error %v", i, test.err, err)
|
||||||
|
}
|
||||||
|
got := int64(ss)
|
||||||
|
if test.want != got {
|
||||||
|
t.Errorf("%d: Want %v got %v", i, test.want, got)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
12
fs/driveletter.go
Normal file
12
fs/driveletter.go
Normal file
@@ -0,0 +1,12 @@
|
|||||||
|
// +build !windows
|
||||||
|
|
||||||
|
package fs
|
||||||
|
|
||||||
|
// isDriveLetter returns a bool indicating whether name is a valid
|
||||||
|
// Windows drive letter
|
||||||
|
//
|
||||||
|
// On non windows platforms we don't have drive letters so we always
|
||||||
|
// return false
|
||||||
|
func isDriveLetter(name string) bool {
|
||||||
|
return false
|
||||||
|
}
|
||||||
13
fs/driveletter_windows.go
Normal file
13
fs/driveletter_windows.go
Normal file
@@ -0,0 +1,13 @@
|
|||||||
|
// +build windows
|
||||||
|
|
||||||
|
package fs
|
||||||
|
|
||||||
|
// isDriveLetter returns a bool indicating whether name is a valid
|
||||||
|
// Windows drive letter
|
||||||
|
func isDriveLetter(name string) bool {
|
||||||
|
if len(name) != 1 {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
c := name[0]
|
||||||
|
return (c >= 'a' && c <= 'z') || (c >= 'A' && c <= 'Z')
|
||||||
|
}
|
||||||
65
fs/fs.go
65
fs/fs.go
@@ -1,11 +1,11 @@
|
|||||||
// File system interface
|
// Generic file system interface for rclone object storage systems
|
||||||
|
|
||||||
package fs
|
package fs
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"fmt"
|
"fmt"
|
||||||
"io"
|
"io"
|
||||||
"log"
|
"log"
|
||||||
|
"path/filepath"
|
||||||
"regexp"
|
"regexp"
|
||||||
"time"
|
"time"
|
||||||
)
|
)
|
||||||
@@ -20,6 +20,8 @@ const (
|
|||||||
var (
|
var (
|
||||||
// Filesystem registry
|
// Filesystem registry
|
||||||
fsRegistry []*FsInfo
|
fsRegistry []*FsInfo
|
||||||
|
// Error returned by NewFs if not found in config file
|
||||||
|
NotFoundInConfigFile = fmt.Errorf("Didn't find section in config file")
|
||||||
)
|
)
|
||||||
|
|
||||||
// Filesystem info
|
// Filesystem info
|
||||||
@@ -79,9 +81,13 @@ type Fs interface {
|
|||||||
Put(in io.Reader, remote string, modTime time.Time, size int64) (Object, error)
|
Put(in io.Reader, remote string, modTime time.Time, size int64) (Object, error)
|
||||||
|
|
||||||
// Make the directory (container, bucket)
|
// Make the directory (container, bucket)
|
||||||
|
//
|
||||||
|
// Shouldn't return an error if it already exists
|
||||||
Mkdir() error
|
Mkdir() error
|
||||||
|
|
||||||
// Remove the directory (container, bucket) if empty
|
// Remove the directory (container, bucket) if empty
|
||||||
|
//
|
||||||
|
// Return an error if it doesn't exist or isn't empty
|
||||||
Rmdir() error
|
Rmdir() error
|
||||||
|
|
||||||
// Precision of the ModTimes in this Fs
|
// Precision of the ModTimes in this Fs
|
||||||
@@ -131,9 +137,40 @@ type Purger interface {
|
|||||||
//
|
//
|
||||||
// Implement this if you have a way of deleting all the files
|
// Implement this if you have a way of deleting all the files
|
||||||
// quicker than just running Remove() on the result of List()
|
// quicker than just running Remove() on the result of List()
|
||||||
|
//
|
||||||
|
// Return an error if it doesn't exist
|
||||||
Purge() error
|
Purge() error
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// An optional interface for error as to whether the operation should be retried
|
||||||
|
//
|
||||||
|
// This should be returned from Update or Put methods as required
|
||||||
|
type Retry interface {
|
||||||
|
error
|
||||||
|
Retry() bool
|
||||||
|
}
|
||||||
|
|
||||||
|
// A type of error
|
||||||
|
type retryError string
|
||||||
|
|
||||||
|
// Error interface
|
||||||
|
func (r retryError) Error() string {
|
||||||
|
return string(r)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Retry interface
|
||||||
|
func (r retryError) Retry() bool {
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
|
||||||
|
// Check interface
|
||||||
|
var _ Retry = retryError("")
|
||||||
|
|
||||||
|
// RetryErrorf makes an error which indicates it would like to be retried
|
||||||
|
func RetryErrorf(format string, a ...interface{}) error {
|
||||||
|
return retryError(fmt.Sprintf(format, a...))
|
||||||
|
}
|
||||||
|
|
||||||
// A channel of Objects
|
// A channel of Objects
|
||||||
type ObjectsChan chan Object
|
type ObjectsChan chan Object
|
||||||
|
|
||||||
@@ -159,9 +196,6 @@ type Dir struct {
|
|||||||
// A channel of Dir objects
|
// A channel of Dir objects
|
||||||
type DirChan chan *Dir
|
type DirChan chan *Dir
|
||||||
|
|
||||||
// Pattern to match a url
|
|
||||||
var matcher = regexp.MustCompile(`^([\w_-]+):(.*)$`)
|
|
||||||
|
|
||||||
// Finds a FsInfo object for the name passed in
|
// Finds a FsInfo object for the name passed in
|
||||||
//
|
//
|
||||||
// Services are looked up in the config file
|
// Services are looked up in the config file
|
||||||
@@ -174,34 +208,43 @@ func Find(name string) (*FsInfo, error) {
|
|||||||
return nil, fmt.Errorf("Didn't find filing system for %q", name)
|
return nil, fmt.Errorf("Didn't find filing system for %q", name)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Pattern to match an rclone url
|
||||||
|
var matcher = regexp.MustCompile(`^([\w_-]+):(.*)$`)
|
||||||
|
|
||||||
// NewFs makes a new Fs object from the path
|
// NewFs makes a new Fs object from the path
|
||||||
//
|
//
|
||||||
// The path is of the form service://path
|
// The path is of the form remote:path
|
||||||
//
|
//
|
||||||
// Services are looked up in the config file
|
// Remotes are looked up in the config file. If the remote isn't
|
||||||
|
// found then NotFoundInConfigFile will be returned.
|
||||||
|
//
|
||||||
|
// On Windows avoid single character remote names as they can be mixed
|
||||||
|
// up with drive letters.
|
||||||
func NewFs(path string) (Fs, error) {
|
func NewFs(path string) (Fs, error) {
|
||||||
parts := matcher.FindStringSubmatch(path)
|
parts := matcher.FindStringSubmatch(path)
|
||||||
fsName, configName, fsPath := "local", "local", path
|
fsName, configName, fsPath := "local", "local", path
|
||||||
if parts != nil {
|
if parts != nil && !isDriveLetter(parts[1]) {
|
||||||
configName, fsPath = parts[1], parts[2]
|
configName, fsPath = parts[1], parts[2]
|
||||||
var err error
|
var err error
|
||||||
fsName, err = ConfigFile.GetValue(configName, "type")
|
fsName, err = ConfigFile.GetValue(configName, "type")
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, fmt.Errorf("Didn't find section in config file for %q", configName)
|
return nil, NotFoundInConfigFile
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
fs, err := Find(fsName)
|
fs, err := Find(fsName)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
|
// change native directory separators to / if there are any
|
||||||
|
fsPath = filepath.ToSlash(fsPath)
|
||||||
return fs.NewFs(configName, fsPath)
|
return fs.NewFs(configName, fsPath)
|
||||||
}
|
}
|
||||||
|
|
||||||
// Outputs log for object
|
// Outputs log for object
|
||||||
func OutputLog(o interface{}, text string, args ...interface{}) {
|
func OutputLog(o interface{}, text string, args ...interface{}) {
|
||||||
description := ""
|
description := ""
|
||||||
if x, ok := o.(fmt.Stringer); ok {
|
if o != nil {
|
||||||
description = x.String() + ": "
|
description = fmt.Sprintf("%v: ", o)
|
||||||
}
|
}
|
||||||
out := fmt.Sprintf(text, args...)
|
out := fmt.Sprintf(text, args...)
|
||||||
log.Print(description + out)
|
log.Print(description + out)
|
||||||
|
|||||||
182
fs/operations.go
182
fs/operations.go
@@ -4,8 +4,11 @@ package fs
|
|||||||
|
|
||||||
import (
|
import (
|
||||||
"fmt"
|
"fmt"
|
||||||
"log"
|
"io"
|
||||||
|
"mime"
|
||||||
|
"path"
|
||||||
"sync"
|
"sync"
|
||||||
|
"time"
|
||||||
)
|
)
|
||||||
|
|
||||||
// Work out modify window for fses passed in - sets Config.ModifyWindow
|
// Work out modify window for fses passed in - sets Config.ModifyWindow
|
||||||
@@ -51,15 +54,16 @@ func CheckMd5sums(src, dst Object) (bool, error) {
|
|||||||
// size, mtime and MD5SUM
|
// size, mtime and MD5SUM
|
||||||
//
|
//
|
||||||
// If the src and dst size are different then it is considered to be
|
// If the src and dst size are different then it is considered to be
|
||||||
// not equal.
|
// not equal. If --size-only is in effect then this is the only check
|
||||||
|
// that is done.
|
||||||
//
|
//
|
||||||
// If the size is the same and the mtime is the same then it is
|
// If the size is the same and the mtime is the same then it is
|
||||||
// considered to be equal. This is the heuristic rsync uses when
|
// considered to be equal. This check is skipped if using --checksum.
|
||||||
// not using --checksum.
|
|
||||||
//
|
//
|
||||||
// If the size is the same and and mtime is different or unreadable
|
// If the size is the same and mtime is different, unreadable or
|
||||||
// and the MD5SUM is the same then the file is considered to be equal.
|
// --checksum is set and the MD5SUM is the same then the file is
|
||||||
// In this case the mtime on the dst is updated.
|
// considered to be equal. In this case the mtime on the dst is
|
||||||
|
// updated if --checksum is not set.
|
||||||
//
|
//
|
||||||
// Otherwise the file is considered to be not equal including if there
|
// Otherwise the file is considered to be not equal including if there
|
||||||
// were errors reading info.
|
// were errors reading info.
|
||||||
@@ -68,19 +72,26 @@ func Equal(src, dst Object) bool {
|
|||||||
Debug(src, "Sizes differ")
|
Debug(src, "Sizes differ")
|
||||||
return false
|
return false
|
||||||
}
|
}
|
||||||
|
if Config.SizeOnly {
|
||||||
// Size the same so check the mtime
|
Debug(src, "Sizes identical")
|
||||||
srcModTime := src.ModTime()
|
|
||||||
dstModTime := dst.ModTime()
|
|
||||||
dt := dstModTime.Sub(srcModTime)
|
|
||||||
ModifyWindow := Config.ModifyWindow
|
|
||||||
if dt >= ModifyWindow || dt <= -ModifyWindow {
|
|
||||||
Debug(src, "Modification times differ by %s: %v, %v", dt, srcModTime, dstModTime)
|
|
||||||
} else {
|
|
||||||
Debug(src, "Size and modification time the same (differ by %s, within tolerance %s)", dt, ModifyWindow)
|
|
||||||
return true
|
return true
|
||||||
}
|
}
|
||||||
|
|
||||||
|
var srcModTime time.Time
|
||||||
|
if !Config.CheckSum {
|
||||||
|
// Size the same so check the mtime
|
||||||
|
srcModTime = src.ModTime()
|
||||||
|
dstModTime := dst.ModTime()
|
||||||
|
dt := dstModTime.Sub(srcModTime)
|
||||||
|
ModifyWindow := Config.ModifyWindow
|
||||||
|
if dt >= ModifyWindow || dt <= -ModifyWindow {
|
||||||
|
Debug(src, "Modification times differ by %s: %v, %v", dt, srcModTime, dstModTime)
|
||||||
|
} else {
|
||||||
|
Debug(src, "Size and modification time the same (differ by %s, within tolerance %s)", dt, ModifyWindow)
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
// mtime is unreadable or different but size is the same so
|
// mtime is unreadable or different but size is the same so
|
||||||
// check the MD5SUM
|
// check the MD5SUM
|
||||||
same, _ := CheckMd5sums(src, dst)
|
same, _ := CheckMd5sums(src, dst)
|
||||||
@@ -89,24 +100,39 @@ func Equal(src, dst Object) bool {
|
|||||||
return false
|
return false
|
||||||
}
|
}
|
||||||
|
|
||||||
// Size and MD5 the same but mtime different so update the
|
if !Config.CheckSum {
|
||||||
// mtime of the dst object here
|
// Size and MD5 the same but mtime different so update the
|
||||||
dst.SetModTime(srcModTime)
|
// mtime of the dst object here
|
||||||
|
dst.SetModTime(srcModTime)
|
||||||
|
}
|
||||||
|
|
||||||
Debug(src, "Size and MD5SUM of src and dst objects identical")
|
Debug(src, "Size and MD5SUM of src and dst objects identical")
|
||||||
return true
|
return true
|
||||||
}
|
}
|
||||||
|
|
||||||
// Used to remove a failed copy
|
// Returns a guess at the mime type from the extension
|
||||||
func removeFailedCopy(dst Object) {
|
func MimeType(o Object) string {
|
||||||
if dst != nil {
|
mimeType := mime.TypeByExtension(path.Ext(o.Remote()))
|
||||||
Debug(dst, "Removing failed copy")
|
if mimeType == "" {
|
||||||
removeErr := dst.Remove()
|
mimeType = "application/octet-stream"
|
||||||
if removeErr != nil {
|
|
||||||
Stats.Error()
|
|
||||||
Log(dst, "Failed to remove failed copy: %s", removeErr)
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
return mimeType
|
||||||
|
}
|
||||||
|
|
||||||
|
// Used to remove a failed copy
|
||||||
|
//
|
||||||
|
// Returns whether the file was succesfully removed or not
|
||||||
|
func removeFailedCopy(dst Object) bool {
|
||||||
|
if dst == nil {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
Debug(dst, "Removing failed copy")
|
||||||
|
removeErr := dst.Remove()
|
||||||
|
if removeErr != nil {
|
||||||
|
Debug(dst, "Failed to remove failed copy: %s", removeErr)
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
return true
|
||||||
}
|
}
|
||||||
|
|
||||||
// Copy src object to dst or f if nil
|
// Copy src object to dst or f if nil
|
||||||
@@ -115,6 +141,10 @@ func removeFailedCopy(dst Object) {
|
|||||||
// call Copy() with dst nil on a pre-existing file then some filing
|
// call Copy() with dst nil on a pre-existing file then some filing
|
||||||
// systems (eg Drive) may duplicate the file.
|
// systems (eg Drive) may duplicate the file.
|
||||||
func Copy(f Fs, dst, src Object) {
|
func Copy(f Fs, dst, src Object) {
|
||||||
|
const maxTries = 10
|
||||||
|
tries := 0
|
||||||
|
doUpdate := dst != nil
|
||||||
|
tryAgain:
|
||||||
in0, err := src.Open()
|
in0, err := src.Open()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
Stats.Error()
|
Stats.Error()
|
||||||
@@ -124,7 +154,7 @@ func Copy(f Fs, dst, src Object) {
|
|||||||
in := NewAccount(in0) // account the transfer
|
in := NewAccount(in0) // account the transfer
|
||||||
|
|
||||||
var actionTaken string
|
var actionTaken string
|
||||||
if dst != nil {
|
if doUpdate {
|
||||||
actionTaken = "Copied (updated existing)"
|
actionTaken = "Copied (updated existing)"
|
||||||
err = dst.Update(in, src.ModTime(), src.Size())
|
err = dst.Update(in, src.ModTime(), src.Size())
|
||||||
} else {
|
} else {
|
||||||
@@ -132,6 +162,17 @@ func Copy(f Fs, dst, src Object) {
|
|||||||
dst, err = f.Put(in, src.Remote(), src.ModTime(), src.Size())
|
dst, err = f.Put(in, src.Remote(), src.ModTime(), src.Size())
|
||||||
}
|
}
|
||||||
inErr := in.Close()
|
inErr := in.Close()
|
||||||
|
// Retry if err returned a retry error
|
||||||
|
if r, ok := err.(Retry); ok && r.Retry() && tries < maxTries {
|
||||||
|
tries++
|
||||||
|
Log(src, "Received error: %v - retrying %d/%d", err, tries, maxTries)
|
||||||
|
if removeFailedCopy(dst) {
|
||||||
|
// If we removed dst, then nil it out and note we are not updating
|
||||||
|
dst = nil
|
||||||
|
doUpdate = false
|
||||||
|
}
|
||||||
|
goto tryAgain
|
||||||
|
}
|
||||||
if err == nil {
|
if err == nil {
|
||||||
err = inErr
|
err = inErr
|
||||||
}
|
}
|
||||||
@@ -225,12 +266,10 @@ func Copier(in ObjectPairChan, fdst Fs, wg *sync.WaitGroup) {
|
|||||||
func DeleteFiles(to_be_deleted ObjectsChan) {
|
func DeleteFiles(to_be_deleted ObjectsChan) {
|
||||||
var wg sync.WaitGroup
|
var wg sync.WaitGroup
|
||||||
wg.Add(Config.Transfers)
|
wg.Add(Config.Transfers)
|
||||||
var fs Fs
|
|
||||||
for i := 0; i < Config.Transfers; i++ {
|
for i := 0; i < Config.Transfers; i++ {
|
||||||
go func() {
|
go func() {
|
||||||
defer wg.Done()
|
defer wg.Done()
|
||||||
for dst := range to_be_deleted {
|
for dst := range to_be_deleted {
|
||||||
fs = dst.Fs()
|
|
||||||
if Config.DryRun {
|
if Config.DryRun {
|
||||||
Debug(dst, "Not deleting as --dry-run")
|
Debug(dst, "Not deleting as --dry-run")
|
||||||
} else {
|
} else {
|
||||||
@@ -247,11 +286,24 @@ func DeleteFiles(to_be_deleted ObjectsChan) {
|
|||||||
}
|
}
|
||||||
}()
|
}()
|
||||||
}
|
}
|
||||||
|
Log(nil, "Waiting for deletions to finish")
|
||||||
Log(fs, "Waiting for deletions to finish")
|
|
||||||
wg.Wait()
|
wg.Wait()
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Read a map of Object.Remote to Object for the given Fs
|
||||||
|
func readFilesMap(fs Fs) map[string]Object {
|
||||||
|
files := make(map[string]Object)
|
||||||
|
for o := range fs.List() {
|
||||||
|
remote := o.Remote()
|
||||||
|
if _, ok := files[remote]; !ok {
|
||||||
|
files[remote] = o
|
||||||
|
} else {
|
||||||
|
Log(o, "Duplicate file detected")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return files
|
||||||
|
}
|
||||||
|
|
||||||
// Syncs fsrc into fdst
|
// Syncs fsrc into fdst
|
||||||
//
|
//
|
||||||
// If Delete is true then it deletes any files in fdst that aren't in fsrc
|
// If Delete is true then it deletes any files in fdst that aren't in fsrc
|
||||||
@@ -266,10 +318,7 @@ func Sync(fdst, fsrc Fs, Delete bool) error {
|
|||||||
|
|
||||||
// Read the destination files first
|
// Read the destination files first
|
||||||
// FIXME could do this in parallel and make it use less memory
|
// FIXME could do this in parallel and make it use less memory
|
||||||
delFiles := make(map[string]Object)
|
delFiles := readFilesMap(fdst)
|
||||||
for dst := range fdst.List() {
|
|
||||||
delFiles[dst.Remote()] = dst
|
|
||||||
}
|
|
||||||
|
|
||||||
// Read source files checking them off against dest files
|
// Read source files checking them off against dest files
|
||||||
to_be_checked := make(ObjectPairChan, Config.Transfers)
|
to_be_checked := make(ObjectPairChan, Config.Transfers)
|
||||||
@@ -334,22 +383,20 @@ func Check(fdst, fsrc Fs) error {
|
|||||||
|
|
||||||
// Read the destination files first
|
// Read the destination files first
|
||||||
// FIXME could do this in parallel and make it use less memory
|
// FIXME could do this in parallel and make it use less memory
|
||||||
dstFiles := make(map[string]Object)
|
dstFiles := readFilesMap(fdst)
|
||||||
for dst := range fdst.List() {
|
|
||||||
dstFiles[dst.Remote()] = dst
|
|
||||||
}
|
|
||||||
|
|
||||||
// Read the source files checking them against dstFiles
|
// Read the source files checking them against dstFiles
|
||||||
// FIXME could do this in parallel and make it use less memory
|
// FIXME could do this in parallel and make it use less memory
|
||||||
srcFiles := make(map[string]Object)
|
srcFiles := readFilesMap(fsrc)
|
||||||
|
|
||||||
|
// Move all the common files into commonFiles and delete then
|
||||||
|
// from srcFiles and dstFiles
|
||||||
commonFiles := make(map[string][]Object)
|
commonFiles := make(map[string][]Object)
|
||||||
for src := range fsrc.List() {
|
for remote, src := range srcFiles {
|
||||||
remote := src.Remote()
|
|
||||||
if dst, ok := dstFiles[remote]; ok {
|
if dst, ok := dstFiles[remote]; ok {
|
||||||
commonFiles[remote] = []Object{dst, src}
|
commonFiles[remote] = []Object{dst, src}
|
||||||
|
delete(srcFiles, remote)
|
||||||
delete(dstFiles, remote)
|
delete(dstFiles, remote)
|
||||||
} else {
|
|
||||||
srcFiles[remote] = src
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -429,14 +476,24 @@ func ListFn(f Fs, fn func(Object)) error {
|
|||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// mutex for synchronized output
|
||||||
|
var outMutex sync.Mutex
|
||||||
|
|
||||||
|
// Synchronized fmt.Fprintf
|
||||||
|
func syncFprintf(w io.Writer, format string, a ...interface{}) (n int, err error) {
|
||||||
|
outMutex.Lock()
|
||||||
|
defer outMutex.Unlock()
|
||||||
|
return fmt.Fprintf(w, format, a...)
|
||||||
|
}
|
||||||
|
|
||||||
// List the Fs to stdout
|
// List the Fs to stdout
|
||||||
//
|
//
|
||||||
// Shows size and path
|
// Shows size and path
|
||||||
//
|
//
|
||||||
// Lists in parallel which may get them out of order
|
// Lists in parallel which may get them out of order
|
||||||
func List(f Fs) error {
|
func List(f Fs, w io.Writer) error {
|
||||||
return ListFn(f, func(o Object) {
|
return ListFn(f, func(o Object) {
|
||||||
fmt.Printf("%9d %s\n", o.Size(), o.Remote())
|
syncFprintf(w, "%9d %s\n", o.Size(), o.Remote())
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -445,12 +502,12 @@ func List(f Fs) error {
|
|||||||
// Shows size, mod time and path
|
// Shows size, mod time and path
|
||||||
//
|
//
|
||||||
// Lists in parallel which may get them out of order
|
// Lists in parallel which may get them out of order
|
||||||
func ListLong(f Fs) error {
|
func ListLong(f Fs, w io.Writer) error {
|
||||||
return ListFn(f, func(o Object) {
|
return ListFn(f, func(o Object) {
|
||||||
Stats.Checking(o)
|
Stats.Checking(o)
|
||||||
modTime := o.ModTime()
|
modTime := o.ModTime()
|
||||||
Stats.DoneChecking(o)
|
Stats.DoneChecking(o)
|
||||||
fmt.Printf("%9d %19s %s\n", o.Size(), modTime.Format("2006-01-02 15:04:05.00000000"), o.Remote())
|
syncFprintf(w, "%9d %s %s\n", o.Size(), modTime.Format("2006-01-02 15:04:05.000000000"), o.Remote())
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -459,7 +516,7 @@ func ListLong(f Fs) error {
|
|||||||
// Produces the same output as the md5sum command
|
// Produces the same output as the md5sum command
|
||||||
//
|
//
|
||||||
// Lists in parallel which may get them out of order
|
// Lists in parallel which may get them out of order
|
||||||
func Md5sum(f Fs) error {
|
func Md5sum(f Fs, w io.Writer) error {
|
||||||
return ListFn(f, func(o Object) {
|
return ListFn(f, func(o Object) {
|
||||||
Stats.Checking(o)
|
Stats.Checking(o)
|
||||||
md5sum, err := o.Md5sum()
|
md5sum, err := o.Md5sum()
|
||||||
@@ -468,14 +525,14 @@ func Md5sum(f Fs) error {
|
|||||||
Debug(o, "Failed to read MD5: %v", err)
|
Debug(o, "Failed to read MD5: %v", err)
|
||||||
md5sum = "UNKNOWN"
|
md5sum = "UNKNOWN"
|
||||||
}
|
}
|
||||||
fmt.Printf("%32s %s\n", md5sum, o.Remote())
|
syncFprintf(w, "%32s %s\n", md5sum, o.Remote())
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
// List the directories/buckets/containers in the Fs to stdout
|
// List the directories/buckets/containers in the Fs to stdout
|
||||||
func ListDir(f Fs) error {
|
func ListDir(f Fs, w io.Writer) error {
|
||||||
for dir := range f.ListDir() {
|
for dir := range f.ListDir() {
|
||||||
fmt.Printf("%12d %13s %9d %s\n", dir.Bytes, dir.When.Format("2006-01-02 15:04:05"), dir.Count, dir.Name)
|
syncFprintf(w, "%12d %13s %9d %s\n", dir.Bytes, dir.When.Format("2006-01-02 15:04:05"), dir.Count, dir.Name)
|
||||||
}
|
}
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
@@ -508,20 +565,21 @@ func Rmdir(f Fs) error {
|
|||||||
//
|
//
|
||||||
// FIXME doesn't delete local directories
|
// FIXME doesn't delete local directories
|
||||||
func Purge(f Fs) error {
|
func Purge(f Fs) error {
|
||||||
|
var err error
|
||||||
if purger, ok := f.(Purger); ok {
|
if purger, ok := f.(Purger); ok {
|
||||||
if Config.DryRun {
|
if Config.DryRun {
|
||||||
Debug(f, "Not purging as --dry-run set")
|
Debug(f, "Not purging as --dry-run set")
|
||||||
} else {
|
} else {
|
||||||
err := purger.Purge()
|
err = purger.Purge()
|
||||||
if err != nil {
|
|
||||||
Stats.Error()
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
|
// DeleteFiles and Rmdir observe --dry-run
|
||||||
DeleteFiles(f.List())
|
DeleteFiles(f.List())
|
||||||
log.Printf("Deleting path")
|
err = Rmdir(f)
|
||||||
Rmdir(f)
|
}
|
||||||
|
if err != nil {
|
||||||
|
Stats.Error()
|
||||||
|
return err
|
||||||
}
|
}
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|||||||
442
fs/operations_test.go
Normal file
442
fs/operations_test.go
Normal file
@@ -0,0 +1,442 @@
|
|||||||
|
// Test rclone by doing real transactions to a storage provider to and
|
||||||
|
// from the local disk
|
||||||
|
|
||||||
|
package fs_test
|
||||||
|
|
||||||
|
import (
|
||||||
|
"bytes"
|
||||||
|
"flag"
|
||||||
|
"io/ioutil"
|
||||||
|
"log"
|
||||||
|
"os"
|
||||||
|
"path"
|
||||||
|
"path/filepath"
|
||||||
|
"regexp"
|
||||||
|
"strings"
|
||||||
|
"testing"
|
||||||
|
"time"
|
||||||
|
|
||||||
|
"github.com/ncw/rclone/fs"
|
||||||
|
"github.com/ncw/rclone/fstest"
|
||||||
|
|
||||||
|
// Active file systems
|
||||||
|
_ "github.com/ncw/rclone/drive"
|
||||||
|
_ "github.com/ncw/rclone/dropbox"
|
||||||
|
_ "github.com/ncw/rclone/googlecloudstorage"
|
||||||
|
_ "github.com/ncw/rclone/local"
|
||||||
|
_ "github.com/ncw/rclone/s3"
|
||||||
|
_ "github.com/ncw/rclone/swift"
|
||||||
|
)
|
||||||
|
|
||||||
|
// Globals
|
||||||
|
var (
|
||||||
|
localName, remoteName string
|
||||||
|
flocal, fremote fs.Fs
|
||||||
|
RemoteName = flag.String("remote", "", "Remote to test with, defaults to local filesystem")
|
||||||
|
SubDir = flag.Bool("subdir", false, "Set to test with a sub directory")
|
||||||
|
Verbose = flag.Bool("verbose", false, "Set to enable logging")
|
||||||
|
finalise func()
|
||||||
|
)
|
||||||
|
|
||||||
|
// Write a file
|
||||||
|
func WriteFile(filePath, content string, t time.Time) {
|
||||||
|
// FIXME make directories?
|
||||||
|
filePath = path.Join(localName, filePath)
|
||||||
|
dirPath := path.Dir(filePath)
|
||||||
|
err := os.MkdirAll(dirPath, 0770)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatalf("Failed to make directories %q: %v", dirPath, err)
|
||||||
|
}
|
||||||
|
err = ioutil.WriteFile(filePath, []byte(content), 0600)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatalf("Failed to write file %q: %v", filePath, err)
|
||||||
|
}
|
||||||
|
err = os.Chtimes(filePath, t, t)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatalf("Failed to chtimes file %q: %v", filePath, err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
var t1 = fstest.Time("2001-02-03T04:05:06.499999999Z")
|
||||||
|
var t2 = fstest.Time("2011-12-25T12:59:59.123456789Z")
|
||||||
|
var t3 = fstest.Time("2011-12-30T12:59:59.000000000Z")
|
||||||
|
|
||||||
|
func TestInit(t *testing.T) {
|
||||||
|
fs.LoadConfig()
|
||||||
|
fs.Config.Verbose = *Verbose
|
||||||
|
fs.Config.Quiet = !*Verbose
|
||||||
|
var err error
|
||||||
|
fremote, finalise, err = fstest.RandomRemote(*RemoteName, *SubDir)
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("Failed to open remote %q: %v", *RemoteName, err)
|
||||||
|
}
|
||||||
|
t.Logf("Testing with remote %v", fremote)
|
||||||
|
|
||||||
|
localName, err = ioutil.TempDir("", "rclone")
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("Failed to create temp dir: %v", err)
|
||||||
|
}
|
||||||
|
localName = filepath.ToSlash(localName)
|
||||||
|
t.Logf("Testing with local %q", localName)
|
||||||
|
flocal, err = fs.NewFs(localName)
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("Failed to make %q: %v", remoteName, err)
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
func TestCalculateModifyWindow(t *testing.T) {
|
||||||
|
fs.CalculateModifyWindow(fremote, flocal)
|
||||||
|
t.Logf("ModifyWindow is %q", fs.Config.ModifyWindow)
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestMkdir(t *testing.T) {
|
||||||
|
fstest.TestMkdir(t, fremote)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Check dry run is working
|
||||||
|
func TestCopyWithDryRun(t *testing.T) {
|
||||||
|
WriteFile("sub dir/hello world", "hello world", t1)
|
||||||
|
|
||||||
|
fs.Config.DryRun = true
|
||||||
|
err := fs.Sync(fremote, flocal, false)
|
||||||
|
fs.Config.DryRun = false
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("Copy failed: %v", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
items := []fstest.Item{
|
||||||
|
{Path: "sub dir/hello world", Size: 11, ModTime: t1, Md5sum: "5eb63bbbe01eeed093cb22bb8f5acdc3"},
|
||||||
|
}
|
||||||
|
|
||||||
|
fstest.CheckListingWithPrecision(t, flocal, items, fs.Config.ModifyWindow)
|
||||||
|
fstest.CheckListingWithPrecision(t, fremote, []fstest.Item{}, fs.Config.ModifyWindow)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Now without dry run
|
||||||
|
func TestCopy(t *testing.T) {
|
||||||
|
err := fs.Sync(fremote, flocal, false)
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("Copy failed: %v", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
items := []fstest.Item{
|
||||||
|
{Path: "sub dir/hello world", Size: 11, ModTime: t1, Md5sum: "5eb63bbbe01eeed093cb22bb8f5acdc3"},
|
||||||
|
}
|
||||||
|
|
||||||
|
fstest.CheckListingWithPrecision(t, flocal, items, fs.Config.ModifyWindow)
|
||||||
|
fstest.CheckListingWithPrecision(t, fremote, items, fs.Config.ModifyWindow)
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestLsd(t *testing.T) {
|
||||||
|
var buf bytes.Buffer
|
||||||
|
err := fs.ListDir(fremote, &buf)
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("ListDir failed: %v", err)
|
||||||
|
}
|
||||||
|
res := buf.String()
|
||||||
|
if !strings.Contains(res, "sub dir\n") {
|
||||||
|
t.Fatalf("Result wrong %q", res)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Now delete the local file and download it
|
||||||
|
func TestCopyAfterDelete(t *testing.T) {
|
||||||
|
err := os.Remove(localName + "/sub dir/hello world")
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("Remove failed: %v", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
items := []fstest.Item{
|
||||||
|
{Path: "sub dir/hello world", Size: 11, ModTime: t1, Md5sum: "5eb63bbbe01eeed093cb22bb8f5acdc3"},
|
||||||
|
}
|
||||||
|
fstest.CheckListingWithPrecision(t, flocal, []fstest.Item{}, fs.Config.ModifyWindow)
|
||||||
|
fstest.CheckListingWithPrecision(t, fremote, items, fs.Config.ModifyWindow)
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestCopyRedownload(t *testing.T) {
|
||||||
|
err := fs.Sync(flocal, fremote, false)
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("Copy failed: %v", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
items := []fstest.Item{
|
||||||
|
{Path: "sub dir/hello world", Size: 11, ModTime: t1, Md5sum: "5eb63bbbe01eeed093cb22bb8f5acdc3"},
|
||||||
|
}
|
||||||
|
fstest.CheckListingWithPrecision(t, flocal, items, fs.Config.ModifyWindow)
|
||||||
|
fstest.CheckListingWithPrecision(t, fremote, items, fs.Config.ModifyWindow)
|
||||||
|
|
||||||
|
// Clean the directory
|
||||||
|
cleanTempDir(t)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Create a file and sync it. Change the last modified date and resync.
|
||||||
|
// If we're only doing sync by size and checksum, we expect nothing to
|
||||||
|
// to be transferred on the second sync.
|
||||||
|
func TestSyncBasedOnCheckSum(t *testing.T) {
|
||||||
|
cleanTempDir(t)
|
||||||
|
fs.Config.CheckSum = true
|
||||||
|
defer func() { fs.Config.CheckSum = false }()
|
||||||
|
|
||||||
|
WriteFile("check sum", "", t1)
|
||||||
|
local_items := []fstest.Item{
|
||||||
|
{Path: "check sum", Size: 0, ModTime: t1, Md5sum: "d41d8cd98f00b204e9800998ecf8427e"},
|
||||||
|
}
|
||||||
|
fstest.CheckListingWithPrecision(t, flocal, local_items, fs.Config.ModifyWindow)
|
||||||
|
|
||||||
|
fs.Stats.ResetCounters()
|
||||||
|
err := fs.Sync(fremote, flocal, true)
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("Initial sync failed: %v", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
// We should have transferred exactly one file.
|
||||||
|
if fs.Stats.GetTransfers() != 1 {
|
||||||
|
t.Fatalf("Sync 1: want 1 transfer, got %d", fs.Stats.GetTransfers())
|
||||||
|
}
|
||||||
|
|
||||||
|
remote_items := local_items
|
||||||
|
fstest.CheckListingWithPrecision(t, fremote, remote_items, fs.Config.ModifyWindow)
|
||||||
|
|
||||||
|
err = os.Chtimes(localName+"/check sum", t2, t2)
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("Chtimes failed: %v", err)
|
||||||
|
}
|
||||||
|
local_items = []fstest.Item{
|
||||||
|
{Path: "check sum", Size: 0, ModTime: t2, Md5sum: "d41d8cd98f00b204e9800998ecf8427e"},
|
||||||
|
}
|
||||||
|
fstest.CheckListingWithPrecision(t, flocal, local_items, fs.Config.ModifyWindow)
|
||||||
|
|
||||||
|
fs.Stats.ResetCounters()
|
||||||
|
err = fs.Sync(fremote, flocal, true)
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("Sync failed: %v", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
// We should have transferred no files
|
||||||
|
if fs.Stats.GetTransfers() != 0 {
|
||||||
|
t.Fatalf("Sync 2: want 0 transfers, got %d", fs.Stats.GetTransfers())
|
||||||
|
}
|
||||||
|
|
||||||
|
fstest.CheckListingWithPrecision(t, flocal, local_items, fs.Config.ModifyWindow)
|
||||||
|
fstest.CheckListingWithPrecision(t, fremote, remote_items, fs.Config.ModifyWindow)
|
||||||
|
|
||||||
|
cleanTempDir(t)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Create a file and sync it. Change the last modified date and the
|
||||||
|
// file contents but not the size. If we're only doing sync by size
|
||||||
|
// only, we expect nothing to to be transferred on the second sync.
|
||||||
|
func TestSyncSizeOnly(t *testing.T) {
|
||||||
|
cleanTempDir(t)
|
||||||
|
fs.Config.SizeOnly = true
|
||||||
|
defer func() { fs.Config.SizeOnly = false }()
|
||||||
|
|
||||||
|
WriteFile("sizeonly", "potato", t1)
|
||||||
|
local_items := []fstest.Item{
|
||||||
|
{Path: "sizeonly", Size: 6, ModTime: t1, Md5sum: "8ee2027983915ec78acc45027d874316"},
|
||||||
|
}
|
||||||
|
fstest.CheckListingWithPrecision(t, flocal, local_items, fs.Config.ModifyWindow)
|
||||||
|
|
||||||
|
fs.Stats.ResetCounters()
|
||||||
|
err := fs.Sync(fremote, flocal, true)
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("Initial sync failed: %v", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
// We should have transferred exactly one file.
|
||||||
|
if fs.Stats.GetTransfers() != 1 {
|
||||||
|
t.Fatalf("Sync 1: want 1 transfer, got %d", fs.Stats.GetTransfers())
|
||||||
|
}
|
||||||
|
|
||||||
|
remote_items := local_items
|
||||||
|
fstest.CheckListingWithPrecision(t, fremote, remote_items, fs.Config.ModifyWindow)
|
||||||
|
|
||||||
|
// Update mtime, md5sum but not length of file
|
||||||
|
WriteFile("sizeonly", "POTATO", t2)
|
||||||
|
local_items = []fstest.Item{
|
||||||
|
{Path: "sizeonly", Size: 6, ModTime: t2, Md5sum: "8ac6f27a282e4938125482607ccfb55f"},
|
||||||
|
}
|
||||||
|
fstest.CheckListingWithPrecision(t, flocal, local_items, fs.Config.ModifyWindow)
|
||||||
|
|
||||||
|
fs.Stats.ResetCounters()
|
||||||
|
err = fs.Sync(fremote, flocal, true)
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("Sync failed: %v", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
// We should have transferred no files
|
||||||
|
if fs.Stats.GetTransfers() != 0 {
|
||||||
|
t.Fatalf("Sync 2: want 0 transfers, got %d", fs.Stats.GetTransfers())
|
||||||
|
}
|
||||||
|
|
||||||
|
fstest.CheckListingWithPrecision(t, flocal, local_items, fs.Config.ModifyWindow)
|
||||||
|
fstest.CheckListingWithPrecision(t, fremote, remote_items, fs.Config.ModifyWindow)
|
||||||
|
|
||||||
|
cleanTempDir(t)
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestSyncAfterChangingModtimeOnly(t *testing.T) {
|
||||||
|
WriteFile("empty space", "", t1)
|
||||||
|
|
||||||
|
err := os.Chtimes(localName+"/empty space", t2, t2)
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("Chtimes failed: %v", err)
|
||||||
|
}
|
||||||
|
err = fs.Sync(fremote, flocal, true)
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("Sync failed: %v", err)
|
||||||
|
}
|
||||||
|
items := []fstest.Item{
|
||||||
|
{Path: "empty space", Size: 0, ModTime: t2, Md5sum: "d41d8cd98f00b204e9800998ecf8427e"},
|
||||||
|
}
|
||||||
|
fstest.CheckListingWithPrecision(t, flocal, items, fs.Config.ModifyWindow)
|
||||||
|
fstest.CheckListingWithPrecision(t, fremote, items, fs.Config.ModifyWindow)
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestSyncAfterAddingAFile(t *testing.T) {
|
||||||
|
WriteFile("potato", "------------------------------------------------------------", t3)
|
||||||
|
err := fs.Sync(fremote, flocal, true)
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("Sync failed: %v", err)
|
||||||
|
}
|
||||||
|
items := []fstest.Item{
|
||||||
|
{Path: "empty space", Size: 0, ModTime: t2, Md5sum: "d41d8cd98f00b204e9800998ecf8427e"},
|
||||||
|
{Path: "potato", Size: 60, ModTime: t3, Md5sum: "d6548b156ea68a4e003e786df99eee76"},
|
||||||
|
}
|
||||||
|
fstest.CheckListingWithPrecision(t, flocal, items, fs.Config.ModifyWindow)
|
||||||
|
fstest.CheckListingWithPrecision(t, fremote, items, fs.Config.ModifyWindow)
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestSyncAfterChangingFilesSizeOnly(t *testing.T) {
|
||||||
|
WriteFile("potato", "smaller but same date", t3)
|
||||||
|
err := fs.Sync(fremote, flocal, true)
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("Sync failed: %v", err)
|
||||||
|
}
|
||||||
|
items := []fstest.Item{
|
||||||
|
{Path: "empty space", Size: 0, ModTime: t2, Md5sum: "d41d8cd98f00b204e9800998ecf8427e"},
|
||||||
|
{Path: "potato", Size: 21, ModTime: t3, Md5sum: "100defcf18c42a1e0dc42a789b107cd2"},
|
||||||
|
}
|
||||||
|
fstest.CheckListingWithPrecision(t, flocal, items, fs.Config.ModifyWindow)
|
||||||
|
fstest.CheckListingWithPrecision(t, fremote, items, fs.Config.ModifyWindow)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Sync after changing a file's contents, modtime but not length
|
||||||
|
func TestSyncAfterChangingContentsOnly(t *testing.T) {
|
||||||
|
WriteFile("potato", "SMALLER BUT SAME DATE", t2)
|
||||||
|
err := fs.Sync(fremote, flocal, true)
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("Sync failed: %v", err)
|
||||||
|
}
|
||||||
|
items := []fstest.Item{
|
||||||
|
{Path: "empty space", Size: 0, ModTime: t2, Md5sum: "d41d8cd98f00b204e9800998ecf8427e"},
|
||||||
|
{Path: "potato", Size: 21, ModTime: t2, Md5sum: "e4cb6955d9106df6263c45fcfc10f163"},
|
||||||
|
}
|
||||||
|
fstest.CheckListingWithPrecision(t, flocal, items, fs.Config.ModifyWindow)
|
||||||
|
fstest.CheckListingWithPrecision(t, fremote, items, fs.Config.ModifyWindow)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Sync after removing a file and adding a file --dry-run
|
||||||
|
func TestSyncAfterRemovingAFileAndAddingAFileDryRun(t *testing.T) {
|
||||||
|
WriteFile("potato2", "------------------------------------------------------------", t1)
|
||||||
|
err := os.Remove(localName + "/potato")
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("Remove failed: %v", err)
|
||||||
|
}
|
||||||
|
fs.Config.DryRun = true
|
||||||
|
err = fs.Sync(fremote, flocal, true)
|
||||||
|
fs.Config.DryRun = false
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("Sync failed: %v", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
before := []fstest.Item{
|
||||||
|
{Path: "empty space", Size: 0, ModTime: t2, Md5sum: "d41d8cd98f00b204e9800998ecf8427e"},
|
||||||
|
{Path: "potato", Size: 21, ModTime: t2, Md5sum: "e4cb6955d9106df6263c45fcfc10f163"},
|
||||||
|
}
|
||||||
|
items := []fstest.Item{
|
||||||
|
{Path: "empty space", Size: 0, ModTime: t2, Md5sum: "d41d8cd98f00b204e9800998ecf8427e"},
|
||||||
|
{Path: "potato2", Size: 60, ModTime: t1, Md5sum: "d6548b156ea68a4e003e786df99eee76"},
|
||||||
|
}
|
||||||
|
fstest.CheckListingWithPrecision(t, flocal, items, fs.Config.ModifyWindow)
|
||||||
|
fstest.CheckListingWithPrecision(t, fremote, before, fs.Config.ModifyWindow)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Sync after removing a file and adding a file
|
||||||
|
func TestSyncAfterRemovingAFileAndAddingAFile(t *testing.T) {
|
||||||
|
err := fs.Sync(fremote, flocal, true)
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("Sync failed: %v", err)
|
||||||
|
}
|
||||||
|
items := []fstest.Item{
|
||||||
|
{Path: "empty space", Size: 0, ModTime: t2, Md5sum: "d41d8cd98f00b204e9800998ecf8427e"},
|
||||||
|
{Path: "potato2", Size: 60, ModTime: t1, Md5sum: "d6548b156ea68a4e003e786df99eee76"},
|
||||||
|
}
|
||||||
|
fstest.CheckListingWithPrecision(t, flocal, items, fs.Config.ModifyWindow)
|
||||||
|
fstest.CheckListingWithPrecision(t, fremote, items, fs.Config.ModifyWindow)
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestLs(t *testing.T) {
|
||||||
|
var buf bytes.Buffer
|
||||||
|
err := fs.List(fremote, &buf)
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("List failed: %v", err)
|
||||||
|
}
|
||||||
|
res := buf.String()
|
||||||
|
if !strings.Contains(res, " 0 empty space\n") {
|
||||||
|
t.Errorf("empty space missing: %q", res)
|
||||||
|
}
|
||||||
|
if !strings.Contains(res, " 60 potato2\n") {
|
||||||
|
t.Errorf("potato2 missing: %q", res)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestLsLong(t *testing.T) {
|
||||||
|
var buf bytes.Buffer
|
||||||
|
err := fs.ListLong(fremote, &buf)
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("List failed: %v", err)
|
||||||
|
}
|
||||||
|
res := buf.String()
|
||||||
|
m1 := regexp.MustCompile(`(?m)^ 0 2011-12-25 12:59:59\.\d{9} empty space$`)
|
||||||
|
if !m1.MatchString(res) {
|
||||||
|
t.Errorf("empty space missing: %q", res)
|
||||||
|
}
|
||||||
|
m2 := regexp.MustCompile(`(?m)^ 60 2001-02-03 04:05:06\.\d{9} potato2$`)
|
||||||
|
if !m2.MatchString(res) {
|
||||||
|
t.Errorf("potato2 missing: %q", res)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestMd5sum(t *testing.T) {
|
||||||
|
var buf bytes.Buffer
|
||||||
|
err := fs.Md5sum(fremote, &buf)
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("List failed: %v", err)
|
||||||
|
}
|
||||||
|
res := buf.String()
|
||||||
|
if !strings.Contains(res, "d41d8cd98f00b204e9800998ecf8427e empty space\n") {
|
||||||
|
t.Errorf("empty space missing: %q", res)
|
||||||
|
}
|
||||||
|
if !strings.Contains(res, "6548b156ea68a4e003e786df99eee76 potato2\n") {
|
||||||
|
t.Errorf("potato2 missing: %q", res)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestCheck(t *testing.T) {
|
||||||
|
}
|
||||||
|
|
||||||
|
// Clean the temporary directory
|
||||||
|
func cleanTempDir(t *testing.T) {
|
||||||
|
t.Logf("Cleaning temporary directory: %q", localName)
|
||||||
|
err := os.RemoveAll(localName)
|
||||||
|
if err != nil {
|
||||||
|
t.Logf("Failed to remove %q: %v", localName, err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestFinalise(t *testing.T) {
|
||||||
|
finalise()
|
||||||
|
|
||||||
|
cleanTempDir(t)
|
||||||
|
}
|
||||||
29
fs/test_all.sh
Executable file
29
fs/test_all.sh
Executable file
@@ -0,0 +1,29 @@
|
|||||||
|
#!/bin/bash
|
||||||
|
|
||||||
|
go install
|
||||||
|
|
||||||
|
REMOTES="
|
||||||
|
TestSwift:
|
||||||
|
TestS3:
|
||||||
|
TestDrive:
|
||||||
|
TestGoogleCloudStorage:
|
||||||
|
TestDropbox:
|
||||||
|
"
|
||||||
|
|
||||||
|
function test_remote {
|
||||||
|
args=$@
|
||||||
|
echo "@go test $args"
|
||||||
|
go test $args || {
|
||||||
|
echo "*** test $args FAILED ***"
|
||||||
|
exit 1
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
test_remote
|
||||||
|
test_remote --subdir
|
||||||
|
for remote in $REMOTES; do
|
||||||
|
test_remote --remote $remote
|
||||||
|
test_remote --remote $remote --subdir
|
||||||
|
done
|
||||||
|
|
||||||
|
echo "All OK"
|
||||||
@@ -1,3 +1,3 @@
|
|||||||
package fs
|
package fs
|
||||||
|
|
||||||
const Version = "v1.02"
|
const Version = "v1.15"
|
||||||
|
|||||||
231
fstest/fstest.go
Normal file
231
fstest/fstest.go
Normal file
@@ -0,0 +1,231 @@
|
|||||||
|
// Utilities for testing the Fs
|
||||||
|
package fstest
|
||||||
|
|
||||||
|
// FIXME put name of test FS in Fs structure
|
||||||
|
|
||||||
|
import (
|
||||||
|
"io/ioutil"
|
||||||
|
"log"
|
||||||
|
"math/rand"
|
||||||
|
"os"
|
||||||
|
"path/filepath"
|
||||||
|
"strings"
|
||||||
|
"testing"
|
||||||
|
"time"
|
||||||
|
|
||||||
|
"github.com/ncw/rclone/fs"
|
||||||
|
)
|
||||||
|
|
||||||
|
// Seed the random number generator
|
||||||
|
func init() {
|
||||||
|
rand.Seed(time.Now().UnixNano())
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
// Represents an item for checking
|
||||||
|
type Item struct {
|
||||||
|
Path string
|
||||||
|
Md5sum string
|
||||||
|
ModTime time.Time
|
||||||
|
Size int64
|
||||||
|
}
|
||||||
|
|
||||||
|
// check the mod time to the given precision
|
||||||
|
func (i *Item) CheckModTime(t *testing.T, obj fs.Object, modTime time.Time, precision time.Duration) {
|
||||||
|
dt := modTime.Sub(i.ModTime)
|
||||||
|
if dt >= precision || dt <= -precision {
|
||||||
|
t.Errorf("%s: Modification time difference too big |%s| > %s (%s vs %s) (precision %s)", obj.Remote(), dt, precision, modTime, i.ModTime, precision)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (i *Item) Check(t *testing.T, obj fs.Object, precision time.Duration) {
|
||||||
|
if obj == nil {
|
||||||
|
t.Fatalf("Object is nil")
|
||||||
|
}
|
||||||
|
// Check attributes
|
||||||
|
Md5sum, err := obj.Md5sum()
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("Failed to read md5sum for %q: %v", obj.Remote(), err)
|
||||||
|
}
|
||||||
|
if i.Md5sum != Md5sum {
|
||||||
|
t.Errorf("%s: Md5sum incorrect - expecting %q got %q", obj.Remote(), i.Md5sum, Md5sum)
|
||||||
|
}
|
||||||
|
if i.Size != obj.Size() {
|
||||||
|
t.Errorf("%s: Size incorrect - expecting %d got %d", obj.Remote(), i.Size, obj.Size())
|
||||||
|
}
|
||||||
|
i.CheckModTime(t, obj, obj.ModTime(), precision)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Represents all items for checking
|
||||||
|
type Items struct {
|
||||||
|
byName map[string]*Item
|
||||||
|
items []Item
|
||||||
|
}
|
||||||
|
|
||||||
|
// Make an Items
|
||||||
|
func NewItems(items []Item) *Items {
|
||||||
|
is := &Items{
|
||||||
|
byName: make(map[string]*Item),
|
||||||
|
items: items,
|
||||||
|
}
|
||||||
|
// Fill up byName
|
||||||
|
for i := range items {
|
||||||
|
is.byName[items[i].Path] = &items[i]
|
||||||
|
}
|
||||||
|
return is
|
||||||
|
}
|
||||||
|
|
||||||
|
// Check off an item
|
||||||
|
func (is *Items) Find(t *testing.T, obj fs.Object, precision time.Duration) {
|
||||||
|
i, ok := is.byName[obj.Remote()]
|
||||||
|
if !ok {
|
||||||
|
t.Errorf("Unexpected file %q", obj.Remote())
|
||||||
|
return
|
||||||
|
}
|
||||||
|
delete(is.byName, obj.Remote())
|
||||||
|
i.Check(t, obj, precision)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Check all done
|
||||||
|
func (is *Items) Done(t *testing.T) {
|
||||||
|
if len(is.byName) != 0 {
|
||||||
|
for name := range is.byName {
|
||||||
|
log.Printf("Not found %q", name)
|
||||||
|
}
|
||||||
|
t.Errorf("%d objects not found", len(is.byName))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Checks the fs to see if it has the expected contents
|
||||||
|
func CheckListingWithPrecision(t *testing.T, f fs.Fs, items []Item, precision time.Duration) {
|
||||||
|
is := NewItems(items)
|
||||||
|
for obj := range f.List() {
|
||||||
|
if obj == nil {
|
||||||
|
t.Errorf("Unexpected nil in List()")
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
is.Find(t, obj, precision)
|
||||||
|
}
|
||||||
|
is.Done(t)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Checks the fs to see if it has the expected contents
|
||||||
|
func CheckListing(t *testing.T, f fs.Fs, items []Item) {
|
||||||
|
precision := f.Precision()
|
||||||
|
CheckListingWithPrecision(t, f, items, precision)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Parse a time string or explode
|
||||||
|
func Time(timeString string) time.Time {
|
||||||
|
t, err := time.Parse(time.RFC3339Nano, timeString)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatalf("Failed to parse time %q: %v", timeString, err)
|
||||||
|
}
|
||||||
|
return t
|
||||||
|
}
|
||||||
|
|
||||||
|
// Create a random string
|
||||||
|
func RandomString(n int) string {
|
||||||
|
source := "abcdefghijklmnopqrstuvwxyz0123456789"
|
||||||
|
out := make([]byte, n)
|
||||||
|
for i := range out {
|
||||||
|
out[i] = source[rand.Intn(len(source))]
|
||||||
|
}
|
||||||
|
return string(out)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Creates a temporary directory name for local remotes
|
||||||
|
func LocalRemote() (path string, err error) {
|
||||||
|
path, err = ioutil.TempDir("", "rclone")
|
||||||
|
if err == nil {
|
||||||
|
// Now remove the directory
|
||||||
|
err = os.Remove(path)
|
||||||
|
}
|
||||||
|
path = filepath.ToSlash(path)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// Make a random bucket or subdirectory name
|
||||||
|
//
|
||||||
|
// Returns a random remote name plus the leaf name
|
||||||
|
func RandomRemoteName(remoteName string) (string, string, error) {
|
||||||
|
var err error
|
||||||
|
var leafName string
|
||||||
|
|
||||||
|
// Make a directory if remote name is null
|
||||||
|
if remoteName == "" {
|
||||||
|
remoteName, err = LocalRemote()
|
||||||
|
if err != nil {
|
||||||
|
return "", "", err
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
if !strings.HasSuffix(remoteName, ":") {
|
||||||
|
remoteName += "/"
|
||||||
|
}
|
||||||
|
leafName = RandomString(32)
|
||||||
|
remoteName += leafName
|
||||||
|
}
|
||||||
|
return remoteName, leafName, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// Make a random bucket or subdirectory on the remote
|
||||||
|
//
|
||||||
|
// Call the finalise function returned to Purge the fs at the end (and
|
||||||
|
// the parent if necessary)
|
||||||
|
func RandomRemote(remoteName string, subdir bool) (fs.Fs, func(), error) {
|
||||||
|
var err error
|
||||||
|
var parentRemote fs.Fs
|
||||||
|
|
||||||
|
remoteName, _, err = RandomRemoteName(remoteName)
|
||||||
|
if err != nil {
|
||||||
|
return nil, nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
if subdir {
|
||||||
|
parentRemote, err = fs.NewFs(remoteName)
|
||||||
|
if err != nil {
|
||||||
|
return nil, nil, err
|
||||||
|
}
|
||||||
|
remoteName += "/" + RandomString(8)
|
||||||
|
}
|
||||||
|
|
||||||
|
remote, err := fs.NewFs(remoteName)
|
||||||
|
if err != nil {
|
||||||
|
return nil, nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
finalise := func() {
|
||||||
|
_ = fs.Purge(remote) // ignore error
|
||||||
|
if parentRemote != nil {
|
||||||
|
err = fs.Purge(parentRemote) // ignore error
|
||||||
|
if err != nil {
|
||||||
|
log.Printf("Failed to purge %v: %v", parentRemote, err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return remote, finalise, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestMkdir(t *testing.T, remote fs.Fs) {
|
||||||
|
err := fs.Mkdir(remote)
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("Mkdir failed: %v", err)
|
||||||
|
}
|
||||||
|
CheckListing(t, remote, []Item{})
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestPurge(t *testing.T, remote fs.Fs) {
|
||||||
|
err := fs.Purge(remote)
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("Purge failed: %v", err)
|
||||||
|
}
|
||||||
|
CheckListing(t, remote, []Item{})
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestRmdir(t *testing.T, remote fs.Fs) {
|
||||||
|
err := fs.Rmdir(remote)
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("Rmdir failed: %v", err)
|
||||||
|
}
|
||||||
|
}
|
||||||
438
fstest/fstests/fstests.go
Normal file
438
fstest/fstests/fstests.go
Normal file
@@ -0,0 +1,438 @@
|
|||||||
|
// Generic tests for testing the Fs and Object interfaces
|
||||||
|
package fstests
|
||||||
|
|
||||||
|
import (
|
||||||
|
"bytes"
|
||||||
|
"crypto/md5"
|
||||||
|
"encoding/hex"
|
||||||
|
"io"
|
||||||
|
"log"
|
||||||
|
"os"
|
||||||
|
"strings"
|
||||||
|
"testing"
|
||||||
|
"time"
|
||||||
|
|
||||||
|
"github.com/ncw/rclone/fs"
|
||||||
|
"github.com/ncw/rclone/fstest"
|
||||||
|
)
|
||||||
|
|
||||||
|
var (
|
||||||
|
remote fs.Fs
|
||||||
|
RemoteName = ""
|
||||||
|
subRemoteName = ""
|
||||||
|
subRemoteLeaf = ""
|
||||||
|
NilObject fs.Object
|
||||||
|
file1 = fstest.Item{
|
||||||
|
ModTime: fstest.Time("2001-02-03T04:05:06.499999999Z"),
|
||||||
|
Path: "file name.txt",
|
||||||
|
}
|
||||||
|
file2 = fstest.Item{
|
||||||
|
ModTime: fstest.Time("2001-02-03T04:05:10.123123123Z"),
|
||||||
|
Path: `hello? sausage/êé/Hello, 世界/ " ' @ < > & ?/z.txt`,
|
||||||
|
}
|
||||||
|
)
|
||||||
|
|
||||||
|
func TestInit(t *testing.T) {
|
||||||
|
var err error
|
||||||
|
fs.LoadConfig()
|
||||||
|
fs.Config.Verbose = false
|
||||||
|
fs.Config.Quiet = true
|
||||||
|
if RemoteName == "" {
|
||||||
|
RemoteName, err = fstest.LocalRemote()
|
||||||
|
if err != nil {
|
||||||
|
log.Fatalf("Failed to create tmp dir: %v", err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
subRemoteName, subRemoteLeaf, err = fstest.RandomRemoteName(RemoteName)
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("Couldn't make remote name: %v", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
remote, err = fs.NewFs(subRemoteName)
|
||||||
|
if err == fs.NotFoundInConfigFile {
|
||||||
|
log.Printf("Didn't find %q in config file - skipping tests", RemoteName)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("Couldn't start FS: %v", err)
|
||||||
|
}
|
||||||
|
fstest.TestMkdir(t, remote)
|
||||||
|
}
|
||||||
|
|
||||||
|
func skipIfNotOk(t *testing.T) {
|
||||||
|
if remote == nil {
|
||||||
|
t.Skip("FS not configured")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// String returns a description of the FS
|
||||||
|
|
||||||
|
func TestFsString(t *testing.T) {
|
||||||
|
skipIfNotOk(t)
|
||||||
|
str := remote.String()
|
||||||
|
if str == "" {
|
||||||
|
t.Fatal("Bad fs.String()")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
type TestFile struct {
|
||||||
|
ModTime time.Time
|
||||||
|
Path string
|
||||||
|
Size int64
|
||||||
|
Md5sum string
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestFsRmdirEmpty(t *testing.T) {
|
||||||
|
skipIfNotOk(t)
|
||||||
|
fstest.TestRmdir(t, remote)
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestFsRmdirNotFound(t *testing.T) {
|
||||||
|
skipIfNotOk(t)
|
||||||
|
err := remote.Rmdir()
|
||||||
|
if err == nil {
|
||||||
|
t.Fatalf("Expecting error on Rmdir non existent")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestFsMkdir(t *testing.T) {
|
||||||
|
skipIfNotOk(t)
|
||||||
|
fstest.TestMkdir(t, remote)
|
||||||
|
fstest.TestMkdir(t, remote)
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestFsListEmpty(t *testing.T) {
|
||||||
|
skipIfNotOk(t)
|
||||||
|
fstest.CheckListing(t, remote, []fstest.Item{})
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestFsListDirEmpty(t *testing.T) {
|
||||||
|
skipIfNotOk(t)
|
||||||
|
for obj := range remote.ListDir() {
|
||||||
|
t.Errorf("Found unexpected item %q", obj.Name)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestFsNewFsObjectNotFound(t *testing.T) {
|
||||||
|
skipIfNotOk(t)
|
||||||
|
if remote.NewFsObject("potato") != nil {
|
||||||
|
t.Fatal("Didn't expect to find object")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func findObject(t *testing.T, Name string) fs.Object {
|
||||||
|
obj := remote.NewFsObject(Name)
|
||||||
|
if obj == nil {
|
||||||
|
t.Fatalf("Object not found: %q", Name)
|
||||||
|
}
|
||||||
|
return obj
|
||||||
|
}
|
||||||
|
|
||||||
|
func testPut(t *testing.T, file *fstest.Item) {
|
||||||
|
buf := bytes.NewBufferString(fstest.RandomString(100))
|
||||||
|
hash := md5.New()
|
||||||
|
in := io.TeeReader(buf, hash)
|
||||||
|
|
||||||
|
file.Size = int64(buf.Len())
|
||||||
|
obj, err := remote.Put(in, file.Path, file.ModTime, file.Size)
|
||||||
|
if err != nil {
|
||||||
|
t.Fatal("Put error", err)
|
||||||
|
}
|
||||||
|
file.Md5sum = hex.EncodeToString(hash.Sum(nil))
|
||||||
|
file.Check(t, obj, remote.Precision())
|
||||||
|
// Re-read the object and check again
|
||||||
|
obj = findObject(t, file.Path)
|
||||||
|
file.Check(t, obj, remote.Precision())
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestFsPutFile1(t *testing.T) {
|
||||||
|
skipIfNotOk(t)
|
||||||
|
testPut(t, &file1)
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestFsPutFile2(t *testing.T) {
|
||||||
|
skipIfNotOk(t)
|
||||||
|
testPut(t, &file2)
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestFsListDirFile2(t *testing.T) {
|
||||||
|
skipIfNotOk(t)
|
||||||
|
found := false
|
||||||
|
for obj := range remote.ListDir() {
|
||||||
|
if obj.Name != `hello? sausage` {
|
||||||
|
t.Errorf("Found unexpected item %q", obj.Name)
|
||||||
|
} else {
|
||||||
|
found = true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if !found {
|
||||||
|
t.Errorf("Didn't find %q", `hello? sausage`)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestFsListDirRoot(t *testing.T) {
|
||||||
|
skipIfNotOk(t)
|
||||||
|
rootRemote, err := fs.NewFs(RemoteName)
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("Failed to make remote %q: %v", RemoteName, err)
|
||||||
|
}
|
||||||
|
found := false
|
||||||
|
for obj := range rootRemote.ListDir() {
|
||||||
|
if obj.Name == subRemoteLeaf {
|
||||||
|
found = true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if !found {
|
||||||
|
t.Errorf("Didn't find %q", subRemoteLeaf)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestFsListRoot(t *testing.T) {
|
||||||
|
skipIfNotOk(t)
|
||||||
|
rootRemote, err := fs.NewFs(RemoteName)
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("Failed to make remote %q: %v", RemoteName, err)
|
||||||
|
}
|
||||||
|
// Should either find file1 and file2 or nothing
|
||||||
|
found1 := false
|
||||||
|
file1 := subRemoteLeaf + "/" + file1.Path
|
||||||
|
found2 := false
|
||||||
|
file2 := subRemoteLeaf + "/" + file2.Path
|
||||||
|
count := 0
|
||||||
|
errors := fs.Stats.GetErrors()
|
||||||
|
for obj := range rootRemote.List() {
|
||||||
|
count++
|
||||||
|
if obj.Remote() == file1 {
|
||||||
|
found1 = true
|
||||||
|
}
|
||||||
|
if obj.Remote() == file2 {
|
||||||
|
found2 = true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
errors -= fs.Stats.GetErrors()
|
||||||
|
if count == 0 {
|
||||||
|
if errors == 0 {
|
||||||
|
t.Error("Expecting error if count==0")
|
||||||
|
}
|
||||||
|
return
|
||||||
|
}
|
||||||
|
if found1 && found2 {
|
||||||
|
if errors != 0 {
|
||||||
|
t.Error("Not expecting error if found")
|
||||||
|
}
|
||||||
|
return
|
||||||
|
}
|
||||||
|
t.Errorf("Didn't find %q (%v) and %q (%v) or no files (count %d)", file1, found1, file2, found2, count)
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestFsListFile1(t *testing.T) {
|
||||||
|
skipIfNotOk(t)
|
||||||
|
fstest.CheckListing(t, remote, []fstest.Item{file1, file2})
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestFsNewFsObject(t *testing.T) {
|
||||||
|
skipIfNotOk(t)
|
||||||
|
obj := findObject(t, file1.Path)
|
||||||
|
file1.Check(t, obj, remote.Precision())
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestFsListFile1and2(t *testing.T) {
|
||||||
|
skipIfNotOk(t)
|
||||||
|
fstest.CheckListing(t, remote, []fstest.Item{file1, file2})
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestFsRmdirFull(t *testing.T) {
|
||||||
|
skipIfNotOk(t)
|
||||||
|
err := remote.Rmdir()
|
||||||
|
if err == nil {
|
||||||
|
t.Fatalf("Expecting error on RMdir on non empty remote")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestFsPrecision(t *testing.T) {
|
||||||
|
skipIfNotOk(t)
|
||||||
|
precision := remote.Precision()
|
||||||
|
if precision > time.Second || precision < 0 {
|
||||||
|
t.Fatalf("Precision out of range %v", precision)
|
||||||
|
}
|
||||||
|
// FIXME check expected precision
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestObjectString(t *testing.T) {
|
||||||
|
skipIfNotOk(t)
|
||||||
|
obj := findObject(t, file1.Path)
|
||||||
|
s := obj.String()
|
||||||
|
if s != file1.Path {
|
||||||
|
t.Errorf("String() wrong %v != %v", s, file1.Path)
|
||||||
|
}
|
||||||
|
obj = NilObject
|
||||||
|
s = obj.String()
|
||||||
|
if s != "<nil>" {
|
||||||
|
t.Errorf("String() wrong %v != %v", s, "<nil>")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestObjectFs(t *testing.T) {
|
||||||
|
skipIfNotOk(t)
|
||||||
|
obj := findObject(t, file1.Path)
|
||||||
|
if obj.Fs() != remote {
|
||||||
|
t.Errorf("Fs is wrong %v != %v", obj.Fs(), remote)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestObjectRemote(t *testing.T) {
|
||||||
|
skipIfNotOk(t)
|
||||||
|
obj := findObject(t, file1.Path)
|
||||||
|
if obj.Remote() != file1.Path {
|
||||||
|
t.Errorf("Remote is wrong %v != %v", obj.Remote(), file1.Path)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestObjectMd5sum(t *testing.T) {
|
||||||
|
skipIfNotOk(t)
|
||||||
|
obj := findObject(t, file1.Path)
|
||||||
|
Md5sum, err := obj.Md5sum()
|
||||||
|
if err != nil {
|
||||||
|
t.Errorf("Error in Md5sum: %v", err)
|
||||||
|
}
|
||||||
|
if Md5sum != file1.Md5sum {
|
||||||
|
t.Errorf("Md5sum is wrong %v != %v", Md5sum, file1.Md5sum)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestObjectModTime(t *testing.T) {
|
||||||
|
skipIfNotOk(t)
|
||||||
|
obj := findObject(t, file1.Path)
|
||||||
|
file1.CheckModTime(t, obj, obj.ModTime(), remote.Precision())
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestObjectSetModTime(t *testing.T) {
|
||||||
|
skipIfNotOk(t)
|
||||||
|
newModTime := fstest.Time("2011-12-13T14:15:16.999999999Z")
|
||||||
|
obj := findObject(t, file1.Path)
|
||||||
|
obj.SetModTime(newModTime)
|
||||||
|
file1.ModTime = newModTime
|
||||||
|
file1.CheckModTime(t, obj, obj.ModTime(), remote.Precision())
|
||||||
|
// And make a new object and read it from there too
|
||||||
|
TestObjectModTime(t)
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestObjectSize(t *testing.T) {
|
||||||
|
skipIfNotOk(t)
|
||||||
|
obj := findObject(t, file1.Path)
|
||||||
|
if obj.Size() != file1.Size {
|
||||||
|
t.Errorf("Size is wrong %v != %v", obj.Size(), file1.Size)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestObjectOpen(t *testing.T) {
|
||||||
|
skipIfNotOk(t)
|
||||||
|
obj := findObject(t, file1.Path)
|
||||||
|
in, err := obj.Open()
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("Open() return error: %v", err)
|
||||||
|
}
|
||||||
|
hash := md5.New()
|
||||||
|
n, err := io.Copy(hash, in)
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("io.Copy() return error: %v", err)
|
||||||
|
}
|
||||||
|
if n != file1.Size {
|
||||||
|
t.Fatalf("Read wrong number of bytes %d != %d", n, file1.Size)
|
||||||
|
}
|
||||||
|
err = in.Close()
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("in.Close() return error: %v", err)
|
||||||
|
}
|
||||||
|
Md5sum := hex.EncodeToString(hash.Sum(nil))
|
||||||
|
if Md5sum != file1.Md5sum {
|
||||||
|
t.Errorf("Md5sum is wrong %v != %v", Md5sum, file1.Md5sum)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestObjectUpdate(t *testing.T) {
|
||||||
|
skipIfNotOk(t)
|
||||||
|
buf := bytes.NewBufferString(fstest.RandomString(200))
|
||||||
|
hash := md5.New()
|
||||||
|
in := io.TeeReader(buf, hash)
|
||||||
|
|
||||||
|
file1.Size = int64(buf.Len())
|
||||||
|
obj := findObject(t, file1.Path)
|
||||||
|
err := obj.Update(in, file1.ModTime, file1.Size)
|
||||||
|
if err != nil {
|
||||||
|
t.Fatal("Update error", err)
|
||||||
|
}
|
||||||
|
file1.Md5sum = hex.EncodeToString(hash.Sum(nil))
|
||||||
|
file1.Check(t, obj, remote.Precision())
|
||||||
|
// Re-read the object and check again
|
||||||
|
obj = findObject(t, file1.Path)
|
||||||
|
file1.Check(t, obj, remote.Precision())
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestObjectStorable(t *testing.T) {
|
||||||
|
skipIfNotOk(t)
|
||||||
|
obj := findObject(t, file1.Path)
|
||||||
|
if !obj.Storable() {
|
||||||
|
t.Fatalf("Expecting %v to be storable", obj)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestLimitedFs(t *testing.T) {
|
||||||
|
skipIfNotOk(t)
|
||||||
|
remoteName := subRemoteName + "/" + file2.Path
|
||||||
|
file2Copy := file2
|
||||||
|
file2Copy.Path = "z.txt"
|
||||||
|
fileRemote, err := fs.NewFs(remoteName)
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("Failed to make remote %q: %v", remoteName, err)
|
||||||
|
}
|
||||||
|
fstest.CheckListing(t, fileRemote, []fstest.Item{file2Copy})
|
||||||
|
_, ok := fileRemote.(*fs.Limited)
|
||||||
|
if !ok {
|
||||||
|
t.Errorf("%v is not a fs.Limited", fileRemote)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestLimitedFsNotFound(t *testing.T) {
|
||||||
|
skipIfNotOk(t)
|
||||||
|
remoteName := subRemoteName + "/not found.txt"
|
||||||
|
fileRemote, err := fs.NewFs(remoteName)
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("Failed to make remote %q: %v", remoteName, err)
|
||||||
|
}
|
||||||
|
fstest.CheckListing(t, fileRemote, []fstest.Item{})
|
||||||
|
_, ok := fileRemote.(*fs.Limited)
|
||||||
|
if ok {
|
||||||
|
t.Errorf("%v is is a fs.Limited", fileRemote)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestObjectRemove(t *testing.T) {
|
||||||
|
skipIfNotOk(t)
|
||||||
|
obj := findObject(t, file1.Path)
|
||||||
|
err := obj.Remove()
|
||||||
|
if err != nil {
|
||||||
|
t.Fatal("Remove error", err)
|
||||||
|
}
|
||||||
|
fstest.CheckListing(t, remote, []fstest.Item{file2})
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestObjectPurge(t *testing.T) {
|
||||||
|
skipIfNotOk(t)
|
||||||
|
fstest.TestPurge(t, remote)
|
||||||
|
err := fs.Purge(remote)
|
||||||
|
if err == nil {
|
||||||
|
t.Fatal("Expecting error after on second purge")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestFinalise(t *testing.T) {
|
||||||
|
skipIfNotOk(t)
|
||||||
|
if strings.HasPrefix(RemoteName, "/") {
|
||||||
|
// Remove temp directory
|
||||||
|
err := os.Remove(RemoteName)
|
||||||
|
if err != nil {
|
||||||
|
log.Printf("Failed to remove %q: %v\n", RemoteName, err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
143
fstest/fstests/gen_tests.go
Normal file
143
fstest/fstests/gen_tests.go
Normal file
@@ -0,0 +1,143 @@
|
|||||||
|
// +build ignore
|
||||||
|
|
||||||
|
// Make the test files from fstests.go
|
||||||
|
package main
|
||||||
|
|
||||||
|
import (
|
||||||
|
"bufio"
|
||||||
|
"html/template"
|
||||||
|
"log"
|
||||||
|
"os"
|
||||||
|
"os/exec"
|
||||||
|
"regexp"
|
||||||
|
"strings"
|
||||||
|
)
|
||||||
|
|
||||||
|
// Search fstests.go and return all the test function names
|
||||||
|
func findTestFunctions() []string {
|
||||||
|
fns := []string{}
|
||||||
|
matcher := regexp.MustCompile(`^func\s+(Test.*?)\(`)
|
||||||
|
|
||||||
|
in, err := os.Open("fstests.go")
|
||||||
|
if err != nil {
|
||||||
|
log.Fatalf("Couldn't open fstests.go: %v", err)
|
||||||
|
}
|
||||||
|
defer in.Close()
|
||||||
|
|
||||||
|
scanner := bufio.NewScanner(in)
|
||||||
|
for scanner.Scan() {
|
||||||
|
line := scanner.Text()
|
||||||
|
matches := matcher.FindStringSubmatch(line)
|
||||||
|
if len(matches) > 0 {
|
||||||
|
fns = append(fns, matches[1])
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := scanner.Err(); err != nil {
|
||||||
|
log.Fatalf("Error scanning file: %v", err)
|
||||||
|
}
|
||||||
|
return fns
|
||||||
|
}
|
||||||
|
|
||||||
|
// Data to substitute
|
||||||
|
type Data struct {
|
||||||
|
Regenerate string
|
||||||
|
FsName string
|
||||||
|
UpperFsName string
|
||||||
|
TestName string
|
||||||
|
ObjectName string
|
||||||
|
Fns []string
|
||||||
|
}
|
||||||
|
|
||||||
|
var testProgram = `
|
||||||
|
// Test {{ .UpperFsName }} filesystem interface
|
||||||
|
//
|
||||||
|
// Automatically generated - DO NOT EDIT
|
||||||
|
// Regenerate with: {{ .Regenerate }}
|
||||||
|
package {{ .FsName }}_test
|
||||||
|
|
||||||
|
import (
|
||||||
|
"testing"
|
||||||
|
|
||||||
|
"github.com/ncw/rclone/fs"
|
||||||
|
"github.com/ncw/rclone/fstest/fstests"
|
||||||
|
"github.com/ncw/rclone/{{ .FsName }}"
|
||||||
|
)
|
||||||
|
|
||||||
|
func init() {
|
||||||
|
fstests.NilObject = fs.Object((*{{ .FsName }}.FsObject{{ .ObjectName }})(nil))
|
||||||
|
fstests.RemoteName = "{{ .TestName }}"
|
||||||
|
}
|
||||||
|
|
||||||
|
// Generic tests for the Fs
|
||||||
|
{{ range $fn := .Fns }}func {{ $fn }}(t *testing.T){ fstests.{{ $fn }}(t) }
|
||||||
|
{{ end }}
|
||||||
|
`
|
||||||
|
|
||||||
|
// Generate test file piping it through gofmt
|
||||||
|
func generateTestProgram(t *template.Template, fns []string, Fsname string) {
|
||||||
|
fsname := strings.ToLower(Fsname)
|
||||||
|
TestName := "Test" + Fsname + ":"
|
||||||
|
outfile := "../../" + fsname + "/" + fsname + "_test.go"
|
||||||
|
// Find last capitalised group to be object name
|
||||||
|
matcher := regexp.MustCompile(`([A-Z][a-z0-9]+)$`)
|
||||||
|
matches := matcher.FindStringSubmatch(Fsname)
|
||||||
|
if len(matches) == 0 {
|
||||||
|
log.Fatalf("Couldn't find object name in %q", Fsname)
|
||||||
|
}
|
||||||
|
ObjectName := matches[1]
|
||||||
|
|
||||||
|
if fsname == "local" {
|
||||||
|
TestName = ""
|
||||||
|
}
|
||||||
|
|
||||||
|
data := Data{
|
||||||
|
Regenerate: "go run gen_tests.go or make gen_tests",
|
||||||
|
FsName: fsname,
|
||||||
|
UpperFsName: Fsname,
|
||||||
|
TestName: TestName,
|
||||||
|
ObjectName: ObjectName,
|
||||||
|
Fns: fns,
|
||||||
|
}
|
||||||
|
|
||||||
|
cmd := exec.Command("gofmt")
|
||||||
|
|
||||||
|
log.Printf("Writing %q", outfile)
|
||||||
|
out, err := os.Create(outfile)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
cmd.Stdout = out
|
||||||
|
|
||||||
|
gofmt, err := cmd.StdinPipe()
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
if err = cmd.Start(); err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
if err = t.Execute(gofmt, data); err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
if err = gofmt.Close(); err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
if err = cmd.Wait(); err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
if err = out.Close(); err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func main() {
|
||||||
|
fns := findTestFunctions()
|
||||||
|
t := template.Must(template.New("main").Parse(testProgram))
|
||||||
|
generateTestProgram(t, fns, "Local")
|
||||||
|
generateTestProgram(t, fns, "Swift")
|
||||||
|
generateTestProgram(t, fns, "S3")
|
||||||
|
generateTestProgram(t, fns, "Drive")
|
||||||
|
generateTestProgram(t, fns, "GoogleCloudStorage")
|
||||||
|
generateTestProgram(t, fns, "Dropbox")
|
||||||
|
log.Printf("Done")
|
||||||
|
}
|
||||||
@@ -5,7 +5,6 @@ import (
|
|||||||
"encoding/json"
|
"encoding/json"
|
||||||
"fmt"
|
"fmt"
|
||||||
"log"
|
"log"
|
||||||
"net/http"
|
|
||||||
|
|
||||||
"code.google.com/p/goauth2/oauth"
|
"code.google.com/p/goauth2/oauth"
|
||||||
"github.com/ncw/rclone/fs"
|
"github.com/ncw/rclone/fs"
|
||||||
@@ -82,7 +81,7 @@ func (auth *Auth) newTransport(name string) (*oauth.Transport, error) {
|
|||||||
|
|
||||||
t := &oauth.Transport{
|
t := &oauth.Transport{
|
||||||
Config: config,
|
Config: config,
|
||||||
Transport: http.DefaultTransport,
|
Transport: fs.Config.Transport(),
|
||||||
}
|
}
|
||||||
|
|
||||||
return t, nil
|
return t, nil
|
||||||
|
|||||||
@@ -17,15 +17,14 @@ import (
|
|||||||
"encoding/hex"
|
"encoding/hex"
|
||||||
"fmt"
|
"fmt"
|
||||||
"io"
|
"io"
|
||||||
"mime"
|
|
||||||
"net/http"
|
"net/http"
|
||||||
"path"
|
"path"
|
||||||
"regexp"
|
"regexp"
|
||||||
"strings"
|
"strings"
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
"code.google.com/p/google-api-go-client/googleapi"
|
"google.golang.org/api/googleapi"
|
||||||
"code.google.com/p/google-api-go-client/storage/v1"
|
"google.golang.org/api/storage/v1"
|
||||||
|
|
||||||
"github.com/ncw/rclone/fs"
|
"github.com/ncw/rclone/fs"
|
||||||
"github.com/ncw/rclone/googleauth"
|
"github.com/ncw/rclone/googleauth"
|
||||||
@@ -34,8 +33,8 @@ import (
|
|||||||
const (
|
const (
|
||||||
rcloneClientId = "202264815644.apps.googleusercontent.com"
|
rcloneClientId = "202264815644.apps.googleusercontent.com"
|
||||||
rcloneClientSecret = "X4Z3ca8xfWDb1Voo-F9a7ZxJ"
|
rcloneClientSecret = "X4Z3ca8xfWDb1Voo-F9a7ZxJ"
|
||||||
RFC3339In = time.RFC3339
|
timeFormatIn = time.RFC3339
|
||||||
RFC3339Out = "2006-01-02T15:04:05.000000000Z07:00"
|
timeFormatOut = "2006-01-02T15:04:05.000000000Z07:00"
|
||||||
metaMtime = "mtime" // key to store mtime under in metadata
|
metaMtime = "mtime" // key to store mtime under in metadata
|
||||||
listChunks = 256 // chunk size to read directory listings
|
listChunks = 256 // chunk size to read directory listings
|
||||||
)
|
)
|
||||||
@@ -43,7 +42,7 @@ const (
|
|||||||
var (
|
var (
|
||||||
// Description of how to auth for this app
|
// Description of how to auth for this app
|
||||||
storageAuth = &googleauth.Auth{
|
storageAuth = &googleauth.Auth{
|
||||||
Scope: storage.DevstorageFull_controlScope,
|
Scope: storage.DevstorageFullControlScope,
|
||||||
DefaultClientId: rcloneClientId,
|
DefaultClientId: rcloneClientId,
|
||||||
DefaultClientSecret: rcloneClientSecret,
|
DefaultClientSecret: rcloneClientSecret,
|
||||||
}
|
}
|
||||||
@@ -215,7 +214,7 @@ func NewFs(name, root string) (fs.Fs, error) {
|
|||||||
// Return an FsObject from a path
|
// Return an FsObject from a path
|
||||||
//
|
//
|
||||||
// May return nil if an error occurred
|
// May return nil if an error occurred
|
||||||
func (f *FsStorage) NewFsObjectWithInfo(remote string, info *storage.Object) fs.Object {
|
func (f *FsStorage) newFsObjectWithInfo(remote string, info *storage.Object) fs.Object {
|
||||||
o := &FsObjectStorage{
|
o := &FsObjectStorage{
|
||||||
storage: f,
|
storage: f,
|
||||||
remote: remote,
|
remote: remote,
|
||||||
@@ -236,7 +235,7 @@ func (f *FsStorage) NewFsObjectWithInfo(remote string, info *storage.Object) fs.
|
|||||||
//
|
//
|
||||||
// May return nil if an error occurred
|
// May return nil if an error occurred
|
||||||
func (f *FsStorage) NewFsObject(remote string) fs.Object {
|
func (f *FsStorage) NewFsObject(remote string) fs.Object {
|
||||||
return f.NewFsObjectWithInfo(remote, nil)
|
return f.newFsObjectWithInfo(remote, nil)
|
||||||
}
|
}
|
||||||
|
|
||||||
// list the objects into the function supplied
|
// list the objects into the function supplied
|
||||||
@@ -255,16 +254,23 @@ func (f *FsStorage) list(directories bool, fn func(string, *storage.Object)) {
|
|||||||
fs.Log(f, "Couldn't read bucket %q: %s", f.bucket, err)
|
fs.Log(f, "Couldn't read bucket %q: %s", f.bucket, err)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
for _, object := range objects.Items {
|
if !directories {
|
||||||
if directories && !strings.HasSuffix(object.Name, "/") {
|
for _, object := range objects.Items {
|
||||||
continue
|
if !strings.HasPrefix(object.Name, f.root) {
|
||||||
|
fs.Log(f, "Odd name received %q", object.Name)
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
remote := object.Name[rootLength:]
|
||||||
|
fn(remote, object)
|
||||||
}
|
}
|
||||||
if !strings.HasPrefix(object.Name, f.root) {
|
} else {
|
||||||
fs.Log(f, "Odd name received %q", object.Name)
|
var object storage.Object
|
||||||
continue
|
for _, prefix := range objects.Prefixes {
|
||||||
|
if !strings.HasSuffix(prefix, "/") {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
fn(prefix[:len(prefix)-1], &object)
|
||||||
}
|
}
|
||||||
remote := object.Name[rootLength:]
|
|
||||||
fn(remote, object)
|
|
||||||
}
|
}
|
||||||
if objects.NextPageToken == "" {
|
if objects.NextPageToken == "" {
|
||||||
break
|
break
|
||||||
@@ -286,7 +292,7 @@ func (f *FsStorage) List() fs.ObjectsChan {
|
|||||||
go func() {
|
go func() {
|
||||||
defer close(out)
|
defer close(out)
|
||||||
f.list(false, func(remote string, object *storage.Object) {
|
f.list(false, func(remote string, object *storage.Object) {
|
||||||
if fs := f.NewFsObjectWithInfo(remote, object); fs != nil {
|
if fs := f.newFsObjectWithInfo(remote, object); fs != nil {
|
||||||
out <- fs
|
out <- fs
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
@@ -352,8 +358,8 @@ func (f *FsStorage) ListDir() fs.DirChan {
|
|||||||
// The new object may have been created if an error is returned
|
// The new object may have been created if an error is returned
|
||||||
func (f *FsStorage) Put(in io.Reader, remote string, modTime time.Time, size int64) (fs.Object, error) {
|
func (f *FsStorage) Put(in io.Reader, remote string, modTime time.Time, size int64) (fs.Object, error) {
|
||||||
// Temporary FsObject under construction
|
// Temporary FsObject under construction
|
||||||
fs := &FsObjectStorage{storage: f, remote: remote}
|
o := &FsObjectStorage{storage: f, remote: remote}
|
||||||
return fs, fs.Update(in, modTime, size)
|
return o, o.Update(in, modTime, size)
|
||||||
}
|
}
|
||||||
|
|
||||||
// Mkdir creates the bucket if it doesn't exist
|
// Mkdir creates the bucket if it doesn't exist
|
||||||
@@ -434,7 +440,7 @@ func (o *FsObjectStorage) setMetaData(info *storage.Object) {
|
|||||||
// read mtime out of metadata if available
|
// read mtime out of metadata if available
|
||||||
mtimeString, ok := info.Metadata[metaMtime]
|
mtimeString, ok := info.Metadata[metaMtime]
|
||||||
if ok {
|
if ok {
|
||||||
modTime, err := time.Parse(RFC3339In, mtimeString)
|
modTime, err := time.Parse(timeFormatIn, mtimeString)
|
||||||
if err == nil {
|
if err == nil {
|
||||||
o.modTime = modTime
|
o.modTime = modTime
|
||||||
return
|
return
|
||||||
@@ -444,7 +450,7 @@ func (o *FsObjectStorage) setMetaData(info *storage.Object) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Fallback to the Updated time
|
// Fallback to the Updated time
|
||||||
modTime, err := time.Parse(RFC3339In, info.Updated)
|
modTime, err := time.Parse(timeFormatIn, info.Updated)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
fs.Log(o, "Bad time decode: %v", err)
|
fs.Log(o, "Bad time decode: %v", err)
|
||||||
} else {
|
} else {
|
||||||
@@ -484,7 +490,7 @@ func (o *FsObjectStorage) ModTime() time.Time {
|
|||||||
// Returns metadata for an object
|
// Returns metadata for an object
|
||||||
func metadataFromModTime(modTime time.Time) map[string]string {
|
func metadataFromModTime(modTime time.Time) map[string]string {
|
||||||
metadata := make(map[string]string, 1)
|
metadata := make(map[string]string, 1)
|
||||||
metadata[metaMtime] = modTime.Format(RFC3339Out)
|
metadata[metaMtime] = modTime.Format(timeFormatOut)
|
||||||
return metadata
|
return metadata
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -496,11 +502,12 @@ func (o *FsObjectStorage) SetModTime(modTime time.Time) {
|
|||||||
Name: o.storage.root + o.remote,
|
Name: o.storage.root + o.remote,
|
||||||
Metadata: metadataFromModTime(modTime),
|
Metadata: metadataFromModTime(modTime),
|
||||||
}
|
}
|
||||||
_, err := o.storage.svc.Objects.Patch(o.storage.bucket, o.storage.root+o.remote, &object).Do()
|
newObject, err := o.storage.svc.Objects.Patch(o.storage.bucket, o.storage.root+o.remote, &object).Do()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
fs.Stats.Error()
|
fs.Stats.Error()
|
||||||
fs.Log(o, "Failed to update remote mtime: %s", err)
|
fs.Log(o, "Failed to update remote mtime: %s", err)
|
||||||
}
|
}
|
||||||
|
o.setMetaData(newObject)
|
||||||
}
|
}
|
||||||
|
|
||||||
// Is this object storable
|
// Is this object storable
|
||||||
@@ -530,7 +537,7 @@ func (o *FsObjectStorage) Open() (in io.ReadCloser, err error) {
|
|||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
if res.StatusCode != 200 {
|
if res.StatusCode != 200 {
|
||||||
res.Body.Close()
|
_ = res.Body.Close() // ignore error
|
||||||
return nil, fmt.Errorf("Bad response: %d: %s", res.StatusCode, res.Status)
|
return nil, fmt.Errorf("Bad response: %d: %s", res.StatusCode, res.Status)
|
||||||
}
|
}
|
||||||
return res.Body, nil
|
return res.Body, nil
|
||||||
@@ -540,24 +547,21 @@ func (o *FsObjectStorage) Open() (in io.ReadCloser, err error) {
|
|||||||
//
|
//
|
||||||
// The new object may have been created if an error is returned
|
// The new object may have been created if an error is returned
|
||||||
func (o *FsObjectStorage) Update(in io.Reader, modTime time.Time, size int64) error {
|
func (o *FsObjectStorage) Update(in io.Reader, modTime time.Time, size int64) error {
|
||||||
// Guess the content type
|
|
||||||
contentType := mime.TypeByExtension(path.Ext(o.remote))
|
|
||||||
if contentType == "" {
|
|
||||||
contentType = "application/octet-stream"
|
|
||||||
}
|
|
||||||
|
|
||||||
object := storage.Object{
|
object := storage.Object{
|
||||||
Bucket: o.storage.bucket,
|
Bucket: o.storage.bucket,
|
||||||
Name: o.storage.root + o.remote,
|
Name: o.storage.root + o.remote,
|
||||||
ContentType: contentType,
|
ContentType: fs.MimeType(o),
|
||||||
Size: uint64(size),
|
Size: uint64(size),
|
||||||
Updated: modTime.Format(RFC3339Out), // Doesn't get set
|
Updated: modTime.Format(timeFormatOut), // Doesn't get set
|
||||||
Metadata: metadataFromModTime(modTime),
|
Metadata: metadataFromModTime(modTime),
|
||||||
}
|
}
|
||||||
newObject, err := o.storage.svc.Objects.Insert(o.storage.bucket, &object).Media(in).Name(object.Name).PredefinedAcl(o.storage.objectAcl).Do()
|
newObject, err := o.storage.svc.Objects.Insert(o.storage.bucket, &object).Media(in).Name(object.Name).PredefinedAcl(o.storage.objectAcl).Do()
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
// Set the metadata for the new object while we have it
|
// Set the metadata for the new object while we have it
|
||||||
o.setMetaData(newObject)
|
o.setMetaData(newObject)
|
||||||
return err
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// Remove an object
|
// Remove an object
|
||||||
|
|||||||
53
googlecloudstorage/googlecloudstorage_test.go
Normal file
53
googlecloudstorage/googlecloudstorage_test.go
Normal file
@@ -0,0 +1,53 @@
|
|||||||
|
// Test GoogleCloudStorage filesystem interface
|
||||||
|
//
|
||||||
|
// Automatically generated - DO NOT EDIT
|
||||||
|
// Regenerate with: go run gen_tests.go or make gen_tests
|
||||||
|
package googlecloudstorage_test
|
||||||
|
|
||||||
|
import (
|
||||||
|
"testing"
|
||||||
|
|
||||||
|
"github.com/ncw/rclone/fs"
|
||||||
|
"github.com/ncw/rclone/fstest/fstests"
|
||||||
|
"github.com/ncw/rclone/googlecloudstorage"
|
||||||
|
)
|
||||||
|
|
||||||
|
func init() {
|
||||||
|
fstests.NilObject = fs.Object((*googlecloudstorage.FsObjectStorage)(nil))
|
||||||
|
fstests.RemoteName = "TestGoogleCloudStorage:"
|
||||||
|
}
|
||||||
|
|
||||||
|
// Generic tests for the Fs
|
||||||
|
func TestInit(t *testing.T) { fstests.TestInit(t) }
|
||||||
|
func TestFsString(t *testing.T) { fstests.TestFsString(t) }
|
||||||
|
func TestFsRmdirEmpty(t *testing.T) { fstests.TestFsRmdirEmpty(t) }
|
||||||
|
func TestFsRmdirNotFound(t *testing.T) { fstests.TestFsRmdirNotFound(t) }
|
||||||
|
func TestFsMkdir(t *testing.T) { fstests.TestFsMkdir(t) }
|
||||||
|
func TestFsListEmpty(t *testing.T) { fstests.TestFsListEmpty(t) }
|
||||||
|
func TestFsListDirEmpty(t *testing.T) { fstests.TestFsListDirEmpty(t) }
|
||||||
|
func TestFsNewFsObjectNotFound(t *testing.T) { fstests.TestFsNewFsObjectNotFound(t) }
|
||||||
|
func TestFsPutFile1(t *testing.T) { fstests.TestFsPutFile1(t) }
|
||||||
|
func TestFsPutFile2(t *testing.T) { fstests.TestFsPutFile2(t) }
|
||||||
|
func TestFsListDirFile2(t *testing.T) { fstests.TestFsListDirFile2(t) }
|
||||||
|
func TestFsListDirRoot(t *testing.T) { fstests.TestFsListDirRoot(t) }
|
||||||
|
func TestFsListRoot(t *testing.T) { fstests.TestFsListRoot(t) }
|
||||||
|
func TestFsListFile1(t *testing.T) { fstests.TestFsListFile1(t) }
|
||||||
|
func TestFsNewFsObject(t *testing.T) { fstests.TestFsNewFsObject(t) }
|
||||||
|
func TestFsListFile1and2(t *testing.T) { fstests.TestFsListFile1and2(t) }
|
||||||
|
func TestFsRmdirFull(t *testing.T) { fstests.TestFsRmdirFull(t) }
|
||||||
|
func TestFsPrecision(t *testing.T) { fstests.TestFsPrecision(t) }
|
||||||
|
func TestObjectString(t *testing.T) { fstests.TestObjectString(t) }
|
||||||
|
func TestObjectFs(t *testing.T) { fstests.TestObjectFs(t) }
|
||||||
|
func TestObjectRemote(t *testing.T) { fstests.TestObjectRemote(t) }
|
||||||
|
func TestObjectMd5sum(t *testing.T) { fstests.TestObjectMd5sum(t) }
|
||||||
|
func TestObjectModTime(t *testing.T) { fstests.TestObjectModTime(t) }
|
||||||
|
func TestObjectSetModTime(t *testing.T) { fstests.TestObjectSetModTime(t) }
|
||||||
|
func TestObjectSize(t *testing.T) { fstests.TestObjectSize(t) }
|
||||||
|
func TestObjectOpen(t *testing.T) { fstests.TestObjectOpen(t) }
|
||||||
|
func TestObjectUpdate(t *testing.T) { fstests.TestObjectUpdate(t) }
|
||||||
|
func TestObjectStorable(t *testing.T) { fstests.TestObjectStorable(t) }
|
||||||
|
func TestLimitedFs(t *testing.T) { fstests.TestLimitedFs(t) }
|
||||||
|
func TestLimitedFsNotFound(t *testing.T) { fstests.TestLimitedFsNotFound(t) }
|
||||||
|
func TestObjectRemove(t *testing.T) { fstests.TestObjectRemove(t) }
|
||||||
|
func TestObjectPurge(t *testing.T) { fstests.TestObjectPurge(t) }
|
||||||
|
func TestFinalise(t *testing.T) { fstests.TestFinalise(t) }
|
||||||
@@ -1,6 +1,11 @@
|
|||||||
// Local filesystem interface
|
// Local filesystem interface
|
||||||
package local
|
package local
|
||||||
|
|
||||||
|
// Note that all rclone paths should be / separated. Anything coming
|
||||||
|
// from the filepath module will have \ separators on windows so
|
||||||
|
// should be converted using filepath.ToSlash. Windows is quite happy
|
||||||
|
// with / separators so there is no need to convert them back.
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"crypto/md5"
|
"crypto/md5"
|
||||||
"encoding/hex"
|
"encoding/hex"
|
||||||
@@ -8,12 +13,12 @@ import (
|
|||||||
"hash"
|
"hash"
|
||||||
"io"
|
"io"
|
||||||
"io/ioutil"
|
"io/ioutil"
|
||||||
"log"
|
|
||||||
"os"
|
"os"
|
||||||
"path"
|
"path"
|
||||||
"path/filepath"
|
"path/filepath"
|
||||||
"sync"
|
"sync"
|
||||||
"time"
|
"time"
|
||||||
|
"unicode/utf8"
|
||||||
|
|
||||||
"github.com/ncw/rclone/fs"
|
"github.com/ncw/rclone/fs"
|
||||||
)
|
)
|
||||||
@@ -28,14 +33,15 @@ func init() {
|
|||||||
|
|
||||||
// FsLocal represents a local filesystem rooted at root
|
// FsLocal represents a local filesystem rooted at root
|
||||||
type FsLocal struct {
|
type FsLocal struct {
|
||||||
root string // The root directory
|
root string // The root directory
|
||||||
precisionOk sync.Once // Whether we need to read the precision
|
precisionOk sync.Once // Whether we need to read the precision
|
||||||
precision time.Duration // precision of local filesystem
|
precision time.Duration // precision of local filesystem
|
||||||
|
warned map[string]struct{} // whether we have warned about this string
|
||||||
}
|
}
|
||||||
|
|
||||||
// FsObjectLocal represents a local filesystem object
|
// FsObjectLocal represents a local filesystem object
|
||||||
type FsObjectLocal struct {
|
type FsObjectLocal struct {
|
||||||
local fs.Fs // The Fs this object is part of
|
local *FsLocal // The Fs this object is part of
|
||||||
remote string // The remote path
|
remote string // The remote path
|
||||||
path string // The local path
|
path string // The local path
|
||||||
info os.FileInfo // Interface for file info (always present)
|
info os.FileInfo // Interface for file info (always present)
|
||||||
@@ -46,8 +52,11 @@ type FsObjectLocal struct {
|
|||||||
|
|
||||||
// NewFs contstructs an FsLocal from the path
|
// NewFs contstructs an FsLocal from the path
|
||||||
func NewFs(name, root string) (fs.Fs, error) {
|
func NewFs(name, root string) (fs.Fs, error) {
|
||||||
root = path.Clean(root)
|
root = filepath.ToSlash(path.Clean(root))
|
||||||
f := &FsLocal{root: root}
|
f := &FsLocal{
|
||||||
|
root: root,
|
||||||
|
warned: make(map[string]struct{}),
|
||||||
|
}
|
||||||
// Check to see if this points to a file
|
// Check to see if this points to a file
|
||||||
fi, err := os.Lstat(f.root)
|
fi, err := os.Lstat(f.root)
|
||||||
if err == nil && fi.Mode().IsRegular() {
|
if err == nil && fi.Mode().IsRegular() {
|
||||||
@@ -69,8 +78,9 @@ func (f *FsLocal) String() string {
|
|||||||
// Return an FsObject from a path
|
// Return an FsObject from a path
|
||||||
//
|
//
|
||||||
// May return nil if an error occurred
|
// May return nil if an error occurred
|
||||||
func (f *FsLocal) NewFsObjectWithInfo(remote string, info os.FileInfo) fs.Object {
|
func (f *FsLocal) newFsObjectWithInfo(remote string, info os.FileInfo) fs.Object {
|
||||||
path := filepath.Join(f.root, remote)
|
remote = filepath.ToSlash(remote)
|
||||||
|
path := path.Join(f.root, remote)
|
||||||
o := &FsObjectLocal{local: f, remote: remote, path: path}
|
o := &FsObjectLocal{local: f, remote: remote, path: path}
|
||||||
if info != nil {
|
if info != nil {
|
||||||
o.info = info
|
o.info = info
|
||||||
@@ -88,7 +98,7 @@ func (f *FsLocal) NewFsObjectWithInfo(remote string, info os.FileInfo) fs.Object
|
|||||||
//
|
//
|
||||||
// May return nil if an error occurred
|
// May return nil if an error occurred
|
||||||
func (f *FsLocal) NewFsObject(remote string) fs.Object {
|
func (f *FsLocal) NewFsObject(remote string) fs.Object {
|
||||||
return f.NewFsObjectWithInfo(remote, nil)
|
return f.newFsObjectWithInfo(remote, nil)
|
||||||
}
|
}
|
||||||
|
|
||||||
// List the path returning a channel of FsObjects
|
// List the path returning a channel of FsObjects
|
||||||
@@ -100,19 +110,19 @@ func (f *FsLocal) List() fs.ObjectsChan {
|
|||||||
err := filepath.Walk(f.root, func(path string, fi os.FileInfo, err error) error {
|
err := filepath.Walk(f.root, func(path string, fi os.FileInfo, err error) error {
|
||||||
if err != nil {
|
if err != nil {
|
||||||
fs.Stats.Error()
|
fs.Stats.Error()
|
||||||
log.Printf("Failed to open directory: %s: %s", path, err)
|
fs.Log(f, "Failed to open directory: %s: %s", path, err)
|
||||||
} else {
|
} else {
|
||||||
remote, err := filepath.Rel(f.root, path)
|
remote, err := filepath.Rel(f.root, path)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
fs.Stats.Error()
|
fs.Stats.Error()
|
||||||
log.Printf("Failed to get relative path %s: %s", path, err)
|
fs.Log(f, "Failed to get relative path %s: %s", path, err)
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
if remote == "." {
|
if remote == "." {
|
||||||
return nil
|
return nil
|
||||||
// remote = ""
|
// remote = ""
|
||||||
}
|
}
|
||||||
if fs := f.NewFsObjectWithInfo(remote, fi); fs != nil {
|
if fs := f.newFsObjectWithInfo(remote, fi); fs != nil {
|
||||||
if fs.Storable() {
|
if fs.Storable() {
|
||||||
out <- fs
|
out <- fs
|
||||||
}
|
}
|
||||||
@@ -122,13 +132,27 @@ func (f *FsLocal) List() fs.ObjectsChan {
|
|||||||
})
|
})
|
||||||
if err != nil {
|
if err != nil {
|
||||||
fs.Stats.Error()
|
fs.Stats.Error()
|
||||||
log.Printf("Failed to open directory: %s: %s", f.root, err)
|
fs.Log(f, "Failed to open directory: %s: %s", f.root, err)
|
||||||
}
|
}
|
||||||
close(out)
|
close(out)
|
||||||
}()
|
}()
|
||||||
return out
|
return out
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// CleanUtf8 makes string a valid UTF-8 string
|
||||||
|
//
|
||||||
|
// Any invalid UTF-8 characters will be replaced with utf8.RuneError
|
||||||
|
func (f *FsLocal) cleanUtf8(name string) string {
|
||||||
|
if utf8.ValidString(name) {
|
||||||
|
return name
|
||||||
|
}
|
||||||
|
if _, ok := f.warned[name]; !ok {
|
||||||
|
fs.Debug(f, "Replacing invalid UTF-8 characters in %q", name)
|
||||||
|
f.warned[name] = struct{}{}
|
||||||
|
}
|
||||||
|
return string([]rune(name))
|
||||||
|
}
|
||||||
|
|
||||||
// Walk the path returning a channel of FsObjects
|
// Walk the path returning a channel of FsObjects
|
||||||
func (f *FsLocal) ListDir() fs.DirChan {
|
func (f *FsLocal) ListDir() fs.DirChan {
|
||||||
out := make(fs.DirChan, fs.Config.Checkers)
|
out := make(fs.DirChan, fs.Config.Checkers)
|
||||||
@@ -137,12 +161,12 @@ func (f *FsLocal) ListDir() fs.DirChan {
|
|||||||
items, err := ioutil.ReadDir(f.root)
|
items, err := ioutil.ReadDir(f.root)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
fs.Stats.Error()
|
fs.Stats.Error()
|
||||||
log.Printf("Couldn't find read directory: %s", err)
|
fs.Log(f, "Couldn't find read directory: %s", err)
|
||||||
} else {
|
} else {
|
||||||
for _, item := range items {
|
for _, item := range items {
|
||||||
if item.IsDir() {
|
if item.IsDir() {
|
||||||
dir := &fs.Dir{
|
dir := &fs.Dir{
|
||||||
Name: item.Name(),
|
Name: f.cleanUtf8(item.Name()),
|
||||||
When: item.ModTime(),
|
When: item.ModTime(),
|
||||||
Bytes: 0,
|
Bytes: 0,
|
||||||
Count: 0,
|
Count: 0,
|
||||||
@@ -152,7 +176,7 @@ func (f *FsLocal) ListDir() fs.DirChan {
|
|||||||
err := filepath.Walk(dirpath, func(path string, fi os.FileInfo, err error) error {
|
err := filepath.Walk(dirpath, func(path string, fi os.FileInfo, err error) error {
|
||||||
if err != nil {
|
if err != nil {
|
||||||
fs.Stats.Error()
|
fs.Stats.Error()
|
||||||
log.Printf("Failed to open directory: %s: %s", path, err)
|
fs.Log(f, "Failed to open directory: %s: %s", path, err)
|
||||||
} else {
|
} else {
|
||||||
dir.Count += 1
|
dir.Count += 1
|
||||||
dir.Bytes += fi.Size()
|
dir.Bytes += fi.Size()
|
||||||
@@ -161,7 +185,7 @@ func (f *FsLocal) ListDir() fs.DirChan {
|
|||||||
})
|
})
|
||||||
if err != nil {
|
if err != nil {
|
||||||
fs.Stats.Error()
|
fs.Stats.Error()
|
||||||
log.Printf("Failed to open directory: %s: %s", dirpath, err)
|
fs.Log(f, "Failed to open directory: %s: %s", dirpath, err)
|
||||||
}
|
}
|
||||||
out <- dir
|
out <- dir
|
||||||
}
|
}
|
||||||
@@ -174,7 +198,7 @@ func (f *FsLocal) ListDir() fs.DirChan {
|
|||||||
|
|
||||||
// Puts the FsObject to the local filesystem
|
// Puts the FsObject to the local filesystem
|
||||||
func (f *FsLocal) Put(in io.Reader, remote string, modTime time.Time, size int64) (fs.Object, error) {
|
func (f *FsLocal) Put(in io.Reader, remote string, modTime time.Time, size int64) (fs.Object, error) {
|
||||||
dstPath := filepath.Join(f.root, remote)
|
dstPath := path.Join(f.root, remote)
|
||||||
// Temporary FsObject under construction - info filled in by Update()
|
// Temporary FsObject under construction - info filled in by Update()
|
||||||
o := &FsObjectLocal{local: f, remote: remote, path: dstPath}
|
o := &FsObjectLocal{local: f, remote: remote, path: dstPath}
|
||||||
err := o.Update(in, modTime, size)
|
err := o.Update(in, modTime, size)
|
||||||
@@ -218,12 +242,15 @@ func (f *FsLocal) readPrecision() (precision time.Duration) {
|
|||||||
}
|
}
|
||||||
path := fd.Name()
|
path := fd.Name()
|
||||||
// fmt.Println("Created temp file", path)
|
// fmt.Println("Created temp file", path)
|
||||||
fd.Close()
|
err = fd.Close()
|
||||||
|
if err != nil {
|
||||||
|
return time.Second
|
||||||
|
}
|
||||||
|
|
||||||
// Delete it on return
|
// Delete it on return
|
||||||
defer func() {
|
defer func() {
|
||||||
// fmt.Println("Remove temp file")
|
// fmt.Println("Remove temp file")
|
||||||
os.Remove(path)
|
_ = os.Remove(path) // ignore error
|
||||||
}()
|
}()
|
||||||
|
|
||||||
// Find the minimum duration we can detect
|
// Find the minimum duration we can detect
|
||||||
@@ -259,6 +286,13 @@ func (f *FsLocal) readPrecision() (precision time.Duration) {
|
|||||||
// deleting all the files quicker than just running Remove() on the
|
// deleting all the files quicker than just running Remove() on the
|
||||||
// result of List()
|
// result of List()
|
||||||
func (f *FsLocal) Purge() error {
|
func (f *FsLocal) Purge() error {
|
||||||
|
fi, err := os.Lstat(f.root)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
if !fi.Mode().IsDir() {
|
||||||
|
return fmt.Errorf("Can't Purge non directory: %q", f.root)
|
||||||
|
}
|
||||||
return os.RemoveAll(f.root)
|
return os.RemoveAll(f.root)
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -279,7 +313,7 @@ func (o *FsObjectLocal) String() string {
|
|||||||
|
|
||||||
// Return the remote path
|
// Return the remote path
|
||||||
func (o *FsObjectLocal) Remote() string {
|
func (o *FsObjectLocal) Remote() string {
|
||||||
return o.remote
|
return o.local.cleanUtf8(o.remote)
|
||||||
}
|
}
|
||||||
|
|
||||||
// Md5sum calculates the Md5sum of a file returning a lowercase hex string
|
// Md5sum calculates the Md5sum of a file returning a lowercase hex string
|
||||||
@@ -325,6 +359,13 @@ func (o *FsObjectLocal) SetModTime(modTime time.Time) {
|
|||||||
err := os.Chtimes(o.path, modTime, modTime)
|
err := os.Chtimes(o.path, modTime, modTime)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
fs.Debug(o, "Failed to set mtime on file: %s", err)
|
fs.Debug(o, "Failed to set mtime on file: %s", err)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
// Re-read metadata
|
||||||
|
err = o.lstat()
|
||||||
|
if err != nil {
|
||||||
|
fs.Debug(o, "Failed to stat: %s", err)
|
||||||
|
return
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -335,7 +376,7 @@ func (o *FsObjectLocal) Storable() bool {
|
|||||||
fs.Debug(o, "Can't transfer non file/directory")
|
fs.Debug(o, "Can't transfer non file/directory")
|
||||||
return false
|
return false
|
||||||
} else if mode&os.ModeDir != 0 {
|
} else if mode&os.ModeDir != 0 {
|
||||||
fs.Debug(o, "FIXME Skipping directory")
|
// fs.Debug(o, "Skipping directory")
|
||||||
return false
|
return false
|
||||||
}
|
}
|
||||||
return true
|
return true
|
||||||
|
|||||||
53
local/local_test.go
Normal file
53
local/local_test.go
Normal file
@@ -0,0 +1,53 @@
|
|||||||
|
// Test Local filesystem interface
|
||||||
|
//
|
||||||
|
// Automatically generated - DO NOT EDIT
|
||||||
|
// Regenerate with: go run gen_tests.go or make gen_tests
|
||||||
|
package local_test
|
||||||
|
|
||||||
|
import (
|
||||||
|
"testing"
|
||||||
|
|
||||||
|
"github.com/ncw/rclone/fs"
|
||||||
|
"github.com/ncw/rclone/fstest/fstests"
|
||||||
|
"github.com/ncw/rclone/local"
|
||||||
|
)
|
||||||
|
|
||||||
|
func init() {
|
||||||
|
fstests.NilObject = fs.Object((*local.FsObjectLocal)(nil))
|
||||||
|
fstests.RemoteName = ""
|
||||||
|
}
|
||||||
|
|
||||||
|
// Generic tests for the Fs
|
||||||
|
func TestInit(t *testing.T) { fstests.TestInit(t) }
|
||||||
|
func TestFsString(t *testing.T) { fstests.TestFsString(t) }
|
||||||
|
func TestFsRmdirEmpty(t *testing.T) { fstests.TestFsRmdirEmpty(t) }
|
||||||
|
func TestFsRmdirNotFound(t *testing.T) { fstests.TestFsRmdirNotFound(t) }
|
||||||
|
func TestFsMkdir(t *testing.T) { fstests.TestFsMkdir(t) }
|
||||||
|
func TestFsListEmpty(t *testing.T) { fstests.TestFsListEmpty(t) }
|
||||||
|
func TestFsListDirEmpty(t *testing.T) { fstests.TestFsListDirEmpty(t) }
|
||||||
|
func TestFsNewFsObjectNotFound(t *testing.T) { fstests.TestFsNewFsObjectNotFound(t) }
|
||||||
|
func TestFsPutFile1(t *testing.T) { fstests.TestFsPutFile1(t) }
|
||||||
|
func TestFsPutFile2(t *testing.T) { fstests.TestFsPutFile2(t) }
|
||||||
|
func TestFsListDirFile2(t *testing.T) { fstests.TestFsListDirFile2(t) }
|
||||||
|
func TestFsListDirRoot(t *testing.T) { fstests.TestFsListDirRoot(t) }
|
||||||
|
func TestFsListRoot(t *testing.T) { fstests.TestFsListRoot(t) }
|
||||||
|
func TestFsListFile1(t *testing.T) { fstests.TestFsListFile1(t) }
|
||||||
|
func TestFsNewFsObject(t *testing.T) { fstests.TestFsNewFsObject(t) }
|
||||||
|
func TestFsListFile1and2(t *testing.T) { fstests.TestFsListFile1and2(t) }
|
||||||
|
func TestFsRmdirFull(t *testing.T) { fstests.TestFsRmdirFull(t) }
|
||||||
|
func TestFsPrecision(t *testing.T) { fstests.TestFsPrecision(t) }
|
||||||
|
func TestObjectString(t *testing.T) { fstests.TestObjectString(t) }
|
||||||
|
func TestObjectFs(t *testing.T) { fstests.TestObjectFs(t) }
|
||||||
|
func TestObjectRemote(t *testing.T) { fstests.TestObjectRemote(t) }
|
||||||
|
func TestObjectMd5sum(t *testing.T) { fstests.TestObjectMd5sum(t) }
|
||||||
|
func TestObjectModTime(t *testing.T) { fstests.TestObjectModTime(t) }
|
||||||
|
func TestObjectSetModTime(t *testing.T) { fstests.TestObjectSetModTime(t) }
|
||||||
|
func TestObjectSize(t *testing.T) { fstests.TestObjectSize(t) }
|
||||||
|
func TestObjectOpen(t *testing.T) { fstests.TestObjectOpen(t) }
|
||||||
|
func TestObjectUpdate(t *testing.T) { fstests.TestObjectUpdate(t) }
|
||||||
|
func TestObjectStorable(t *testing.T) { fstests.TestObjectStorable(t) }
|
||||||
|
func TestLimitedFs(t *testing.T) { fstests.TestLimitedFs(t) }
|
||||||
|
func TestLimitedFsNotFound(t *testing.T) { fstests.TestLimitedFsNotFound(t) }
|
||||||
|
func TestObjectRemove(t *testing.T) { fstests.TestObjectRemove(t) }
|
||||||
|
func TestObjectPurge(t *testing.T) { fstests.TestObjectPurge(t) }
|
||||||
|
func TestFinalise(t *testing.T) { fstests.TestFinalise(t) }
|
||||||
77
make_manual.py
Executable file
77
make_manual.py
Executable file
@@ -0,0 +1,77 @@
|
|||||||
|
#!/usr/bin/python
|
||||||
|
"""
|
||||||
|
Make single page versions of the documentation for release and
|
||||||
|
conversion into man pages etc.
|
||||||
|
"""
|
||||||
|
|
||||||
|
import os
|
||||||
|
import re
|
||||||
|
from datetime import datetime
|
||||||
|
|
||||||
|
docpath = "docs/content"
|
||||||
|
outfile = "MANUAL.md"
|
||||||
|
|
||||||
|
# Order to add docs segments to make outfile
|
||||||
|
docs = [
|
||||||
|
"about.md",
|
||||||
|
"install.md",
|
||||||
|
"docs.md",
|
||||||
|
"drive.md",
|
||||||
|
"s3.md",
|
||||||
|
"swift.md",
|
||||||
|
"dropbox.md",
|
||||||
|
"googlecloudstorage.md",
|
||||||
|
"local.md",
|
||||||
|
"changelog.md",
|
||||||
|
"bugs.md",
|
||||||
|
"licence.md",
|
||||||
|
"authors.md",
|
||||||
|
"contact.md",
|
||||||
|
]
|
||||||
|
|
||||||
|
# Docs which aren't made into outfile
|
||||||
|
ignore_docs = [
|
||||||
|
"downloads.md",
|
||||||
|
]
|
||||||
|
|
||||||
|
def read_doc(doc):
|
||||||
|
"""Read file as a string"""
|
||||||
|
path = os.path.join(docpath, doc)
|
||||||
|
with open(path) as fd:
|
||||||
|
contents = fd.read()
|
||||||
|
parts = contents.split("---\n", 2)
|
||||||
|
if len(parts) != 3:
|
||||||
|
raise ValueError("Couldn't find --- markers: found %d parts" % len(parts))
|
||||||
|
contents = parts[2].strip()+"\n\n"
|
||||||
|
# Remove icons
|
||||||
|
contents = re.sub(r'<i class="fa.*?</i>\s*', "", contents)
|
||||||
|
# Make [...](/links/) absolute
|
||||||
|
contents = re.sub(r'\((\/.*?\/)\)', r"(http://rclone.org\1)", contents)
|
||||||
|
return contents
|
||||||
|
|
||||||
|
def check_docs(docpath):
|
||||||
|
"""Check all the docs are in docpath"""
|
||||||
|
files = set(f for f in os.listdir(docpath) if f.endswith(".md"))
|
||||||
|
files -= set(ignore_docs)
|
||||||
|
docs_set = set(docs)
|
||||||
|
if files == docs_set:
|
||||||
|
return
|
||||||
|
print "Files on disk but not in docs variable: %s" % ", ".join(files - docs_set)
|
||||||
|
print "Files in docs variable but not on disk: %s" % ", ".join(docs_set - files)
|
||||||
|
raise ValueError("Missing files")
|
||||||
|
|
||||||
|
def main():
|
||||||
|
check_docs(docpath)
|
||||||
|
with open(outfile, "w") as out:
|
||||||
|
out.write("""\
|
||||||
|
%% rclone(1) User Manual
|
||||||
|
%% Nick Craig-Wood
|
||||||
|
%% %s
|
||||||
|
|
||||||
|
""" % datetime.now().strftime("%b %d, %Y"))
|
||||||
|
for doc in docs:
|
||||||
|
out.write(read_doc(doc))
|
||||||
|
print "Written '%s'" % outfile
|
||||||
|
|
||||||
|
if __name__ == "__main__":
|
||||||
|
main()
|
||||||
62
notes.txt
62
notes.txt
@@ -1,3 +1,26 @@
|
|||||||
|
Perhaps make Md5sum() and Modtime() optional. Define the zero values
|
||||||
|
"" and 0. Make it so we can support remotes which can't do those.
|
||||||
|
|
||||||
|
Fix the docs
|
||||||
|
* factor the README.md into the docs directory
|
||||||
|
* create it as part of make by assembling other parts
|
||||||
|
* write long docs about each flag
|
||||||
|
|
||||||
|
Change lsd command so it doesn't show -1
|
||||||
|
* Make sure all Fses show -1 for objects Zero for dates etc
|
||||||
|
* Make test?
|
||||||
|
|
||||||
|
Put the TestRemote names into the Fs description
|
||||||
|
Make test_all.sh use the TestRemote name automatically
|
||||||
|
|
||||||
|
Run errcheck and go vet in the make file
|
||||||
|
.. Also race detector?
|
||||||
|
.. go tool vet -shadow
|
||||||
|
|
||||||
|
Get rid of Storable?
|
||||||
|
|
||||||
|
Write developer manual
|
||||||
|
|
||||||
Todo
|
Todo
|
||||||
* FIXME: More -dry-run checks for object transfer
|
* FIXME: More -dry-run checks for object transfer
|
||||||
* Might be quicker to check md5sums first? for swift <-> swift certainly, and maybe for small files
|
* Might be quicker to check md5sums first? for swift <-> swift certainly, and maybe for small files
|
||||||
@@ -5,24 +28,23 @@ Todo
|
|||||||
* if object.PseudoDirectory {
|
* if object.PseudoDirectory {
|
||||||
* fmt.Printf("%9s %19s %s\n", "Directory", "-", fs.Remote())
|
* fmt.Printf("%9s %19s %s\n", "Directory", "-", fs.Remote())
|
||||||
* Make Account wrapper
|
* Make Account wrapper
|
||||||
* limit bandwidth for a pool of all individual connectinos
|
|
||||||
* do timeouts by setting a limit, seeing whether io has happened
|
|
||||||
and resetting it if it has
|
|
||||||
* make Account do progress meter
|
* make Account do progress meter
|
||||||
* Make logging controllable with flags (mostly done)
|
|
||||||
* -timeout: Make all timeouts be settable with command line parameters
|
* -timeout: Make all timeouts be settable with command line parameters
|
||||||
* Windows paths? Do we need to translate / and \?
|
|
||||||
* Make a fs.Errorf and count errors and log them at a different level
|
|
||||||
* Add max object size to fs metadata - 5GB for swift, infinite for local, ? for s3
|
* Add max object size to fs metadata - 5GB for swift, infinite for local, ? for s3
|
||||||
* tie into -max-size flag
|
* tie into -max-size flag
|
||||||
* FIXME Make NewFs to return err.IsAnObject so can put the LimitedFs
|
* FIXME Make NewFs to return err.IsAnObject so can put the LimitedFs
|
||||||
creation in common code? Or try for as much as possible?
|
creation in common code? Or try for as much as possible?
|
||||||
* FIXME Account all the transactons (ls etc) using a different
|
* FIXME Account all the transactons (ls etc) using a different
|
||||||
Roundtripper wrapper which wraps the transactions?
|
Roundtripper wrapper which wraps the transactions?
|
||||||
* FIXME write tests for local file system
|
|
||||||
* FIXME implement tests for single file operations in rclonetest
|
More rsync features
|
||||||
* Need to make directory objects otherwise can't upload an empty directory
|
* include
|
||||||
* Or could upload empty directories only?
|
* exclude
|
||||||
|
* max size
|
||||||
|
* -c, --checksum skip based on checksum, not mod-time & size
|
||||||
|
|
||||||
|
Ideas for flags
|
||||||
|
* --retries N flag which would make rclone retry a sync until successful or it tried N times.
|
||||||
|
|
||||||
Ideas
|
Ideas
|
||||||
* could do encryption - put IV into metadata?
|
* could do encryption - put IV into metadata?
|
||||||
@@ -35,26 +57,6 @@ Ideas
|
|||||||
* control times sync (which is slow with some remotes) with -a --archive flag?
|
* control times sync (which is slow with some remotes) with -a --archive flag?
|
||||||
* Copy a glob pattern - could do with LimitedFs
|
* Copy a glob pattern - could do with LimitedFs
|
||||||
|
|
||||||
s3
|
|
||||||
* Can maybe set last modified?
|
|
||||||
* https://forums.aws.amazon.com/message.jspa?messageID=214062
|
|
||||||
* Otherwise can set metadata
|
|
||||||
* Returns etag and last modified in bucket list
|
|
||||||
|
|
||||||
Bugs
|
Bugs
|
||||||
* Non verbose - not sure number transferred got counted up? CHECK
|
* Non verbose - not sure number transferred got counted up? CHECK
|
||||||
* When doing copy it recurses the whole of the destination FS which isn't necessary
|
* When doing copy it recurses the whole of the destination FS which isn't necessary
|
||||||
|
|
||||||
Making a release
|
|
||||||
* go install -v ./...
|
|
||||||
* go test ./...
|
|
||||||
* rclonetest/test.sh
|
|
||||||
* make tag
|
|
||||||
* edit README.md
|
|
||||||
* git commit fs/version.go README.md docs/content/downloads.md
|
|
||||||
* make retag
|
|
||||||
* . ~/bin/go-cross
|
|
||||||
* make cross
|
|
||||||
* make upload
|
|
||||||
* make upload_website
|
|
||||||
* git push --tags origin master
|
|
||||||
|
|||||||
76
rclone.go
76
rclone.go
@@ -28,8 +28,9 @@ import (
|
|||||||
var (
|
var (
|
||||||
// Flags
|
// Flags
|
||||||
cpuprofile = pflag.StringP("cpuprofile", "", "", "Write cpu profile to file")
|
cpuprofile = pflag.StringP("cpuprofile", "", "", "Write cpu profile to file")
|
||||||
statsInterval = pflag.DurationP("stats", "", time.Minute*1, "Interval to print stats")
|
statsInterval = pflag.DurationP("stats", "", time.Minute*1, "Interval to print stats (0 to disable)")
|
||||||
version = pflag.BoolP("version", "V", false, "Print the version number")
|
version = pflag.BoolP("version", "V", false, "Print the version number")
|
||||||
|
logFile = pflag.StringP("log-file", "", "", "Log everything to this file")
|
||||||
)
|
)
|
||||||
|
|
||||||
type Command struct {
|
type Command struct {
|
||||||
@@ -58,10 +59,10 @@ func (cmd *Command) checkArgs(args []string) {
|
|||||||
var Commands = []Command{
|
var Commands = []Command{
|
||||||
{
|
{
|
||||||
Name: "copy",
|
Name: "copy",
|
||||||
ArgsHelp: "source://path dest://path",
|
ArgsHelp: "source:path dest:path",
|
||||||
Help: `
|
Help: `
|
||||||
Copy the source to the destination. Doesn't transfer
|
Copy the source to the destination. Doesn't transfer
|
||||||
unchanged files, testing first by modification time then by
|
unchanged files, testing by size and modification time or
|
||||||
MD5SUM. Doesn't delete files from the destination.`,
|
MD5SUM. Doesn't delete files from the destination.`,
|
||||||
Run: func(fdst, fsrc fs.Fs) {
|
Run: func(fdst, fsrc fs.Fs) {
|
||||||
err := fs.Sync(fdst, fsrc, false)
|
err := fs.Sync(fdst, fsrc, false)
|
||||||
@@ -74,13 +75,13 @@ var Commands = []Command{
|
|||||||
},
|
},
|
||||||
{
|
{
|
||||||
Name: "sync",
|
Name: "sync",
|
||||||
ArgsHelp: "source://path dest://path",
|
ArgsHelp: "source:path dest:path",
|
||||||
Help: `
|
Help: `
|
||||||
Sync the source to the destination. Doesn't transfer
|
Sync the source to the destination, changing the destination
|
||||||
unchanged files, testing first by modification time then by
|
only. Doesn't transfer unchanged files, testing by size and
|
||||||
MD5SUM. Deletes any files that exist in source that don't
|
modification time or MD5SUM. Destination is updated to match
|
||||||
exist in destination. Since this can cause data loss, test
|
source, including deleting files if necessary. Since this can
|
||||||
first with the --dry-run flag.`,
|
cause data loss, test first with the --dry-run flag.`,
|
||||||
Run: func(fdst, fsrc fs.Fs) {
|
Run: func(fdst, fsrc fs.Fs) {
|
||||||
err := fs.Sync(fdst, fsrc, true)
|
err := fs.Sync(fdst, fsrc, true)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
@@ -92,11 +93,11 @@ var Commands = []Command{
|
|||||||
},
|
},
|
||||||
{
|
{
|
||||||
Name: "ls",
|
Name: "ls",
|
||||||
ArgsHelp: "[remote://path]",
|
ArgsHelp: "[remote:path]",
|
||||||
Help: `
|
Help: `
|
||||||
List all the objects in the the path with size and path.`,
|
List all the objects in the the path with size and path.`,
|
||||||
Run: func(fdst, fsrc fs.Fs) {
|
Run: func(fdst, fsrc fs.Fs) {
|
||||||
err := fs.List(fdst)
|
err := fs.List(fdst, os.Stdout)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Fatalf("Failed to list: %v", err)
|
log.Fatalf("Failed to list: %v", err)
|
||||||
}
|
}
|
||||||
@@ -106,11 +107,11 @@ var Commands = []Command{
|
|||||||
},
|
},
|
||||||
{
|
{
|
||||||
Name: "lsd",
|
Name: "lsd",
|
||||||
ArgsHelp: "[remote://path]",
|
ArgsHelp: "[remote:path]",
|
||||||
Help: `
|
Help: `
|
||||||
List all directories/containers/buckets in the the path.`,
|
List all directories/containers/buckets in the the path.`,
|
||||||
Run: func(fdst, fsrc fs.Fs) {
|
Run: func(fdst, fsrc fs.Fs) {
|
||||||
err := fs.ListDir(fdst)
|
err := fs.ListDir(fdst, os.Stdout)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Fatalf("Failed to listdir: %v", err)
|
log.Fatalf("Failed to listdir: %v", err)
|
||||||
}
|
}
|
||||||
@@ -120,11 +121,12 @@ var Commands = []Command{
|
|||||||
},
|
},
|
||||||
{
|
{
|
||||||
Name: "lsl",
|
Name: "lsl",
|
||||||
ArgsHelp: "[remote://path]",
|
ArgsHelp: "[remote:path]",
|
||||||
Help: `
|
Help: `
|
||||||
List all the objects in the the path with modification time, size and path.`,
|
List all the objects in the the path with modification time,
|
||||||
|
size and path.`,
|
||||||
Run: func(fdst, fsrc fs.Fs) {
|
Run: func(fdst, fsrc fs.Fs) {
|
||||||
err := fs.ListLong(fdst)
|
err := fs.ListLong(fdst, os.Stdout)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Fatalf("Failed to list long: %v", err)
|
log.Fatalf("Failed to list long: %v", err)
|
||||||
}
|
}
|
||||||
@@ -134,11 +136,12 @@ var Commands = []Command{
|
|||||||
},
|
},
|
||||||
{
|
{
|
||||||
Name: "md5sum",
|
Name: "md5sum",
|
||||||
ArgsHelp: "[remote://path]",
|
ArgsHelp: "[remote:path]",
|
||||||
Help: `
|
Help: `
|
||||||
Produces an md5sum file for all the objects in the path.`,
|
Produces an md5sum file for all the objects in the path. This
|
||||||
|
is in the same format as the standard md5sum tool produces.`,
|
||||||
Run: func(fdst, fsrc fs.Fs) {
|
Run: func(fdst, fsrc fs.Fs) {
|
||||||
err := fs.Md5sum(fdst)
|
err := fs.Md5sum(fdst, os.Stdout)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Fatalf("Failed to list: %v", err)
|
log.Fatalf("Failed to list: %v", err)
|
||||||
}
|
}
|
||||||
@@ -148,7 +151,7 @@ var Commands = []Command{
|
|||||||
},
|
},
|
||||||
{
|
{
|
||||||
Name: "mkdir",
|
Name: "mkdir",
|
||||||
ArgsHelp: "remote://path",
|
ArgsHelp: "remote:path",
|
||||||
Help: `
|
Help: `
|
||||||
Make the path if it doesn't already exist`,
|
Make the path if it doesn't already exist`,
|
||||||
Run: func(fdst, fsrc fs.Fs) {
|
Run: func(fdst, fsrc fs.Fs) {
|
||||||
@@ -162,7 +165,7 @@ var Commands = []Command{
|
|||||||
},
|
},
|
||||||
{
|
{
|
||||||
Name: "rmdir",
|
Name: "rmdir",
|
||||||
ArgsHelp: "remote://path",
|
ArgsHelp: "remote:path",
|
||||||
Help: `
|
Help: `
|
||||||
Remove the path. Note that you can't remove a path with
|
Remove the path. Note that you can't remove a path with
|
||||||
objects in it, use purge for that.`,
|
objects in it, use purge for that.`,
|
||||||
@@ -177,7 +180,7 @@ var Commands = []Command{
|
|||||||
},
|
},
|
||||||
{
|
{
|
||||||
Name: "purge",
|
Name: "purge",
|
||||||
ArgsHelp: "remote://path",
|
ArgsHelp: "remote:path",
|
||||||
Help: `
|
Help: `
|
||||||
Remove the path and all of its contents.`,
|
Remove the path and all of its contents.`,
|
||||||
Run: func(fdst, fsrc fs.Fs) {
|
Run: func(fdst, fsrc fs.Fs) {
|
||||||
@@ -191,7 +194,7 @@ var Commands = []Command{
|
|||||||
},
|
},
|
||||||
{
|
{
|
||||||
Name: "check",
|
Name: "check",
|
||||||
ArgsHelp: "source://path dest://path",
|
ArgsHelp: "source:path dest:path",
|
||||||
Help: `
|
Help: `
|
||||||
Checks the files in the source and destination match. It
|
Checks the files in the source and destination match. It
|
||||||
compares sizes and MD5SUMs and prints a report of files which
|
compares sizes and MD5SUMs and prints a report of files which
|
||||||
@@ -240,7 +243,8 @@ Subcommands:
|
|||||||
fmt.Fprintf(os.Stderr, "Options:\n")
|
fmt.Fprintf(os.Stderr, "Options:\n")
|
||||||
pflag.PrintDefaults()
|
pflag.PrintDefaults()
|
||||||
fmt.Fprintf(os.Stderr, `
|
fmt.Fprintf(os.Stderr, `
|
||||||
It is only necessary to use a unique prefix of the subcommand, eg 'up' for 'upload'.
|
It is only necessary to use a unique prefix of the subcommand, eg 'up'
|
||||||
|
for 'upload'.
|
||||||
`)
|
`)
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -265,7 +269,11 @@ func ParseFlags() {
|
|||||||
fs.Stats.Error()
|
fs.Stats.Error()
|
||||||
log.Fatal(err)
|
log.Fatal(err)
|
||||||
}
|
}
|
||||||
pprof.StartCPUProfile(f)
|
err = pprof.StartCPUProfile(f)
|
||||||
|
if err != nil {
|
||||||
|
fs.Stats.Error()
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
defer pprof.StopCPUProfile()
|
defer pprof.StopCPUProfile()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -319,6 +327,9 @@ func NewFs(remote string) fs.Fs {
|
|||||||
|
|
||||||
// Print the stats every statsInterval
|
// Print the stats every statsInterval
|
||||||
func StartStats() {
|
func StartStats() {
|
||||||
|
if *statsInterval <= 0 {
|
||||||
|
return
|
||||||
|
}
|
||||||
go func() {
|
go func() {
|
||||||
ch := time.Tick(*statsInterval)
|
ch := time.Tick(*statsInterval)
|
||||||
for {
|
for {
|
||||||
@@ -336,6 +347,17 @@ func main() {
|
|||||||
}
|
}
|
||||||
command, args := ParseCommand()
|
command, args := ParseCommand()
|
||||||
|
|
||||||
|
// Log file output
|
||||||
|
if *logFile != "" {
|
||||||
|
f, err := os.OpenFile(*logFile, os.O_WRONLY|os.O_CREATE|os.O_APPEND, 0640)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatalf("Failed to open log file: %v", err)
|
||||||
|
}
|
||||||
|
f.Seek(0, os.SEEK_END)
|
||||||
|
log.SetOutput(f)
|
||||||
|
redirectStderr(f)
|
||||||
|
}
|
||||||
|
|
||||||
// Make source and destination fs
|
// Make source and destination fs
|
||||||
var fdst, fsrc fs.Fs
|
var fdst, fsrc fs.Fs
|
||||||
if len(args) >= 1 {
|
if len(args) >= 1 {
|
||||||
@@ -356,10 +378,10 @@ func main() {
|
|||||||
if command.Run != nil {
|
if command.Run != nil {
|
||||||
command.Run(fdst, fsrc)
|
command.Run(fdst, fsrc)
|
||||||
if !command.NoStats {
|
if !command.NoStats {
|
||||||
fmt.Println(fs.Stats)
|
fmt.Fprintln(os.Stderr, fs.Stats)
|
||||||
}
|
}
|
||||||
if fs.Config.Verbose {
|
if fs.Config.Verbose {
|
||||||
log.Printf("*** Go routines at exit %d\n", runtime.NumGoroutine())
|
fs.Debug(nil, "Go routines at exit %d\n", runtime.NumGoroutine())
|
||||||
}
|
}
|
||||||
if fs.Stats.Errored() {
|
if fs.Stats.Errored() {
|
||||||
os.Exit(1)
|
os.Exit(1)
|
||||||
|
|||||||
@@ -1,419 +0,0 @@
|
|||||||
// Test rclone by doing real transactions to a storage provider to and
|
|
||||||
// from the local disk
|
|
||||||
|
|
||||||
package main
|
|
||||||
|
|
||||||
import (
|
|
||||||
"fmt"
|
|
||||||
"io/ioutil"
|
|
||||||
"log"
|
|
||||||
"math/rand"
|
|
||||||
"os"
|
|
||||||
"path"
|
|
||||||
"strings"
|
|
||||||
"time"
|
|
||||||
|
|
||||||
"github.com/ncw/rclone/fs"
|
|
||||||
"github.com/ogier/pflag"
|
|
||||||
|
|
||||||
// Active file systems
|
|
||||||
_ "github.com/ncw/rclone/drive"
|
|
||||||
_ "github.com/ncw/rclone/dropbox"
|
|
||||||
_ "github.com/ncw/rclone/googlecloudstorage"
|
|
||||||
_ "github.com/ncw/rclone/local"
|
|
||||||
_ "github.com/ncw/rclone/s3"
|
|
||||||
_ "github.com/ncw/rclone/swift"
|
|
||||||
)
|
|
||||||
|
|
||||||
// Globals
|
|
||||||
var (
|
|
||||||
localName, remoteName string
|
|
||||||
version = pflag.BoolP("version", "V", false, "Print the version number")
|
|
||||||
subDir = pflag.BoolP("subdir", "S", false, "Test with a sub directory")
|
|
||||||
)
|
|
||||||
|
|
||||||
// Represents an item for checking
|
|
||||||
type Item struct {
|
|
||||||
Path string
|
|
||||||
Md5sum string
|
|
||||||
ModTime time.Time
|
|
||||||
Size int64
|
|
||||||
}
|
|
||||||
|
|
||||||
// Represents all items for checking
|
|
||||||
type Items struct {
|
|
||||||
byName map[string]*Item
|
|
||||||
items []Item
|
|
||||||
}
|
|
||||||
|
|
||||||
// Make an Items
|
|
||||||
func NewItems(items []Item) *Items {
|
|
||||||
is := &Items{
|
|
||||||
byName: make(map[string]*Item),
|
|
||||||
items: items,
|
|
||||||
}
|
|
||||||
// Fill up byName
|
|
||||||
for i := range items {
|
|
||||||
is.byName[items[i].Path] = &items[i]
|
|
||||||
}
|
|
||||||
return is
|
|
||||||
}
|
|
||||||
|
|
||||||
// Check off an item
|
|
||||||
func (is *Items) Find(obj fs.Object) {
|
|
||||||
i, ok := is.byName[obj.Remote()]
|
|
||||||
if !ok {
|
|
||||||
log.Fatalf("Unexpected file %q", obj.Remote())
|
|
||||||
}
|
|
||||||
delete(is.byName, obj.Remote())
|
|
||||||
// Check attributes
|
|
||||||
Md5sum, err := obj.Md5sum()
|
|
||||||
if err != nil {
|
|
||||||
log.Fatalf("Failed to read md5sum for %q: %v", obj.Remote(), err)
|
|
||||||
}
|
|
||||||
if i.Md5sum != Md5sum {
|
|
||||||
log.Fatalf("%s: Md5sum incorrect - expecting %q got %q", obj.Remote(), i.Md5sum, Md5sum)
|
|
||||||
}
|
|
||||||
if i.Size != obj.Size() {
|
|
||||||
log.Fatalf("%s: Size incorrect - expecting %d got %d", obj.Remote(), i.Size, obj.Size())
|
|
||||||
}
|
|
||||||
// check the mod time to the given precision
|
|
||||||
modTime := obj.ModTime()
|
|
||||||
dt := modTime.Sub(i.ModTime)
|
|
||||||
if dt >= fs.Config.ModifyWindow || dt <= -fs.Config.ModifyWindow {
|
|
||||||
log.Fatalf("%s: Modification time difference too big |%s| > %s (%s vs %s)", obj.Remote(), dt, fs.Config.ModifyWindow, modTime, i.ModTime)
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
// Check all done
|
|
||||||
func (is *Items) Done() {
|
|
||||||
if len(is.byName) != 0 {
|
|
||||||
for name := range is.byName {
|
|
||||||
log.Printf("Not found %q", name)
|
|
||||||
}
|
|
||||||
log.Fatalf("%d objects not found", len(is.byName))
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Checks the fs to see if it has the expected contents
|
|
||||||
func CheckListing(f fs.Fs, items []Item) {
|
|
||||||
is := NewItems(items)
|
|
||||||
for obj := range f.List() {
|
|
||||||
is.Find(obj)
|
|
||||||
}
|
|
||||||
is.Done()
|
|
||||||
}
|
|
||||||
|
|
||||||
// Parse a time string or explode
|
|
||||||
func Time(timeString string) time.Time {
|
|
||||||
t, err := time.Parse(time.RFC3339Nano, timeString)
|
|
||||||
if err != nil {
|
|
||||||
log.Fatalf("Failed to parse time %q: %v", timeString, err)
|
|
||||||
}
|
|
||||||
return t
|
|
||||||
}
|
|
||||||
|
|
||||||
// Write a file
|
|
||||||
func WriteFile(filePath, content string, t time.Time) {
|
|
||||||
// FIXME make directories?
|
|
||||||
filePath = path.Join(localName, filePath)
|
|
||||||
dirPath := path.Dir(filePath)
|
|
||||||
err := os.MkdirAll(dirPath, 0770)
|
|
||||||
if err != nil {
|
|
||||||
log.Fatalf("Failed to make directories %q: %v", dirPath, err)
|
|
||||||
}
|
|
||||||
err = ioutil.WriteFile(filePath, []byte(content), 0600)
|
|
||||||
if err != nil {
|
|
||||||
log.Fatalf("Failed to write file %q: %v", filePath, err)
|
|
||||||
}
|
|
||||||
err = os.Chtimes(filePath, t, t)
|
|
||||||
if err != nil {
|
|
||||||
log.Fatalf("Failed to chtimes file %q: %v", filePath, err)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Create a random string
|
|
||||||
func RandomString(n int) string {
|
|
||||||
source := "abcdefghijklmnopqrstuvwxyz0123456789"
|
|
||||||
out := make([]byte, n)
|
|
||||||
for i := range out {
|
|
||||||
out[i] = source[rand.Intn(len(source))]
|
|
||||||
}
|
|
||||||
return string(out)
|
|
||||||
}
|
|
||||||
|
|
||||||
func TestMkdir(flocal, fremote fs.Fs) {
|
|
||||||
err := fs.Mkdir(fremote)
|
|
||||||
if err != nil {
|
|
||||||
log.Fatalf("Mkdir failed: %v", err)
|
|
||||||
}
|
|
||||||
items := []Item{}
|
|
||||||
CheckListing(flocal, items)
|
|
||||||
CheckListing(fremote, items)
|
|
||||||
}
|
|
||||||
|
|
||||||
var t1 = Time("2001-02-03T04:05:06.499999999Z")
|
|
||||||
var t2 = Time("2011-12-25T12:59:59.123456789Z")
|
|
||||||
var t3 = Time("2011-12-30T12:59:59.000000000Z")
|
|
||||||
|
|
||||||
func TestCopy(flocal, fremote fs.Fs) {
|
|
||||||
WriteFile("sub dir/hello world", "hello world", t1)
|
|
||||||
|
|
||||||
// Check dry run is working
|
|
||||||
log.Printf("Copy with --dry-run")
|
|
||||||
fs.Config.DryRun = true
|
|
||||||
err := fs.Sync(fremote, flocal, false)
|
|
||||||
fs.Config.DryRun = false
|
|
||||||
if err != nil {
|
|
||||||
log.Fatalf("Copy failed: %v", err)
|
|
||||||
}
|
|
||||||
|
|
||||||
items := []Item{
|
|
||||||
{Path: "sub dir/hello world", Size: 11, ModTime: t1, Md5sum: "5eb63bbbe01eeed093cb22bb8f5acdc3"},
|
|
||||||
}
|
|
||||||
|
|
||||||
CheckListing(flocal, items)
|
|
||||||
CheckListing(fremote, []Item{})
|
|
||||||
|
|
||||||
// Now without dry run
|
|
||||||
|
|
||||||
log.Printf("Copy")
|
|
||||||
err = fs.Sync(fremote, flocal, false)
|
|
||||||
if err != nil {
|
|
||||||
log.Fatalf("Copy failed: %v", err)
|
|
||||||
}
|
|
||||||
|
|
||||||
CheckListing(flocal, items)
|
|
||||||
CheckListing(fremote, items)
|
|
||||||
|
|
||||||
// Now delete the local file and download it
|
|
||||||
|
|
||||||
err = os.Remove(localName + "/sub dir/hello world")
|
|
||||||
if err != nil {
|
|
||||||
log.Fatalf("Remove failed: %v", err)
|
|
||||||
}
|
|
||||||
|
|
||||||
CheckListing(flocal, []Item{})
|
|
||||||
CheckListing(fremote, items)
|
|
||||||
|
|
||||||
log.Printf("Copy - redownload")
|
|
||||||
err = fs.Sync(flocal, fremote, false)
|
|
||||||
if err != nil {
|
|
||||||
log.Fatalf("Copy failed: %v", err)
|
|
||||||
}
|
|
||||||
|
|
||||||
CheckListing(flocal, items)
|
|
||||||
CheckListing(fremote, items)
|
|
||||||
|
|
||||||
// Clean the directory
|
|
||||||
cleanTempDir()
|
|
||||||
}
|
|
||||||
|
|
||||||
func TestSync(flocal, fremote fs.Fs) {
|
|
||||||
WriteFile("empty space", "", t1)
|
|
||||||
|
|
||||||
log.Printf("Sync after changing file modtime only")
|
|
||||||
err := os.Chtimes(localName+"/empty space", t2, t2)
|
|
||||||
if err != nil {
|
|
||||||
log.Fatalf("Chtimes failed: %v", err)
|
|
||||||
}
|
|
||||||
err = fs.Sync(fremote, flocal, true)
|
|
||||||
if err != nil {
|
|
||||||
log.Fatalf("Sync failed: %v", err)
|
|
||||||
}
|
|
||||||
items := []Item{
|
|
||||||
{Path: "empty space", Size: 0, ModTime: t2, Md5sum: "d41d8cd98f00b204e9800998ecf8427e"},
|
|
||||||
}
|
|
||||||
CheckListing(flocal, items)
|
|
||||||
CheckListing(fremote, items)
|
|
||||||
|
|
||||||
// ------------------------------------------------------------
|
|
||||||
|
|
||||||
log.Printf("Sync after adding a file")
|
|
||||||
WriteFile("potato", "------------------------------------------------------------", t3)
|
|
||||||
err = fs.Sync(fremote, flocal, true)
|
|
||||||
if err != nil {
|
|
||||||
log.Fatalf("Sync failed: %v", err)
|
|
||||||
}
|
|
||||||
items = []Item{
|
|
||||||
{Path: "empty space", Size: 0, ModTime: t2, Md5sum: "d41d8cd98f00b204e9800998ecf8427e"},
|
|
||||||
{Path: "potato", Size: 60, ModTime: t3, Md5sum: "d6548b156ea68a4e003e786df99eee76"},
|
|
||||||
}
|
|
||||||
CheckListing(flocal, items)
|
|
||||||
CheckListing(fremote, items)
|
|
||||||
|
|
||||||
// ------------------------------------------------------------
|
|
||||||
|
|
||||||
log.Printf("Sync after changing a file's size only")
|
|
||||||
WriteFile("potato", "smaller but same date", t3)
|
|
||||||
err = fs.Sync(fremote, flocal, true)
|
|
||||||
if err != nil {
|
|
||||||
log.Fatalf("Sync failed: %v", err)
|
|
||||||
}
|
|
||||||
items = []Item{
|
|
||||||
{Path: "empty space", Size: 0, ModTime: t2, Md5sum: "d41d8cd98f00b204e9800998ecf8427e"},
|
|
||||||
{Path: "potato", Size: 21, ModTime: t3, Md5sum: "100defcf18c42a1e0dc42a789b107cd2"},
|
|
||||||
}
|
|
||||||
CheckListing(flocal, items)
|
|
||||||
CheckListing(fremote, items)
|
|
||||||
|
|
||||||
// ------------------------------------------------------------
|
|
||||||
|
|
||||||
log.Printf("Sync after removing a file and adding a file --dry-run")
|
|
||||||
WriteFile("potato2", "------------------------------------------------------------", t1)
|
|
||||||
err = os.Remove(localName + "/potato")
|
|
||||||
if err != nil {
|
|
||||||
log.Fatalf("Remove failed: %v", err)
|
|
||||||
}
|
|
||||||
fs.Config.DryRun = true
|
|
||||||
err = fs.Sync(fremote, flocal, true)
|
|
||||||
fs.Config.DryRun = false
|
|
||||||
if err != nil {
|
|
||||||
log.Fatalf("Sync failed: %v", err)
|
|
||||||
}
|
|
||||||
|
|
||||||
before := []Item{
|
|
||||||
{Path: "empty space", Size: 0, ModTime: t2, Md5sum: "d41d8cd98f00b204e9800998ecf8427e"},
|
|
||||||
{Path: "potato", Size: 21, ModTime: t3, Md5sum: "100defcf18c42a1e0dc42a789b107cd2"},
|
|
||||||
}
|
|
||||||
items = []Item{
|
|
||||||
{Path: "empty space", Size: 0, ModTime: t2, Md5sum: "d41d8cd98f00b204e9800998ecf8427e"},
|
|
||||||
{Path: "potato2", Size: 60, ModTime: t1, Md5sum: "d6548b156ea68a4e003e786df99eee76"},
|
|
||||||
}
|
|
||||||
CheckListing(flocal, items)
|
|
||||||
CheckListing(fremote, before)
|
|
||||||
|
|
||||||
log.Printf("Sync after removing a file and adding a file")
|
|
||||||
err = fs.Sync(fremote, flocal, true)
|
|
||||||
if err != nil {
|
|
||||||
log.Fatalf("Sync failed: %v", err)
|
|
||||||
}
|
|
||||||
CheckListing(flocal, items)
|
|
||||||
CheckListing(fremote, items)
|
|
||||||
}
|
|
||||||
|
|
||||||
func TestLs(flocal, fremote fs.Fs) {
|
|
||||||
// Underlying List has been tested above, so we just make sure it runs
|
|
||||||
err := fs.List(fremote)
|
|
||||||
if err != nil {
|
|
||||||
log.Fatalf("List failed: %v", err)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func TestLsd(flocal, fremote fs.Fs) {
|
|
||||||
}
|
|
||||||
|
|
||||||
func TestCheck(flocal, fremote fs.Fs) {
|
|
||||||
}
|
|
||||||
|
|
||||||
func TestPurge(fremote fs.Fs) {
|
|
||||||
err := fs.Purge(fremote)
|
|
||||||
if err != nil {
|
|
||||||
log.Fatalf("Purge failed: %v", err)
|
|
||||||
}
|
|
||||||
unexpected := 0
|
|
||||||
for obj := range fremote.List() {
|
|
||||||
unexpected++
|
|
||||||
log.Printf("Found unexpected item %s", obj.Remote())
|
|
||||||
}
|
|
||||||
if unexpected != 0 {
|
|
||||||
log.Fatalf("exiting as found %d unexpected items", unexpected)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func TestRmdir(flocal, fremote fs.Fs) {
|
|
||||||
err := fs.Rmdir(fremote)
|
|
||||||
if err != nil {
|
|
||||||
log.Fatalf("Rmdir failed: %v", err)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func syntaxError() {
|
|
||||||
fmt.Fprintf(os.Stderr, `Test rclone with a remote to find bugs in either - %s.
|
|
||||||
|
|
||||||
Syntax: [options] remote:
|
|
||||||
|
|
||||||
Need a remote: as argument. This will create a random container or
|
|
||||||
directory under it and perform tests on it, deleting it at the end.
|
|
||||||
|
|
||||||
Options:
|
|
||||||
|
|
||||||
`, fs.Version)
|
|
||||||
pflag.PrintDefaults()
|
|
||||||
}
|
|
||||||
|
|
||||||
// Clean the temporary directory
|
|
||||||
func cleanTempDir() {
|
|
||||||
log.Printf("Cleaning temporary directory: %q", localName)
|
|
||||||
err := os.RemoveAll(localName)
|
|
||||||
if err != nil {
|
|
||||||
log.Printf("Failed to remove %q: %v", localName, err)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func main() {
|
|
||||||
pflag.Usage = syntaxError
|
|
||||||
pflag.Parse()
|
|
||||||
if *version {
|
|
||||||
fmt.Printf("rclonetest %s\n", fs.Version)
|
|
||||||
os.Exit(0)
|
|
||||||
}
|
|
||||||
fs.LoadConfig()
|
|
||||||
rand.Seed(time.Now().UnixNano())
|
|
||||||
args := pflag.Args()
|
|
||||||
|
|
||||||
if len(args) != 1 {
|
|
||||||
syntaxError()
|
|
||||||
os.Exit(1)
|
|
||||||
}
|
|
||||||
|
|
||||||
remoteName = args[0]
|
|
||||||
if !strings.HasSuffix(remoteName, ":") {
|
|
||||||
remoteName += "/"
|
|
||||||
}
|
|
||||||
remoteName += RandomString(32)
|
|
||||||
var parentRemote fs.Fs
|
|
||||||
if *subDir {
|
|
||||||
var err error
|
|
||||||
parentRemote, err = fs.NewFs(remoteName)
|
|
||||||
if err != nil {
|
|
||||||
log.Fatalf("Failed to make parent %q: %v", remoteName, err)
|
|
||||||
}
|
|
||||||
remoteName += "/" + RandomString(8)
|
|
||||||
}
|
|
||||||
log.Printf("Testing with remote %q", remoteName)
|
|
||||||
var err error
|
|
||||||
localName, err = ioutil.TempDir("", "rclone")
|
|
||||||
if err != nil {
|
|
||||||
log.Fatalf("Failed to create temp dir: %v", err)
|
|
||||||
}
|
|
||||||
log.Printf("Testing with local %q", localName)
|
|
||||||
|
|
||||||
fremote, err := fs.NewFs(remoteName)
|
|
||||||
if err != nil {
|
|
||||||
log.Fatalf("Failed to make %q: %v", remoteName, err)
|
|
||||||
}
|
|
||||||
flocal, err := fs.NewFs(localName)
|
|
||||||
if err != nil {
|
|
||||||
log.Fatalf("Failed to make %q: %v", remoteName, err)
|
|
||||||
}
|
|
||||||
|
|
||||||
fs.CalculateModifyWindow(fremote, flocal)
|
|
||||||
|
|
||||||
TestMkdir(flocal, fremote)
|
|
||||||
TestCopy(flocal, fremote)
|
|
||||||
TestSync(flocal, fremote)
|
|
||||||
TestLs(flocal, fremote)
|
|
||||||
TestLsd(flocal, fremote)
|
|
||||||
TestCheck(flocal, fremote)
|
|
||||||
TestPurge(fremote)
|
|
||||||
//TestRmdir(flocal, fremote)
|
|
||||||
|
|
||||||
if parentRemote != nil {
|
|
||||||
TestPurge(parentRemote)
|
|
||||||
}
|
|
||||||
|
|
||||||
cleanTempDir()
|
|
||||||
log.Printf("Tests OK")
|
|
||||||
}
|
|
||||||
@@ -1,25 +0,0 @@
|
|||||||
#!/bin/bash
|
|
||||||
|
|
||||||
go install
|
|
||||||
|
|
||||||
REMOTES="
|
|
||||||
memstore:
|
|
||||||
s3:
|
|
||||||
drive2:
|
|
||||||
gcs:
|
|
||||||
dropbox:
|
|
||||||
/tmp/z
|
|
||||||
"
|
|
||||||
|
|
||||||
function test_remote {
|
|
||||||
args=$@
|
|
||||||
rclonetest $args || {
|
|
||||||
echo "*** rclonetest $args FAILED ***"
|
|
||||||
exit 1
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
for remote in $REMOTES; do
|
|
||||||
test_remote $remote
|
|
||||||
test_remote --subdir $remote
|
|
||||||
done
|
|
||||||
15
redirect_stderr.go
Normal file
15
redirect_stderr.go
Normal file
@@ -0,0 +1,15 @@
|
|||||||
|
// Log the panic to the log file - for oses which can't do this
|
||||||
|
|
||||||
|
//+build !windows,!unix
|
||||||
|
|
||||||
|
package main
|
||||||
|
|
||||||
|
import (
|
||||||
|
"log"
|
||||||
|
"os"
|
||||||
|
)
|
||||||
|
|
||||||
|
// redirectStderr to the file passed in
|
||||||
|
func redirectStderr(f *os.File) {
|
||||||
|
log.Printf("Can't redirect stderr to file")
|
||||||
|
}
|
||||||
19
redirect_stderr_unix.go
Normal file
19
redirect_stderr_unix.go
Normal file
@@ -0,0 +1,19 @@
|
|||||||
|
// Log the panic under unix to the log file
|
||||||
|
|
||||||
|
//+build unix
|
||||||
|
|
||||||
|
package main
|
||||||
|
|
||||||
|
import (
|
||||||
|
"log"
|
||||||
|
"os"
|
||||||
|
"syscall"
|
||||||
|
)
|
||||||
|
|
||||||
|
// redirectStderr to the file passed in
|
||||||
|
func redirectStderr(f *os.File) {
|
||||||
|
err := syscall.Dup2(int(f.Fd()), int(os.Stderr.Fd()))
|
||||||
|
if err != nil {
|
||||||
|
log.Fatalf("Failed to redirect stderr to file: %v", err)
|
||||||
|
}
|
||||||
|
}
|
||||||
39
redirect_stderr_windows.go
Normal file
39
redirect_stderr_windows.go
Normal file
@@ -0,0 +1,39 @@
|
|||||||
|
// Log the panic under windows to the log file
|
||||||
|
//
|
||||||
|
// Code from minix, via
|
||||||
|
//
|
||||||
|
// http://play.golang.org/p/kLtct7lSUg
|
||||||
|
|
||||||
|
//+build windows
|
||||||
|
|
||||||
|
package main
|
||||||
|
|
||||||
|
import (
|
||||||
|
"log"
|
||||||
|
"os"
|
||||||
|
"syscall"
|
||||||
|
)
|
||||||
|
|
||||||
|
var (
|
||||||
|
kernel32 = syscall.MustLoadDLL("kernel32.dll")
|
||||||
|
procSetStdHandle = kernel32.MustFindProc("SetStdHandle")
|
||||||
|
)
|
||||||
|
|
||||||
|
func setStdHandle(stdhandle int32, handle syscall.Handle) error {
|
||||||
|
r0, _, e1 := syscall.Syscall(procSetStdHandle.Addr(), 2, uintptr(stdhandle), uintptr(handle), 0)
|
||||||
|
if r0 == 0 {
|
||||||
|
if e1 != 0 {
|
||||||
|
return error(e1)
|
||||||
|
}
|
||||||
|
return syscall.EINVAL
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// redirectStderr to the file passed in
|
||||||
|
func redirectStderr(f *os.File) {
|
||||||
|
err := setStdHandle(syscall.STD_ERROR_HANDLE, syscall.Handle(f.Fd()))
|
||||||
|
if err != nil {
|
||||||
|
log.Fatalf("Failed to redirect stderr to file: %v", err)
|
||||||
|
}
|
||||||
|
}
|
||||||
106
s3/s3.go
106
s3/s3.go
@@ -7,7 +7,6 @@ import (
|
|||||||
"errors"
|
"errors"
|
||||||
"fmt"
|
"fmt"
|
||||||
"io"
|
"io"
|
||||||
"mime"
|
|
||||||
"net/http"
|
"net/http"
|
||||||
"path"
|
"path"
|
||||||
"regexp"
|
"regexp"
|
||||||
@@ -101,7 +100,8 @@ func init() {
|
|||||||
|
|
||||||
// Constants
|
// Constants
|
||||||
const (
|
const (
|
||||||
metaMtime = "X-Amz-Meta-Mtime" // the meta key to store mtime in
|
metaMtime = "X-Amz-Meta-Mtime" // the meta key to store mtime in
|
||||||
|
listChunkSize = 1024 // number of items to read at once
|
||||||
)
|
)
|
||||||
|
|
||||||
// FsS3 represents a remote s3 server
|
// FsS3 represents a remote s3 server
|
||||||
@@ -184,6 +184,7 @@ func s3Connection(name string) (*s3.S3, error) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
c := s3.New(auth, region)
|
c := s3.New(auth, region)
|
||||||
|
c.Client = fs.Config.Client()
|
||||||
return c, nil
|
return c, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -227,7 +228,7 @@ func NewFs(name, root string) (fs.Fs, error) {
|
|||||||
// Return an FsObject from a path
|
// Return an FsObject from a path
|
||||||
//
|
//
|
||||||
// May return nil if an error occurred
|
// May return nil if an error occurred
|
||||||
func (f *FsS3) NewFsObjectWithInfo(remote string, info *s3.Key) fs.Object {
|
func (f *FsS3) newFsObjectWithInfo(remote string, info *s3.Key) fs.Object {
|
||||||
o := &FsObjectS3{
|
o := &FsObjectS3{
|
||||||
s3: f,
|
s3: f,
|
||||||
remote: remote,
|
remote: remote,
|
||||||
@@ -256,7 +257,7 @@ func (f *FsS3) NewFsObjectWithInfo(remote string, info *s3.Key) fs.Object {
|
|||||||
//
|
//
|
||||||
// May return nil if an error occurred
|
// May return nil if an error occurred
|
||||||
func (f *FsS3) NewFsObject(remote string) fs.Object {
|
func (f *FsS3) NewFsObject(remote string) fs.Object {
|
||||||
return f.NewFsObjectWithInfo(remote, nil)
|
return f.newFsObjectWithInfo(remote, nil)
|
||||||
}
|
}
|
||||||
|
|
||||||
// list the objects into the function supplied
|
// list the objects into the function supplied
|
||||||
@@ -267,33 +268,46 @@ func (f *FsS3) list(directories bool, fn func(string, *s3.Key)) {
|
|||||||
if directories {
|
if directories {
|
||||||
delimiter = "/"
|
delimiter = "/"
|
||||||
}
|
}
|
||||||
// FIXME need to implement ALL loop
|
marker := ""
|
||||||
objects, err := f.b.List(f.root, delimiter, "", 10000)
|
for {
|
||||||
if err != nil {
|
objects, err := f.b.List(f.root, delimiter, marker, listChunkSize)
|
||||||
fs.Stats.Error()
|
if err != nil {
|
||||||
fs.Log(f, "Couldn't read bucket %q: %s", f.bucket, err)
|
fs.Stats.Error()
|
||||||
} else {
|
fs.Log(f, "Couldn't read bucket %q: %s", f.bucket, err)
|
||||||
rootLength := len(f.root)
|
|
||||||
if directories {
|
|
||||||
for _, remote := range objects.CommonPrefixes {
|
|
||||||
if !strings.HasPrefix(remote, f.root) {
|
|
||||||
fs.Log(f, "Odd name received %q", remote)
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
remote := remote[rootLength:]
|
|
||||||
fn(remote, &s3.Key{Key: remote})
|
|
||||||
}
|
|
||||||
} else {
|
} else {
|
||||||
for i := range objects.Contents {
|
rootLength := len(f.root)
|
||||||
object := &objects.Contents[i]
|
if directories {
|
||||||
if !strings.HasPrefix(object.Key, f.root) {
|
for _, remote := range objects.CommonPrefixes {
|
||||||
fs.Log(f, "Odd name received %q", object.Key)
|
if !strings.HasPrefix(remote, f.root) {
|
||||||
continue
|
fs.Log(f, "Odd name received %q", remote)
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
remote := remote[rootLength:]
|
||||||
|
if strings.HasSuffix(remote, "/") {
|
||||||
|
remote = remote[:len(remote)-1]
|
||||||
|
}
|
||||||
|
fn(remote, &s3.Key{Key: remote})
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
for i := range objects.Contents {
|
||||||
|
object := &objects.Contents[i]
|
||||||
|
if !strings.HasPrefix(object.Key, f.root) {
|
||||||
|
fs.Log(f, "Odd name received %q", object.Key)
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
remote := object.Key[rootLength:]
|
||||||
|
fn(remote, object)
|
||||||
}
|
}
|
||||||
remote := object.Key[rootLength:]
|
|
||||||
fn(remote, object)
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
if !objects.IsTruncated {
|
||||||
|
break
|
||||||
|
}
|
||||||
|
// Use NextMarker if set, otherwise use last Key
|
||||||
|
marker = objects.NextMarker
|
||||||
|
if marker == "" {
|
||||||
|
marker = objects.Contents[len(objects.Contents)-1].Key
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -309,7 +323,7 @@ func (f *FsS3) List() fs.ObjectsChan {
|
|||||||
go func() {
|
go func() {
|
||||||
defer close(out)
|
defer close(out)
|
||||||
f.list(false, func(remote string, object *s3.Key) {
|
f.list(false, func(remote string, object *s3.Key) {
|
||||||
if fs := f.NewFsObjectWithInfo(remote, object); fs != nil {
|
if fs := f.newFsObjectWithInfo(remote, object); fs != nil {
|
||||||
out <- fs
|
out <- fs
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
@@ -406,9 +420,17 @@ func (o *FsObjectS3) Remote() string {
|
|||||||
return o.remote
|
return o.remote
|
||||||
}
|
}
|
||||||
|
|
||||||
|
var matchMd5 = regexp.MustCompile(`^[0-9a-f]{32}$`)
|
||||||
|
|
||||||
// Md5sum returns the Md5sum of an object returning a lowercase hex string
|
// Md5sum returns the Md5sum of an object returning a lowercase hex string
|
||||||
func (o *FsObjectS3) Md5sum() (string, error) {
|
func (o *FsObjectS3) Md5sum() (string, error) {
|
||||||
return strings.Trim(strings.ToLower(o.etag), `"`), nil
|
etag := strings.Trim(strings.ToLower(o.etag), `"`)
|
||||||
|
// Check the etag is a valid md5sum
|
||||||
|
if !matchMd5.MatchString(etag) {
|
||||||
|
fs.Debug(o, "Invalid md5sum (probably multipart uploaded) - ignoring: %q", etag)
|
||||||
|
return "", nil
|
||||||
|
}
|
||||||
|
return etag, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// Size returns the size of an object in bytes
|
// Size returns the size of an object in bytes
|
||||||
@@ -418,13 +440,28 @@ func (o *FsObjectS3) Size() int64 {
|
|||||||
|
|
||||||
// readMetaData gets the metadata if it hasn't already been fetched
|
// readMetaData gets the metadata if it hasn't already been fetched
|
||||||
//
|
//
|
||||||
|
// if we get a 404 error then we retry a few times for eventual
|
||||||
|
// consistency reasons
|
||||||
|
//
|
||||||
// it also sets the info
|
// it also sets the info
|
||||||
func (o *FsObjectS3) readMetaData() (err error) {
|
func (o *FsObjectS3) readMetaData() (err error) {
|
||||||
if o.meta != nil {
|
if o.meta != nil {
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
var headers s3.Headers
|
||||||
|
|
||||||
headers, err := o.s3.b.Head(o.s3.root+o.remote, nil)
|
// Try reading the metadata a few times (with exponential
|
||||||
|
// backoff) to get around eventual consistency on 404 error
|
||||||
|
for tries := uint(0); tries < 10; tries++ {
|
||||||
|
headers, err = o.s3.b.Head(o.s3.root+o.remote, nil)
|
||||||
|
if s3Err, ok := err.(*s3.Error); ok {
|
||||||
|
if s3Err.StatusCode == http.StatusNotFound {
|
||||||
|
time.Sleep(5 * time.Millisecond << tries)
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
}
|
||||||
|
break
|
||||||
|
}
|
||||||
if err != nil {
|
if err != nil {
|
||||||
fs.Debug(o, "Failed to read info: %s", err)
|
fs.Debug(o, "Failed to read info: %s", err)
|
||||||
return err
|
return err
|
||||||
@@ -507,17 +544,12 @@ func (o *FsObjectS3) Update(in io.Reader, modTime time.Time, size int64) error {
|
|||||||
metaMtime: swift.TimeToFloatString(modTime),
|
metaMtime: swift.TimeToFloatString(modTime),
|
||||||
}
|
}
|
||||||
|
|
||||||
// Guess the content type
|
_, err := o.s3.b.PutReaderHeaders(o.s3.root+o.remote, in, size, fs.MimeType(o), o.s3.perm, headers)
|
||||||
contentType := mime.TypeByExtension(path.Ext(o.remote))
|
|
||||||
if contentType == "" {
|
|
||||||
contentType = "application/octet-stream"
|
|
||||||
}
|
|
||||||
|
|
||||||
_, err := o.s3.b.PutReaderHeaders(o.s3.root+o.remote, in, size, contentType, o.s3.perm, headers)
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
// Read the metadata from the newly created object
|
// Read the metadata from the newly created object
|
||||||
|
o.meta = nil // wipe old metadata
|
||||||
err = o.readMetaData()
|
err = o.readMetaData()
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|||||||
53
s3/s3_test.go
Normal file
53
s3/s3_test.go
Normal file
@@ -0,0 +1,53 @@
|
|||||||
|
// Test S3 filesystem interface
|
||||||
|
//
|
||||||
|
// Automatically generated - DO NOT EDIT
|
||||||
|
// Regenerate with: go run gen_tests.go or make gen_tests
|
||||||
|
package s3_test
|
||||||
|
|
||||||
|
import (
|
||||||
|
"testing"
|
||||||
|
|
||||||
|
"github.com/ncw/rclone/fs"
|
||||||
|
"github.com/ncw/rclone/fstest/fstests"
|
||||||
|
"github.com/ncw/rclone/s3"
|
||||||
|
)
|
||||||
|
|
||||||
|
func init() {
|
||||||
|
fstests.NilObject = fs.Object((*s3.FsObjectS3)(nil))
|
||||||
|
fstests.RemoteName = "TestS3:"
|
||||||
|
}
|
||||||
|
|
||||||
|
// Generic tests for the Fs
|
||||||
|
func TestInit(t *testing.T) { fstests.TestInit(t) }
|
||||||
|
func TestFsString(t *testing.T) { fstests.TestFsString(t) }
|
||||||
|
func TestFsRmdirEmpty(t *testing.T) { fstests.TestFsRmdirEmpty(t) }
|
||||||
|
func TestFsRmdirNotFound(t *testing.T) { fstests.TestFsRmdirNotFound(t) }
|
||||||
|
func TestFsMkdir(t *testing.T) { fstests.TestFsMkdir(t) }
|
||||||
|
func TestFsListEmpty(t *testing.T) { fstests.TestFsListEmpty(t) }
|
||||||
|
func TestFsListDirEmpty(t *testing.T) { fstests.TestFsListDirEmpty(t) }
|
||||||
|
func TestFsNewFsObjectNotFound(t *testing.T) { fstests.TestFsNewFsObjectNotFound(t) }
|
||||||
|
func TestFsPutFile1(t *testing.T) { fstests.TestFsPutFile1(t) }
|
||||||
|
func TestFsPutFile2(t *testing.T) { fstests.TestFsPutFile2(t) }
|
||||||
|
func TestFsListDirFile2(t *testing.T) { fstests.TestFsListDirFile2(t) }
|
||||||
|
func TestFsListDirRoot(t *testing.T) { fstests.TestFsListDirRoot(t) }
|
||||||
|
func TestFsListRoot(t *testing.T) { fstests.TestFsListRoot(t) }
|
||||||
|
func TestFsListFile1(t *testing.T) { fstests.TestFsListFile1(t) }
|
||||||
|
func TestFsNewFsObject(t *testing.T) { fstests.TestFsNewFsObject(t) }
|
||||||
|
func TestFsListFile1and2(t *testing.T) { fstests.TestFsListFile1and2(t) }
|
||||||
|
func TestFsRmdirFull(t *testing.T) { fstests.TestFsRmdirFull(t) }
|
||||||
|
func TestFsPrecision(t *testing.T) { fstests.TestFsPrecision(t) }
|
||||||
|
func TestObjectString(t *testing.T) { fstests.TestObjectString(t) }
|
||||||
|
func TestObjectFs(t *testing.T) { fstests.TestObjectFs(t) }
|
||||||
|
func TestObjectRemote(t *testing.T) { fstests.TestObjectRemote(t) }
|
||||||
|
func TestObjectMd5sum(t *testing.T) { fstests.TestObjectMd5sum(t) }
|
||||||
|
func TestObjectModTime(t *testing.T) { fstests.TestObjectModTime(t) }
|
||||||
|
func TestObjectSetModTime(t *testing.T) { fstests.TestObjectSetModTime(t) }
|
||||||
|
func TestObjectSize(t *testing.T) { fstests.TestObjectSize(t) }
|
||||||
|
func TestObjectOpen(t *testing.T) { fstests.TestObjectOpen(t) }
|
||||||
|
func TestObjectUpdate(t *testing.T) { fstests.TestObjectUpdate(t) }
|
||||||
|
func TestObjectStorable(t *testing.T) { fstests.TestObjectStorable(t) }
|
||||||
|
func TestLimitedFs(t *testing.T) { fstests.TestLimitedFs(t) }
|
||||||
|
func TestLimitedFsNotFound(t *testing.T) { fstests.TestLimitedFsNotFound(t) }
|
||||||
|
func TestObjectRemove(t *testing.T) { fstests.TestObjectRemove(t) }
|
||||||
|
func TestObjectPurge(t *testing.T) { fstests.TestObjectPurge(t) }
|
||||||
|
func TestFinalise(t *testing.T) { fstests.TestFinalise(t) }
|
||||||
@@ -44,6 +44,12 @@ func init() {
|
|||||||
Help: "Memset Memstore UK v2",
|
Help: "Memset Memstore UK v2",
|
||||||
Value: "https://auth.storage.memset.com/v2.0",
|
Value: "https://auth.storage.memset.com/v2.0",
|
||||||
}},
|
}},
|
||||||
|
}, {
|
||||||
|
Name: "tenant",
|
||||||
|
Help: "Tenant name - optional",
|
||||||
|
}, {
|
||||||
|
Name: "region",
|
||||||
|
Help: "Region name - optional",
|
||||||
},
|
},
|
||||||
// snet = flag.Bool("swift-snet", false, "Use internal service network") // FIXME not implemented
|
// snet = flag.Bool("swift-snet", false, "Use internal service network") // FIXME not implemented
|
||||||
},
|
},
|
||||||
@@ -107,10 +113,15 @@ func swiftConnection(name string) (*swift.Connection, error) {
|
|||||||
return nil, errors.New("auth not found")
|
return nil, errors.New("auth not found")
|
||||||
}
|
}
|
||||||
c := &swift.Connection{
|
c := &swift.Connection{
|
||||||
UserName: userName,
|
UserName: userName,
|
||||||
ApiKey: apiKey,
|
ApiKey: apiKey,
|
||||||
AuthUrl: authUrl,
|
AuthUrl: authUrl,
|
||||||
UserAgent: fs.UserAgent,
|
UserAgent: fs.UserAgent,
|
||||||
|
Tenant: fs.ConfigFile.MustValue(name, "tenant"),
|
||||||
|
Region: fs.ConfigFile.MustValue(name, "region"),
|
||||||
|
ConnectTimeout: 10 * fs.Config.ConnectTimeout, // Use the timeouts in the transport
|
||||||
|
Timeout: 10 * fs.Config.Timeout, // Use the timeouts in the transport
|
||||||
|
Transport: fs.Config.Transport(),
|
||||||
}
|
}
|
||||||
err := c.Authenticate()
|
err := c.Authenticate()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
@@ -157,7 +168,7 @@ func NewFs(name, root string) (fs.Fs, error) {
|
|||||||
// Return an FsObject from a path
|
// Return an FsObject from a path
|
||||||
//
|
//
|
||||||
// May return nil if an error occurred
|
// May return nil if an error occurred
|
||||||
func (f *FsSwift) NewFsObjectWithInfo(remote string, info *swift.Object) fs.Object {
|
func (f *FsSwift) newFsObjectWithInfo(remote string, info *swift.Object) fs.Object {
|
||||||
fs := &FsObjectSwift{
|
fs := &FsObjectSwift{
|
||||||
swift: f,
|
swift: f,
|
||||||
remote: remote,
|
remote: remote,
|
||||||
@@ -179,7 +190,7 @@ func (f *FsSwift) NewFsObjectWithInfo(remote string, info *swift.Object) fs.Obje
|
|||||||
//
|
//
|
||||||
// May return nil if an error occurred
|
// May return nil if an error occurred
|
||||||
func (f *FsSwift) NewFsObject(remote string) fs.Object {
|
func (f *FsSwift) NewFsObject(remote string) fs.Object {
|
||||||
return f.NewFsObjectWithInfo(remote, nil)
|
return f.newFsObjectWithInfo(remote, nil)
|
||||||
}
|
}
|
||||||
|
|
||||||
// list the objects into the function supplied
|
// list the objects into the function supplied
|
||||||
@@ -201,8 +212,11 @@ func (f *FsSwift) list(directories bool, fn func(string, *swift.Object)) {
|
|||||||
for i := range objects {
|
for i := range objects {
|
||||||
object := &objects[i]
|
object := &objects[i]
|
||||||
// FIXME if there are no directories, swift gives back the files for some reason!
|
// FIXME if there are no directories, swift gives back the files for some reason!
|
||||||
if directories && !strings.HasSuffix(object.Name, "/") {
|
if directories {
|
||||||
continue
|
if !strings.HasSuffix(object.Name, "/") {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
object.Name = object.Name[:len(object.Name)-1]
|
||||||
}
|
}
|
||||||
if !strings.HasPrefix(object.Name, f.root) {
|
if !strings.HasPrefix(object.Name, f.root) {
|
||||||
fs.Log(f, "Odd name received %q", object.Name)
|
fs.Log(f, "Odd name received %q", object.Name)
|
||||||
@@ -233,7 +247,7 @@ func (f *FsSwift) List() fs.ObjectsChan {
|
|||||||
go func() {
|
go func() {
|
||||||
defer close(out)
|
defer close(out)
|
||||||
f.list(false, func(remote string, object *swift.Object) {
|
f.list(false, func(remote string, object *swift.Object) {
|
||||||
if fs := f.NewFsObjectWithInfo(remote, object); fs != nil {
|
if fs := f.newFsObjectWithInfo(remote, object); fs != nil {
|
||||||
out <- fs
|
out <- fs
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
@@ -413,6 +427,7 @@ func (o *FsObjectSwift) Update(in io.Reader, modTime time.Time, size int64) erro
|
|||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
// Read the metadata from the newly created object
|
// Read the metadata from the newly created object
|
||||||
|
o.meta = nil // wipe old metadata
|
||||||
err = o.readMetaData()
|
err = o.readMetaData()
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|||||||
53
swift/swift_test.go
Normal file
53
swift/swift_test.go
Normal file
@@ -0,0 +1,53 @@
|
|||||||
|
// Test Swift filesystem interface
|
||||||
|
//
|
||||||
|
// Automatically generated - DO NOT EDIT
|
||||||
|
// Regenerate with: go run gen_tests.go or make gen_tests
|
||||||
|
package swift_test
|
||||||
|
|
||||||
|
import (
|
||||||
|
"testing"
|
||||||
|
|
||||||
|
"github.com/ncw/rclone/fs"
|
||||||
|
"github.com/ncw/rclone/fstest/fstests"
|
||||||
|
"github.com/ncw/rclone/swift"
|
||||||
|
)
|
||||||
|
|
||||||
|
func init() {
|
||||||
|
fstests.NilObject = fs.Object((*swift.FsObjectSwift)(nil))
|
||||||
|
fstests.RemoteName = "TestSwift:"
|
||||||
|
}
|
||||||
|
|
||||||
|
// Generic tests for the Fs
|
||||||
|
func TestInit(t *testing.T) { fstests.TestInit(t) }
|
||||||
|
func TestFsString(t *testing.T) { fstests.TestFsString(t) }
|
||||||
|
func TestFsRmdirEmpty(t *testing.T) { fstests.TestFsRmdirEmpty(t) }
|
||||||
|
func TestFsRmdirNotFound(t *testing.T) { fstests.TestFsRmdirNotFound(t) }
|
||||||
|
func TestFsMkdir(t *testing.T) { fstests.TestFsMkdir(t) }
|
||||||
|
func TestFsListEmpty(t *testing.T) { fstests.TestFsListEmpty(t) }
|
||||||
|
func TestFsListDirEmpty(t *testing.T) { fstests.TestFsListDirEmpty(t) }
|
||||||
|
func TestFsNewFsObjectNotFound(t *testing.T) { fstests.TestFsNewFsObjectNotFound(t) }
|
||||||
|
func TestFsPutFile1(t *testing.T) { fstests.TestFsPutFile1(t) }
|
||||||
|
func TestFsPutFile2(t *testing.T) { fstests.TestFsPutFile2(t) }
|
||||||
|
func TestFsListDirFile2(t *testing.T) { fstests.TestFsListDirFile2(t) }
|
||||||
|
func TestFsListDirRoot(t *testing.T) { fstests.TestFsListDirRoot(t) }
|
||||||
|
func TestFsListRoot(t *testing.T) { fstests.TestFsListRoot(t) }
|
||||||
|
func TestFsListFile1(t *testing.T) { fstests.TestFsListFile1(t) }
|
||||||
|
func TestFsNewFsObject(t *testing.T) { fstests.TestFsNewFsObject(t) }
|
||||||
|
func TestFsListFile1and2(t *testing.T) { fstests.TestFsListFile1and2(t) }
|
||||||
|
func TestFsRmdirFull(t *testing.T) { fstests.TestFsRmdirFull(t) }
|
||||||
|
func TestFsPrecision(t *testing.T) { fstests.TestFsPrecision(t) }
|
||||||
|
func TestObjectString(t *testing.T) { fstests.TestObjectString(t) }
|
||||||
|
func TestObjectFs(t *testing.T) { fstests.TestObjectFs(t) }
|
||||||
|
func TestObjectRemote(t *testing.T) { fstests.TestObjectRemote(t) }
|
||||||
|
func TestObjectMd5sum(t *testing.T) { fstests.TestObjectMd5sum(t) }
|
||||||
|
func TestObjectModTime(t *testing.T) { fstests.TestObjectModTime(t) }
|
||||||
|
func TestObjectSetModTime(t *testing.T) { fstests.TestObjectSetModTime(t) }
|
||||||
|
func TestObjectSize(t *testing.T) { fstests.TestObjectSize(t) }
|
||||||
|
func TestObjectOpen(t *testing.T) { fstests.TestObjectOpen(t) }
|
||||||
|
func TestObjectUpdate(t *testing.T) { fstests.TestObjectUpdate(t) }
|
||||||
|
func TestObjectStorable(t *testing.T) { fstests.TestObjectStorable(t) }
|
||||||
|
func TestLimitedFs(t *testing.T) { fstests.TestLimitedFs(t) }
|
||||||
|
func TestLimitedFsNotFound(t *testing.T) { fstests.TestLimitedFsNotFound(t) }
|
||||||
|
func TestObjectRemove(t *testing.T) { fstests.TestObjectRemove(t) }
|
||||||
|
func TestObjectPurge(t *testing.T) { fstests.TestObjectPurge(t) }
|
||||||
|
func TestFinalise(t *testing.T) { fstests.TestFinalise(t) }
|
||||||
Reference in New Issue
Block a user