mirror of
https://github.com/rclone/rclone.git
synced 2025-01-20 13:18:41 +01:00
Rename log functions and factor into own file
This commit is contained in:
parent
40c02989f1
commit
80f53176d9
2
Makefile
2
Makefile
@ -34,7 +34,7 @@ quicktest:
|
|||||||
# Do source code quality checks
|
# Do source code quality checks
|
||||||
check: rclone
|
check: rclone
|
||||||
ifdef GO_LATEST
|
ifdef GO_LATEST
|
||||||
go vet $(GO_FILES)
|
go tool vet -printfuncs Debugf,Infof,Logf,Errorf . 2>&1 | grep -E -v ^vendor/ ; test $$? -eq 1
|
||||||
errcheck $(GO_FILES)
|
errcheck $(GO_FILES)
|
||||||
find . -name \*.go | grep -v /vendor/ | xargs goimports -d | grep . ; test $$? -eq 1
|
find . -name \*.go | grep -v /vendor/ | xargs goimports -d | grep . ; test $$? -eq 1
|
||||||
go list ./... | grep -v /vendor/ | xargs -i golint {} | grep -E -v '(StorageUrl|CdnUrl)' ; test $$? -eq 1
|
go list ./... | grep -v /vendor/ | xargs -i golint {} | grep -E -v '(StorageUrl|CdnUrl)' ; test $$? -eq 1
|
||||||
|
@ -157,14 +157,14 @@ func (f *Fs) shouldRetry(resp *http.Response, err error) (bool, error) {
|
|||||||
if resp != nil {
|
if resp != nil {
|
||||||
if resp.StatusCode == 401 {
|
if resp.StatusCode == 401 {
|
||||||
f.tokenRenewer.Invalidate()
|
f.tokenRenewer.Invalidate()
|
||||||
fs.Debug(f, "401 error received - invalidating token")
|
fs.Debugf(f, "401 error received - invalidating token")
|
||||||
return true, err
|
return true, err
|
||||||
}
|
}
|
||||||
// Work around receiving this error sporadically on authentication
|
// Work around receiving this error sporadically on authentication
|
||||||
//
|
//
|
||||||
// HTTP code 403: "403 Forbidden", reponse body: {"message":"Authorization header requires 'Credential' parameter. Authorization header requires 'Signature' parameter. Authorization header requires 'SignedHeaders' parameter. Authorization header requires existence of either a 'X-Amz-Date' or a 'Date' header. Authorization=Bearer"}
|
// HTTP code 403: "403 Forbidden", reponse body: {"message":"Authorization header requires 'Credential' parameter. Authorization header requires 'Signature' parameter. Authorization header requires 'SignedHeaders' parameter. Authorization header requires existence of either a 'X-Amz-Date' or a 'Date' header. Authorization=Bearer"}
|
||||||
if resp.StatusCode == 403 && strings.Contains(err.Error(), "Authorization header requires") {
|
if resp.StatusCode == 403 && strings.Contains(err.Error(), "Authorization header requires") {
|
||||||
fs.Log(f, "403 \"Authorization header requires...\" error received - retry")
|
fs.Logf(f, "403 \"Authorization header requires...\" error received - retry")
|
||||||
return true, err
|
return true, err
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -280,7 +280,7 @@ func (f *Fs) NewObject(remote string) (fs.Object, error) {
|
|||||||
|
|
||||||
// FindLeaf finds a directory of name leaf in the folder with ID pathID
|
// FindLeaf finds a directory of name leaf in the folder with ID pathID
|
||||||
func (f *Fs) FindLeaf(pathID, leaf string) (pathIDOut string, found bool, err error) {
|
func (f *Fs) FindLeaf(pathID, leaf string) (pathIDOut string, found bool, err error) {
|
||||||
//fs.Debug(f, "FindLeaf(%q, %q)", pathID, leaf)
|
//fs.Debugf(f, "FindLeaf(%q, %q)", pathID, leaf)
|
||||||
folder := acd.FolderFromId(pathID, f.c.Nodes)
|
folder := acd.FolderFromId(pathID, f.c.Nodes)
|
||||||
var resp *http.Response
|
var resp *http.Response
|
||||||
var subFolder *acd.Folder
|
var subFolder *acd.Folder
|
||||||
@ -290,18 +290,18 @@ func (f *Fs) FindLeaf(pathID, leaf string) (pathIDOut string, found bool, err er
|
|||||||
})
|
})
|
||||||
if err != nil {
|
if err != nil {
|
||||||
if err == acd.ErrorNodeNotFound {
|
if err == acd.ErrorNodeNotFound {
|
||||||
//fs.Debug(f, "...Not found")
|
//fs.Debugf(f, "...Not found")
|
||||||
return "", false, nil
|
return "", false, nil
|
||||||
}
|
}
|
||||||
//fs.Debug(f, "...Error %v", err)
|
//fs.Debugf(f, "...Error %v", err)
|
||||||
return "", false, err
|
return "", false, err
|
||||||
}
|
}
|
||||||
if subFolder.Status != nil && *subFolder.Status != statusAvailable {
|
if subFolder.Status != nil && *subFolder.Status != statusAvailable {
|
||||||
fs.Debug(f, "Ignoring folder %q in state %q", leaf, *subFolder.Status)
|
fs.Debugf(f, "Ignoring folder %q in state %q", leaf, *subFolder.Status)
|
||||||
time.Sleep(1 * time.Second) // FIXME wait for problem to go away!
|
time.Sleep(1 * time.Second) // FIXME wait for problem to go away!
|
||||||
return "", false, nil
|
return "", false, nil
|
||||||
}
|
}
|
||||||
//fs.Debug(f, "...Found(%q, %v)", *subFolder.Id, leaf)
|
//fs.Debugf(f, "...Found(%q, %v)", *subFolder.Id, leaf)
|
||||||
return *subFolder.Id, true, nil
|
return *subFolder.Id, true, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -397,7 +397,7 @@ func (f *Fs) listAll(dirID string, title string, directoriesOnly bool, filesOnly
|
|||||||
|
|
||||||
// ListDir reads the directory specified by the job into out, returning any more jobs
|
// ListDir reads the directory specified by the job into out, returning any more jobs
|
||||||
func (f *Fs) ListDir(out fs.ListOpts, job dircache.ListDirJob) (jobs []dircache.ListDirJob, err error) {
|
func (f *Fs) ListDir(out fs.ListOpts, job dircache.ListDirJob) (jobs []dircache.ListDirJob, err error) {
|
||||||
fs.Debug(f, "Reading %q", job.Path)
|
fs.Debugf(f, "Reading %q", job.Path)
|
||||||
maxTries := fs.Config.LowLevelRetries
|
maxTries := fs.Config.LowLevelRetries
|
||||||
for tries := 1; tries <= maxTries; tries++ {
|
for tries := 1; tries <= maxTries; tries++ {
|
||||||
_, err = f.listAll(job.DirID, "", false, false, func(node *acd.Node) bool {
|
_, err = f.listAll(job.DirID, "", false, false, func(node *acd.Node) bool {
|
||||||
@ -433,7 +433,7 @@ func (f *Fs) ListDir(out fs.ListOpts, job dircache.ListDirJob) (jobs []dircache.
|
|||||||
return false
|
return false
|
||||||
})
|
})
|
||||||
if fs.IsRetryError(err) {
|
if fs.IsRetryError(err) {
|
||||||
fs.Debug(f, "Directory listing error for %q: %v - low level retry %d/%d", job.Path, err, tries, maxTries)
|
fs.Debugf(f, "Directory listing error for %q: %v - low level retry %d/%d", job.Path, err, tries, maxTries)
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
if err != nil {
|
if err != nil {
|
||||||
@ -441,7 +441,7 @@ func (f *Fs) ListDir(out fs.ListOpts, job dircache.ListDirJob) (jobs []dircache.
|
|||||||
}
|
}
|
||||||
break
|
break
|
||||||
}
|
}
|
||||||
fs.Debug(f, "Finished reading %q", job.Path)
|
fs.Debugf(f, "Finished reading %q", job.Path)
|
||||||
return jobs, err
|
return jobs, err
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -488,13 +488,13 @@ func (f *Fs) checkUpload(resp *http.Response, in io.Reader, src fs.ObjectInfo, i
|
|||||||
buf := make([]byte, 1)
|
buf := make([]byte, 1)
|
||||||
n, err := in.Read(buf)
|
n, err := in.Read(buf)
|
||||||
if !(n == 0 && err == io.EOF) {
|
if !(n == 0 && err == io.EOF) {
|
||||||
fs.Debug(src, "Upload error detected but didn't finish upload: %v (%q)", inErr, httpStatus)
|
fs.Debugf(src, "Upload error detected but didn't finish upload: %v (%q)", inErr, httpStatus)
|
||||||
return false, inInfo, inErr
|
return false, inInfo, inErr
|
||||||
}
|
}
|
||||||
|
|
||||||
// Don't wait for uploads - assume they will appear later
|
// Don't wait for uploads - assume they will appear later
|
||||||
if *uploadWaitPerGB <= 0 {
|
if *uploadWaitPerGB <= 0 {
|
||||||
fs.Debug(src, "Upload error detected but waiting disabled: %v (%q)", inErr, httpStatus)
|
fs.Debugf(src, "Upload error detected but waiting disabled: %v (%q)", inErr, httpStatus)
|
||||||
return false, inInfo, inErr
|
return false, inInfo, inErr
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -505,27 +505,27 @@ func (f *Fs) checkUpload(resp *http.Response, in io.Reader, src fs.ObjectInfo, i
|
|||||||
const sleepTime = 5 * time.Second // sleep between tries
|
const sleepTime = 5 * time.Second // sleep between tries
|
||||||
retries := int((timeToWait + sleepTime - 1) / sleepTime) // number of retries, rounded up
|
retries := int((timeToWait + sleepTime - 1) / sleepTime) // number of retries, rounded up
|
||||||
|
|
||||||
fs.Debug(src, "Error detected after finished upload - waiting to see if object was uploaded correctly: %v (%q)", inErr, httpStatus)
|
fs.Debugf(src, "Error detected after finished upload - waiting to see if object was uploaded correctly: %v (%q)", inErr, httpStatus)
|
||||||
remote := src.Remote()
|
remote := src.Remote()
|
||||||
for i := 1; i <= retries; i++ {
|
for i := 1; i <= retries; i++ {
|
||||||
o, err := f.NewObject(remote)
|
o, err := f.NewObject(remote)
|
||||||
if err == fs.ErrorObjectNotFound {
|
if err == fs.ErrorObjectNotFound {
|
||||||
fs.Debug(src, "Object not found - waiting (%d/%d)", i, retries)
|
fs.Debugf(src, "Object not found - waiting (%d/%d)", i, retries)
|
||||||
} else if err != nil {
|
} else if err != nil {
|
||||||
fs.Debug(src, "Object returned error - waiting (%d/%d): %v", i, retries, err)
|
fs.Debugf(src, "Object returned error - waiting (%d/%d): %v", i, retries, err)
|
||||||
} else {
|
} else {
|
||||||
if src.Size() == o.Size() {
|
if src.Size() == o.Size() {
|
||||||
fs.Debug(src, "Object found with correct size %d after waiting (%d/%d) - %v - returning with no error", src.Size(), i, retries, sleepTime*time.Duration(i-1))
|
fs.Debugf(src, "Object found with correct size %d after waiting (%d/%d) - %v - returning with no error", src.Size(), i, retries, sleepTime*time.Duration(i-1))
|
||||||
info = &acd.File{
|
info = &acd.File{
|
||||||
Node: o.(*Object).info,
|
Node: o.(*Object).info,
|
||||||
}
|
}
|
||||||
return true, info, nil
|
return true, info, nil
|
||||||
}
|
}
|
||||||
fs.Debug(src, "Object found but wrong size %d vs %d - waiting (%d/%d)", src.Size(), o.Size(), i, retries)
|
fs.Debugf(src, "Object found but wrong size %d vs %d - waiting (%d/%d)", src.Size(), o.Size(), i, retries)
|
||||||
}
|
}
|
||||||
time.Sleep(sleepTime)
|
time.Sleep(sleepTime)
|
||||||
}
|
}
|
||||||
fs.Debug(src, "Giving up waiting for object - returning original error: %v (%q)", inErr, httpStatus)
|
fs.Debugf(src, "Giving up waiting for object - returning original error: %v (%q)", inErr, httpStatus)
|
||||||
return false, inInfo, inErr
|
return false, inInfo, inErr
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -558,7 +558,7 @@ func (f *Fs) Put(in io.Reader, src fs.ObjectInfo) (fs.Object, error) {
|
|||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
if size > warnFileSize {
|
if size > warnFileSize {
|
||||||
fs.Debug(f, "Warning: file %q may fail because it is too big. Use --max-size=%dM to skip large files.", remote, warnFileSize>>20)
|
fs.Debugf(f, "Warning: file %q may fail because it is too big. Use --max-size=%dM to skip large files.", remote, warnFileSize>>20)
|
||||||
}
|
}
|
||||||
folder := acd.FolderFromId(directoryID, o.fs.c.Nodes)
|
folder := acd.FolderFromId(directoryID, o.fs.c.Nodes)
|
||||||
var info *acd.File
|
var info *acd.File
|
||||||
@ -607,7 +607,7 @@ func (f *Fs) Move(src fs.Object, remote string) (fs.Object, error) {
|
|||||||
// go test -v -run '^Test(Setup|Init|FsMkdir|FsPutFile1|FsPutFile2|FsUpdateFile1|FsMove)$'
|
// go test -v -run '^Test(Setup|Init|FsMkdir|FsPutFile1|FsPutFile2|FsUpdateFile1|FsMove)$'
|
||||||
srcObj, ok := src.(*Object)
|
srcObj, ok := src.(*Object)
|
||||||
if !ok {
|
if !ok {
|
||||||
fs.Debug(src, "Can't move - not same remote type")
|
fs.Debugf(src, "Can't move - not same remote type")
|
||||||
return nil, fs.ErrorCantMove
|
return nil, fs.ErrorCantMove
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -648,7 +648,7 @@ func (f *Fs) Move(src fs.Object, remote string) (fs.Object, error) {
|
|||||||
// finished if src not found and dst found
|
// finished if src not found and dst found
|
||||||
break
|
break
|
||||||
}
|
}
|
||||||
fs.Debug(src, "Wait for directory listing to update after move %d/%d", i, fs.Config.LowLevelRetries)
|
fs.Debugf(src, "Wait for directory listing to update after move %d/%d", i, fs.Config.LowLevelRetries)
|
||||||
time.Sleep(1 * time.Second)
|
time.Sleep(1 * time.Second)
|
||||||
}
|
}
|
||||||
return dstObj, nil
|
return dstObj, nil
|
||||||
@ -672,19 +672,19 @@ func (f *Fs) DirMove(src fs.Fs) (err error) {
|
|||||||
// go test -v -run '^Test(Setup|Init|FsMkdir|FsPutFile1|FsPutFile2|FsUpdateFile1|FsDirMove)$
|
// go test -v -run '^Test(Setup|Init|FsMkdir|FsPutFile1|FsPutFile2|FsUpdateFile1|FsDirMove)$
|
||||||
srcFs, ok := src.(*Fs)
|
srcFs, ok := src.(*Fs)
|
||||||
if !ok {
|
if !ok {
|
||||||
fs.Debug(src, "DirMove error: not same remote type")
|
fs.Debugf(src, "DirMove error: not same remote type")
|
||||||
return fs.ErrorCantDirMove
|
return fs.ErrorCantDirMove
|
||||||
}
|
}
|
||||||
|
|
||||||
// Check if destination exists
|
// Check if destination exists
|
||||||
if f.dirCache.FoundRoot() {
|
if f.dirCache.FoundRoot() {
|
||||||
fs.Debug(src, "DirMove error: destination exists")
|
fs.Debugf(src, "DirMove error: destination exists")
|
||||||
return fs.ErrorDirExists
|
return fs.ErrorDirExists
|
||||||
}
|
}
|
||||||
|
|
||||||
// Refuse to move to or from the root
|
// Refuse to move to or from the root
|
||||||
if f.root == "" || srcFs.root == "" {
|
if f.root == "" || srcFs.root == "" {
|
||||||
fs.Debug(src, "DirMove error: Can't move root")
|
fs.Debugf(src, "DirMove error: Can't move root")
|
||||||
return errors.New("can't move root directory")
|
return errors.New("can't move root directory")
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -697,12 +697,12 @@ func (f *Fs) DirMove(src fs.Fs) (err error) {
|
|||||||
// Find the ID of the source and make a node from it
|
// Find the ID of the source and make a node from it
|
||||||
err = srcFs.dirCache.FindRoot(false)
|
err = srcFs.dirCache.FindRoot(false)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
fs.Debug(src, "DirMove error: error finding src root: %v", err)
|
fs.Debugf(src, "DirMove error: error finding src root: %v", err)
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
srcDirectoryID, err := srcFs.dirCache.RootParentID()
|
srcDirectoryID, err := srcFs.dirCache.RootParentID()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
fs.Debug(src, "DirMove error: error finding src RootParentID: %v", err)
|
fs.Debugf(src, "DirMove error: error finding src RootParentID: %v", err)
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
srcLeaf, _ := dircache.SplitPath(srcFs.root)
|
srcLeaf, _ := dircache.SplitPath(srcFs.root)
|
||||||
@ -715,12 +715,12 @@ func (f *Fs) DirMove(src fs.Fs) (err error) {
|
|||||||
return srcFs.shouldRetry(nil, err)
|
return srcFs.shouldRetry(nil, err)
|
||||||
})
|
})
|
||||||
if err != nil {
|
if err != nil {
|
||||||
fs.Debug(src, "DirMove error: error reading src metadata: %v", err)
|
fs.Debugf(src, "DirMove error: error reading src metadata: %v", err)
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
err = json.Unmarshal([]byte(jsonStr), &srcInfo)
|
err = json.Unmarshal([]byte(jsonStr), &srcInfo)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
fs.Debug(src, "DirMove error: error reading unpacking src metadata: %v", err)
|
fs.Debugf(src, "DirMove error: error reading unpacking src metadata: %v", err)
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -759,7 +759,7 @@ func (f *Fs) purgeCheck(dir string, check bool) error {
|
|||||||
empty = false
|
empty = false
|
||||||
return true
|
return true
|
||||||
default:
|
default:
|
||||||
fs.Debug("Found ASSET %s", *node.Id)
|
fs.Debugf("Found ASSET %s", *node.Id)
|
||||||
}
|
}
|
||||||
return false
|
return false
|
||||||
})
|
})
|
||||||
@ -817,7 +817,7 @@ func (f *Fs) Hashes() fs.HashSet {
|
|||||||
//func (f *Fs) Copy(src fs.Object, remote string) (fs.Object, error) {
|
//func (f *Fs) Copy(src fs.Object, remote string) (fs.Object, error) {
|
||||||
// srcObj, ok := src.(*Object)
|
// srcObj, ok := src.(*Object)
|
||||||
// if !ok {
|
// if !ok {
|
||||||
// fs.Debug(src, "Can't copy - not same remote type")
|
// fs.Debugf(src, "Can't copy - not same remote type")
|
||||||
// return nil, fs.ErrorCantCopy
|
// return nil, fs.ErrorCantCopy
|
||||||
// }
|
// }
|
||||||
// srcFs := srcObj.fs
|
// srcFs := srcObj.fs
|
||||||
@ -917,12 +917,12 @@ func (o *Object) readMetaData() (err error) {
|
|||||||
func (o *Object) ModTime() time.Time {
|
func (o *Object) ModTime() time.Time {
|
||||||
err := o.readMetaData()
|
err := o.readMetaData()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
fs.Log(o, "Failed to read metadata: %v", err)
|
fs.Logf(o, "Failed to read metadata: %v", err)
|
||||||
return time.Now()
|
return time.Now()
|
||||||
}
|
}
|
||||||
modTime, err := time.Parse(timeFormat, *o.info.ModifiedDate)
|
modTime, err := time.Parse(timeFormat, *o.info.ModifiedDate)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
fs.Log(o, "Failed to read mtime from object: %v", err)
|
fs.Logf(o, "Failed to read mtime from object: %v", err)
|
||||||
return time.Now()
|
return time.Now()
|
||||||
}
|
}
|
||||||
return modTime
|
return modTime
|
||||||
@ -943,7 +943,7 @@ func (o *Object) Storable() bool {
|
|||||||
func (o *Object) Open(options ...fs.OpenOption) (in io.ReadCloser, err error) {
|
func (o *Object) Open(options ...fs.OpenOption) (in io.ReadCloser, err error) {
|
||||||
bigObject := o.Size() >= int64(tempLinkThreshold)
|
bigObject := o.Size() >= int64(tempLinkThreshold)
|
||||||
if bigObject {
|
if bigObject {
|
||||||
fs.Debug(o, "Dowloading large object via tempLink")
|
fs.Debugf(o, "Dowloading large object via tempLink")
|
||||||
}
|
}
|
||||||
file := acd.File{Node: o.info}
|
file := acd.File{Node: o.info}
|
||||||
var resp *http.Response
|
var resp *http.Response
|
||||||
@ -1051,30 +1051,30 @@ func (f *Fs) removeParent(info *acd.Node, parentID string) error {
|
|||||||
// moveNode moves the node given from the srcLeaf,srcDirectoryID to
|
// moveNode moves the node given from the srcLeaf,srcDirectoryID to
|
||||||
// the dstLeaf,dstDirectoryID
|
// the dstLeaf,dstDirectoryID
|
||||||
func (f *Fs) moveNode(name, dstLeaf, dstDirectoryID string, srcInfo *acd.Node, srcLeaf, srcDirectoryID string, useDirErrorMsgs bool) (err error) {
|
func (f *Fs) moveNode(name, dstLeaf, dstDirectoryID string, srcInfo *acd.Node, srcLeaf, srcDirectoryID string, useDirErrorMsgs bool) (err error) {
|
||||||
// fs.Debug(name, "moveNode dst(%q,%s) <- src(%q,%s)", dstLeaf, dstDirectoryID, srcLeaf, srcDirectoryID)
|
// fs.Debugf(name, "moveNode dst(%q,%s) <- src(%q,%s)", dstLeaf, dstDirectoryID, srcLeaf, srcDirectoryID)
|
||||||
cantMove := fs.ErrorCantMove
|
cantMove := fs.ErrorCantMove
|
||||||
if useDirErrorMsgs {
|
if useDirErrorMsgs {
|
||||||
cantMove = fs.ErrorCantDirMove
|
cantMove = fs.ErrorCantDirMove
|
||||||
}
|
}
|
||||||
|
|
||||||
if len(srcInfo.Parents) > 1 && srcLeaf != dstLeaf {
|
if len(srcInfo.Parents) > 1 && srcLeaf != dstLeaf {
|
||||||
fs.Debug(name, "Move error: object is attached to multiple parents and should be renamed. This would change the name of the node in all parents.")
|
fs.Debugf(name, "Move error: object is attached to multiple parents and should be renamed. This would change the name of the node in all parents.")
|
||||||
return cantMove
|
return cantMove
|
||||||
}
|
}
|
||||||
|
|
||||||
if srcLeaf != dstLeaf {
|
if srcLeaf != dstLeaf {
|
||||||
// fs.Debug(name, "renaming")
|
// fs.Debugf(name, "renaming")
|
||||||
_, err = f.renameNode(srcInfo, dstLeaf)
|
_, err = f.renameNode(srcInfo, dstLeaf)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
fs.Debug(name, "Move: quick path rename failed: %v", err)
|
fs.Debugf(name, "Move: quick path rename failed: %v", err)
|
||||||
goto OnConflict
|
goto OnConflict
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if srcDirectoryID != dstDirectoryID {
|
if srcDirectoryID != dstDirectoryID {
|
||||||
// fs.Debug(name, "trying parent replace: %s -> %s", oldParentID, newParentID)
|
// fs.Debugf(name, "trying parent replace: %s -> %s", oldParentID, newParentID)
|
||||||
err = f.replaceParent(srcInfo, srcDirectoryID, dstDirectoryID)
|
err = f.replaceParent(srcInfo, srcDirectoryID, dstDirectoryID)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
fs.Debug(name, "Move: quick path parent replace failed: %v", err)
|
fs.Debugf(name, "Move: quick path parent replace failed: %v", err)
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -1082,38 +1082,38 @@ func (f *Fs) moveNode(name, dstLeaf, dstDirectoryID string, srcInfo *acd.Node, s
|
|||||||
return nil
|
return nil
|
||||||
|
|
||||||
OnConflict:
|
OnConflict:
|
||||||
fs.Debug(name, "Could not directly rename file, presumably because there was a file with the same name already. Instead, the file will now be trashed where such operations do not cause errors. It will be restored to the correct parent after. If any of the subsequent calls fails, the rename/move will be in an invalid state.")
|
fs.Debugf(name, "Could not directly rename file, presumably because there was a file with the same name already. Instead, the file will now be trashed where such operations do not cause errors. It will be restored to the correct parent after. If any of the subsequent calls fails, the rename/move will be in an invalid state.")
|
||||||
|
|
||||||
// fs.Debug(name, "Trashing file")
|
// fs.Debugf(name, "Trashing file")
|
||||||
err = f.removeNode(srcInfo)
|
err = f.removeNode(srcInfo)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
fs.Debug(name, "Move: remove node failed: %v", err)
|
fs.Debugf(name, "Move: remove node failed: %v", err)
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
// fs.Debug(name, "Renaming file")
|
// fs.Debugf(name, "Renaming file")
|
||||||
_, err = f.renameNode(srcInfo, dstLeaf)
|
_, err = f.renameNode(srcInfo, dstLeaf)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
fs.Debug(name, "Move: rename node failed: %v", err)
|
fs.Debugf(name, "Move: rename node failed: %v", err)
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
// note: replacing parent is forbidden by API, modifying them individually is
|
// note: replacing parent is forbidden by API, modifying them individually is
|
||||||
// okay though
|
// okay though
|
||||||
// fs.Debug(name, "Adding target parent")
|
// fs.Debugf(name, "Adding target parent")
|
||||||
err = f.addParent(srcInfo, dstDirectoryID)
|
err = f.addParent(srcInfo, dstDirectoryID)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
fs.Debug(name, "Move: addParent failed: %v", err)
|
fs.Debugf(name, "Move: addParent failed: %v", err)
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
// fs.Debug(name, "removing original parent")
|
// fs.Debugf(name, "removing original parent")
|
||||||
err = f.removeParent(srcInfo, srcDirectoryID)
|
err = f.removeParent(srcInfo, srcDirectoryID)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
fs.Debug(name, "Move: removeParent failed: %v", err)
|
fs.Debugf(name, "Move: removeParent failed: %v", err)
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
// fs.Debug(name, "Restoring")
|
// fs.Debugf(name, "Restoring")
|
||||||
_, err = f.restoreNode(srcInfo)
|
_, err = f.restoreNode(srcInfo)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
fs.Debug(name, "Move: restoreNode node failed: %v", err)
|
fs.Debugf(name, "Move: restoreNode node failed: %v", err)
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
return nil
|
return nil
|
||||||
|
32
b2/b2.go
32
b2/b2.go
@ -173,12 +173,12 @@ func (f *Fs) shouldRetryNoReauth(resp *http.Response, err error) (bool, error) {
|
|||||||
var err error
|
var err error
|
||||||
retryAfter, err = strconv.Atoi(retryAfterString)
|
retryAfter, err = strconv.Atoi(retryAfterString)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
fs.ErrorLog(f, "Malformed %s header %q: %v", retryAfterHeader, retryAfterString, err)
|
fs.Errorf(f, "Malformed %s header %q: %v", retryAfterHeader, retryAfterString, err)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
retryAfterDuration := time.Duration(retryAfter) * time.Second
|
retryAfterDuration := time.Duration(retryAfter) * time.Second
|
||||||
if f.pacer.GetSleep() < retryAfterDuration {
|
if f.pacer.GetSleep() < retryAfterDuration {
|
||||||
fs.Debug(f, "Setting sleep to %v after error: %v", retryAfterDuration, err)
|
fs.Debugf(f, "Setting sleep to %v after error: %v", retryAfterDuration, err)
|
||||||
// We set 1/2 the value here because the pacer will double it immediately
|
// We set 1/2 the value here because the pacer will double it immediately
|
||||||
f.pacer.SetSleep(retryAfterDuration / 2)
|
f.pacer.SetSleep(retryAfterDuration / 2)
|
||||||
}
|
}
|
||||||
@ -191,7 +191,7 @@ func (f *Fs) shouldRetryNoReauth(resp *http.Response, err error) (bool, error) {
|
|||||||
// deserve to be retried. It returns the err as a convenience
|
// deserve to be retried. It returns the err as a convenience
|
||||||
func (f *Fs) shouldRetry(resp *http.Response, err error) (bool, error) {
|
func (f *Fs) shouldRetry(resp *http.Response, err error) (bool, error) {
|
||||||
if resp != nil && resp.StatusCode == 401 {
|
if resp != nil && resp.StatusCode == 401 {
|
||||||
fs.Debug(f, "Unauthorized: %v", err)
|
fs.Debugf(f, "Unauthorized: %v", err)
|
||||||
// Reauth
|
// Reauth
|
||||||
authErr := f.authorizeAccount()
|
authErr := f.authorizeAccount()
|
||||||
if authErr != nil {
|
if authErr != nil {
|
||||||
@ -208,7 +208,7 @@ func errorHandler(resp *http.Response) error {
|
|||||||
errResponse := new(api.Error)
|
errResponse := new(api.Error)
|
||||||
err := rest.DecodeJSON(resp, &errResponse)
|
err := rest.DecodeJSON(resp, &errResponse)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
fs.Debug(nil, "Couldn't decode error response: %v", err)
|
fs.Debugf(nil, "Couldn't decode error response: %v", err)
|
||||||
}
|
}
|
||||||
if errResponse.Code == "" {
|
if errResponse.Code == "" {
|
||||||
errResponse.Code = "unknown"
|
errResponse.Code = "unknown"
|
||||||
@ -259,7 +259,7 @@ func NewFs(name, root string) (fs.Fs, error) {
|
|||||||
if *b2TestMode != "" {
|
if *b2TestMode != "" {
|
||||||
testMode := strings.TrimSpace(*b2TestMode)
|
testMode := strings.TrimSpace(*b2TestMode)
|
||||||
f.srv.SetHeader(testModeHeader, testMode)
|
f.srv.SetHeader(testModeHeader, testMode)
|
||||||
fs.Debug(f, "Setting test header \"%s: %s\"", testModeHeader, testMode)
|
fs.Debugf(f, "Setting test header \"%s: %s\"", testModeHeader, testMode)
|
||||||
}
|
}
|
||||||
// Fill up the buffer tokens
|
// Fill up the buffer tokens
|
||||||
for i := 0; i < fs.Config.Transfers; i++ {
|
for i := 0; i < fs.Config.Transfers; i++ {
|
||||||
@ -373,7 +373,7 @@ func (f *Fs) getUploadBlock() []byte {
|
|||||||
if buf == nil {
|
if buf == nil {
|
||||||
buf = make([]byte, chunkSize)
|
buf = make([]byte, chunkSize)
|
||||||
}
|
}
|
||||||
// fs.Debug(f, "Getting upload block %p", buf)
|
// fs.Debugf(f, "Getting upload block %p", buf)
|
||||||
return buf
|
return buf
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -383,7 +383,7 @@ func (f *Fs) putUploadBlock(buf []byte) {
|
|||||||
if len(buf) != int(chunkSize) {
|
if len(buf) != int(chunkSize) {
|
||||||
panic("bad blocksize returned to pool")
|
panic("bad blocksize returned to pool")
|
||||||
}
|
}
|
||||||
// fs.Debug(f, "Returning upload block %p", buf)
|
// fs.Debugf(f, "Returning upload block %p", buf)
|
||||||
f.bufferTokens <- buf
|
f.bufferTokens <- buf
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -490,7 +490,7 @@ func (f *Fs) list(dir string, level int, prefix string, limit int, hidden bool,
|
|||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
if !strings.HasPrefix(file.Name, f.root) {
|
if !strings.HasPrefix(file.Name, f.root) {
|
||||||
fs.Log(f, "Odd name received %q", file.Name)
|
fs.Logf(f, "Odd name received %q", file.Name)
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
remote := file.Name[len(f.root):]
|
remote := file.Name[len(f.root):]
|
||||||
@ -700,7 +700,7 @@ func (f *Fs) Mkdir(dir string) error {
|
|||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
if getBucketErr != fs.ErrorDirNotFound {
|
if getBucketErr != fs.ErrorDirNotFound {
|
||||||
fs.Debug(f, "Error checking bucket exists: %v", getBucketErr)
|
fs.Debugf(f, "Error checking bucket exists: %v", getBucketErr)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -807,13 +807,13 @@ func (f *Fs) purge(oldOnly bool) error {
|
|||||||
fs.Stats.Checking(remote)
|
fs.Stats.Checking(remote)
|
||||||
if oldOnly && last != remote {
|
if oldOnly && last != remote {
|
||||||
if object.Action == "hide" {
|
if object.Action == "hide" {
|
||||||
fs.Debug(remote, "Deleting current version (id %q) as it is a hide marker", object.ID)
|
fs.Debugf(remote, "Deleting current version (id %q) as it is a hide marker", object.ID)
|
||||||
toBeDeleted <- object
|
toBeDeleted <- object
|
||||||
} else {
|
} else {
|
||||||
fs.Debug(remote, "Not deleting current version (id %q) %q", object.ID, object.Action)
|
fs.Debugf(remote, "Not deleting current version (id %q) %q", object.ID, object.Action)
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
fs.Debug(remote, "Deleting (id %q)", object.ID)
|
fs.Debugf(remote, "Deleting (id %q)", object.ID)
|
||||||
toBeDeleted <- object
|
toBeDeleted <- object
|
||||||
}
|
}
|
||||||
last = remote
|
last = remote
|
||||||
@ -986,7 +986,7 @@ func (o *Object) parseTimeString(timeString string) (err error) {
|
|||||||
}
|
}
|
||||||
unixMilliseconds, err := strconv.ParseInt(timeString, 10, 64)
|
unixMilliseconds, err := strconv.ParseInt(timeString, 10, 64)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
fs.Debug(o, "Failed to parse mod time string %q: %v", timeString, err)
|
fs.Debugf(o, "Failed to parse mod time string %q: %v", timeString, err)
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
o.modTime = time.Unix(unixMilliseconds/1E3, (unixMilliseconds%1E3)*1E6).UTC()
|
o.modTime = time.Unix(unixMilliseconds/1E3, (unixMilliseconds%1E3)*1E6).UTC()
|
||||||
@ -1108,12 +1108,12 @@ func (o *Object) Open(options ...fs.OpenOption) (in io.ReadCloser, err error) {
|
|||||||
// Read sha1 from header if it isn't set
|
// Read sha1 from header if it isn't set
|
||||||
if o.sha1 == "" {
|
if o.sha1 == "" {
|
||||||
o.sha1 = resp.Header.Get(sha1Header)
|
o.sha1 = resp.Header.Get(sha1Header)
|
||||||
fs.Debug(o, "Reading sha1 from header - %q", o.sha1)
|
fs.Debugf(o, "Reading sha1 from header - %q", o.sha1)
|
||||||
// if sha1 header is "none" (in big files), then need
|
// if sha1 header is "none" (in big files), then need
|
||||||
// to read it from the metadata
|
// to read it from the metadata
|
||||||
if o.sha1 == "none" {
|
if o.sha1 == "none" {
|
||||||
o.sha1 = resp.Header.Get(sha1InfoHeader)
|
o.sha1 = resp.Header.Get(sha1InfoHeader)
|
||||||
fs.Debug(o, "Reading sha1 from info - %q", o.sha1)
|
fs.Debugf(o, "Reading sha1 from info - %q", o.sha1)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
// Don't check length or hash on partial content
|
// Don't check length or hash on partial content
|
||||||
@ -1299,7 +1299,7 @@ func (o *Object) Update(in io.Reader, src fs.ObjectInfo) (err error) {
|
|||||||
retry, err := o.fs.shouldRetry(resp, err)
|
retry, err := o.fs.shouldRetry(resp, err)
|
||||||
// On retryable error clear UploadURL
|
// On retryable error clear UploadURL
|
||||||
if retry {
|
if retry {
|
||||||
fs.Debug(o, "Clearing upload URL because of error: %v", err)
|
fs.Debugf(o, "Clearing upload URL because of error: %v", err)
|
||||||
upload = nil
|
upload = nil
|
||||||
}
|
}
|
||||||
return retry, err
|
return retry, err
|
||||||
|
16
b2/upload.go
16
b2/upload.go
@ -132,7 +132,7 @@ func (up *largeUpload) transferChunk(part int64, body []byte) error {
|
|||||||
up.sha1s[part-1] = calculatedSHA1
|
up.sha1s[part-1] = calculatedSHA1
|
||||||
size := int64(len(body))
|
size := int64(len(body))
|
||||||
err := up.f.pacer.Call(func() (bool, error) {
|
err := up.f.pacer.Call(func() (bool, error) {
|
||||||
fs.Debug(up.o, "Sending chunk %d length %d", part, len(body))
|
fs.Debugf(up.o, "Sending chunk %d length %d", part, len(body))
|
||||||
|
|
||||||
// Get upload URL
|
// Get upload URL
|
||||||
upload, err := up.getUploadURL()
|
upload, err := up.getUploadURL()
|
||||||
@ -181,16 +181,16 @@ func (up *largeUpload) transferChunk(part int64, body []byte) error {
|
|||||||
retry, err := up.f.shouldRetry(resp, err)
|
retry, err := up.f.shouldRetry(resp, err)
|
||||||
// On retryable error clear PartUploadURL
|
// On retryable error clear PartUploadURL
|
||||||
if retry {
|
if retry {
|
||||||
fs.Debug(up.o, "Clearing part upload URL because of error: %v", err)
|
fs.Debugf(up.o, "Clearing part upload URL because of error: %v", err)
|
||||||
upload = nil
|
upload = nil
|
||||||
}
|
}
|
||||||
up.returnUploadURL(upload)
|
up.returnUploadURL(upload)
|
||||||
return retry, err
|
return retry, err
|
||||||
})
|
})
|
||||||
if err != nil {
|
if err != nil {
|
||||||
fs.Debug(up.o, "Error sending chunk %d: %v", part, err)
|
fs.Debugf(up.o, "Error sending chunk %d: %v", part, err)
|
||||||
} else {
|
} else {
|
||||||
fs.Debug(up.o, "Done sending chunk %d", part)
|
fs.Debugf(up.o, "Done sending chunk %d", part)
|
||||||
}
|
}
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
@ -235,7 +235,7 @@ func (up *largeUpload) cancel() error {
|
|||||||
|
|
||||||
// Upload uploads the chunks from the input
|
// Upload uploads the chunks from the input
|
||||||
func (up *largeUpload) Upload() error {
|
func (up *largeUpload) Upload() error {
|
||||||
fs.Debug(up.o, "Starting upload of large file in %d chunks (id %q)", up.parts, up.id)
|
fs.Debugf(up.o, "Starting upload of large file in %d chunks (id %q)", up.parts, up.id)
|
||||||
remaining := up.size
|
remaining := up.size
|
||||||
errs := make(chan error, 1)
|
errs := make(chan error, 1)
|
||||||
var wg sync.WaitGroup
|
var wg sync.WaitGroup
|
||||||
@ -289,14 +289,14 @@ outer:
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
if err != nil {
|
if err != nil {
|
||||||
fs.Debug(up.o, "Cancelling large file upload due to error: %v", err)
|
fs.Debugf(up.o, "Cancelling large file upload due to error: %v", err)
|
||||||
cancelErr := up.cancel()
|
cancelErr := up.cancel()
|
||||||
if cancelErr != nil {
|
if cancelErr != nil {
|
||||||
fs.ErrorLog(up.o, "Failed to cancel large file upload: %v", cancelErr)
|
fs.Errorf(up.o, "Failed to cancel large file upload: %v", cancelErr)
|
||||||
}
|
}
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
// Check any errors
|
// Check any errors
|
||||||
fs.Debug(up.o, "Finishing large file upload")
|
fs.Debugf(up.o, "Finishing large file upload")
|
||||||
return up.finish()
|
return up.finish()
|
||||||
}
|
}
|
||||||
|
24
cmd/cmd.go
24
cmd/cmd.go
@ -247,22 +247,22 @@ func Run(Retry bool, showStats bool, cmd *cobra.Command, f func() error) {
|
|||||||
err = f()
|
err = f()
|
||||||
if !Retry || (err == nil && !fs.Stats.Errored()) {
|
if !Retry || (err == nil && !fs.Stats.Errored()) {
|
||||||
if try > 1 {
|
if try > 1 {
|
||||||
fs.ErrorLog(nil, "Attempt %d/%d succeeded", try, *retries)
|
fs.Errorf(nil, "Attempt %d/%d succeeded", try, *retries)
|
||||||
}
|
}
|
||||||
break
|
break
|
||||||
}
|
}
|
||||||
if fs.IsFatalError(err) {
|
if fs.IsFatalError(err) {
|
||||||
fs.ErrorLog(nil, "Fatal error received - not attempting retries")
|
fs.Errorf(nil, "Fatal error received - not attempting retries")
|
||||||
break
|
break
|
||||||
}
|
}
|
||||||
if fs.IsNoRetryError(err) {
|
if fs.IsNoRetryError(err) {
|
||||||
fs.ErrorLog(nil, "Can't retry this error - not attempting retries")
|
fs.Errorf(nil, "Can't retry this error - not attempting retries")
|
||||||
break
|
break
|
||||||
}
|
}
|
||||||
if err != nil {
|
if err != nil {
|
||||||
fs.ErrorLog(nil, "Attempt %d/%d failed with %d errors and: %v", try, *retries, fs.Stats.GetErrors(), err)
|
fs.Errorf(nil, "Attempt %d/%d failed with %d errors and: %v", try, *retries, fs.Stats.GetErrors(), err)
|
||||||
} else {
|
} else {
|
||||||
fs.ErrorLog(nil, "Attempt %d/%d failed with %d errors", try, *retries, fs.Stats.GetErrors())
|
fs.Errorf(nil, "Attempt %d/%d failed with %d errors", try, *retries, fs.Stats.GetErrors())
|
||||||
}
|
}
|
||||||
if try < *retries {
|
if try < *retries {
|
||||||
fs.Stats.ResetErrors()
|
fs.Stats.ResetErrors()
|
||||||
@ -275,10 +275,10 @@ func Run(Retry bool, showStats bool, cmd *cobra.Command, f func() error) {
|
|||||||
log.Fatalf("Failed to %s: %v", cmd.Name(), err)
|
log.Fatalf("Failed to %s: %v", cmd.Name(), err)
|
||||||
}
|
}
|
||||||
if showStats && (!fs.Config.Quiet || fs.Stats.Errored() || *statsInterval > 0) {
|
if showStats && (!fs.Config.Quiet || fs.Stats.Errored() || *statsInterval > 0) {
|
||||||
fs.Log(nil, "%s", fs.Stats)
|
fs.Logf(nil, "%s", fs.Stats)
|
||||||
}
|
}
|
||||||
if fs.Config.Verbose {
|
if fs.Config.Verbose {
|
||||||
fs.Debug(nil, "Go routines at exit %d\n", runtime.NumGoroutine())
|
fs.Debugf(nil, "Go routines at exit %d\n", runtime.NumGoroutine())
|
||||||
}
|
}
|
||||||
if fs.Stats.Errored() {
|
if fs.Stats.Errored() {
|
||||||
os.Exit(1)
|
os.Exit(1)
|
||||||
@ -330,7 +330,7 @@ func initConfig() {
|
|||||||
}
|
}
|
||||||
_, err = f.Seek(0, os.SEEK_END)
|
_, err = f.Seek(0, os.SEEK_END)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
fs.ErrorLog(nil, "Failed to seek log file to end: %v", err)
|
fs.Errorf(nil, "Failed to seek log file to end: %v", err)
|
||||||
}
|
}
|
||||||
log.SetOutput(f)
|
log.SetOutput(f)
|
||||||
fs.DebugLogger.SetOutput(f)
|
fs.DebugLogger.SetOutput(f)
|
||||||
@ -341,11 +341,11 @@ func initConfig() {
|
|||||||
fs.LoadConfig()
|
fs.LoadConfig()
|
||||||
|
|
||||||
// Write the args for debug purposes
|
// Write the args for debug purposes
|
||||||
fs.Debug("rclone", "Version %q starting with parameters %q", fs.Version, os.Args)
|
fs.Debugf("rclone", "Version %q starting with parameters %q", fs.Version, os.Args)
|
||||||
|
|
||||||
// Setup CPU profiling if desired
|
// Setup CPU profiling if desired
|
||||||
if *cpuProfile != "" {
|
if *cpuProfile != "" {
|
||||||
fs.Log(nil, "Creating CPU profile %q\n", *cpuProfile)
|
fs.Logf(nil, "Creating CPU profile %q\n", *cpuProfile)
|
||||||
f, err := os.Create(*cpuProfile)
|
f, err := os.Create(*cpuProfile)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
fs.Stats.Error()
|
fs.Stats.Error()
|
||||||
@ -362,7 +362,7 @@ func initConfig() {
|
|||||||
// Setup memory profiling if desired
|
// Setup memory profiling if desired
|
||||||
if *memProfile != "" {
|
if *memProfile != "" {
|
||||||
defer func() {
|
defer func() {
|
||||||
fs.Log(nil, "Saving Memory profile %q\n", *memProfile)
|
fs.Logf(nil, "Saving Memory profile %q\n", *memProfile)
|
||||||
f, err := os.Create(*memProfile)
|
f, err := os.Create(*memProfile)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
fs.Stats.Error()
|
fs.Stats.Error()
|
||||||
@ -382,7 +382,7 @@ func initConfig() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
if m, _ := regexp.MatchString("^(bits|bytes)$", *dataRateUnit); m == false {
|
if m, _ := regexp.MatchString("^(bits|bytes)$", *dataRateUnit); m == false {
|
||||||
fs.ErrorLog(nil, "Invalid unit passed to --stats-unit. Defaulting to bytes.")
|
fs.Errorf(nil, "Invalid unit passed to --stats-unit. Defaulting to bytes.")
|
||||||
fs.Config.DataRateUnit = "bytes"
|
fs.Config.DataRateUnit = "bytes"
|
||||||
} else {
|
} else {
|
||||||
fs.Config.DataRateUnit = *dataRateUnit
|
fs.Config.DataRateUnit = *dataRateUnit
|
||||||
|
@ -41,8 +41,8 @@ var commandDefintion = &cobra.Command{
|
|||||||
runtime.GC()
|
runtime.GC()
|
||||||
runtime.ReadMemStats(&after)
|
runtime.ReadMemStats(&after)
|
||||||
usedMemory := after.Alloc - before.Alloc
|
usedMemory := after.Alloc - before.Alloc
|
||||||
fs.Log(nil, "%d objects took %d bytes, %.1f bytes/object", len(objs), usedMemory, float64(usedMemory)/float64(len(objs)))
|
fs.Logf(nil, "%d objects took %d bytes, %.1f bytes/object", len(objs), usedMemory, float64(usedMemory)/float64(len(objs)))
|
||||||
fs.Log(nil, "System memory changed from %d to %d bytes a change of %d bytes", before.Sys, after.Sys, after.Sys-before.Sys)
|
fs.Logf(nil, "System memory changed from %d to %d bytes a change of %d bytes", before.Sys, after.Sys, after.Sys-before.Sys)
|
||||||
return nil
|
return nil
|
||||||
})
|
})
|
||||||
},
|
},
|
||||||
|
@ -70,13 +70,13 @@ func (d *Dir) readDir() error {
|
|||||||
defer d.mu.Unlock()
|
defer d.mu.Unlock()
|
||||||
when := time.Now()
|
when := time.Now()
|
||||||
if d.read.IsZero() {
|
if d.read.IsZero() {
|
||||||
fs.Debug(d.path, "Reading directory")
|
fs.Debugf(d.path, "Reading directory")
|
||||||
} else {
|
} else {
|
||||||
age := when.Sub(d.read)
|
age := when.Sub(d.read)
|
||||||
if age < dirCacheTime {
|
if age < dirCacheTime {
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
fs.Debug(d.path, "Re-reading directory (%v old)", age)
|
fs.Debugf(d.path, "Re-reading directory (%v old)", age)
|
||||||
}
|
}
|
||||||
objs, dirs, err := fs.NewLister().SetLevel(1).Start(d.f, d.path).GetAll()
|
objs, dirs, err := fs.NewLister().SetLevel(1).Start(d.f, d.path).GetAll()
|
||||||
if err == fs.ErrorDirNotFound {
|
if err == fs.ErrorDirNotFound {
|
||||||
@ -152,7 +152,7 @@ var _ fusefs.Node = (*Dir)(nil)
|
|||||||
|
|
||||||
// Attr updates the attribes of a directory
|
// Attr updates the attribes of a directory
|
||||||
func (d *Dir) Attr(ctx context.Context, a *fuse.Attr) error {
|
func (d *Dir) Attr(ctx context.Context, a *fuse.Attr) error {
|
||||||
fs.Debug(d.path, "Dir.Attr")
|
fs.Debugf(d.path, "Dir.Attr")
|
||||||
a.Gid = gid
|
a.Gid = gid
|
||||||
a.Uid = uid
|
a.Uid = uid
|
||||||
a.Mode = os.ModeDir | dirPerms
|
a.Mode = os.ModeDir | dirPerms
|
||||||
@ -200,15 +200,15 @@ var _ fusefs.NodeRequestLookuper = (*Dir)(nil)
|
|||||||
// Lookup need not to handle the names "." and "..".
|
// Lookup need not to handle the names "." and "..".
|
||||||
func (d *Dir) Lookup(ctx context.Context, req *fuse.LookupRequest, resp *fuse.LookupResponse) (node fusefs.Node, err error) {
|
func (d *Dir) Lookup(ctx context.Context, req *fuse.LookupRequest, resp *fuse.LookupResponse) (node fusefs.Node, err error) {
|
||||||
path := path.Join(d.path, req.Name)
|
path := path.Join(d.path, req.Name)
|
||||||
fs.Debug(path, "Dir.Lookup")
|
fs.Debugf(path, "Dir.Lookup")
|
||||||
item, err := d.lookupNode(req.Name)
|
item, err := d.lookupNode(req.Name)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
if err != fuse.ENOENT {
|
if err != fuse.ENOENT {
|
||||||
fs.ErrorLog(path, "Dir.Lookup error: %v", err)
|
fs.Errorf(path, "Dir.Lookup error: %v", err)
|
||||||
}
|
}
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
fs.Debug(path, "Dir.Lookup OK")
|
fs.Debugf(path, "Dir.Lookup OK")
|
||||||
return item.node, nil
|
return item.node, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -217,10 +217,10 @@ var _ fusefs.HandleReadDirAller = (*Dir)(nil)
|
|||||||
|
|
||||||
// ReadDirAll reads the contents of the directory
|
// ReadDirAll reads the contents of the directory
|
||||||
func (d *Dir) ReadDirAll(ctx context.Context) (dirents []fuse.Dirent, err error) {
|
func (d *Dir) ReadDirAll(ctx context.Context) (dirents []fuse.Dirent, err error) {
|
||||||
fs.Debug(d.path, "Dir.ReadDirAll")
|
fs.Debugf(d.path, "Dir.ReadDirAll")
|
||||||
err = d.readDir()
|
err = d.readDir()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
fs.Debug(d.path, "Dir.ReadDirAll error: %v", err)
|
fs.Debugf(d.path, "Dir.ReadDirAll error: %v", err)
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
d.mu.RLock()
|
d.mu.RLock()
|
||||||
@ -242,12 +242,12 @@ func (d *Dir) ReadDirAll(ctx context.Context) (dirents []fuse.Dirent, err error)
|
|||||||
}
|
}
|
||||||
default:
|
default:
|
||||||
err = errors.Errorf("unknown type %T", item)
|
err = errors.Errorf("unknown type %T", item)
|
||||||
fs.ErrorLog(d.path, "Dir.ReadDirAll error: %v", err)
|
fs.Errorf(d.path, "Dir.ReadDirAll error: %v", err)
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
dirents = append(dirents, dirent)
|
dirents = append(dirents, dirent)
|
||||||
}
|
}
|
||||||
fs.Debug(d.path, "Dir.ReadDirAll OK with %d entries", len(dirents))
|
fs.Debugf(d.path, "Dir.ReadDirAll OK with %d entries", len(dirents))
|
||||||
return dirents, nil
|
return dirents, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -256,16 +256,16 @@ var _ fusefs.NodeCreater = (*Dir)(nil)
|
|||||||
// Create makes a new file
|
// Create makes a new file
|
||||||
func (d *Dir) Create(ctx context.Context, req *fuse.CreateRequest, resp *fuse.CreateResponse) (fusefs.Node, fusefs.Handle, error) {
|
func (d *Dir) Create(ctx context.Context, req *fuse.CreateRequest, resp *fuse.CreateResponse) (fusefs.Node, fusefs.Handle, error) {
|
||||||
path := path.Join(d.path, req.Name)
|
path := path.Join(d.path, req.Name)
|
||||||
fs.Debug(path, "Dir.Create")
|
fs.Debugf(path, "Dir.Create")
|
||||||
src := newCreateInfo(d.f, path)
|
src := newCreateInfo(d.f, path)
|
||||||
// This gets added to the directory when the file is written
|
// This gets added to the directory when the file is written
|
||||||
file := newFile(d, nil)
|
file := newFile(d, nil)
|
||||||
fh, err := newWriteFileHandle(d, file, src)
|
fh, err := newWriteFileHandle(d, file, src)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
fs.ErrorLog(path, "Dir.Create error: %v", err)
|
fs.Errorf(path, "Dir.Create error: %v", err)
|
||||||
return nil, nil, err
|
return nil, nil, err
|
||||||
}
|
}
|
||||||
fs.Debug(path, "Dir.Create OK")
|
fs.Debugf(path, "Dir.Create OK")
|
||||||
return file, fh, nil
|
return file, fh, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -274,10 +274,10 @@ var _ fusefs.NodeMkdirer = (*Dir)(nil)
|
|||||||
// Mkdir creates a new directory
|
// Mkdir creates a new directory
|
||||||
func (d *Dir) Mkdir(ctx context.Context, req *fuse.MkdirRequest) (fusefs.Node, error) {
|
func (d *Dir) Mkdir(ctx context.Context, req *fuse.MkdirRequest) (fusefs.Node, error) {
|
||||||
path := path.Join(d.path, req.Name)
|
path := path.Join(d.path, req.Name)
|
||||||
fs.Debug(path, "Dir.Mkdir")
|
fs.Debugf(path, "Dir.Mkdir")
|
||||||
err := d.f.Mkdir(path)
|
err := d.f.Mkdir(path)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
fs.ErrorLog(path, "Dir.Mkdir failed to create directory: %v", err)
|
fs.Errorf(path, "Dir.Mkdir failed to create directory: %v", err)
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
fsDir := &fs.Dir{
|
fsDir := &fs.Dir{
|
||||||
@ -286,7 +286,7 @@ func (d *Dir) Mkdir(ctx context.Context, req *fuse.MkdirRequest) (fusefs.Node, e
|
|||||||
}
|
}
|
||||||
dir := newDir(d.f, fsDir)
|
dir := newDir(d.f, fsDir)
|
||||||
d.addObject(fsDir, dir)
|
d.addObject(fsDir, dir)
|
||||||
fs.Debug(path, "Dir.Mkdir OK")
|
fs.Debugf(path, "Dir.Mkdir OK")
|
||||||
return dir, nil
|
return dir, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -297,17 +297,17 @@ var _ fusefs.NodeRemover = (*Dir)(nil)
|
|||||||
// may correspond to a file (unlink) or to a directory (rmdir).
|
// may correspond to a file (unlink) or to a directory (rmdir).
|
||||||
func (d *Dir) Remove(ctx context.Context, req *fuse.RemoveRequest) error {
|
func (d *Dir) Remove(ctx context.Context, req *fuse.RemoveRequest) error {
|
||||||
path := path.Join(d.path, req.Name)
|
path := path.Join(d.path, req.Name)
|
||||||
fs.Debug(path, "Dir.Remove")
|
fs.Debugf(path, "Dir.Remove")
|
||||||
item, err := d.lookupNode(req.Name)
|
item, err := d.lookupNode(req.Name)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
fs.ErrorLog(path, "Dir.Remove error: %v", err)
|
fs.Errorf(path, "Dir.Remove error: %v", err)
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
switch x := item.o.(type) {
|
switch x := item.o.(type) {
|
||||||
case fs.Object:
|
case fs.Object:
|
||||||
err = x.Remove()
|
err = x.Remove()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
fs.ErrorLog(path, "Dir.Remove file error: %v", err)
|
fs.Errorf(path, "Dir.Remove file error: %v", err)
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
case *fs.Dir:
|
case *fs.Dir:
|
||||||
@ -315,27 +315,27 @@ func (d *Dir) Remove(ctx context.Context, req *fuse.RemoveRequest) error {
|
|||||||
dir := item.node.(*Dir)
|
dir := item.node.(*Dir)
|
||||||
empty, err := dir.isEmpty()
|
empty, err := dir.isEmpty()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
fs.ErrorLog(path, "Dir.Remove dir error: %v", err)
|
fs.Errorf(path, "Dir.Remove dir error: %v", err)
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
if !empty {
|
if !empty {
|
||||||
// return fuse.ENOTEMPTY - doesn't exist though so use EEXIST
|
// return fuse.ENOTEMPTY - doesn't exist though so use EEXIST
|
||||||
fs.ErrorLog(path, "Dir.Remove not empty")
|
fs.Errorf(path, "Dir.Remove not empty")
|
||||||
return fuse.EEXIST
|
return fuse.EEXIST
|
||||||
}
|
}
|
||||||
// remove directory
|
// remove directory
|
||||||
err = d.f.Rmdir(path)
|
err = d.f.Rmdir(path)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
fs.ErrorLog(path, "Dir.Remove failed to remove directory: %v", err)
|
fs.Errorf(path, "Dir.Remove failed to remove directory: %v", err)
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
default:
|
default:
|
||||||
fs.ErrorLog(path, "Dir.Remove unknown type %T", item)
|
fs.Errorf(path, "Dir.Remove unknown type %T", item)
|
||||||
return errors.Errorf("unknown type %T", item)
|
return errors.Errorf("unknown type %T", item)
|
||||||
}
|
}
|
||||||
// Remove the item from the directory listing
|
// Remove the item from the directory listing
|
||||||
d.delObject(req.Name)
|
d.delObject(req.Name)
|
||||||
fs.Debug(path, "Dir.Remove OK")
|
fs.Debugf(path, "Dir.Remove OK")
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -348,14 +348,14 @@ func (d *Dir) Rename(ctx context.Context, req *fuse.RenameRequest, newDir fusefs
|
|||||||
destDir, ok := newDir.(*Dir)
|
destDir, ok := newDir.(*Dir)
|
||||||
if !ok {
|
if !ok {
|
||||||
err := errors.Errorf("Unknown Dir type %T", newDir)
|
err := errors.Errorf("Unknown Dir type %T", newDir)
|
||||||
fs.ErrorLog(oldPath, "Dir.Rename error: %v", err)
|
fs.Errorf(oldPath, "Dir.Rename error: %v", err)
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
newPath := path.Join(destDir.path, req.NewName)
|
newPath := path.Join(destDir.path, req.NewName)
|
||||||
fs.Debug(oldPath, "Dir.Rename to %q", newPath)
|
fs.Debugf(oldPath, "Dir.Rename to %q", newPath)
|
||||||
oldItem, err := d.lookupNode(req.OldName)
|
oldItem, err := d.lookupNode(req.OldName)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
fs.ErrorLog(oldPath, "Dir.Rename error: %v", err)
|
fs.Errorf(oldPath, "Dir.Rename error: %v", err)
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
var newObj fs.BasicInfo
|
var newObj fs.BasicInfo
|
||||||
@ -365,12 +365,12 @@ func (d *Dir) Rename(ctx context.Context, req *fuse.RenameRequest, newDir fusefs
|
|||||||
do, ok := d.f.(fs.Mover)
|
do, ok := d.f.(fs.Mover)
|
||||||
if !ok {
|
if !ok {
|
||||||
err := errors.Errorf("Fs %q can't Move files", d.f)
|
err := errors.Errorf("Fs %q can't Move files", d.f)
|
||||||
fs.ErrorLog(oldPath, "Dir.Rename error: %v", err)
|
fs.Errorf(oldPath, "Dir.Rename error: %v", err)
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
newObject, err := do.Move(oldObject, newPath)
|
newObject, err := do.Move(oldObject, newPath)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
fs.ErrorLog(oldPath, "Dir.Rename error: %v", err)
|
fs.Errorf(oldPath, "Dir.Rename error: %v", err)
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
newObj = newObject
|
newObj = newObject
|
||||||
@ -378,22 +378,22 @@ func (d *Dir) Rename(ctx context.Context, req *fuse.RenameRequest, newDir fusefs
|
|||||||
oldDir := oldItem.node.(*Dir)
|
oldDir := oldItem.node.(*Dir)
|
||||||
empty, err := oldDir.isEmpty()
|
empty, err := oldDir.isEmpty()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
fs.ErrorLog(oldPath, "Dir.Rename dir error: %v", err)
|
fs.Errorf(oldPath, "Dir.Rename dir error: %v", err)
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
if !empty {
|
if !empty {
|
||||||
// return fuse.ENOTEMPTY - doesn't exist though so use EEXIST
|
// return fuse.ENOTEMPTY - doesn't exist though so use EEXIST
|
||||||
fs.ErrorLog(oldPath, "Dir.Rename can't rename non empty directory")
|
fs.Errorf(oldPath, "Dir.Rename can't rename non empty directory")
|
||||||
return fuse.EEXIST
|
return fuse.EEXIST
|
||||||
}
|
}
|
||||||
err = d.f.Rmdir(oldPath)
|
err = d.f.Rmdir(oldPath)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
fs.ErrorLog(oldPath, "Dir.Rename failed to remove directory: %v", err)
|
fs.Errorf(oldPath, "Dir.Rename failed to remove directory: %v", err)
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
err = d.f.Mkdir(newPath)
|
err = d.f.Mkdir(newPath)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
fs.ErrorLog(newPath, "Dir.Rename failed to create directory: %v", err)
|
fs.Errorf(newPath, "Dir.Rename failed to create directory: %v", err)
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
newObj = &fs.Dir{
|
newObj = &fs.Dir{
|
||||||
@ -402,7 +402,7 @@ func (d *Dir) Rename(ctx context.Context, req *fuse.RenameRequest, newDir fusefs
|
|||||||
}
|
}
|
||||||
default:
|
default:
|
||||||
err = errors.Errorf("unknown type %T", oldItem)
|
err = errors.Errorf("unknown type %T", oldItem)
|
||||||
fs.ErrorLog(d.path, "Dir.ReadDirAll error: %v", err)
|
fs.Errorf(d.path, "Dir.ReadDirAll error: %v", err)
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -417,7 +417,7 @@ func (d *Dir) Rename(ctx context.Context, req *fuse.RenameRequest, newDir fusefs
|
|||||||
// fallback to Copy/Delete if no Move?
|
// fallback to Copy/Delete if no Move?
|
||||||
// if dir is empty then can move it
|
// if dir is empty then can move it
|
||||||
|
|
||||||
fs.ErrorLog(newPath, "Dir.Rename renamed from %q", oldPath)
|
fs.Errorf(newPath, "Dir.Rename renamed from %q", oldPath)
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -45,7 +45,7 @@ var _ fusefs.Node = (*File)(nil)
|
|||||||
func (f *File) Attr(ctx context.Context, a *fuse.Attr) error {
|
func (f *File) Attr(ctx context.Context, a *fuse.Attr) error {
|
||||||
f.mu.Lock()
|
f.mu.Lock()
|
||||||
defer f.mu.Unlock()
|
defer f.mu.Unlock()
|
||||||
fs.Debug(f.o, "File.Attr")
|
fs.Debugf(f.o, "File.Attr")
|
||||||
a.Gid = gid
|
a.Gid = gid
|
||||||
a.Uid = uid
|
a.Uid = uid
|
||||||
a.Mode = filePerms
|
a.Mode = filePerms
|
||||||
@ -111,7 +111,7 @@ func (f *File) Open(ctx context.Context, req *fuse.OpenRequest, resp *fuse.OpenR
|
|||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
|
|
||||||
fs.Debug(o, "File.Open")
|
fs.Debugf(o, "File.Open")
|
||||||
|
|
||||||
switch {
|
switch {
|
||||||
case req.Flags.IsReadOnly():
|
case req.Flags.IsReadOnly():
|
||||||
|
@ -23,7 +23,7 @@ var _ fusefs.FS = (*FS)(nil)
|
|||||||
|
|
||||||
// Root returns the root node
|
// Root returns the root node
|
||||||
func (f *FS) Root() (fusefs.Node, error) {
|
func (f *FS) Root() (fusefs.Node, error) {
|
||||||
fs.Debug(f.f, "Root()")
|
fs.Debugf(f.f, "Root()")
|
||||||
fsDir := &fs.Dir{
|
fsDir := &fs.Dir{
|
||||||
Name: "",
|
Name: "",
|
||||||
When: time.Now(),
|
When: time.Now(),
|
||||||
@ -75,7 +75,7 @@ func mountOptions(device string) (options []fuse.MountOption) {
|
|||||||
// returns an error, and an error channel for the serve process to
|
// returns an error, and an error channel for the serve process to
|
||||||
// report an error when fusermount is called.
|
// report an error when fusermount is called.
|
||||||
func mount(f fs.Fs, mountpoint string) (<-chan error, error) {
|
func mount(f fs.Fs, mountpoint string) (<-chan error, error) {
|
||||||
fs.Debug(f, "Mounting on %q", mountpoint)
|
fs.Debugf(f, "Mounting on %q", mountpoint)
|
||||||
c, err := fuse.Mount(mountpoint, mountOptions(f.Name()+":"+f.Root())...)
|
c, err := fuse.Mount(mountpoint, mountOptions(f.Name()+":"+f.Root())...)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
|
@ -143,7 +143,7 @@ mount won't do that, so will be less reliable than the rclone command.
|
|||||||
func Mount(f fs.Fs, mountpoint string) error {
|
func Mount(f fs.Fs, mountpoint string) error {
|
||||||
if debugFUSE {
|
if debugFUSE {
|
||||||
fuse.Debug = func(msg interface{}) {
|
fuse.Debug = func(msg interface{}) {
|
||||||
fs.Debug("fuse", "%v", msg)
|
fs.Debugf("fuse", "%v", msg)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -50,23 +50,23 @@ func (fh *ReadFileHandle) seek(offset int64, reopen bool) error {
|
|||||||
// Can we seek it directly?
|
// Can we seek it directly?
|
||||||
oldReader := fh.r.GetReader()
|
oldReader := fh.r.GetReader()
|
||||||
if do, ok := oldReader.(io.Seeker); !reopen && ok {
|
if do, ok := oldReader.(io.Seeker); !reopen && ok {
|
||||||
fs.Debug(fh.o, "ReadFileHandle.seek from %d to %d (io.Seeker)", fh.offset, offset)
|
fs.Debugf(fh.o, "ReadFileHandle.seek from %d to %d (io.Seeker)", fh.offset, offset)
|
||||||
_, err := do.Seek(offset, 0)
|
_, err := do.Seek(offset, 0)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
fs.Debug(fh.o, "ReadFileHandle.Read io.Seeker failed: %v", err)
|
fs.Debugf(fh.o, "ReadFileHandle.Read io.Seeker failed: %v", err)
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
fs.Debug(fh.o, "ReadFileHandle.seek from %d to %d", fh.offset, offset)
|
fs.Debugf(fh.o, "ReadFileHandle.seek from %d to %d", fh.offset, offset)
|
||||||
// if not re-open with a seek
|
// if not re-open with a seek
|
||||||
r, err := fh.o.Open(&fs.SeekOption{Offset: offset})
|
r, err := fh.o.Open(&fs.SeekOption{Offset: offset})
|
||||||
if err != nil {
|
if err != nil {
|
||||||
fs.Debug(fh.o, "ReadFileHandle.Read seek failed: %v", err)
|
fs.Debugf(fh.o, "ReadFileHandle.Read seek failed: %v", err)
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
err = oldReader.Close()
|
err = oldReader.Close()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
fs.Debug(fh.o, "ReadFileHandle.Read seek close old failed: %v", err)
|
fs.Debugf(fh.o, "ReadFileHandle.Read seek close old failed: %v", err)
|
||||||
}
|
}
|
||||||
// fh.r = fs.NewAccount(r, fh.o) // account the transfer
|
// fh.r = fs.NewAccount(r, fh.o) // account the transfer
|
||||||
fh.r.UpdateReader(r)
|
fh.r.UpdateReader(r)
|
||||||
@ -79,9 +79,9 @@ func (fh *ReadFileHandle) seek(offset int64, reopen bool) error {
|
|||||||
func (fh *ReadFileHandle) Read(ctx context.Context, req *fuse.ReadRequest, resp *fuse.ReadResponse) (err error) {
|
func (fh *ReadFileHandle) Read(ctx context.Context, req *fuse.ReadRequest, resp *fuse.ReadResponse) (err error) {
|
||||||
fh.mu.Lock()
|
fh.mu.Lock()
|
||||||
defer fh.mu.Unlock()
|
defer fh.mu.Unlock()
|
||||||
fs.Debug(fh.o, "ReadFileHandle.Read size %d offset %d", req.Size, req.Offset)
|
fs.Debugf(fh.o, "ReadFileHandle.Read size %d offset %d", req.Size, req.Offset)
|
||||||
if fh.closed {
|
if fh.closed {
|
||||||
fs.ErrorLog(fh.o, "ReadFileHandle.Read error: %v", errClosedFileHandle)
|
fs.Errorf(fh.o, "ReadFileHandle.Read error: %v", errClosedFileHandle)
|
||||||
return errClosedFileHandle
|
return errClosedFileHandle
|
||||||
}
|
}
|
||||||
doSeek := req.Offset != fh.offset
|
doSeek := req.Offset != fh.offset
|
||||||
@ -96,7 +96,7 @@ func (fh *ReadFileHandle) Read(ctx context.Context, req *fuse.ReadRequest, resp
|
|||||||
// file - if so just return EOF leaving the underlying
|
// file - if so just return EOF leaving the underlying
|
||||||
// file in an unchanged state.
|
// file in an unchanged state.
|
||||||
if req.Offset >= fh.o.Size() {
|
if req.Offset >= fh.o.Size() {
|
||||||
fs.Debug(fh.o, "ReadFileHandle.Read attempt to read beyond end of file: %d > %d", req.Offset, fh.o.Size())
|
fs.Debugf(fh.o, "ReadFileHandle.Read attempt to read beyond end of file: %d > %d", req.Offset, fh.o.Size())
|
||||||
resp.Data = nil
|
resp.Data = nil
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
@ -129,16 +129,16 @@ func (fh *ReadFileHandle) Read(ctx context.Context, req *fuse.ReadRequest, resp
|
|||||||
break
|
break
|
||||||
}
|
}
|
||||||
retries++
|
retries++
|
||||||
fs.ErrorLog(fh.o, "ReadFileHandle.Read error: low level retry %d/%d: %v", retries, fs.Config.LowLevelRetries, err)
|
fs.Errorf(fh.o, "ReadFileHandle.Read error: low level retry %d/%d: %v", retries, fs.Config.LowLevelRetries, err)
|
||||||
doSeek = true
|
doSeek = true
|
||||||
doReopen = true
|
doReopen = true
|
||||||
}
|
}
|
||||||
if err != nil {
|
if err != nil {
|
||||||
fs.ErrorLog(fh.o, "ReadFileHandle.Read error: %v", err)
|
fs.Errorf(fh.o, "ReadFileHandle.Read error: %v", err)
|
||||||
} else {
|
} else {
|
||||||
resp.Data = buf[:n]
|
resp.Data = buf[:n]
|
||||||
fh.offset = newOffset
|
fh.offset = newOffset
|
||||||
fs.Debug(fh.o, "ReadFileHandle.Read OK")
|
fs.Debugf(fh.o, "ReadFileHandle.Read OK")
|
||||||
}
|
}
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
@ -165,7 +165,7 @@ var _ fusefs.HandleFlusher = (*ReadFileHandle)(nil)
|
|||||||
func (fh *ReadFileHandle) Flush(ctx context.Context, req *fuse.FlushRequest) error {
|
func (fh *ReadFileHandle) Flush(ctx context.Context, req *fuse.FlushRequest) error {
|
||||||
fh.mu.Lock()
|
fh.mu.Lock()
|
||||||
defer fh.mu.Unlock()
|
defer fh.mu.Unlock()
|
||||||
fs.Debug(fh.o, "ReadFileHandle.Flush")
|
fs.Debugf(fh.o, "ReadFileHandle.Flush")
|
||||||
|
|
||||||
// Ignore the Flush as there is nothing we can sensibly do and
|
// Ignore the Flush as there is nothing we can sensibly do and
|
||||||
// it seems quite common for Flush to be called from
|
// it seems quite common for Flush to be called from
|
||||||
@ -174,17 +174,17 @@ func (fh *ReadFileHandle) Flush(ctx context.Context, req *fuse.FlushRequest) err
|
|||||||
// If Read hasn't been called then ignore the Flush - Release
|
// If Read hasn't been called then ignore the Flush - Release
|
||||||
// will pick it up
|
// will pick it up
|
||||||
if !fh.readCalled {
|
if !fh.readCalled {
|
||||||
fs.Debug(fh.o, "ReadFileHandle.Flush ignoring flush on unread handle")
|
fs.Debugf(fh.o, "ReadFileHandle.Flush ignoring flush on unread handle")
|
||||||
return nil
|
return nil
|
||||||
|
|
||||||
}
|
}
|
||||||
err := fh.close()
|
err := fh.close()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
fs.ErrorLog(fh.o, "ReadFileHandle.Flush error: %v", err)
|
fs.Errorf(fh.o, "ReadFileHandle.Flush error: %v", err)
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
fs.Debug(fh.o, "ReadFileHandle.Flush OK")
|
fs.Debugf(fh.o, "ReadFileHandle.Flush OK")
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -198,15 +198,15 @@ func (fh *ReadFileHandle) Release(ctx context.Context, req *fuse.ReleaseRequest)
|
|||||||
fh.mu.Lock()
|
fh.mu.Lock()
|
||||||
defer fh.mu.Unlock()
|
defer fh.mu.Unlock()
|
||||||
if fh.closed {
|
if fh.closed {
|
||||||
fs.Debug(fh.o, "ReadFileHandle.Release nothing to do")
|
fs.Debugf(fh.o, "ReadFileHandle.Release nothing to do")
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
fs.Debug(fh.o, "ReadFileHandle.Release closing")
|
fs.Debugf(fh.o, "ReadFileHandle.Release closing")
|
||||||
err := fh.close()
|
err := fh.close()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
fs.ErrorLog(fh.o, "ReadFileHandle.Release error: %v", err)
|
fs.Errorf(fh.o, "ReadFileHandle.Release error: %v", err)
|
||||||
} else {
|
} else {
|
||||||
fs.Debug(fh.o, "ReadFileHandle.Release OK")
|
fs.Debugf(fh.o, "ReadFileHandle.Release OK")
|
||||||
}
|
}
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
@ -54,11 +54,11 @@ var _ fusefs.HandleWriter = (*WriteFileHandle)(nil)
|
|||||||
|
|
||||||
// Write data to the file handle
|
// Write data to the file handle
|
||||||
func (fh *WriteFileHandle) Write(ctx context.Context, req *fuse.WriteRequest, resp *fuse.WriteResponse) error {
|
func (fh *WriteFileHandle) Write(ctx context.Context, req *fuse.WriteRequest, resp *fuse.WriteResponse) error {
|
||||||
fs.Debug(fh.remote, "WriteFileHandle.Write len=%d", len(req.Data))
|
fs.Debugf(fh.remote, "WriteFileHandle.Write len=%d", len(req.Data))
|
||||||
fh.mu.Lock()
|
fh.mu.Lock()
|
||||||
defer fh.mu.Unlock()
|
defer fh.mu.Unlock()
|
||||||
if fh.closed {
|
if fh.closed {
|
||||||
fs.ErrorLog(fh.remote, "WriteFileHandle.Write error: %v", errClosedFileHandle)
|
fs.Errorf(fh.remote, "WriteFileHandle.Write error: %v", errClosedFileHandle)
|
||||||
return errClosedFileHandle
|
return errClosedFileHandle
|
||||||
}
|
}
|
||||||
fh.writeCalled = true
|
fh.writeCalled = true
|
||||||
@ -67,10 +67,10 @@ func (fh *WriteFileHandle) Write(ctx context.Context, req *fuse.WriteRequest, re
|
|||||||
resp.Size = n
|
resp.Size = n
|
||||||
fh.file.written(int64(n))
|
fh.file.written(int64(n))
|
||||||
if err != nil {
|
if err != nil {
|
||||||
fs.ErrorLog(fh.remote, "WriteFileHandle.Write error: %v", err)
|
fs.Errorf(fh.remote, "WriteFileHandle.Write error: %v", err)
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
fs.Debug(fh.remote, "WriteFileHandle.Write OK (%d bytes written)", n)
|
fs.Debugf(fh.remote, "WriteFileHandle.Write OK (%d bytes written)", n)
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -119,19 +119,19 @@ var _ fusefs.HandleFlusher = (*WriteFileHandle)(nil)
|
|||||||
func (fh *WriteFileHandle) Flush(ctx context.Context, req *fuse.FlushRequest) error {
|
func (fh *WriteFileHandle) Flush(ctx context.Context, req *fuse.FlushRequest) error {
|
||||||
fh.mu.Lock()
|
fh.mu.Lock()
|
||||||
defer fh.mu.Unlock()
|
defer fh.mu.Unlock()
|
||||||
fs.Debug(fh.remote, "WriteFileHandle.Flush")
|
fs.Debugf(fh.remote, "WriteFileHandle.Flush")
|
||||||
// If Write hasn't been called then ignore the Flush - Release
|
// If Write hasn't been called then ignore the Flush - Release
|
||||||
// will pick it up
|
// will pick it up
|
||||||
if !fh.writeCalled {
|
if !fh.writeCalled {
|
||||||
fs.Debug(fh.remote, "WriteFileHandle.Flush ignoring flush on unwritten handle")
|
fs.Debugf(fh.remote, "WriteFileHandle.Flush ignoring flush on unwritten handle")
|
||||||
return nil
|
return nil
|
||||||
|
|
||||||
}
|
}
|
||||||
err := fh.close()
|
err := fh.close()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
fs.ErrorLog(fh.remote, "WriteFileHandle.Flush error: %v", err)
|
fs.Errorf(fh.remote, "WriteFileHandle.Flush error: %v", err)
|
||||||
} else {
|
} else {
|
||||||
fs.Debug(fh.remote, "WriteFileHandle.Flush OK")
|
fs.Debugf(fh.remote, "WriteFileHandle.Flush OK")
|
||||||
}
|
}
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
@ -146,15 +146,15 @@ func (fh *WriteFileHandle) Release(ctx context.Context, req *fuse.ReleaseRequest
|
|||||||
fh.mu.Lock()
|
fh.mu.Lock()
|
||||||
defer fh.mu.Unlock()
|
defer fh.mu.Unlock()
|
||||||
if fh.closed {
|
if fh.closed {
|
||||||
fs.Debug(fh.remote, "WriteFileHandle.Release nothing to do")
|
fs.Debugf(fh.remote, "WriteFileHandle.Release nothing to do")
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
fs.Debug(fh.remote, "WriteFileHandle.Release closing")
|
fs.Debugf(fh.remote, "WriteFileHandle.Release closing")
|
||||||
err := fh.close()
|
err := fh.close()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
fs.ErrorLog(fh.remote, "WriteFileHandle.Release error: %v", err)
|
fs.Errorf(fh.remote, "WriteFileHandle.Release error: %v", err)
|
||||||
} else {
|
} else {
|
||||||
fs.Debug(fh.remote, "WriteFileHandle.Release OK")
|
fs.Debugf(fh.remote, "WriteFileHandle.Release OK")
|
||||||
}
|
}
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
@ -12,5 +12,5 @@ import (
|
|||||||
|
|
||||||
// redirectStderr to the file passed in
|
// redirectStderr to the file passed in
|
||||||
func redirectStderr(f *os.File) {
|
func redirectStderr(f *os.File) {
|
||||||
fs.ErrorLog(nil, "Can't redirect stderr to file")
|
fs.Errorf(nil, "Can't redirect stderr to file")
|
||||||
}
|
}
|
||||||
|
@ -269,7 +269,7 @@ func (f *Fs) DirMove(src fs.Fs) error {
|
|||||||
}
|
}
|
||||||
srcFs, ok := src.(*Fs)
|
srcFs, ok := src.(*Fs)
|
||||||
if !ok {
|
if !ok {
|
||||||
fs.Debug(srcFs, "Can't move directory - not same remote type")
|
fs.Debugf(srcFs, "Can't move directory - not same remote type")
|
||||||
return fs.ErrorCantDirMove
|
return fs.ErrorCantDirMove
|
||||||
}
|
}
|
||||||
return do(srcFs.Fs)
|
return do(srcFs.Fs)
|
||||||
@ -345,7 +345,7 @@ func (o *Object) Remote() string {
|
|||||||
remote := o.Object.Remote()
|
remote := o.Object.Remote()
|
||||||
decryptedName, err := o.f.cipher.DecryptFileName(remote)
|
decryptedName, err := o.f.cipher.DecryptFileName(remote)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
fs.Debug(remote, "Undecryptable file name: %v", err)
|
fs.Debugf(remote, "Undecryptable file name: %v", err)
|
||||||
return remote
|
return remote
|
||||||
}
|
}
|
||||||
return decryptedName
|
return decryptedName
|
||||||
@ -355,7 +355,7 @@ func (o *Object) Remote() string {
|
|||||||
func (o *Object) Size() int64 {
|
func (o *Object) Size() int64 {
|
||||||
size, err := o.f.cipher.DecryptedSize(o.Object.Size())
|
size, err := o.f.cipher.DecryptedSize(o.Object.Size())
|
||||||
if err != nil {
|
if err != nil {
|
||||||
fs.Debug(o, "Bad size for decrypt: %v", err)
|
fs.Debugf(o, "Bad size for decrypt: %v", err)
|
||||||
}
|
}
|
||||||
return size
|
return size
|
||||||
}
|
}
|
||||||
@ -375,7 +375,7 @@ func (o *Object) Open(options ...fs.OpenOption) (rc io.ReadCloser, err error) {
|
|||||||
offset = x.Offset
|
offset = x.Offset
|
||||||
default:
|
default:
|
||||||
if option.Mandatory() {
|
if option.Mandatory() {
|
||||||
fs.Log(o, "Unsupported mandatory option: %v", option)
|
fs.Logf(o, "Unsupported mandatory option: %v", option)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -408,7 +408,7 @@ func (f *Fs) newDir(dir *fs.Dir) *fs.Dir {
|
|||||||
remote := dir.Name
|
remote := dir.Name
|
||||||
decryptedRemote, err := f.cipher.DecryptDirName(remote)
|
decryptedRemote, err := f.cipher.DecryptDirName(remote)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
fs.Debug(remote, "Undecryptable dir name: %v", err)
|
fs.Debugf(remote, "Undecryptable dir name: %v", err)
|
||||||
} else {
|
} else {
|
||||||
new.Name = decryptedRemote
|
new.Name = decryptedRemote
|
||||||
}
|
}
|
||||||
@ -493,11 +493,11 @@ func (lo *ListOpts) Add(obj fs.Object) (abort bool) {
|
|||||||
remote := obj.Remote()
|
remote := obj.Remote()
|
||||||
decryptedRemote, err := lo.f.cipher.DecryptFileName(remote)
|
decryptedRemote, err := lo.f.cipher.DecryptFileName(remote)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
fs.Debug(remote, "Skipping undecryptable file name: %v", err)
|
fs.Debugf(remote, "Skipping undecryptable file name: %v", err)
|
||||||
return lo.ListOpts.IsFinished()
|
return lo.ListOpts.IsFinished()
|
||||||
}
|
}
|
||||||
if *cryptShowMapping {
|
if *cryptShowMapping {
|
||||||
fs.Log(decryptedRemote, "Encrypts to %q", remote)
|
fs.Logf(decryptedRemote, "Encrypts to %q", remote)
|
||||||
}
|
}
|
||||||
return lo.ListOpts.Add(lo.f.newObject(obj))
|
return lo.ListOpts.Add(lo.f.newObject(obj))
|
||||||
}
|
}
|
||||||
@ -509,11 +509,11 @@ func (lo *ListOpts) AddDir(dir *fs.Dir) (abort bool) {
|
|||||||
remote := dir.Name
|
remote := dir.Name
|
||||||
decryptedRemote, err := lo.f.cipher.DecryptDirName(remote)
|
decryptedRemote, err := lo.f.cipher.DecryptDirName(remote)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
fs.Debug(remote, "Skipping undecryptable dir name: %v", err)
|
fs.Debugf(remote, "Skipping undecryptable dir name: %v", err)
|
||||||
return lo.ListOpts.IsFinished()
|
return lo.ListOpts.IsFinished()
|
||||||
}
|
}
|
||||||
if *cryptShowMapping {
|
if *cryptShowMapping {
|
||||||
fs.Log(decryptedRemote, "Encrypts to %q", remote)
|
fs.Logf(decryptedRemote, "Encrypts to %q", remote)
|
||||||
}
|
}
|
||||||
return lo.ListOpts.AddDir(lo.f.newDir(dir))
|
return lo.ListOpts.AddDir(lo.f.newDir(dir))
|
||||||
}
|
}
|
||||||
@ -523,7 +523,7 @@ func (lo *ListOpts) AddDir(dir *fs.Dir) (abort bool) {
|
|||||||
func (lo *ListOpts) IncludeDirectory(remote string) bool {
|
func (lo *ListOpts) IncludeDirectory(remote string) bool {
|
||||||
decryptedRemote, err := lo.f.cipher.DecryptDirName(remote)
|
decryptedRemote, err := lo.f.cipher.DecryptDirName(remote)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
fs.Debug(remote, "Not including undecryptable directory name: %v", err)
|
fs.Debugf(remote, "Not including undecryptable directory name: %v", err)
|
||||||
return false
|
return false
|
||||||
}
|
}
|
||||||
return lo.ListOpts.IncludeDirectory(decryptedRemote)
|
return lo.ListOpts.IncludeDirectory(decryptedRemote)
|
||||||
|
@ -38,7 +38,7 @@ func listDir(f ListDirer, out fs.ListOpts, dirID string, path string) {
|
|||||||
jobs, err := f.ListDir(out, job)
|
jobs, err := f.ListDir(out, job)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
out.SetError(err)
|
out.SetError(err)
|
||||||
fs.Debug(f, "Error reading %s: %s", path, err)
|
fs.Debugf(f, "Error reading %s: %s", path, err)
|
||||||
} else {
|
} else {
|
||||||
traversing.Add(len(jobs))
|
traversing.Add(len(jobs))
|
||||||
go func() {
|
go func() {
|
||||||
|
@ -440,7 +440,7 @@ func (f *Fs) findExportFormat(filepath string, item *drive.File) (extension, lin
|
|||||||
// Warn about unknown export formats
|
// Warn about unknown export formats
|
||||||
for mimeType := range item.ExportLinks {
|
for mimeType := range item.ExportLinks {
|
||||||
if _, ok := mimeTypeToExtension[mimeType]; !ok {
|
if _, ok := mimeTypeToExtension[mimeType]; !ok {
|
||||||
fs.Debug(filepath, "Unknown export type %q - ignoring", mimeType)
|
fs.Debugf(filepath, "Unknown export type %q - ignoring", mimeType)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -458,7 +458,7 @@ func (f *Fs) findExportFormat(filepath string, item *drive.File) (extension, lin
|
|||||||
|
|
||||||
// ListDir reads the directory specified by the job into out, returning any more jobs
|
// ListDir reads the directory specified by the job into out, returning any more jobs
|
||||||
func (f *Fs) ListDir(out fs.ListOpts, job dircache.ListDirJob) (jobs []dircache.ListDirJob, err error) {
|
func (f *Fs) ListDir(out fs.ListOpts, job dircache.ListDirJob) (jobs []dircache.ListDirJob, err error) {
|
||||||
fs.Debug(f, "Reading %q", job.Path)
|
fs.Debugf(f, "Reading %q", job.Path)
|
||||||
_, err = f.listAll(job.DirID, "", false, false, func(item *drive.File) bool {
|
_, err = f.listAll(job.DirID, "", false, false, func(item *drive.File) bool {
|
||||||
remote := job.Path + item.Title
|
remote := job.Path + item.Title
|
||||||
switch {
|
switch {
|
||||||
@ -493,7 +493,7 @@ func (f *Fs) ListDir(out fs.ListOpts, job dircache.ListDirJob) (jobs []dircache.
|
|||||||
// If item has export links then it is a google doc
|
// If item has export links then it is a google doc
|
||||||
extension, link := f.findExportFormat(remote, item)
|
extension, link := f.findExportFormat(remote, item)
|
||||||
if extension == "" {
|
if extension == "" {
|
||||||
fs.Debug(remote, "No export formats found")
|
fs.Debugf(remote, "No export formats found")
|
||||||
} else {
|
} else {
|
||||||
o, err := f.newObjectWithInfo(remote+"."+extension, item)
|
o, err := f.newObjectWithInfo(remote+"."+extension, item)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
@ -509,15 +509,15 @@ func (f *Fs) ListDir(out fs.ListOpts, job dircache.ListDirJob) (jobs []dircache.
|
|||||||
return true
|
return true
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
fs.Debug(f, "Skip google document: %q", remote)
|
fs.Debugf(f, "Skip google document: %q", remote)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
default:
|
default:
|
||||||
fs.Debug(remote, "Ignoring unknown object")
|
fs.Debugf(remote, "Ignoring unknown object")
|
||||||
}
|
}
|
||||||
return false
|
return false
|
||||||
})
|
})
|
||||||
fs.Debug(f, "Finished reading %q", job.Path)
|
fs.Debugf(f, "Finished reading %q", job.Path)
|
||||||
return jobs, err
|
return jobs, err
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -679,7 +679,7 @@ func (f *Fs) Precision() time.Duration {
|
|||||||
func (f *Fs) Copy(src fs.Object, remote string) (fs.Object, error) {
|
func (f *Fs) Copy(src fs.Object, remote string) (fs.Object, error) {
|
||||||
srcObj, ok := src.(*Object)
|
srcObj, ok := src.(*Object)
|
||||||
if !ok {
|
if !ok {
|
||||||
fs.Debug(src, "Can't copy - not same remote type")
|
fs.Debugf(src, "Can't copy - not same remote type")
|
||||||
return nil, fs.ErrorCantCopy
|
return nil, fs.ErrorCantCopy
|
||||||
}
|
}
|
||||||
if srcObj.isDocument {
|
if srcObj.isDocument {
|
||||||
@ -744,7 +744,7 @@ func (f *Fs) Purge() error {
|
|||||||
func (f *Fs) Move(src fs.Object, remote string) (fs.Object, error) {
|
func (f *Fs) Move(src fs.Object, remote string) (fs.Object, error) {
|
||||||
srcObj, ok := src.(*Object)
|
srcObj, ok := src.(*Object)
|
||||||
if !ok {
|
if !ok {
|
||||||
fs.Debug(src, "Can't move - not same remote type")
|
fs.Debugf(src, "Can't move - not same remote type")
|
||||||
return nil, fs.ErrorCantMove
|
return nil, fs.ErrorCantMove
|
||||||
}
|
}
|
||||||
if srcObj.isDocument {
|
if srcObj.isDocument {
|
||||||
@ -788,7 +788,7 @@ func (f *Fs) Move(src fs.Object, remote string) (fs.Object, error) {
|
|||||||
func (f *Fs) DirMove(src fs.Fs) error {
|
func (f *Fs) DirMove(src fs.Fs) error {
|
||||||
srcFs, ok := src.(*Fs)
|
srcFs, ok := src.(*Fs)
|
||||||
if !ok {
|
if !ok {
|
||||||
fs.Debug(srcFs, "Can't move directory - not same remote type")
|
fs.Debugf(srcFs, "Can't move directory - not same remote type")
|
||||||
return fs.ErrorCantDirMove
|
return fs.ErrorCantDirMove
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -799,7 +799,7 @@ func (f *Fs) DirMove(src fs.Fs) error {
|
|||||||
|
|
||||||
// Refuse to move to or from the root
|
// Refuse to move to or from the root
|
||||||
if f.root == "" || srcFs.root == "" {
|
if f.root == "" || srcFs.root == "" {
|
||||||
fs.Debug(src, "DirMove error: Can't move root")
|
fs.Debugf(src, "DirMove error: Can't move root")
|
||||||
return errors.New("can't move root directory")
|
return errors.New("can't move root directory")
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -871,12 +871,12 @@ func (o *Object) Size() int64 {
|
|||||||
// how big it is
|
// how big it is
|
||||||
_, res, err := o.httpResponse("HEAD", nil)
|
_, res, err := o.httpResponse("HEAD", nil)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
fs.ErrorLog(o, "Error reading size: %v", err)
|
fs.Errorf(o, "Error reading size: %v", err)
|
||||||
return 0
|
return 0
|
||||||
}
|
}
|
||||||
_ = res.Body.Close()
|
_ = res.Body.Close()
|
||||||
o.bytes = res.ContentLength
|
o.bytes = res.ContentLength
|
||||||
// fs.Debug(o, "Read size of document: %v", o.bytes)
|
// fs.Debugf(o, "Read size of document: %v", o.bytes)
|
||||||
}
|
}
|
||||||
return o.bytes
|
return o.bytes
|
||||||
}
|
}
|
||||||
@ -929,12 +929,12 @@ func (o *Object) readMetaData() (err error) {
|
|||||||
func (o *Object) ModTime() time.Time {
|
func (o *Object) ModTime() time.Time {
|
||||||
err := o.readMetaData()
|
err := o.readMetaData()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
fs.Log(o, "Failed to read metadata: %v", err)
|
fs.Logf(o, "Failed to read metadata: %v", err)
|
||||||
return time.Now()
|
return time.Now()
|
||||||
}
|
}
|
||||||
modTime, err := time.Parse(timeFormatIn, o.modifiedDate)
|
modTime, err := time.Parse(timeFormatIn, o.modifiedDate)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
fs.Log(o, "Failed to read mtime from object: %v", err)
|
fs.Logf(o, "Failed to read mtime from object: %v", err)
|
||||||
return time.Now()
|
return time.Now()
|
||||||
}
|
}
|
||||||
return modTime
|
return modTime
|
||||||
@ -1012,7 +1012,7 @@ func (file *openFile) Read(p []byte) (n int, err error) {
|
|||||||
func (file *openFile) Close() (err error) {
|
func (file *openFile) Close() (err error) {
|
||||||
// If end of file, update bytes read
|
// If end of file, update bytes read
|
||||||
if file.eof {
|
if file.eof {
|
||||||
// fs.Debug(file.o, "Updating size of doc after download to %v", file.bytes)
|
// fs.Debugf(file.o, "Updating size of doc after download to %v", file.bytes)
|
||||||
file.o.bytes = file.bytes
|
file.o.bytes = file.bytes
|
||||||
}
|
}
|
||||||
return file.in.Close()
|
return file.in.Close()
|
||||||
@ -1103,7 +1103,7 @@ func (o *Object) Remove() error {
|
|||||||
func (o *Object) MimeType() string {
|
func (o *Object) MimeType() string {
|
||||||
err := o.readMetaData()
|
err := o.readMetaData()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
fs.Log(o, "Failed to read metadata: %v", err)
|
fs.Logf(o, "Failed to read metadata: %v", err)
|
||||||
return ""
|
return ""
|
||||||
}
|
}
|
||||||
return o.mimeType
|
return o.mimeType
|
||||||
|
@ -209,7 +209,7 @@ func (rx *resumableUpload) Upload() (*drive.File, error) {
|
|||||||
|
|
||||||
// Transfer the chunk
|
// Transfer the chunk
|
||||||
err = rx.f.pacer.Call(func() (bool, error) {
|
err = rx.f.pacer.Call(func() (bool, error) {
|
||||||
fs.Debug(rx.remote, "Sending chunk %d length %d", start, reqSize)
|
fs.Debugf(rx.remote, "Sending chunk %d length %d", start, reqSize)
|
||||||
StatusCode, err = rx.transferChunk(start, buf)
|
StatusCode, err = rx.transferChunk(start, buf)
|
||||||
again, err := shouldRetry(err)
|
again, err := shouldRetry(err)
|
||||||
if StatusCode == statusResumeIncomplete || StatusCode == http.StatusCreated || StatusCode == http.StatusOK {
|
if StatusCode == statusResumeIncomplete || StatusCode == http.StatusCreated || StatusCode == http.StatusOK {
|
||||||
|
@ -279,7 +279,7 @@ func (f *Fs) list(out fs.ListOpts, dir string) {
|
|||||||
out.SetError(err)
|
out.SetError(err)
|
||||||
break
|
break
|
||||||
}
|
}
|
||||||
fs.Debug(f, "%d delta entries received", len(deltaPage.Entries))
|
fs.Debugf(f, "%d delta entries received", len(deltaPage.Entries))
|
||||||
for i := range deltaPage.Entries {
|
for i := range deltaPage.Entries {
|
||||||
deltaEntry := &deltaPage.Entries[i]
|
deltaEntry := &deltaPage.Entries[i]
|
||||||
entry := deltaEntry.Entry
|
entry := deltaEntry.Entry
|
||||||
@ -287,7 +287,7 @@ func (f *Fs) list(out fs.ListOpts, dir string) {
|
|||||||
// This notifies of a deleted object
|
// This notifies of a deleted object
|
||||||
} else {
|
} else {
|
||||||
if len(entry.Path) <= 1 || entry.Path[0] != '/' {
|
if len(entry.Path) <= 1 || entry.Path[0] != '/' {
|
||||||
fs.Log(f, "dropbox API inconsistency: a path should always start with a slash and be at least 2 characters: %s", entry.Path)
|
fs.Logf(f, "dropbox API inconsistency: a path should always start with a slash and be at least 2 characters: %s", entry.Path)
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -500,7 +500,7 @@ func (f *Fs) Precision() time.Duration {
|
|||||||
func (f *Fs) Copy(src fs.Object, remote string) (fs.Object, error) {
|
func (f *Fs) Copy(src fs.Object, remote string) (fs.Object, error) {
|
||||||
srcObj, ok := src.(*Object)
|
srcObj, ok := src.(*Object)
|
||||||
if !ok {
|
if !ok {
|
||||||
fs.Debug(src, "Can't copy - not same remote type")
|
fs.Debugf(src, "Can't copy - not same remote type")
|
||||||
return nil, fs.ErrorCantCopy
|
return nil, fs.ErrorCantCopy
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -543,7 +543,7 @@ func (f *Fs) Purge() error {
|
|||||||
func (f *Fs) Move(src fs.Object, remote string) (fs.Object, error) {
|
func (f *Fs) Move(src fs.Object, remote string) (fs.Object, error) {
|
||||||
srcObj, ok := src.(*Object)
|
srcObj, ok := src.(*Object)
|
||||||
if !ok {
|
if !ok {
|
||||||
fs.Debug(src, "Can't move - not same remote type")
|
fs.Debugf(src, "Can't move - not same remote type")
|
||||||
return nil, fs.ErrorCantMove
|
return nil, fs.ErrorCantMove
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -573,7 +573,7 @@ func (f *Fs) Move(src fs.Object, remote string) (fs.Object, error) {
|
|||||||
func (f *Fs) DirMove(src fs.Fs) error {
|
func (f *Fs) DirMove(src fs.Fs) error {
|
||||||
srcFs, ok := src.(*Fs)
|
srcFs, ok := src.(*Fs)
|
||||||
if !ok {
|
if !ok {
|
||||||
fs.Debug(srcFs, "Can't move directory - not same remote type")
|
fs.Debugf(srcFs, "Can't move directory - not same remote type")
|
||||||
return fs.ErrorCantDirMove
|
return fs.ErrorCantDirMove
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -699,7 +699,7 @@ func (o *Object) readMetaData() (err error) {
|
|||||||
func (o *Object) ModTime() time.Time {
|
func (o *Object) ModTime() time.Time {
|
||||||
err := o.readMetaData()
|
err := o.readMetaData()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
fs.Log(o, "Failed to read metadata: %v", err)
|
fs.Logf(o, "Failed to read metadata: %v", err)
|
||||||
return time.Now()
|
return time.Now()
|
||||||
}
|
}
|
||||||
return o.modTime
|
return o.modTime
|
||||||
@ -728,7 +728,7 @@ func (o *Object) Open(options ...fs.OpenOption) (in io.ReadCloser, err error) {
|
|||||||
offset = x.Offset
|
offset = x.Offset
|
||||||
default:
|
default:
|
||||||
if option.Mandatory() {
|
if option.Mandatory() {
|
||||||
fs.Log(o, "Unsupported mandatory option: %v", option)
|
fs.Logf(o, "Unsupported mandatory option: %v", option)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -752,7 +752,7 @@ func (o *Object) Open(options ...fs.OpenOption) (in io.ReadCloser, err error) {
|
|||||||
func (o *Object) Update(in io.Reader, src fs.ObjectInfo) error {
|
func (o *Object) Update(in io.Reader, src fs.ObjectInfo) error {
|
||||||
remote := o.remotePath()
|
remote := o.remotePath()
|
||||||
if ignoredFiles.MatchString(remote) {
|
if ignoredFiles.MatchString(remote) {
|
||||||
fs.Log(o, "File name disallowed - not uploading")
|
fs.Logf(o, "File name disallowed - not uploading")
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
entry, err := o.fs.db.UploadByChunk(ioutil.NopCloser(in), int(uploadChunkSize), remote, true, "")
|
entry, err := o.fs.db.UploadByChunk(ioutil.NopCloser(in), int(uploadChunkSize), remote, true, "")
|
||||||
@ -773,7 +773,7 @@ func (o *Object) Remove() error {
|
|||||||
func (o *Object) MimeType() string {
|
func (o *Object) MimeType() string {
|
||||||
err := o.readMetaData()
|
err := o.readMetaData()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
fs.Log(o, "Failed to read metadata: %v", err)
|
fs.Logf(o, "Failed to read metadata: %v", err)
|
||||||
return ""
|
return ""
|
||||||
}
|
}
|
||||||
return o.mimeType
|
return o.mimeType
|
||||||
|
@ -52,7 +52,7 @@ func (tree *nameTreeNode) getTreeNode(path string) *nameTreeNode {
|
|||||||
for _, component := range strings.Split(path, "/") {
|
for _, component := range strings.Split(path, "/") {
|
||||||
if len(component) == 0 {
|
if len(component) == 0 {
|
||||||
fs.Stats.Error()
|
fs.Stats.Error()
|
||||||
fs.ErrorLog(tree, "getTreeNode: path component is empty (full path %q)", path)
|
fs.Errorf(tree, "getTreeNode: path component is empty (full path %q)", path)
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -79,7 +79,7 @@ func (tree *nameTreeNode) PutCaseCorrectPath(caseCorrectPath string) {
|
|||||||
for _, component := range strings.Split(caseCorrectPath, "/") {
|
for _, component := range strings.Split(caseCorrectPath, "/") {
|
||||||
if len(component) == 0 {
|
if len(component) == 0 {
|
||||||
fs.Stats.Error()
|
fs.Stats.Error()
|
||||||
fs.ErrorLog(tree, "PutCaseCorrectPath: path component is empty (full path %q)", caseCorrectPath)
|
fs.Errorf(tree, "PutCaseCorrectPath: path component is empty (full path %q)", caseCorrectPath)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
lowercase := strings.ToLower(component)
|
lowercase := strings.ToLower(component)
|
||||||
@ -96,7 +96,7 @@ func (tree *nameTreeNode) PutCaseCorrectPath(caseCorrectPath string) {
|
|||||||
func (tree *nameTreeNode) PutCaseCorrectDirectoryName(parentPath string, caseCorrectDirectoryName string) {
|
func (tree *nameTreeNode) PutCaseCorrectDirectoryName(parentPath string, caseCorrectDirectoryName string) {
|
||||||
if len(caseCorrectDirectoryName) == 0 {
|
if len(caseCorrectDirectoryName) == 0 {
|
||||||
fs.Stats.Error()
|
fs.Stats.Error()
|
||||||
fs.ErrorLog(tree, "PutCaseCorrectDirectoryName: empty caseCorrectDirectoryName is not allowed (parentPath: %q)", parentPath)
|
fs.Errorf(tree, "PutCaseCorrectDirectoryName: empty caseCorrectDirectoryName is not allowed (parentPath: %q)", parentPath)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -113,7 +113,7 @@ func (tree *nameTreeNode) PutCaseCorrectDirectoryName(parentPath string, caseCor
|
|||||||
} else {
|
} else {
|
||||||
if len(directory.CaseCorrectName) > 0 {
|
if len(directory.CaseCorrectName) > 0 {
|
||||||
fs.Stats.Error()
|
fs.Stats.Error()
|
||||||
fs.ErrorLog(tree, "PutCaseCorrectDirectoryName: directory %q is already exists under parent path %q", caseCorrectDirectoryName, parentPath)
|
fs.Errorf(tree, "PutCaseCorrectDirectoryName: directory %q is already exists under parent path %q", caseCorrectDirectoryName, parentPath)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -129,7 +129,7 @@ func (tree *nameTreeNode) PutFile(parentPath string, caseCorrectFileName string,
|
|||||||
|
|
||||||
if node.Files[caseCorrectFileName] != nil {
|
if node.Files[caseCorrectFileName] != nil {
|
||||||
fs.Stats.Error()
|
fs.Stats.Error()
|
||||||
fs.ErrorLog(tree, "PutFile: file %q is already exists at %q", caseCorrectFileName, parentPath)
|
fs.Errorf(tree, "PutFile: file %q is already exists at %q", caseCorrectFileName, parentPath)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -148,7 +148,7 @@ func (tree *nameTreeNode) GetPathWithCorrectCase(path string) *string {
|
|||||||
for _, component := range strings.Split(path, "/") {
|
for _, component := range strings.Split(path, "/") {
|
||||||
if component == "" {
|
if component == "" {
|
||||||
fs.Stats.Error()
|
fs.Stats.Error()
|
||||||
fs.ErrorLog(tree, "GetPathWithCorrectCase: path component is empty (full path %q)", path)
|
fs.Errorf(tree, "GetPathWithCorrectCase: path component is empty (full path %q)", path)
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -188,7 +188,7 @@ func (tree *nameTreeNode) walkFilesRec(currentPath string, walkFunc nameTreeFile
|
|||||||
caseCorrectName := directory.CaseCorrectName
|
caseCorrectName := directory.CaseCorrectName
|
||||||
if caseCorrectName == "" {
|
if caseCorrectName == "" {
|
||||||
fs.Stats.Error()
|
fs.Stats.Error()
|
||||||
fs.ErrorLog(tree, "WalkFiles: exact name of the directory %q is unknown (parent path: %q)", lowerCaseName, currentPath)
|
fs.Errorf(tree, "WalkFiles: exact name of the directory %q is unknown (parent path: %q)", lowerCaseName, currentPath)
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -33,7 +33,7 @@ func startTokenBucket() {
|
|||||||
|
|
||||||
if currLimit.bandwidth > 0 {
|
if currLimit.bandwidth > 0 {
|
||||||
tokenBucket = tb.NewBucket(int64(currLimit.bandwidth), 100*time.Millisecond)
|
tokenBucket = tb.NewBucket(int64(currLimit.bandwidth), 100*time.Millisecond)
|
||||||
Log(nil, "Starting bandwidth limiter at %vBytes/s", &currLimit.bandwidth)
|
Logf(nil, "Starting bandwidth limiter at %vBytes/s", &currLimit.bandwidth)
|
||||||
|
|
||||||
// Start the SIGUSR2 signal handler to toggle bandwidth.
|
// Start the SIGUSR2 signal handler to toggle bandwidth.
|
||||||
// This function does nothing in windows systems.
|
// This function does nothing in windows systems.
|
||||||
@ -60,17 +60,17 @@ func startTokenTicker() {
|
|||||||
if tokenBucket != nil {
|
if tokenBucket != nil {
|
||||||
err := tokenBucket.Close()
|
err := tokenBucket.Close()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
Log(nil, "Error closing token bucket: %v", err)
|
Logf(nil, "Error closing token bucket: %v", err)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Set new bandwidth. If unlimited, set tokenbucket to nil.
|
// Set new bandwidth. If unlimited, set tokenbucket to nil.
|
||||||
if limitNow.bandwidth > 0 {
|
if limitNow.bandwidth > 0 {
|
||||||
tokenBucket = tb.NewBucket(int64(limitNow.bandwidth), 100*time.Millisecond)
|
tokenBucket = tb.NewBucket(int64(limitNow.bandwidth), 100*time.Millisecond)
|
||||||
Log(nil, "Scheduled bandwidth change. Limit set to %vBytes/s", &limitNow.bandwidth)
|
Logf(nil, "Scheduled bandwidth change. Limit set to %vBytes/s", &limitNow.bandwidth)
|
||||||
} else {
|
} else {
|
||||||
tokenBucket = nil
|
tokenBucket = nil
|
||||||
Log(nil, "Scheduled bandwidth change. Bandwidth limits disabled")
|
Logf(nil, "Scheduled bandwidth change. Bandwidth limits disabled")
|
||||||
}
|
}
|
||||||
|
|
||||||
currLimit = limitNow
|
currLimit = limitNow
|
||||||
@ -203,7 +203,7 @@ Elapsed time: %10v
|
|||||||
|
|
||||||
// Log outputs the StatsInfo to the log
|
// Log outputs the StatsInfo to the log
|
||||||
func (s *StatsInfo) Log() {
|
func (s *StatsInfo) Log() {
|
||||||
Log(nil, "%v\n", s)
|
Logf(nil, "%v\n", s)
|
||||||
}
|
}
|
||||||
|
|
||||||
// Bytes updates the stats for bytes bytes
|
// Bytes updates the stats for bytes bytes
|
||||||
@ -355,7 +355,7 @@ func NewAccountSizeNameWithBuffer(in io.ReadCloser, size int64, name string) *Ac
|
|||||||
const buffers = memUsed / bufSize
|
const buffers = memUsed / bufSize
|
||||||
newIn, err := newAsyncReader(in, buffers, bufSize)
|
newIn, err := newAsyncReader(in, buffers, bufSize)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
ErrorLog(name, "Failed to make buffer: %v", err)
|
Errorf(name, "Failed to make buffer: %v", err)
|
||||||
} else {
|
} else {
|
||||||
in = newIn
|
in = newIn
|
||||||
}
|
}
|
||||||
@ -591,7 +591,7 @@ func (a *accountStream) Read(p []byte) (n int, err error) {
|
|||||||
func AccountByPart(obj Object) *Account {
|
func AccountByPart(obj Object) *Account {
|
||||||
acc := Stats.inProgress.get(obj.Remote())
|
acc := Stats.inProgress.get(obj.Remote())
|
||||||
if acc == nil {
|
if acc == nil {
|
||||||
Debug(obj, "Didn't find object to account part transfer")
|
Debugf(obj, "Didn't find object to account part transfer")
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
acc.disableWholeFileAccounting()
|
acc.disableWholeFileAccounting()
|
||||||
|
@ -27,7 +27,7 @@ func startSignalHandler() {
|
|||||||
s = "enabled"
|
s = "enabled"
|
||||||
}
|
}
|
||||||
tokenBucketMu.Unlock()
|
tokenBucketMu.Unlock()
|
||||||
Log(nil, "Bandwidth limit %s by user", s)
|
Logf(nil, "Bandwidth limit %s by user", s)
|
||||||
}
|
}
|
||||||
}()
|
}()
|
||||||
}
|
}
|
||||||
|
26
fs/config.go
26
fs/config.go
@ -275,10 +275,10 @@ func makeConfigPath() string {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Default to ./.rclone.conf (current working directory)
|
// Default to ./.rclone.conf (current working directory)
|
||||||
ErrorLog(nil, "Couldn't find home directory or read HOME or XDG_CONFIG_HOME environment variables.")
|
Errorf(nil, "Couldn't find home directory or read HOME or XDG_CONFIG_HOME environment variables.")
|
||||||
ErrorLog(nil, "Defaulting to storing config in current directory.")
|
Errorf(nil, "Defaulting to storing config in current directory.")
|
||||||
ErrorLog(nil, "Use -config flag to workaround.")
|
Errorf(nil, "Use -config flag to workaround.")
|
||||||
ErrorLog(nil, "Error was: %v", err)
|
Errorf(nil, "Error was: %v", err)
|
||||||
return hiddenConfigFileName
|
return hiddenConfigFileName
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -357,7 +357,7 @@ func LoadConfig() {
|
|||||||
var err error
|
var err error
|
||||||
configData, err = loadConfigFile()
|
configData, err = loadConfigFile()
|
||||||
if err == errorConfigFileNotFound {
|
if err == errorConfigFileNotFound {
|
||||||
Log(nil, "Config file %q not found - using defaults", ConfigPath)
|
Logf(nil, "Config file %q not found - using defaults", ConfigPath)
|
||||||
configData, _ = goconfig.LoadFromReader(&bytes.Buffer{})
|
configData, _ = goconfig.LoadFromReader(&bytes.Buffer{})
|
||||||
} else if err != nil {
|
} else if err != nil {
|
||||||
log.Fatalf("Failed to load config file %q: %v", ConfigPath, err)
|
log.Fatalf("Failed to load config file %q: %v", ConfigPath, err)
|
||||||
@ -432,7 +432,7 @@ func loadConfigFile() (*goconfig.ConfigFile, error) {
|
|||||||
fmt.Println("Using RCLONE_CONFIG_PASS returned:", err)
|
fmt.Println("Using RCLONE_CONFIG_PASS returned:", err)
|
||||||
envpw = ""
|
envpw = ""
|
||||||
} else {
|
} else {
|
||||||
Debug(nil, "Using RCLONE_CONFIG_PASS password.")
|
Debugf(nil, "Using RCLONE_CONFIG_PASS password.")
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if len(configKey) == 0 {
|
if len(configKey) == 0 {
|
||||||
@ -456,7 +456,7 @@ func loadConfigFile() (*goconfig.ConfigFile, error) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Retry
|
// Retry
|
||||||
ErrorLog(nil, "Couldn't decrypt configuration, most likely wrong password.")
|
Errorf(nil, "Couldn't decrypt configuration, most likely wrong password.")
|
||||||
configKey = nil
|
configKey = nil
|
||||||
envpw = ""
|
envpw = ""
|
||||||
}
|
}
|
||||||
@ -560,7 +560,7 @@ func SaveConfig() {
|
|||||||
}
|
}
|
||||||
err = os.Chmod(ConfigPath, 0600)
|
err = os.Chmod(ConfigPath, 0600)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
ErrorLog(nil, "Failed to set permissions on config file: %v", err)
|
Errorf(nil, "Failed to set permissions on config file: %v", err)
|
||||||
}
|
}
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
@ -607,7 +607,7 @@ func SaveConfig() {
|
|||||||
|
|
||||||
err = os.Chmod(ConfigPath, 0600)
|
err = os.Chmod(ConfigPath, 0600)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
ErrorLog(nil, "Failed to set permissions on config file: %v", err)
|
Errorf(nil, "Failed to set permissions on config file: %v", err)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -809,7 +809,7 @@ func OkRemote(name string) bool {
|
|||||||
configData.DeleteSection(name)
|
configData.DeleteSection(name)
|
||||||
return true
|
return true
|
||||||
default:
|
default:
|
||||||
ErrorLog(nil, "Bad choice %c", i)
|
Errorf(nil, "Bad choice %c", i)
|
||||||
}
|
}
|
||||||
return false
|
return false
|
||||||
}
|
}
|
||||||
@ -868,7 +868,7 @@ func ChooseOption(o *Option) string {
|
|||||||
case 'n':
|
case 'n':
|
||||||
return ""
|
return ""
|
||||||
default:
|
default:
|
||||||
ErrorLog(nil, "Bad choice %c", i)
|
Errorf(nil, "Bad choice %c", i)
|
||||||
}
|
}
|
||||||
return MustObscure(password)
|
return MustObscure(password)
|
||||||
}
|
}
|
||||||
@ -1128,7 +1128,7 @@ func ConfigFileGetBool(section, key string, defaultVal ...bool) bool {
|
|||||||
if found {
|
if found {
|
||||||
newBool, err := strconv.ParseBool(newValue)
|
newBool, err := strconv.ParseBool(newValue)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
ErrorLog(nil, "Couldn't parse %q into bool - ignoring: %v", envKey, err)
|
Errorf(nil, "Couldn't parse %q into bool - ignoring: %v", envKey, err)
|
||||||
} else {
|
} else {
|
||||||
defaultVal = []bool{newBool}
|
defaultVal = []bool{newBool}
|
||||||
}
|
}
|
||||||
@ -1146,7 +1146,7 @@ func ConfigFileGetInt(section, key string, defaultVal ...int) int {
|
|||||||
if found {
|
if found {
|
||||||
newInt, err := strconv.Atoi(newValue)
|
newInt, err := strconv.Atoi(newValue)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
ErrorLog(nil, "Couldn't parse %q into int - ignoring: %v", envKey, err)
|
Errorf(nil, "Couldn't parse %q into int - ignoring: %v", envKey, err)
|
||||||
} else {
|
} else {
|
||||||
defaultVal = []int{newInt}
|
defaultVal = []int{newInt}
|
||||||
}
|
}
|
||||||
|
@ -232,7 +232,7 @@ func NewFilter() (f *Filter, err error) {
|
|||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
f.ModTimeTo = time.Now().Add(-duration)
|
f.ModTimeTo = time.Now().Add(-duration)
|
||||||
Debug(nil, "--min-age %v to %v", duration, f.ModTimeTo)
|
Debugf(nil, "--min-age %v to %v", duration, f.ModTimeTo)
|
||||||
}
|
}
|
||||||
if *maxAge != "" {
|
if *maxAge != "" {
|
||||||
duration, err := ParseDuration(*maxAge)
|
duration, err := ParseDuration(*maxAge)
|
||||||
@ -243,7 +243,7 @@ func NewFilter() (f *Filter, err error) {
|
|||||||
if !f.ModTimeTo.IsZero() && f.ModTimeTo.Before(f.ModTimeFrom) {
|
if !f.ModTimeTo.IsZero() && f.ModTimeTo.Before(f.ModTimeFrom) {
|
||||||
return nil, errors.New("argument --min-age can't be larger than --max-age")
|
return nil, errors.New("argument --min-age can't be larger than --max-age")
|
||||||
}
|
}
|
||||||
Debug(nil, "--max-age %v to %v", duration, f.ModTimeFrom)
|
Debugf(nil, "--max-age %v to %v", duration, f.ModTimeFrom)
|
||||||
}
|
}
|
||||||
if *dumpFilters {
|
if *dumpFilters {
|
||||||
fmt.Println("--- start filters ---")
|
fmt.Println("--- start filters ---")
|
||||||
|
36
fs/fs.go
36
fs/fs.go
@ -2,11 +2,9 @@
|
|||||||
package fs
|
package fs
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"fmt"
|
|
||||||
"io"
|
"io"
|
||||||
"log"
|
"log"
|
||||||
"math"
|
"math"
|
||||||
"os"
|
|
||||||
"path/filepath"
|
"path/filepath"
|
||||||
"regexp"
|
"regexp"
|
||||||
"sort"
|
"sort"
|
||||||
@ -614,40 +612,6 @@ func NewFs(path string) (Fs, error) {
|
|||||||
return fsInfo.NewFs(configName, fsPath)
|
return fsInfo.NewFs(configName, fsPath)
|
||||||
}
|
}
|
||||||
|
|
||||||
// DebugLogger - logs to Stdout
|
|
||||||
var DebugLogger = log.New(os.Stdout, "", log.LstdFlags)
|
|
||||||
|
|
||||||
// makeLog produces a log string from the arguments passed in
|
|
||||||
func makeLog(o interface{}, text string, args ...interface{}) string {
|
|
||||||
out := fmt.Sprintf(text, args...)
|
|
||||||
if o == nil {
|
|
||||||
return out
|
|
||||||
}
|
|
||||||
return fmt.Sprintf("%v: %s", o, out)
|
|
||||||
}
|
|
||||||
|
|
||||||
// Debug writes debugging output for this Object or Fs
|
|
||||||
func Debug(o interface{}, text string, args ...interface{}) {
|
|
||||||
if Config.Verbose {
|
|
||||||
DebugLogger.Print(makeLog(o, text, args...))
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Log writes log output for this Object or Fs. This should be
|
|
||||||
// considered to be Info level logging.
|
|
||||||
func Log(o interface{}, text string, args ...interface{}) {
|
|
||||||
if !Config.Quiet {
|
|
||||||
log.Print(makeLog(o, text, args...))
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// ErrorLog writes error log output for this Object or Fs. It
|
|
||||||
// unconditionally logs a message regardless of Config.Quiet or
|
|
||||||
// Config.Verbose.
|
|
||||||
func ErrorLog(o interface{}, text string, args ...interface{}) {
|
|
||||||
log.Print(makeLog(o, text, args...))
|
|
||||||
}
|
|
||||||
|
|
||||||
// CheckClose is a utility function used to check the return from
|
// CheckClose is a utility function used to check the return from
|
||||||
// Close in a defer statement.
|
// Close in a defer statement.
|
||||||
func CheckClose(c io.Closer, err *error) {
|
func CheckClose(c io.Closer, err *error) {
|
||||||
|
26
fs/http.go
26
fs/http.go
@ -170,13 +170,13 @@ func checkServerTime(req *http.Request, resp *http.Response) {
|
|||||||
}
|
}
|
||||||
date, err := http.ParseTime(dateString)
|
date, err := http.ParseTime(dateString)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
Debug(nil, "Couldn't parse Date: from server %s: %q: %v", host, dateString, err)
|
Debugf(nil, "Couldn't parse Date: from server %s: %q: %v", host, dateString, err)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
dt := time.Since(date)
|
dt := time.Since(date)
|
||||||
const window = 5 * 60 * time.Second
|
const window = 5 * 60 * time.Second
|
||||||
if dt > window || dt < -window {
|
if dt > window || dt < -window {
|
||||||
Log(nil, "Time may be set wrong - time from %q is %v different from this computer", host, dt)
|
Logf(nil, "Time may be set wrong - time from %q is %v different from this computer", host, dt)
|
||||||
}
|
}
|
||||||
checkedHostMu.Lock()
|
checkedHostMu.Lock()
|
||||||
checkedHost[host] = struct{}{}
|
checkedHost[host] = struct{}{}
|
||||||
@ -219,30 +219,30 @@ func cleanAuth(buf []byte) []byte {
|
|||||||
func (t *Transport) RoundTrip(req *http.Request) (resp *http.Response, err error) {
|
func (t *Transport) RoundTrip(req *http.Request) (resp *http.Response, err error) {
|
||||||
// Force user agent
|
// Force user agent
|
||||||
req.Header.Set("User-Agent", UserAgent)
|
req.Header.Set("User-Agent", UserAgent)
|
||||||
// Log request
|
// Logf request
|
||||||
if t.logHeader || t.logBody || t.logAuth {
|
if t.logHeader || t.logBody || t.logAuth {
|
||||||
buf, _ := httputil.DumpRequestOut(req, t.logBody)
|
buf, _ := httputil.DumpRequestOut(req, t.logBody)
|
||||||
if !t.logAuth {
|
if !t.logAuth {
|
||||||
buf = cleanAuth(buf)
|
buf = cleanAuth(buf)
|
||||||
}
|
}
|
||||||
Debug(nil, "%s", separatorReq)
|
Debugf(nil, "%s", separatorReq)
|
||||||
Debug(nil, "%s (req %p)", "HTTP REQUEST", req)
|
Debugf(nil, "%s (req %p)", "HTTP REQUEST", req)
|
||||||
Debug(nil, "%s", string(buf))
|
Debugf(nil, "%s", string(buf))
|
||||||
Debug(nil, "%s", separatorReq)
|
Debugf(nil, "%s", separatorReq)
|
||||||
}
|
}
|
||||||
// Do round trip
|
// Do round trip
|
||||||
resp, err = t.Transport.RoundTrip(req)
|
resp, err = t.Transport.RoundTrip(req)
|
||||||
// Log response
|
// Logf response
|
||||||
if t.logHeader || t.logBody || t.logAuth {
|
if t.logHeader || t.logBody || t.logAuth {
|
||||||
Debug(nil, "%s", separatorResp)
|
Debugf(nil, "%s", separatorResp)
|
||||||
Debug(nil, "%s (req %p)", "HTTP RESPONSE", req)
|
Debugf(nil, "%s (req %p)", "HTTP RESPONSE", req)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
Debug(nil, "Error: %v", err)
|
Debugf(nil, "Error: %v", err)
|
||||||
} else {
|
} else {
|
||||||
buf, _ := httputil.DumpResponse(resp, t.logBody)
|
buf, _ := httputil.DumpResponse(resp, t.logBody)
|
||||||
Debug(nil, "%s", string(buf))
|
Debugf(nil, "%s", string(buf))
|
||||||
}
|
}
|
||||||
Debug(nil, "%s", separatorResp)
|
Debugf(nil, "%s", separatorResp)
|
||||||
}
|
}
|
||||||
if err == nil {
|
if err == nil {
|
||||||
checkServerTime(req, resp)
|
checkServerTime(req, resp)
|
||||||
|
67
fs/log.go
Normal file
67
fs/log.go
Normal file
@ -0,0 +1,67 @@
|
|||||||
|
// Logging for rclone
|
||||||
|
|
||||||
|
package fs
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"log"
|
||||||
|
"os"
|
||||||
|
)
|
||||||
|
|
||||||
|
// LogLevel describes rclone's logs. These are a subset of the syslog log levels.
|
||||||
|
type LogLevel byte
|
||||||
|
|
||||||
|
// Log levels - a subset of the syslog logs
|
||||||
|
const (
|
||||||
|
LogLevelEmergency = iota
|
||||||
|
LogLevelAlert
|
||||||
|
LogLevelCritical
|
||||||
|
LogLevelError // Error - can't be suppressed
|
||||||
|
LogLevelWarning
|
||||||
|
LogLevelNotice // Normal logging, -q suppresses
|
||||||
|
LogLevelInfo // Transfers, needs -v
|
||||||
|
LogLevelDebug // Debug level, needs -vv
|
||||||
|
)
|
||||||
|
|
||||||
|
// Outside world interface
|
||||||
|
|
||||||
|
// DebugLogger - logs to Stdout
|
||||||
|
var DebugLogger = log.New(os.Stdout, "", log.LstdFlags)
|
||||||
|
|
||||||
|
// makeLog produces a log string from the arguments passed in
|
||||||
|
func makeLog(o interface{}, text string, args ...interface{}) string {
|
||||||
|
out := fmt.Sprintf(text, args...)
|
||||||
|
if o == nil {
|
||||||
|
return out
|
||||||
|
}
|
||||||
|
return fmt.Sprintf("%v: %s", o, out)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Errorf writes error log output for this Object or Fs. It
|
||||||
|
// unconditionally logs a message regardless of Config.Quiet or
|
||||||
|
// Config.Verbose.
|
||||||
|
func Errorf(o interface{}, text string, args ...interface{}) {
|
||||||
|
log.Print(makeLog(o, text, args...))
|
||||||
|
}
|
||||||
|
|
||||||
|
// Logf writes log output for this Object or Fs. This should be
|
||||||
|
// considered to be Info level logging.
|
||||||
|
func Logf(o interface{}, text string, args ...interface{}) {
|
||||||
|
if !Config.Quiet {
|
||||||
|
log.Print(makeLog(o, text, args...))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Infof writes info on transfers for this Object or Fs
|
||||||
|
func Infof(o interface{}, text string, args ...interface{}) {
|
||||||
|
if Config.Verbose {
|
||||||
|
DebugLogger.Print(makeLog(o, text, args...))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Debugf writes debugging output for this Object or Fs
|
||||||
|
func Debugf(o interface{}, text string, args ...interface{}) {
|
||||||
|
if Config.Verbose {
|
||||||
|
DebugLogger.Print(makeLog(o, text, args...))
|
||||||
|
}
|
||||||
|
}
|
152
fs/operations.go
152
fs/operations.go
@ -32,12 +32,12 @@ func CalculateModifyWindow(fs ...Fs) {
|
|||||||
Config.ModifyWindow = precision
|
Config.ModifyWindow = precision
|
||||||
}
|
}
|
||||||
if precision == ModTimeNotSupported {
|
if precision == ModTimeNotSupported {
|
||||||
Debug(f, "Modify window not supported")
|
Debugf(f, "Modify window not supported")
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
Debug(fs[0], "Modify window is %s", Config.ModifyWindow)
|
Debugf(fs[0], "Modify window is %s", Config.ModifyWindow)
|
||||||
}
|
}
|
||||||
|
|
||||||
// HashEquals checks to see if src == dst, but ignores empty strings
|
// HashEquals checks to see if src == dst, but ignores empty strings
|
||||||
@ -64,7 +64,7 @@ func HashEquals(src, dst string) bool {
|
|||||||
// If an error is returned it will return equal as false
|
// If an error is returned it will return equal as false
|
||||||
func CheckHashes(src, dst Object) (equal bool, hash HashType, err error) {
|
func CheckHashes(src, dst Object) (equal bool, hash HashType, err error) {
|
||||||
common := src.Fs().Hashes().Overlap(dst.Fs().Hashes())
|
common := src.Fs().Hashes().Overlap(dst.Fs().Hashes())
|
||||||
// Debug(nil, "Shared hashes: %v", common)
|
// Debugf(nil, "Shared hashes: %v", common)
|
||||||
if common.Count() == 0 {
|
if common.Count() == 0 {
|
||||||
return true, HashNone, nil
|
return true, HashNone, nil
|
||||||
}
|
}
|
||||||
@ -72,7 +72,7 @@ func CheckHashes(src, dst Object) (equal bool, hash HashType, err error) {
|
|||||||
srcHash, err := src.Hash(hash)
|
srcHash, err := src.Hash(hash)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
Stats.Error()
|
Stats.Error()
|
||||||
ErrorLog(src, "Failed to calculate src hash: %v", err)
|
Errorf(src, "Failed to calculate src hash: %v", err)
|
||||||
return false, hash, err
|
return false, hash, err
|
||||||
}
|
}
|
||||||
if srcHash == "" {
|
if srcHash == "" {
|
||||||
@ -81,7 +81,7 @@ func CheckHashes(src, dst Object) (equal bool, hash HashType, err error) {
|
|||||||
dstHash, err := dst.Hash(hash)
|
dstHash, err := dst.Hash(hash)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
Stats.Error()
|
Stats.Error()
|
||||||
ErrorLog(dst, "Failed to calculate dst hash: %v", err)
|
Errorf(dst, "Failed to calculate dst hash: %v", err)
|
||||||
return false, hash, err
|
return false, hash, err
|
||||||
}
|
}
|
||||||
if dstHash == "" {
|
if dstHash == "" {
|
||||||
@ -115,12 +115,12 @@ func Equal(src, dst Object) bool {
|
|||||||
func equal(src, dst Object, sizeOnly, checkSum bool) bool {
|
func equal(src, dst Object, sizeOnly, checkSum bool) bool {
|
||||||
if !Config.IgnoreSize {
|
if !Config.IgnoreSize {
|
||||||
if src.Size() != dst.Size() {
|
if src.Size() != dst.Size() {
|
||||||
Debug(src, "Sizes differ")
|
Debugf(src, "Sizes differ")
|
||||||
return false
|
return false
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if sizeOnly {
|
if sizeOnly {
|
||||||
Debug(src, "Sizes identical")
|
Debugf(src, "Sizes identical")
|
||||||
return true
|
return true
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -131,20 +131,20 @@ func equal(src, dst Object, sizeOnly, checkSum bool) bool {
|
|||||||
// Check the hash
|
// Check the hash
|
||||||
same, hash, _ := CheckHashes(src, dst)
|
same, hash, _ := CheckHashes(src, dst)
|
||||||
if !same {
|
if !same {
|
||||||
Debug(src, "%v differ", hash)
|
Debugf(src, "%v differ", hash)
|
||||||
return false
|
return false
|
||||||
}
|
}
|
||||||
if hash == HashNone {
|
if hash == HashNone {
|
||||||
Debug(src, "Size of src and dst objects identical")
|
Debugf(src, "Size of src and dst objects identical")
|
||||||
} else {
|
} else {
|
||||||
Debug(src, "Size and %v of src and dst objects identical", hash)
|
Debugf(src, "Size and %v of src and dst objects identical", hash)
|
||||||
}
|
}
|
||||||
return true
|
return true
|
||||||
}
|
}
|
||||||
|
|
||||||
// Sizes the same so check the mtime
|
// Sizes the same so check the mtime
|
||||||
if Config.ModifyWindow == ModTimeNotSupported {
|
if Config.ModifyWindow == ModTimeNotSupported {
|
||||||
Debug(src, "Sizes identical")
|
Debugf(src, "Sizes identical")
|
||||||
return true
|
return true
|
||||||
}
|
}
|
||||||
srcModTime := src.ModTime()
|
srcModTime := src.ModTime()
|
||||||
@ -152,16 +152,16 @@ func equal(src, dst Object, sizeOnly, checkSum bool) bool {
|
|||||||
dt := dstModTime.Sub(srcModTime)
|
dt := dstModTime.Sub(srcModTime)
|
||||||
ModifyWindow := Config.ModifyWindow
|
ModifyWindow := Config.ModifyWindow
|
||||||
if dt < ModifyWindow && dt > -ModifyWindow {
|
if dt < ModifyWindow && dt > -ModifyWindow {
|
||||||
Debug(src, "Size and modification time the same (differ by %s, within tolerance %s)", dt, ModifyWindow)
|
Debugf(src, "Size and modification time the same (differ by %s, within tolerance %s)", dt, ModifyWindow)
|
||||||
return true
|
return true
|
||||||
}
|
}
|
||||||
|
|
||||||
Debug(src, "Modification times differ by %s: %v, %v", dt, srcModTime, dstModTime)
|
Debugf(src, "Modification times differ by %s: %v, %v", dt, srcModTime, dstModTime)
|
||||||
|
|
||||||
// Check if the hashes are the same
|
// Check if the hashes are the same
|
||||||
same, hash, _ := CheckHashes(src, dst)
|
same, hash, _ := CheckHashes(src, dst)
|
||||||
if !same {
|
if !same {
|
||||||
Debug(src, "%v differ", hash)
|
Debugf(src, "%v differ", hash)
|
||||||
return false
|
return false
|
||||||
}
|
}
|
||||||
if hash == HashNone {
|
if hash == HashNone {
|
||||||
@ -175,13 +175,13 @@ func equal(src, dst Object, sizeOnly, checkSum bool) bool {
|
|||||||
// mtime of the dst object here
|
// mtime of the dst object here
|
||||||
err := dst.SetModTime(srcModTime)
|
err := dst.SetModTime(srcModTime)
|
||||||
if err == ErrorCantSetModTime {
|
if err == ErrorCantSetModTime {
|
||||||
Debug(src, "src and dst identical but can't set mod time without re-uploading")
|
Debugf(src, "src and dst identical but can't set mod time without re-uploading")
|
||||||
return false
|
return false
|
||||||
} else if err != nil {
|
} else if err != nil {
|
||||||
Stats.Error()
|
Stats.Error()
|
||||||
ErrorLog(dst, "Failed to set modification time: %v", err)
|
Errorf(dst, "Failed to set modification time: %v", err)
|
||||||
} else {
|
} else {
|
||||||
Debug(src, "Updated modification time in destination")
|
Debugf(src, "Updated modification time in destination")
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return true
|
return true
|
||||||
@ -202,7 +202,7 @@ func MimeType(o ObjectInfo) (mimeType string) {
|
|||||||
// Read the MimeType from the optional interface if available
|
// Read the MimeType from the optional interface if available
|
||||||
if do, ok := o.(MimeTyper); ok {
|
if do, ok := o.(MimeTyper); ok {
|
||||||
mimeType = do.MimeType()
|
mimeType = do.MimeType()
|
||||||
Debug(o, "Read MimeType as %q", mimeType)
|
Debugf(o, "Read MimeType as %q", mimeType)
|
||||||
if mimeType != "" {
|
if mimeType != "" {
|
||||||
return mimeType
|
return mimeType
|
||||||
}
|
}
|
||||||
@ -217,10 +217,10 @@ func removeFailedCopy(dst Object) bool {
|
|||||||
if dst == nil {
|
if dst == nil {
|
||||||
return false
|
return false
|
||||||
}
|
}
|
||||||
Debug(dst, "Removing failed copy")
|
Debugf(dst, "Removing failed copy")
|
||||||
removeErr := dst.Remove()
|
removeErr := dst.Remove()
|
||||||
if removeErr != nil {
|
if removeErr != nil {
|
||||||
Debug(dst, "Failed to remove failed copy: %s", removeErr)
|
Debugf(dst, "Failed to remove failed copy: %s", removeErr)
|
||||||
return false
|
return false
|
||||||
}
|
}
|
||||||
return true
|
return true
|
||||||
@ -241,7 +241,7 @@ func (o *overrideRemoteObject) Remote() string {
|
|||||||
// remote as the name of the new object.
|
// remote as the name of the new object.
|
||||||
func Copy(f Fs, dst Object, remote string, src Object) (err error) {
|
func Copy(f Fs, dst Object, remote string, src Object) (err error) {
|
||||||
if Config.DryRun {
|
if Config.DryRun {
|
||||||
Log(src, "Not copying as --dry-run")
|
Logf(src, "Not copying as --dry-run")
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
maxTries := Config.LowLevelRetries
|
maxTries := Config.LowLevelRetries
|
||||||
@ -290,7 +290,7 @@ func Copy(f Fs, dst Object, remote string, src Object) (err error) {
|
|||||||
}
|
}
|
||||||
// Retry if err returned a retry error
|
// Retry if err returned a retry error
|
||||||
if IsRetryError(err) || ShouldRetry(err) {
|
if IsRetryError(err) || ShouldRetry(err) {
|
||||||
Debug(src, "Received error: %v - low level retry %d/%d", err, tries, maxTries)
|
Debugf(src, "Received error: %v - low level retry %d/%d", err, tries, maxTries)
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
// otherwise finish
|
// otherwise finish
|
||||||
@ -298,7 +298,7 @@ func Copy(f Fs, dst Object, remote string, src Object) (err error) {
|
|||||||
}
|
}
|
||||||
if err != nil {
|
if err != nil {
|
||||||
Stats.Error()
|
Stats.Error()
|
||||||
ErrorLog(src, "Failed to copy: %v", err)
|
Errorf(src, "Failed to copy: %v", err)
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -306,7 +306,7 @@ func Copy(f Fs, dst Object, remote string, src Object) (err error) {
|
|||||||
if !Config.IgnoreSize && src.Size() != dst.Size() {
|
if !Config.IgnoreSize && src.Size() != dst.Size() {
|
||||||
Stats.Error()
|
Stats.Error()
|
||||||
err = errors.Errorf("corrupted on transfer: sizes differ %d vs %d", src.Size(), dst.Size())
|
err = errors.Errorf("corrupted on transfer: sizes differ %d vs %d", src.Size(), dst.Size())
|
||||||
ErrorLog(dst, "%v", err)
|
Errorf(dst, "%v", err)
|
||||||
removeFailedCopy(dst)
|
removeFailedCopy(dst)
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
@ -315,7 +315,7 @@ func Copy(f Fs, dst Object, remote string, src Object) (err error) {
|
|||||||
// TODO(klauspost): This could be extended, so we always create a hash type matching
|
// TODO(klauspost): This could be extended, so we always create a hash type matching
|
||||||
// the destination, and calculate it while sending.
|
// the destination, and calculate it while sending.
|
||||||
common := src.Fs().Hashes().Overlap(dst.Fs().Hashes())
|
common := src.Fs().Hashes().Overlap(dst.Fs().Hashes())
|
||||||
// Debug(src, "common hashes: %v", common)
|
// Debugf(src, "common hashes: %v", common)
|
||||||
if !Config.SizeOnly && common.Count() > 0 {
|
if !Config.SizeOnly && common.Count() > 0 {
|
||||||
// Get common hash type
|
// Get common hash type
|
||||||
hashType := common.GetOne()
|
hashType := common.GetOne()
|
||||||
@ -324,24 +324,24 @@ func Copy(f Fs, dst Object, remote string, src Object) (err error) {
|
|||||||
srcSum, err = src.Hash(hashType)
|
srcSum, err = src.Hash(hashType)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
Stats.Error()
|
Stats.Error()
|
||||||
ErrorLog(src, "Failed to read src hash: %v", err)
|
Errorf(src, "Failed to read src hash: %v", err)
|
||||||
} else if srcSum != "" {
|
} else if srcSum != "" {
|
||||||
var dstSum string
|
var dstSum string
|
||||||
dstSum, err = dst.Hash(hashType)
|
dstSum, err = dst.Hash(hashType)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
Stats.Error()
|
Stats.Error()
|
||||||
ErrorLog(dst, "Failed to read hash: %v", err)
|
Errorf(dst, "Failed to read hash: %v", err)
|
||||||
} else if !Config.IgnoreSize && !HashEquals(srcSum, dstSum) {
|
} else if !Config.IgnoreSize && !HashEquals(srcSum, dstSum) {
|
||||||
Stats.Error()
|
Stats.Error()
|
||||||
err = errors.Errorf("corrupted on transfer: %v hash differ %q vs %q", hashType, srcSum, dstSum)
|
err = errors.Errorf("corrupted on transfer: %v hash differ %q vs %q", hashType, srcSum, dstSum)
|
||||||
ErrorLog(dst, "%v", err)
|
Errorf(dst, "%v", err)
|
||||||
removeFailedCopy(dst)
|
removeFailedCopy(dst)
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
Debug(src, actionTaken)
|
Debugf(src, actionTaken)
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -349,7 +349,7 @@ func Copy(f Fs, dst Object, remote string, src Object) (err error) {
|
|||||||
// remote as the name of the new object.
|
// remote as the name of the new object.
|
||||||
func Move(fdst Fs, dst Object, remote string, src Object) (err error) {
|
func Move(fdst Fs, dst Object, remote string, src Object) (err error) {
|
||||||
if Config.DryRun {
|
if Config.DryRun {
|
||||||
Log(src, "Not moving as --dry-run")
|
Logf(src, "Not moving as --dry-run")
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
// See if we have Move available
|
// See if we have Move available
|
||||||
@ -365,20 +365,20 @@ func Move(fdst Fs, dst Object, remote string, src Object) (err error) {
|
|||||||
_, err := doMove(src, remote)
|
_, err := doMove(src, remote)
|
||||||
switch err {
|
switch err {
|
||||||
case nil:
|
case nil:
|
||||||
Debug(src, "Moved (server side)")
|
Debugf(src, "Moved (server side)")
|
||||||
return nil
|
return nil
|
||||||
case ErrorCantMove:
|
case ErrorCantMove:
|
||||||
Debug(src, "Can't move, switching to copy")
|
Debugf(src, "Can't move, switching to copy")
|
||||||
default:
|
default:
|
||||||
Stats.Error()
|
Stats.Error()
|
||||||
ErrorLog(dst, "Couldn't move: %v", err)
|
Errorf(dst, "Couldn't move: %v", err)
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
// Move not found or didn't work so copy dst <- src
|
// Move not found or didn't work so copy dst <- src
|
||||||
err = Copy(fdst, dst, remote, src)
|
err = Copy(fdst, dst, remote, src)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
ErrorLog(src, "Not deleting source as copy failed: %v", err)
|
Errorf(src, "Not deleting source as copy failed: %v", err)
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
// Delete src if no error on copy
|
// Delete src if no error on copy
|
||||||
@ -408,7 +408,7 @@ func deleteFileWithBackupDir(dst Object, backupDir Fs) (err error) {
|
|||||||
action, actioned, actioning = "move into backup dir", "Moved into backup dir", "moving into backup dir"
|
action, actioned, actioning = "move into backup dir", "Moved into backup dir", "moving into backup dir"
|
||||||
}
|
}
|
||||||
if Config.DryRun {
|
if Config.DryRun {
|
||||||
Log(dst, "Not %s as --dry-run", actioning)
|
Logf(dst, "Not %s as --dry-run", actioning)
|
||||||
} else if backupDir != nil {
|
} else if backupDir != nil {
|
||||||
if !SameConfig(dst.Fs(), backupDir) {
|
if !SameConfig(dst.Fs(), backupDir) {
|
||||||
err = errors.New("parameter to --backup-dir has to be on the same remote as destination")
|
err = errors.New("parameter to --backup-dir has to be on the same remote as destination")
|
||||||
@ -422,9 +422,9 @@ func deleteFileWithBackupDir(dst Object, backupDir Fs) (err error) {
|
|||||||
}
|
}
|
||||||
if err != nil {
|
if err != nil {
|
||||||
Stats.Error()
|
Stats.Error()
|
||||||
ErrorLog(dst, "Couldn't %s: %v", action, err)
|
Errorf(dst, "Couldn't %s: %v", action, err)
|
||||||
} else {
|
} else {
|
||||||
Debug(dst, actioned)
|
Debugf(dst, actioned)
|
||||||
}
|
}
|
||||||
Stats.DoneChecking(dst.Remote())
|
Stats.DoneChecking(dst.Remote())
|
||||||
return err
|
return err
|
||||||
@ -458,7 +458,7 @@ func deleteFilesWithBackupDir(toBeDeleted ObjectsChan, backupDir Fs) error {
|
|||||||
}
|
}
|
||||||
}()
|
}()
|
||||||
}
|
}
|
||||||
Log(nil, "Waiting for deletions to finish")
|
Logf(nil, "Waiting for deletions to finish")
|
||||||
wg.Wait()
|
wg.Wait()
|
||||||
if errorCount > 0 {
|
if errorCount > 0 {
|
||||||
return errors.Errorf("failed to delete %d files", errorCount)
|
return errors.Errorf("failed to delete %d files", errorCount)
|
||||||
@ -501,7 +501,7 @@ func readFilesFn(fs Fs, includeAll bool, dir string, add func(Object) error) (er
|
|||||||
list.SetError(err)
|
list.SetError(err)
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
Debug(o, "Excluded from sync (and deletion)")
|
Debugf(o, "Excluded from sync (and deletion)")
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return list.Error()
|
return list.Error()
|
||||||
@ -550,13 +550,13 @@ func ListDirSorted(fs Fs, includeAll bool, dir string) (entries DirEntries, err
|
|||||||
if includeAll || Config.Filter.IncludeObject(o) {
|
if includeAll || Config.Filter.IncludeObject(o) {
|
||||||
entries = append(entries, o)
|
entries = append(entries, o)
|
||||||
} else {
|
} else {
|
||||||
Debug(o, "Excluded from sync (and deletion)")
|
Debugf(o, "Excluded from sync (and deletion)")
|
||||||
}
|
}
|
||||||
} else if dir != nil {
|
} else if dir != nil {
|
||||||
if includeAll || Config.Filter.IncludeDirectory(dir.Remote()) {
|
if includeAll || Config.Filter.IncludeDirectory(dir.Remote()) {
|
||||||
entries = append(entries, dir)
|
entries = append(entries, dir)
|
||||||
} else {
|
} else {
|
||||||
Debug(dir, "Excluded from sync (and deletion)")
|
Debugf(dir, "Excluded from sync (and deletion)")
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
// finishd since err, o, dir == nil
|
// finishd since err, o, dir == nil
|
||||||
@ -587,10 +587,10 @@ func readFilesMap(fs Fs, includeAll bool, dir string) (files map[string]Object,
|
|||||||
if _, ok := files[remote]; !ok {
|
if _, ok := files[remote]; !ok {
|
||||||
files[remote] = o
|
files[remote] = o
|
||||||
if _, ok := normalised[normalisedRemote]; ok {
|
if _, ok := normalised[normalisedRemote]; ok {
|
||||||
Log(o, "Warning: File found with same name but different case on %v", o.Fs())
|
Logf(o, "Warning: File found with same name but different case on %v", o.Fs())
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
Log(o, "Duplicate file detected")
|
Logf(o, "Duplicate file detected")
|
||||||
}
|
}
|
||||||
normalised[normalisedRemote] = struct{}{}
|
normalised[normalisedRemote] = struct{}{}
|
||||||
return nil
|
return nil
|
||||||
@ -609,13 +609,13 @@ func readFilesMaps(fdst Fs, fdstIncludeAll bool, fsrc Fs, fsrcIncludeAll bool, d
|
|||||||
|
|
||||||
list := func(fs Fs, includeAll bool, pMap *map[string]Object, pErr *error) {
|
list := func(fs Fs, includeAll bool, pMap *map[string]Object, pErr *error) {
|
||||||
defer wg.Done()
|
defer wg.Done()
|
||||||
Log(fs, "Building file list")
|
Logf(fs, "Building file list")
|
||||||
files, listErr := readFilesMap(fs, includeAll, dir)
|
files, listErr := readFilesMap(fs, includeAll, dir)
|
||||||
if listErr != nil {
|
if listErr != nil {
|
||||||
ErrorLog(fs, "Error building file list: %v", listErr)
|
Errorf(fs, "Error building file list: %v", listErr)
|
||||||
*pErr = listErr
|
*pErr = listErr
|
||||||
} else {
|
} else {
|
||||||
Debug(fs, "Done building file list")
|
Debugf(fs, "Done building file list")
|
||||||
*pMap = files
|
*pMap = files
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -673,7 +673,7 @@ func checkIdentical(dst, src Object) (differ bool, noHash bool) {
|
|||||||
defer Stats.DoneChecking(src.Remote())
|
defer Stats.DoneChecking(src.Remote())
|
||||||
if src.Size() != dst.Size() {
|
if src.Size() != dst.Size() {
|
||||||
Stats.Error()
|
Stats.Error()
|
||||||
ErrorLog(src, "Sizes differ")
|
Errorf(src, "Sizes differ")
|
||||||
return true, false
|
return true, false
|
||||||
}
|
}
|
||||||
if !Config.SizeOnly {
|
if !Config.SizeOnly {
|
||||||
@ -687,11 +687,11 @@ func checkIdentical(dst, src Object) (differ bool, noHash bool) {
|
|||||||
}
|
}
|
||||||
if !same {
|
if !same {
|
||||||
Stats.Error()
|
Stats.Error()
|
||||||
ErrorLog(src, "%v differ", hash)
|
Errorf(src, "%v differ", hash)
|
||||||
return true, false
|
return true, false
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
Debug(src, "OK")
|
Debugf(src, "OK")
|
||||||
return false, false
|
return false, false
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -718,17 +718,17 @@ func Check(fdst, fsrc Fs) error {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
Log(fdst, "%d files not in %v", len(dstFiles), fsrc)
|
Logf(fdst, "%d files not in %v", len(dstFiles), fsrc)
|
||||||
for _, dst := range dstFiles {
|
for _, dst := range dstFiles {
|
||||||
Stats.Error()
|
Stats.Error()
|
||||||
ErrorLog(dst, "File not in %v", fsrc)
|
Errorf(dst, "File not in %v", fsrc)
|
||||||
atomic.AddInt32(&differences, 1)
|
atomic.AddInt32(&differences, 1)
|
||||||
}
|
}
|
||||||
|
|
||||||
Log(fsrc, "%d files not in %s", len(srcFiles), fdst)
|
Logf(fsrc, "%d files not in %s", len(srcFiles), fdst)
|
||||||
for _, src := range srcFiles {
|
for _, src := range srcFiles {
|
||||||
Stats.Error()
|
Stats.Error()
|
||||||
ErrorLog(src, "File not in %v", fdst)
|
Errorf(src, "File not in %v", fdst)
|
||||||
atomic.AddInt32(&differences, 1)
|
atomic.AddInt32(&differences, 1)
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -757,11 +757,11 @@ func Check(fdst, fsrc Fs) error {
|
|||||||
}()
|
}()
|
||||||
}
|
}
|
||||||
|
|
||||||
Log(fdst, "Waiting for checks to finish")
|
Logf(fdst, "Waiting for checks to finish")
|
||||||
checkerWg.Wait()
|
checkerWg.Wait()
|
||||||
Log(fdst, "%d differences found", Stats.GetErrors())
|
Logf(fdst, "%d differences found", Stats.GetErrors())
|
||||||
if noHashes > 0 {
|
if noHashes > 0 {
|
||||||
Log(fdst, "%d hashes could not be checked", noHashes)
|
Logf(fdst, "%d hashes could not be checked", noHashes)
|
||||||
}
|
}
|
||||||
if differences > 0 {
|
if differences > 0 {
|
||||||
return errors.Errorf("%d differences found", differences)
|
return errors.Errorf("%d differences found", differences)
|
||||||
@ -865,7 +865,7 @@ func hashLister(ht HashType, f Fs, w io.Writer) error {
|
|||||||
if err == ErrHashUnsupported {
|
if err == ErrHashUnsupported {
|
||||||
sum = "UNSUPPORTED"
|
sum = "UNSUPPORTED"
|
||||||
} else if err != nil {
|
} else if err != nil {
|
||||||
Debug(o, "Failed to read %v: %v", ht, err)
|
Debugf(o, "Failed to read %v: %v", ht, err)
|
||||||
sum = "ERROR"
|
sum = "ERROR"
|
||||||
}
|
}
|
||||||
syncFprintf(w, "%*s %s\n", HashWidth[ht], sum, o.Remote())
|
syncFprintf(w, "%*s %s\n", HashWidth[ht], sum, o.Remote())
|
||||||
@ -906,7 +906,7 @@ func ListDir(f Fs, w io.Writer) error {
|
|||||||
// Mkdir makes a destination directory or container
|
// Mkdir makes a destination directory or container
|
||||||
func Mkdir(f Fs, dir string) error {
|
func Mkdir(f Fs, dir string) error {
|
||||||
if Config.DryRun {
|
if Config.DryRun {
|
||||||
Log(f, "Not making directory as dry run is set")
|
Logf(f, "Not making directory as dry run is set")
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
err := f.Mkdir(dir)
|
err := f.Mkdir(dir)
|
||||||
@ -922,9 +922,9 @@ func Mkdir(f Fs, dir string) error {
|
|||||||
func TryRmdir(f Fs, dir string) error {
|
func TryRmdir(f Fs, dir string) error {
|
||||||
if Config.DryRun {
|
if Config.DryRun {
|
||||||
if dir != "" {
|
if dir != "" {
|
||||||
Log(dir, "Not deleting as dry run is set")
|
Logf(dir, "Not deleting as dry run is set")
|
||||||
} else {
|
} else {
|
||||||
Log(f, "Not deleting as dry run is set")
|
Logf(f, "Not deleting as dry run is set")
|
||||||
}
|
}
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
@ -948,7 +948,7 @@ func Purge(f Fs) error {
|
|||||||
if doPurge := f.Features().Purge; doPurge != nil {
|
if doPurge := f.Features().Purge; doPurge != nil {
|
||||||
doFallbackPurge = false
|
doFallbackPurge = false
|
||||||
if Config.DryRun {
|
if Config.DryRun {
|
||||||
Log(f, "Not purging as --dry-run set")
|
Logf(f, "Not purging as --dry-run set")
|
||||||
} else {
|
} else {
|
||||||
err = doPurge()
|
err = doPurge()
|
||||||
if err == ErrorCantPurge {
|
if err == ErrorCantPurge {
|
||||||
@ -1006,12 +1006,12 @@ func dedupeRename(remote string, objs []Object) {
|
|||||||
newObj, err := doMove(o, newName)
|
newObj, err := doMove(o, newName)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
Stats.Error()
|
Stats.Error()
|
||||||
ErrorLog(o, "Failed to rename: %v", err)
|
Errorf(o, "Failed to rename: %v", err)
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
Log(newObj, "renamed from: %v", o)
|
Logf(newObj, "renamed from: %v", o)
|
||||||
} else {
|
} else {
|
||||||
Log(remote, "Not renaming to %q as --dry-run", newName)
|
Logf(remote, "Not renaming to %q as --dry-run", newName)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -1024,7 +1024,7 @@ func dedupeDeleteAllButOne(keep int, remote string, objs []Object) {
|
|||||||
}
|
}
|
||||||
_ = DeleteFile(o)
|
_ = DeleteFile(o)
|
||||||
}
|
}
|
||||||
Log(remote, "Deleted %d extra copies", len(objs)-1)
|
Logf(remote, "Deleted %d extra copies", len(objs)-1)
|
||||||
}
|
}
|
||||||
|
|
||||||
// dedupeDeleteIdentical deletes all but one of identical (by hash) copies
|
// dedupeDeleteIdentical deletes all but one of identical (by hash) copies
|
||||||
@ -1042,7 +1042,7 @@ func dedupeDeleteIdentical(remote string, objs []Object) []Object {
|
|||||||
objs = nil
|
objs = nil
|
||||||
for md5sum, hashObjs := range byHash {
|
for md5sum, hashObjs := range byHash {
|
||||||
if len(hashObjs) > 1 {
|
if len(hashObjs) > 1 {
|
||||||
Log(remote, "Deleting %d/%d identical duplicates (md5sum %q)", len(hashObjs)-1, len(hashObjs), md5sum)
|
Logf(remote, "Deleting %d/%d identical duplicates (md5sum %q)", len(hashObjs)-1, len(hashObjs), md5sum)
|
||||||
for _, o := range hashObjs[1:] {
|
for _, o := range hashObjs[1:] {
|
||||||
_ = DeleteFile(o)
|
_ = DeleteFile(o)
|
||||||
}
|
}
|
||||||
@ -1145,7 +1145,7 @@ var _ pflag.Value = (*DeduplicateMode)(nil)
|
|||||||
// delete all but one or rename them to be different. Only useful with
|
// delete all but one or rename them to be different. Only useful with
|
||||||
// Google Drive which can have duplicate file names.
|
// Google Drive which can have duplicate file names.
|
||||||
func Deduplicate(f Fs, mode DeduplicateMode) error {
|
func Deduplicate(f Fs, mode DeduplicateMode) error {
|
||||||
Log(f, "Looking for duplicates using %v mode.", mode)
|
Logf(f, "Looking for duplicates using %v mode.", mode)
|
||||||
files := map[string][]Object{}
|
files := map[string][]Object{}
|
||||||
list := NewLister().Start(f, "")
|
list := NewLister().Start(f, "")
|
||||||
for {
|
for {
|
||||||
@ -1162,10 +1162,10 @@ func Deduplicate(f Fs, mode DeduplicateMode) error {
|
|||||||
}
|
}
|
||||||
for remote, objs := range files {
|
for remote, objs := range files {
|
||||||
if len(objs) > 1 {
|
if len(objs) > 1 {
|
||||||
Log(remote, "Found %d duplicates - deleting identical copies", len(objs))
|
Logf(remote, "Found %d duplicates - deleting identical copies", len(objs))
|
||||||
objs = dedupeDeleteIdentical(remote, objs)
|
objs = dedupeDeleteIdentical(remote, objs)
|
||||||
if len(objs) <= 1 {
|
if len(objs) <= 1 {
|
||||||
Log(remote, "All duplicates removed")
|
Logf(remote, "All duplicates removed")
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
switch mode {
|
switch mode {
|
||||||
@ -1206,7 +1206,7 @@ func listToChan(list *Lister) ObjectsChan {
|
|||||||
if err != nil {
|
if err != nil {
|
||||||
if err != ErrorDirNotFound {
|
if err != ErrorDirNotFound {
|
||||||
Stats.Error()
|
Stats.Error()
|
||||||
ErrorLog(nil, "Failed to list: %v", err)
|
Errorf(nil, "Failed to list: %v", err)
|
||||||
}
|
}
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
@ -1229,7 +1229,7 @@ func CleanUp(f Fs) error {
|
|||||||
return errors.Errorf("%v doesn't support cleanup", f)
|
return errors.Errorf("%v doesn't support cleanup", f)
|
||||||
}
|
}
|
||||||
if Config.DryRun {
|
if Config.DryRun {
|
||||||
Log(f, "Not running cleanup as --dry-run set")
|
Logf(f, "Not running cleanup as --dry-run set")
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
return doCleanUp()
|
return doCleanUp()
|
||||||
@ -1276,7 +1276,7 @@ func Cat(f Fs, w io.Writer, offset, count int64) error {
|
|||||||
in, err := o.Open(options...)
|
in, err := o.Open(options...)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
Stats.Error()
|
Stats.Error()
|
||||||
ErrorLog(o, "Failed to open: %v", err)
|
Errorf(o, "Failed to open: %v", err)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
if count >= 0 {
|
if count >= 0 {
|
||||||
@ -1291,7 +1291,7 @@ func Cat(f Fs, w io.Writer, offset, count int64) error {
|
|||||||
err = in.Close()
|
err = in.Close()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
Stats.Error()
|
Stats.Error()
|
||||||
ErrorLog(o, "Failed to close: %v", err)
|
Errorf(o, "Failed to close: %v", err)
|
||||||
}
|
}
|
||||||
}()
|
}()
|
||||||
// take the lock just before we output stuff, so at the last possible moment
|
// take the lock just before we output stuff, so at the last possible moment
|
||||||
@ -1300,7 +1300,7 @@ func Cat(f Fs, w io.Writer, offset, count int64) error {
|
|||||||
_, err = io.Copy(w, in)
|
_, err = io.Copy(w, in)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
Stats.Error()
|
Stats.Error()
|
||||||
ErrorLog(o, "Failed to send to output: %v", err)
|
Errorf(o, "Failed to send to output: %v", err)
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
@ -1315,7 +1315,7 @@ func Rmdirs(f Fs, dir string) error {
|
|||||||
o, dir, err := list.Get()
|
o, dir, err := list.Get()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
Stats.Error()
|
Stats.Error()
|
||||||
ErrorLog(f, "Failed to list: %v", err)
|
Errorf(f, "Failed to list: %v", err)
|
||||||
return err
|
return err
|
||||||
} else if dir != nil {
|
} else if dir != nil {
|
||||||
// add a new directory as empty
|
// add a new directory as empty
|
||||||
@ -1357,7 +1357,7 @@ func Rmdirs(f Fs, dir string) error {
|
|||||||
err := TryRmdir(f, dir)
|
err := TryRmdir(f, dir)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
Stats.Error()
|
Stats.Error()
|
||||||
ErrorLog(dir, "Failed to rmdir: %v", err)
|
Errorf(dir, "Failed to rmdir: %v", err)
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -473,7 +473,7 @@ func TestCheck(t *testing.T) {
|
|||||||
defer r.Finalise()
|
defer r.Finalise()
|
||||||
|
|
||||||
check := func(i int, wantErrors int64) {
|
check := func(i int, wantErrors int64) {
|
||||||
fs.Debug(r.fremote, "%d: Starting check test", i)
|
fs.Debugf(r.fremote, "%d: Starting check test", i)
|
||||||
oldErrors := fs.Stats.GetErrors()
|
oldErrors := fs.Stats.GetErrors()
|
||||||
err := fs.Check(r.flocal, r.fremote)
|
err := fs.Check(r.flocal, r.fremote)
|
||||||
gotErrors := fs.Stats.GetErrors() - oldErrors
|
gotErrors := fs.Stats.GetErrors() - oldErrors
|
||||||
@ -486,7 +486,7 @@ func TestCheck(t *testing.T) {
|
|||||||
if wantErrors != gotErrors {
|
if wantErrors != gotErrors {
|
||||||
t.Errorf("%d: Expecting %d errors but got %d", i, wantErrors, gotErrors)
|
t.Errorf("%d: Expecting %d errors but got %d", i, wantErrors, gotErrors)
|
||||||
}
|
}
|
||||||
fs.Debug(r.fremote, "%d: Ending check test", i)
|
fs.Debugf(r.fremote, "%d: Ending check test", i)
|
||||||
}
|
}
|
||||||
|
|
||||||
file1 := r.WriteBoth("rutabaga", "is tasty", t3)
|
file1 := r.WriteBoth("rutabaga", "is tasty", t3)
|
||||||
|
72
fs/sync.go
72
fs/sync.go
@ -72,17 +72,17 @@ func newSyncCopyMove(fdst, fsrc Fs, deleteMode DeleteMode, DoMove bool) (*syncCo
|
|||||||
trackRenamesCh: make(chan Object, Config.Checkers),
|
trackRenamesCh: make(chan Object, Config.Checkers),
|
||||||
}
|
}
|
||||||
if s.noTraverse && s.deleteMode != DeleteModeOff {
|
if s.noTraverse && s.deleteMode != DeleteModeOff {
|
||||||
ErrorLog(nil, "Ignoring --no-traverse with sync")
|
Errorf(nil, "Ignoring --no-traverse with sync")
|
||||||
s.noTraverse = false
|
s.noTraverse = false
|
||||||
}
|
}
|
||||||
if s.trackRenames {
|
if s.trackRenames {
|
||||||
// Don't track renames for remotes without server-side move support.
|
// Don't track renames for remotes without server-side move support.
|
||||||
if !CanServerSideMove(fdst) {
|
if !CanServerSideMove(fdst) {
|
||||||
ErrorLog(fdst, "Ignoring --track-renames as the destination does not support server-side move or copy")
|
Errorf(fdst, "Ignoring --track-renames as the destination does not support server-side move or copy")
|
||||||
s.trackRenames = false
|
s.trackRenames = false
|
||||||
}
|
}
|
||||||
if s.commonHash == HashNone {
|
if s.commonHash == HashNone {
|
||||||
ErrorLog(fdst, "Ignoring --track-renames as the source and destination do not have a common hash")
|
Errorf(fdst, "Ignoring --track-renames as the source and destination do not have a common hash")
|
||||||
s.trackRenames = false
|
s.trackRenames = false
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -92,7 +92,7 @@ func newSyncCopyMove(fdst, fsrc Fs, deleteMode DeleteMode, DoMove bool) (*syncCo
|
|||||||
s.deleteMode = DeleteModeAfter
|
s.deleteMode = DeleteModeAfter
|
||||||
}
|
}
|
||||||
if s.noTraverse {
|
if s.noTraverse {
|
||||||
ErrorLog(nil, "Ignoring --no-traverse with --track-renames")
|
Errorf(nil, "Ignoring --no-traverse with --track-renames")
|
||||||
s.noTraverse = false
|
s.noTraverse = false
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -195,17 +195,17 @@ func (s *syncCopyMove) readDstFiles() {
|
|||||||
// transferred or not.
|
// transferred or not.
|
||||||
func NeedTransfer(dst, src Object) bool {
|
func NeedTransfer(dst, src Object) bool {
|
||||||
if dst == nil {
|
if dst == nil {
|
||||||
Debug(src, "Couldn't find file - need to transfer")
|
Debugf(src, "Couldn't find file - need to transfer")
|
||||||
return true
|
return true
|
||||||
}
|
}
|
||||||
// If we should ignore existing files, don't transfer
|
// If we should ignore existing files, don't transfer
|
||||||
if Config.IgnoreExisting {
|
if Config.IgnoreExisting {
|
||||||
Debug(src, "Destination exists, skipping")
|
Debugf(src, "Destination exists, skipping")
|
||||||
return false
|
return false
|
||||||
}
|
}
|
||||||
// If we should upload unconditionally
|
// If we should upload unconditionally
|
||||||
if Config.IgnoreTimes {
|
if Config.IgnoreTimes {
|
||||||
Debug(src, "Transferring unconditionally as --ignore-times is in use")
|
Debugf(src, "Transferring unconditionally as --ignore-times is in use")
|
||||||
return true
|
return true
|
||||||
}
|
}
|
||||||
// If UpdateOlder is in effect, skip if dst is newer than src
|
// If UpdateOlder is in effect, skip if dst is newer than src
|
||||||
@ -223,21 +223,21 @@ func NeedTransfer(dst, src Object) bool {
|
|||||||
}
|
}
|
||||||
switch {
|
switch {
|
||||||
case dt >= modifyWindow:
|
case dt >= modifyWindow:
|
||||||
Debug(src, "Destination is newer than source, skipping")
|
Debugf(src, "Destination is newer than source, skipping")
|
||||||
return false
|
return false
|
||||||
case dt <= -modifyWindow:
|
case dt <= -modifyWindow:
|
||||||
Debug(src, "Destination is older than source, transferring")
|
Debugf(src, "Destination is older than source, transferring")
|
||||||
default:
|
default:
|
||||||
if src.Size() == dst.Size() {
|
if src.Size() == dst.Size() {
|
||||||
Debug(src, "Destination mod time is within %v of source and sizes identical, skipping", modifyWindow)
|
Debugf(src, "Destination mod time is within %v of source and sizes identical, skipping", modifyWindow)
|
||||||
return false
|
return false
|
||||||
}
|
}
|
||||||
Debug(src, "Destination mod time is within %v of source but sizes differ, transferring", modifyWindow)
|
Debugf(src, "Destination mod time is within %v of source but sizes differ, transferring", modifyWindow)
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
// Check to see if changed or not
|
// Check to see if changed or not
|
||||||
if Equal(src, dst) {
|
if Equal(src, dst) {
|
||||||
Debug(src, "Unchanged skipping")
|
Debugf(src, "Unchanged skipping")
|
||||||
return false
|
return false
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -392,7 +392,7 @@ func (s *syncCopyMove) startCheckers() {
|
|||||||
// This stops the background checkers
|
// This stops the background checkers
|
||||||
func (s *syncCopyMove) stopCheckers() {
|
func (s *syncCopyMove) stopCheckers() {
|
||||||
close(s.toBeChecked)
|
close(s.toBeChecked)
|
||||||
Log(s.fdst, "Waiting for checks to finish")
|
Logf(s.fdst, "Waiting for checks to finish")
|
||||||
s.checkerWg.Wait()
|
s.checkerWg.Wait()
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -407,7 +407,7 @@ func (s *syncCopyMove) startTransfers() {
|
|||||||
// This stops the background transfers
|
// This stops the background transfers
|
||||||
func (s *syncCopyMove) stopTransfers() {
|
func (s *syncCopyMove) stopTransfers() {
|
||||||
close(s.toBeUploaded)
|
close(s.toBeUploaded)
|
||||||
Log(s.fdst, "Waiting for transfers to finish")
|
Logf(s.fdst, "Waiting for transfers to finish")
|
||||||
s.transfersWg.Wait()
|
s.transfersWg.Wait()
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -428,7 +428,7 @@ func (s *syncCopyMove) stopRenamers() {
|
|||||||
return
|
return
|
||||||
}
|
}
|
||||||
close(s.toBeRenamed)
|
close(s.toBeRenamed)
|
||||||
Log(s.fdst, "Waiting for renames to finish")
|
Logf(s.fdst, "Waiting for renames to finish")
|
||||||
s.renamerWg.Wait()
|
s.renamerWg.Wait()
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -484,7 +484,7 @@ func (s *syncCopyMove) stopDeleters() {
|
|||||||
// have been found have been removed from dstFiles already.
|
// have been found have been removed from dstFiles already.
|
||||||
func (s *syncCopyMove) deleteFiles(checkSrcMap bool) error {
|
func (s *syncCopyMove) deleteFiles(checkSrcMap bool) error {
|
||||||
if Stats.Errored() {
|
if Stats.Errored() {
|
||||||
ErrorLog(s.fdst, "%v", ErrorNotDeleting)
|
Errorf(s.fdst, "%v", ErrorNotDeleting)
|
||||||
return ErrorNotDeleting
|
return ErrorNotDeleting
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -515,7 +515,7 @@ func (s *syncCopyMove) renameHash(obj Object) (hash string) {
|
|||||||
var err error
|
var err error
|
||||||
hash, err = obj.Hash(s.commonHash)
|
hash, err = obj.Hash(s.commonHash)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
Debug(obj, "Hash failed: %v", err)
|
Debugf(obj, "Hash failed: %v", err)
|
||||||
return ""
|
return ""
|
||||||
}
|
}
|
||||||
if hash == "" {
|
if hash == "" {
|
||||||
@ -551,7 +551,7 @@ func (s *syncCopyMove) popRenameMap(hash string) (dst Object) {
|
|||||||
// makeRenameMap builds a map of the destination files by hash that
|
// makeRenameMap builds a map of the destination files by hash that
|
||||||
// match sizes in the slice of objects in s.renameCheck
|
// match sizes in the slice of objects in s.renameCheck
|
||||||
func (s *syncCopyMove) makeRenameMap() {
|
func (s *syncCopyMove) makeRenameMap() {
|
||||||
Debug(s.fdst, "Making map for --track-renames")
|
Debugf(s.fdst, "Making map for --track-renames")
|
||||||
|
|
||||||
// first make a map of possible sizes we need to check
|
// first make a map of possible sizes we need to check
|
||||||
possibleSizes := map[int64]struct{}{}
|
possibleSizes := map[int64]struct{}{}
|
||||||
@ -584,7 +584,7 @@ func (s *syncCopyMove) makeRenameMap() {
|
|||||||
}()
|
}()
|
||||||
}
|
}
|
||||||
wg.Wait()
|
wg.Wait()
|
||||||
Debug(s.fdst, "Finished making map for --track-renames")
|
Debugf(s.fdst, "Finished making map for --track-renames")
|
||||||
}
|
}
|
||||||
|
|
||||||
// tryRename renames a src object when doing track renames if
|
// tryRename renames a src object when doing track renames if
|
||||||
@ -605,7 +605,7 @@ func (s *syncCopyMove) tryRename(src Object) bool {
|
|||||||
|
|
||||||
err := MoveFile(s.fdst, s.fdst, src.Remote(), dst.Remote())
|
err := MoveFile(s.fdst, s.fdst, src.Remote(), dst.Remote())
|
||||||
if err != nil {
|
if err != nil {
|
||||||
Debug(src, "Failed to rename to %q: %v", dst.Remote(), err)
|
Debugf(src, "Failed to rename to %q: %v", dst.Remote(), err)
|
||||||
return false
|
return false
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -614,7 +614,7 @@ func (s *syncCopyMove) tryRename(src Object) bool {
|
|||||||
delete(s.dstFiles, dst.Remote())
|
delete(s.dstFiles, dst.Remote())
|
||||||
s.dstFilesMu.Unlock()
|
s.dstFilesMu.Unlock()
|
||||||
|
|
||||||
Debug(src, "Renamed from %q", dst.Remote())
|
Debugf(src, "Renamed from %q", dst.Remote())
|
||||||
return true
|
return true
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -627,7 +627,7 @@ func (s *syncCopyMove) tryRename(src Object) bool {
|
|||||||
// dir is the start directory, "" for root
|
// dir is the start directory, "" for root
|
||||||
func (s *syncCopyMove) runRecursive() error {
|
func (s *syncCopyMove) runRecursive() error {
|
||||||
if Same(s.fdst, s.fsrc) {
|
if Same(s.fdst, s.fsrc) {
|
||||||
ErrorLog(s.fdst, "Nothing to do as source and destination are the same")
|
Errorf(s.fdst, "Nothing to do as source and destination are the same")
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -694,7 +694,7 @@ func (s *syncCopyMove) runRecursive() error {
|
|||||||
if err != nil {
|
if err != nil {
|
||||||
dst = nil
|
dst = nil
|
||||||
if err != ErrorObjectNotFound {
|
if err != ErrorObjectNotFound {
|
||||||
Debug(src, "Error making NewObject: %v", err)
|
Debugf(src, "Error making NewObject: %v", err)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
@ -739,7 +739,7 @@ func (s *syncCopyMove) runRecursive() error {
|
|||||||
// Delete files during or after
|
// Delete files during or after
|
||||||
if s.deleteMode == DeleteModeDuring || s.deleteMode == DeleteModeAfter {
|
if s.deleteMode == DeleteModeDuring || s.deleteMode == DeleteModeAfter {
|
||||||
if err != nil {
|
if err != nil {
|
||||||
ErrorLog(s.fdst, "%v", ErrorNotDeleting)
|
Errorf(s.fdst, "%v", ErrorNotDeleting)
|
||||||
} else {
|
} else {
|
||||||
err = s.deleteFiles(false)
|
err = s.deleteFiles(false)
|
||||||
}
|
}
|
||||||
@ -775,7 +775,7 @@ func (s *syncCopyMove) runDirAtATime() error {
|
|||||||
}
|
}
|
||||||
|
|
||||||
if Same(s.fdst, s.fsrc) {
|
if Same(s.fdst, s.fsrc) {
|
||||||
ErrorLog(s.fdst, "Nothing to do as source and destination are the same")
|
Errorf(s.fdst, "Nothing to do as source and destination are the same")
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -859,7 +859,7 @@ func (s *syncCopyMove) runDirAtATime() error {
|
|||||||
// Delete files after
|
// Delete files after
|
||||||
if s.deleteMode == DeleteModeAfter {
|
if s.deleteMode == DeleteModeAfter {
|
||||||
if s.currentError() != nil {
|
if s.currentError() != nil {
|
||||||
ErrorLog(s.fdst, "%v", ErrorNotDeleting)
|
Errorf(s.fdst, "%v", ErrorNotDeleting)
|
||||||
} else {
|
} else {
|
||||||
s.processError(s.deleteFiles(false))
|
s.processError(s.deleteFiles(false))
|
||||||
}
|
}
|
||||||
@ -941,7 +941,7 @@ func (s *syncCopyMove) transfer(dst, src BasicInfo, job listDirJob, jobs *[]list
|
|||||||
} else {
|
} else {
|
||||||
// FIXME src is file, dst is directory
|
// FIXME src is file, dst is directory
|
||||||
err := errors.New("can't overwrite directory with file")
|
err := errors.New("can't overwrite directory with file")
|
||||||
ErrorLog(srcX, "%v", err)
|
Errorf(srcX, "%v", err)
|
||||||
s.processError(err)
|
s.processError(err)
|
||||||
}
|
}
|
||||||
case *Dir:
|
case *Dir:
|
||||||
@ -958,7 +958,7 @@ func (s *syncCopyMove) transfer(dst, src BasicInfo, job listDirJob, jobs *[]list
|
|||||||
} else {
|
} else {
|
||||||
// FIXME src is dir, dst is file
|
// FIXME src is dir, dst is file
|
||||||
err := errors.New("can't overwrite file with directory")
|
err := errors.New("can't overwrite file with directory")
|
||||||
ErrorLog(dstX, "%v", err)
|
Errorf(dstX, "%v", err)
|
||||||
s.processError(err)
|
s.processError(err)
|
||||||
}
|
}
|
||||||
default:
|
default:
|
||||||
@ -1027,7 +1027,7 @@ func (s *syncCopyMove) _runDirAtATime(job listDirJob) (jobs []listDirJob) {
|
|||||||
iSrc-- // retry the src
|
iSrc-- // retry the src
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
// Debug(nil, "src = %v, dst = %v", src, dst)
|
// Debugf(nil, "src = %v, dst = %v", src, dst)
|
||||||
switch {
|
switch {
|
||||||
case src == nil:
|
case src == nil:
|
||||||
s.dstOnly(dst, job, &jobs)
|
s.dstOnly(dst, job, &jobs)
|
||||||
@ -1096,27 +1096,27 @@ func moveDir(fdst, fsrc Fs) error {
|
|||||||
// MoveDir moves fsrc into fdst
|
// MoveDir moves fsrc into fdst
|
||||||
func MoveDir(fdst, fsrc Fs) error {
|
func MoveDir(fdst, fsrc Fs) error {
|
||||||
if Same(fdst, fsrc) {
|
if Same(fdst, fsrc) {
|
||||||
ErrorLog(fdst, "Nothing to do as source and destination are the same")
|
Errorf(fdst, "Nothing to do as source and destination are the same")
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// First attempt to use DirMover if exists, same Fs and no filters are active
|
// First attempt to use DirMover if exists, same Fs and no filters are active
|
||||||
if fdstDirMove := fdst.Features().DirMove; fdstDirMove != nil && SameConfig(fsrc, fdst) && Config.Filter.InActive() {
|
if fdstDirMove := fdst.Features().DirMove; fdstDirMove != nil && SameConfig(fsrc, fdst) && Config.Filter.InActive() {
|
||||||
if Config.DryRun {
|
if Config.DryRun {
|
||||||
Log(fdst, "Not doing server side directory move as --dry-run")
|
Logf(fdst, "Not doing server side directory move as --dry-run")
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
Debug(fdst, "Using server side directory move")
|
Debugf(fdst, "Using server side directory move")
|
||||||
err := fdstDirMove(fsrc)
|
err := fdstDirMove(fsrc)
|
||||||
switch err {
|
switch err {
|
||||||
case ErrorCantDirMove, ErrorDirExists:
|
case ErrorCantDirMove, ErrorDirExists:
|
||||||
Debug(fdst, "Server side directory move failed - fallback to file moves: %v", err)
|
Debugf(fdst, "Server side directory move failed - fallback to file moves: %v", err)
|
||||||
case nil:
|
case nil:
|
||||||
Debug(fdst, "Server side directory move succeeded")
|
Debugf(fdst, "Server side directory move succeeded")
|
||||||
return nil
|
return nil
|
||||||
default:
|
default:
|
||||||
Stats.Error()
|
Stats.Error()
|
||||||
ErrorLog(fdst, "Server side directory move failed: %v", err)
|
Errorf(fdst, "Server side directory move failed: %v", err)
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -1124,7 +1124,7 @@ func MoveDir(fdst, fsrc Fs) error {
|
|||||||
// The two remotes mustn't overlap if we didn't do server side move
|
// The two remotes mustn't overlap if we didn't do server side move
|
||||||
if Overlapping(fdst, fsrc) {
|
if Overlapping(fdst, fsrc) {
|
||||||
err := ErrorCantMoveOverlapping
|
err := ErrorCantMoveOverlapping
|
||||||
ErrorLog(fdst, "%v", err)
|
Errorf(fdst, "%v", err)
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -321,7 +321,7 @@ func TestSyncAfterChangingModtimeOnlyWithNoUpdateModTime(t *testing.T) {
|
|||||||
defer r.Finalise()
|
defer r.Finalise()
|
||||||
|
|
||||||
if r.fremote.Hashes().Count() == 0 {
|
if r.fremote.Hashes().Count() == 0 {
|
||||||
t.Log("Can't check this if no hashes supported")
|
t.Logf("Can't check this if no hashes supported")
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -339,7 +339,7 @@ func (f *Fs) list(dir string, level int, fn listFn) error {
|
|||||||
}
|
}
|
||||||
for _, object := range objects.Items {
|
for _, object := range objects.Items {
|
||||||
if !strings.HasPrefix(object.Name, root) {
|
if !strings.HasPrefix(object.Name, root) {
|
||||||
fs.Log(f, "Odd name received %q", object.Name)
|
fs.Logf(f, "Odd name received %q", object.Name)
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
remote := object.Name[rootLength:]
|
remote := object.Name[rootLength:]
|
||||||
@ -505,7 +505,7 @@ func (f *Fs) Precision() time.Duration {
|
|||||||
func (f *Fs) Copy(src fs.Object, remote string) (fs.Object, error) {
|
func (f *Fs) Copy(src fs.Object, remote string) (fs.Object, error) {
|
||||||
srcObj, ok := src.(*Object)
|
srcObj, ok := src.(*Object)
|
||||||
if !ok {
|
if !ok {
|
||||||
fs.Debug(src, "Can't copy - not same remote type")
|
fs.Debugf(src, "Can't copy - not same remote type")
|
||||||
return nil, fs.ErrorCantCopy
|
return nil, fs.ErrorCantCopy
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -575,7 +575,7 @@ func (o *Object) setMetaData(info *storage.Object) {
|
|||||||
// Read md5sum
|
// Read md5sum
|
||||||
md5sumData, err := base64.StdEncoding.DecodeString(info.Md5Hash)
|
md5sumData, err := base64.StdEncoding.DecodeString(info.Md5Hash)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
fs.Log(o, "Bad MD5 decode: %v", err)
|
fs.Logf(o, "Bad MD5 decode: %v", err)
|
||||||
} else {
|
} else {
|
||||||
o.md5sum = hex.EncodeToString(md5sumData)
|
o.md5sum = hex.EncodeToString(md5sumData)
|
||||||
}
|
}
|
||||||
@ -588,13 +588,13 @@ func (o *Object) setMetaData(info *storage.Object) {
|
|||||||
o.modTime = modTime
|
o.modTime = modTime
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
fs.Debug(o, "Failed to read mtime from metadata: %s", err)
|
fs.Debugf(o, "Failed to read mtime from metadata: %s", err)
|
||||||
}
|
}
|
||||||
|
|
||||||
// Fallback to the Updated time
|
// Fallback to the Updated time
|
||||||
modTime, err := time.Parse(timeFormatIn, info.Updated)
|
modTime, err := time.Parse(timeFormatIn, info.Updated)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
fs.Log(o, "Bad time decode: %v", err)
|
fs.Logf(o, "Bad time decode: %v", err)
|
||||||
} else {
|
} else {
|
||||||
o.modTime = modTime
|
o.modTime = modTime
|
||||||
}
|
}
|
||||||
@ -627,7 +627,7 @@ func (o *Object) readMetaData() (err error) {
|
|||||||
func (o *Object) ModTime() time.Time {
|
func (o *Object) ModTime() time.Time {
|
||||||
err := o.readMetaData()
|
err := o.readMetaData()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
// fs.Log(o, "Failed to read metadata: %v", err)
|
// fs.Logf(o, "Failed to read metadata: %v", err)
|
||||||
return time.Now()
|
return time.Now()
|
||||||
}
|
}
|
||||||
return o.modTime
|
return o.modTime
|
||||||
|
@ -127,7 +127,7 @@ func (f *Fs) getCredentials() (err error) {
|
|||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
// fs.Debug(f, "Got credentials %+v", result)
|
// fs.Debugf(f, "Got credentials %+v", result)
|
||||||
if result.Token == "" || result.Endpoint == "" || result.Expires == "" {
|
if result.Token == "" || result.Endpoint == "" || result.Expires == "" {
|
||||||
return errors.New("couldn't read token, result and expired from credentials")
|
return errors.New("couldn't read token, result and expired from credentials")
|
||||||
}
|
}
|
||||||
@ -137,7 +137,7 @@ func (f *Fs) getCredentials() (err error) {
|
|||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
f.expires = expires
|
f.expires = expires
|
||||||
fs.Debug(f, "Got swift credentials (expiry %v in %v)", f.expires, f.expires.Sub(time.Now()))
|
fs.Debugf(f, "Got swift credentials (expiry %v in %v)", f.expires, f.expires.Sub(time.Now()))
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -309,7 +309,7 @@ func (f *Fs) cleanRemote(name string) string {
|
|||||||
if !utf8.ValidString(name) {
|
if !utf8.ValidString(name) {
|
||||||
f.wmu.Lock()
|
f.wmu.Lock()
|
||||||
if _, ok := f.warned[name]; !ok {
|
if _, ok := f.warned[name]; !ok {
|
||||||
fs.Debug(f, "Replacing invalid UTF-8 characters in %q", name)
|
fs.Debugf(f, "Replacing invalid UTF-8 characters in %q", name)
|
||||||
f.warned[name] = struct{}{}
|
f.warned[name] = struct{}{}
|
||||||
}
|
}
|
||||||
f.wmu.Unlock()
|
f.wmu.Unlock()
|
||||||
@ -446,7 +446,7 @@ func (f *Fs) Purge() error {
|
|||||||
func (f *Fs) Move(src fs.Object, remote string) (fs.Object, error) {
|
func (f *Fs) Move(src fs.Object, remote string) (fs.Object, error) {
|
||||||
srcObj, ok := src.(*Object)
|
srcObj, ok := src.(*Object)
|
||||||
if !ok {
|
if !ok {
|
||||||
fs.Debug(src, "Can't move - not same remote type")
|
fs.Debugf(src, "Can't move - not same remote type")
|
||||||
return nil, fs.ErrorCantMove
|
return nil, fs.ErrorCantMove
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -496,7 +496,7 @@ func (f *Fs) Move(src fs.Object, remote string) (fs.Object, error) {
|
|||||||
func (f *Fs) DirMove(src fs.Fs) error {
|
func (f *Fs) DirMove(src fs.Fs) error {
|
||||||
srcFs, ok := src.(*Fs)
|
srcFs, ok := src.(*Fs)
|
||||||
if !ok {
|
if !ok {
|
||||||
fs.Debug(srcFs, "Can't move directory - not same remote type")
|
fs.Debugf(srcFs, "Can't move directory - not same remote type")
|
||||||
return fs.ErrorCantDirMove
|
return fs.ErrorCantDirMove
|
||||||
}
|
}
|
||||||
// Check if source exists
|
// Check if source exists
|
||||||
@ -601,24 +601,24 @@ func (o *Object) Storable() bool {
|
|||||||
// Check for control characters in the remote name and show non storable
|
// Check for control characters in the remote name and show non storable
|
||||||
for _, c := range o.Remote() {
|
for _, c := range o.Remote() {
|
||||||
if c >= 0x00 && c < 0x20 || c == 0x7F {
|
if c >= 0x00 && c < 0x20 || c == 0x7F {
|
||||||
fs.Debug(o.fs, "Can't store file with control characters: %q", o.Remote())
|
fs.Debugf(o.fs, "Can't store file with control characters: %q", o.Remote())
|
||||||
return false
|
return false
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
mode := o.info.Mode()
|
mode := o.info.Mode()
|
||||||
// On windows a file with os.ModeSymlink represents a file with reparse points
|
// On windows a file with os.ModeSymlink represents a file with reparse points
|
||||||
if runtime.GOOS == "windows" && (mode&os.ModeSymlink) != 0 {
|
if runtime.GOOS == "windows" && (mode&os.ModeSymlink) != 0 {
|
||||||
fs.Debug(o, "Clearing symlink bit to allow a file with reparse points to be copied")
|
fs.Debugf(o, "Clearing symlink bit to allow a file with reparse points to be copied")
|
||||||
mode &^= os.ModeSymlink
|
mode &^= os.ModeSymlink
|
||||||
}
|
}
|
||||||
if mode&os.ModeSymlink != 0 {
|
if mode&os.ModeSymlink != 0 {
|
||||||
fs.Debug(o, "Can't follow symlink without -L/--copy-links")
|
fs.Debugf(o, "Can't follow symlink without -L/--copy-links")
|
||||||
return false
|
return false
|
||||||
} else if mode&(os.ModeNamedPipe|os.ModeSocket|os.ModeDevice) != 0 {
|
} else if mode&(os.ModeNamedPipe|os.ModeSocket|os.ModeDevice) != 0 {
|
||||||
fs.Debug(o, "Can't transfer non file/directory")
|
fs.Debugf(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, "Skipping directory")
|
// fs.Debugf(o, "Skipping directory")
|
||||||
return false
|
return false
|
||||||
}
|
}
|
||||||
return true
|
return true
|
||||||
@ -662,7 +662,7 @@ func (o *Object) Open(options ...fs.OpenOption) (in io.ReadCloser, err error) {
|
|||||||
offset = x.Offset
|
offset = x.Offset
|
||||||
default:
|
default:
|
||||||
if option.Mandatory() {
|
if option.Mandatory() {
|
||||||
fs.Log(o, "Unsupported mandatory option: %v", option)
|
fs.Logf(o, "Unsupported mandatory option: %v", option)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -714,9 +714,9 @@ func (o *Object) Update(in io.Reader, src fs.ObjectInfo) error {
|
|||||||
err = closeErr
|
err = closeErr
|
||||||
}
|
}
|
||||||
if err != nil {
|
if err != nil {
|
||||||
fs.Debug(o, "Removing partially written file on error: %v", err)
|
fs.Debugf(o, "Removing partially written file on error: %v", err)
|
||||||
if removeErr := os.Remove(o.path); removeErr != nil {
|
if removeErr := os.Remove(o.path); removeErr != nil {
|
||||||
fs.ErrorLog(o, "Failed to remove partially written file: %v", removeErr)
|
fs.Errorf(o, "Failed to remove partially written file: %v", removeErr)
|
||||||
}
|
}
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
@ -859,7 +859,7 @@ func cleanWindowsName(f *Fs, name string) string {
|
|||||||
if name2 != original && f != nil {
|
if name2 != original && f != nil {
|
||||||
f.wmu.Lock()
|
f.wmu.Lock()
|
||||||
if _, ok := f.warned[name]; !ok {
|
if _, ok := f.warned[name]; !ok {
|
||||||
fs.Debug(f, "Replacing invalid characters in %q to %q", name, name2)
|
fs.Debugf(f, "Replacing invalid characters in %q to %q", name, name2)
|
||||||
f.warned[name] = struct{}{}
|
f.warned[name] = struct{}{}
|
||||||
}
|
}
|
||||||
f.wmu.Unlock()
|
f.wmu.Unlock()
|
||||||
|
@ -23,7 +23,7 @@ func readDevice(fi os.FileInfo) uint64 {
|
|||||||
}
|
}
|
||||||
statT, ok := fi.Sys().(*syscall.Stat_t)
|
statT, ok := fi.Sys().(*syscall.Stat_t)
|
||||||
if !ok {
|
if !ok {
|
||||||
fs.Debug(fi.Name(), "Type assertion fi.Sys().(*syscall.Stat_t) failed from: %#v", fi.Sys())
|
fs.Debugf(fi.Name(), "Type assertion fi.Sys().(*syscall.Stat_t) failed from: %#v", fi.Sys())
|
||||||
return devUnset
|
return devUnset
|
||||||
}
|
}
|
||||||
return uint64(statT.Dev)
|
return uint64(statT.Dev)
|
||||||
|
@ -98,9 +98,9 @@ func putToken(name string, token *oauth2.Token) error {
|
|||||||
if tokenString != old {
|
if tokenString != old {
|
||||||
err = fs.ConfigSetValueAndSave(name, fs.ConfigToken, tokenString)
|
err = fs.ConfigSetValueAndSave(name, fs.ConfigToken, tokenString)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
fs.ErrorLog(nil, "Failed to save new token in config file: %v", err)
|
fs.Errorf(nil, "Failed to save new token in config file: %v", err)
|
||||||
} else {
|
} else {
|
||||||
fs.Debug(name, "Saved new token in config file")
|
fs.Debugf(name, "Saved new token in config file")
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return nil
|
return nil
|
||||||
@ -370,7 +370,7 @@ type authServer struct {
|
|||||||
|
|
||||||
// startWebServer runs an internal web server to receive config details
|
// startWebServer runs an internal web server to receive config details
|
||||||
func (s *authServer) Start() {
|
func (s *authServer) Start() {
|
||||||
fs.Debug(nil, "Starting auth server on %s", s.bindAddress)
|
fs.Debugf(nil, "Starting auth server on %s", s.bindAddress)
|
||||||
mux := http.NewServeMux()
|
mux := http.NewServeMux()
|
||||||
server := &http.Server{
|
server := &http.Server{
|
||||||
Addr: s.bindAddress,
|
Addr: s.bindAddress,
|
||||||
@ -387,15 +387,15 @@ func (s *authServer) Start() {
|
|||||||
})
|
})
|
||||||
mux.HandleFunc("/", func(w http.ResponseWriter, req *http.Request) {
|
mux.HandleFunc("/", func(w http.ResponseWriter, req *http.Request) {
|
||||||
w.Header().Set("Content-Type", "text/html")
|
w.Header().Set("Content-Type", "text/html")
|
||||||
fs.Debug(nil, "Received request on auth server")
|
fs.Debugf(nil, "Received request on auth server")
|
||||||
code := req.FormValue("code")
|
code := req.FormValue("code")
|
||||||
if code != "" {
|
if code != "" {
|
||||||
state := req.FormValue("state")
|
state := req.FormValue("state")
|
||||||
if state != s.state {
|
if state != s.state {
|
||||||
fs.Debug(nil, "State did not match: want %q got %q", s.state, state)
|
fs.Debugf(nil, "State did not match: want %q got %q", s.state, state)
|
||||||
fmt.Fprintf(w, "<h1>Failure</h1>\n<p>Auth state doesn't match</p>")
|
fmt.Fprintf(w, "<h1>Failure</h1>\n<p>Auth state doesn't match</p>")
|
||||||
} else {
|
} else {
|
||||||
fs.Debug(nil, "Successfully got code")
|
fs.Debugf(nil, "Successfully got code")
|
||||||
if s.code != nil {
|
if s.code != nil {
|
||||||
fmt.Fprintf(w, "<h1>Success</h1>\n<p>Go back to rclone to continue</p>")
|
fmt.Fprintf(w, "<h1>Success</h1>\n<p>Go back to rclone to continue</p>")
|
||||||
s.code <- code
|
s.code <- code
|
||||||
@ -405,7 +405,7 @@ func (s *authServer) Start() {
|
|||||||
}
|
}
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
fs.Debug(nil, "No code found on request")
|
fs.Debugf(nil, "No code found on request")
|
||||||
w.WriteHeader(500)
|
w.WriteHeader(500)
|
||||||
fmt.Fprintf(w, "<h1>Failed!</h1>\nNo code found returned by remote server.")
|
fmt.Fprintf(w, "<h1>Failed!</h1>\nNo code found returned by remote server.")
|
||||||
|
|
||||||
@ -417,11 +417,11 @@ func (s *authServer) Start() {
|
|||||||
log.Fatalf("Failed to start auth webserver: %v", err)
|
log.Fatalf("Failed to start auth webserver: %v", err)
|
||||||
}
|
}
|
||||||
err = server.Serve(s.listener)
|
err = server.Serve(s.listener)
|
||||||
fs.Debug(nil, "Closed auth server with error: %v", err)
|
fs.Debugf(nil, "Closed auth server with error: %v", err)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (s *authServer) Stop() {
|
func (s *authServer) Stop() {
|
||||||
fs.Debug(nil, "Closing auth server")
|
fs.Debugf(nil, "Closing auth server")
|
||||||
if s.code != nil {
|
if s.code != nil {
|
||||||
close(s.code)
|
close(s.code)
|
||||||
s.code = nil
|
s.code = nil
|
||||||
|
@ -39,16 +39,16 @@ func (r *Renew) renewOnExpiry() {
|
|||||||
<-expiry
|
<-expiry
|
||||||
uploads := atomic.LoadInt32(&r.uploads)
|
uploads := atomic.LoadInt32(&r.uploads)
|
||||||
if uploads != 0 {
|
if uploads != 0 {
|
||||||
fs.Debug(r.name, "Token expired - %d uploads in progress - refreshing", uploads)
|
fs.Debugf(r.name, "Token expired - %d uploads in progress - refreshing", uploads)
|
||||||
// Do a transaction
|
// Do a transaction
|
||||||
err := r.run()
|
err := r.run()
|
||||||
if err == nil {
|
if err == nil {
|
||||||
fs.Debug(r.name, "Token refresh successful")
|
fs.Debugf(r.name, "Token refresh successful")
|
||||||
} else {
|
} else {
|
||||||
fs.ErrorLog(r.name, "Token refresh failed: %v", err)
|
fs.Errorf(r.name, "Token refresh failed: %v", err)
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
fs.Debug(r.name, "Token expired but no uploads in progress - doing nothing")
|
fs.Debugf(r.name, "Token expired but no uploads in progress - doing nothing")
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -169,7 +169,7 @@ func errorHandler(resp *http.Response) error {
|
|||||||
errResponse := new(api.Error)
|
errResponse := new(api.Error)
|
||||||
err := rest.DecodeJSON(resp, &errResponse)
|
err := rest.DecodeJSON(resp, &errResponse)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
fs.Debug(nil, "Couldn't decode error response: %v", err)
|
fs.Debugf(nil, "Couldn't decode error response: %v", err)
|
||||||
}
|
}
|
||||||
if errResponse.ErrorInfo.Code == "" {
|
if errResponse.ErrorInfo.Code == "" {
|
||||||
errResponse.ErrorInfo.Code = resp.Status
|
errResponse.ErrorInfo.Code = resp.Status
|
||||||
@ -272,7 +272,7 @@ func (f *Fs) NewObject(remote string) (fs.Object, error) {
|
|||||||
|
|
||||||
// FindLeaf finds a directory of name leaf in the folder with ID pathID
|
// FindLeaf finds a directory of name leaf in the folder with ID pathID
|
||||||
func (f *Fs) FindLeaf(pathID, leaf string) (pathIDOut string, found bool, err error) {
|
func (f *Fs) FindLeaf(pathID, leaf string) (pathIDOut string, found bool, err error) {
|
||||||
// fs.Debug(f, "FindLeaf(%q, %q)", pathID, leaf)
|
// fs.Debugf(f, "FindLeaf(%q, %q)", pathID, leaf)
|
||||||
parent, ok := f.dirCache.GetInv(pathID)
|
parent, ok := f.dirCache.GetInv(pathID)
|
||||||
if !ok {
|
if !ok {
|
||||||
return "", false, errors.New("couldn't find parent ID")
|
return "", false, errors.New("couldn't find parent ID")
|
||||||
@ -299,7 +299,7 @@ func (f *Fs) FindLeaf(pathID, leaf string) (pathIDOut string, found bool, err er
|
|||||||
|
|
||||||
// CreateDir makes a directory with pathID as parent and name leaf
|
// CreateDir makes a directory with pathID as parent and name leaf
|
||||||
func (f *Fs) CreateDir(pathID, leaf string) (newID string, err error) {
|
func (f *Fs) CreateDir(pathID, leaf string) (newID string, err error) {
|
||||||
// fs.Debug(f, "CreateDir(%q, %q)\n", pathID, leaf)
|
// fs.Debugf(f, "CreateDir(%q, %q)\n", pathID, leaf)
|
||||||
var resp *http.Response
|
var resp *http.Response
|
||||||
var info *api.Item
|
var info *api.Item
|
||||||
opts := rest.Opts{
|
opts := rest.Opts{
|
||||||
@ -386,7 +386,7 @@ OUTER:
|
|||||||
|
|
||||||
// ListDir reads the directory specified by the job into out, returning any more jobs
|
// ListDir reads the directory specified by the job into out, returning any more jobs
|
||||||
func (f *Fs) ListDir(out fs.ListOpts, job dircache.ListDirJob) (jobs []dircache.ListDirJob, err error) {
|
func (f *Fs) ListDir(out fs.ListOpts, job dircache.ListDirJob) (jobs []dircache.ListDirJob, err error) {
|
||||||
fs.Debug(f, "Reading %q", job.Path)
|
fs.Debugf(f, "Reading %q", job.Path)
|
||||||
_, err = f.listAll(job.DirID, false, false, func(info *api.Item) bool {
|
_, err = f.listAll(job.DirID, false, false, func(info *api.Item) bool {
|
||||||
remote := job.Path + info.Name
|
remote := job.Path + info.Name
|
||||||
if info.Folder != nil {
|
if info.Folder != nil {
|
||||||
@ -419,7 +419,7 @@ func (f *Fs) ListDir(out fs.ListOpts, job dircache.ListDirJob) (jobs []dircache.
|
|||||||
}
|
}
|
||||||
return false
|
return false
|
||||||
})
|
})
|
||||||
fs.Debug(f, "Finished reading %q", job.Path)
|
fs.Debugf(f, "Finished reading %q", job.Path)
|
||||||
return jobs, err
|
return jobs, err
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -592,7 +592,7 @@ func (f *Fs) waitForJob(location string, o *Object) error {
|
|||||||
func (f *Fs) Copy(src fs.Object, remote string) (fs.Object, error) {
|
func (f *Fs) Copy(src fs.Object, remote string) (fs.Object, error) {
|
||||||
srcObj, ok := src.(*Object)
|
srcObj, ok := src.(*Object)
|
||||||
if !ok {
|
if !ok {
|
||||||
fs.Debug(src, "Can't copy - not same remote type")
|
fs.Debugf(src, "Can't copy - not same remote type")
|
||||||
return nil, fs.ErrorCantCopy
|
return nil, fs.ErrorCantCopy
|
||||||
}
|
}
|
||||||
err := srcObj.readMetaData()
|
err := srcObj.readMetaData()
|
||||||
@ -700,7 +700,7 @@ func (o *Object) Hash(t fs.HashType) (string, error) {
|
|||||||
func (o *Object) Size() int64 {
|
func (o *Object) Size() int64 {
|
||||||
err := o.readMetaData()
|
err := o.readMetaData()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
fs.Log(o, "Failed to read metadata: %v", err)
|
fs.Logf(o, "Failed to read metadata: %v", err)
|
||||||
return 0
|
return 0
|
||||||
}
|
}
|
||||||
return o.size
|
return o.size
|
||||||
@ -764,7 +764,7 @@ func (o *Object) readMetaData() (err error) {
|
|||||||
func (o *Object) ModTime() time.Time {
|
func (o *Object) ModTime() time.Time {
|
||||||
err := o.readMetaData()
|
err := o.readMetaData()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
fs.Log(o, "Failed to read metadata: %v", err)
|
fs.Logf(o, "Failed to read metadata: %v", err)
|
||||||
return time.Now()
|
return time.Now()
|
||||||
}
|
}
|
||||||
return o.modTime
|
return o.modTime
|
||||||
@ -883,7 +883,7 @@ func (o *Object) uploadMultipart(in io.Reader, size int64) (err error) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Create upload session
|
// Create upload session
|
||||||
fs.Debug(o, "Starting multipart upload")
|
fs.Debugf(o, "Starting multipart upload")
|
||||||
session, err := o.createUploadSession()
|
session, err := o.createUploadSession()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
@ -893,10 +893,10 @@ func (o *Object) uploadMultipart(in io.Reader, size int64) (err error) {
|
|||||||
// Cancel the session if something went wrong
|
// Cancel the session if something went wrong
|
||||||
defer func() {
|
defer func() {
|
||||||
if err != nil {
|
if err != nil {
|
||||||
fs.Debug(o, "Cancelling multipart upload")
|
fs.Debugf(o, "Cancelling multipart upload")
|
||||||
cancelErr := o.cancelUploadSession(uploadURL)
|
cancelErr := o.cancelUploadSession(uploadURL)
|
||||||
if cancelErr != nil {
|
if cancelErr != nil {
|
||||||
fs.Log(o, "Failed to cancel multipart upload: %v", err)
|
fs.Logf(o, "Failed to cancel multipart upload: %v", err)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}()
|
}()
|
||||||
@ -915,7 +915,7 @@ func (o *Object) uploadMultipart(in io.Reader, size int64) (err error) {
|
|||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
fs.Debug(o, "Uploading segment %d/%d size %d", position, size, n)
|
fs.Debugf(o, "Uploading segment %d/%d size %d", position, size, n)
|
||||||
err = o.uploadFragment(uploadURL, position, size, buf)
|
err = o.uploadFragment(uploadURL, position, size, buf)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
|
@ -208,7 +208,7 @@ func (p *Pacer) beginCall() {
|
|||||||
p.mu.Lock()
|
p.mu.Lock()
|
||||||
// Restart the timer
|
// Restart the timer
|
||||||
go func(t time.Duration) {
|
go func(t time.Duration) {
|
||||||
// fs.Debug(f, "New sleep for %v at %v", t, time.Now())
|
// fs.Debugf(f, "New sleep for %v at %v", t, time.Now())
|
||||||
time.Sleep(t)
|
time.Sleep(t)
|
||||||
p.pacer <- struct{}{}
|
p.pacer <- struct{}{}
|
||||||
}(p.sleepTime)
|
}(p.sleepTime)
|
||||||
@ -236,7 +236,7 @@ func (p *Pacer) defaultPacer(retry bool) {
|
|||||||
p.sleepTime = p.maxSleep
|
p.sleepTime = p.maxSleep
|
||||||
}
|
}
|
||||||
if p.sleepTime != oldSleepTime {
|
if p.sleepTime != oldSleepTime {
|
||||||
fs.Debug("pacer", "Rate limited, increasing sleep to %v", p.sleepTime)
|
fs.Debugf("pacer", "Rate limited, increasing sleep to %v", p.sleepTime)
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
p.sleepTime = (p.sleepTime<<p.decayConstant - p.sleepTime) >> p.decayConstant
|
p.sleepTime = (p.sleepTime<<p.decayConstant - p.sleepTime) >> p.decayConstant
|
||||||
@ -244,7 +244,7 @@ func (p *Pacer) defaultPacer(retry bool) {
|
|||||||
p.sleepTime = p.minSleep
|
p.sleepTime = p.minSleep
|
||||||
}
|
}
|
||||||
if p.sleepTime != oldSleepTime {
|
if p.sleepTime != oldSleepTime {
|
||||||
fs.Debug("pacer", "Reducing sleep to %v", p.sleepTime)
|
fs.Debugf("pacer", "Reducing sleep to %v", p.sleepTime)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -263,7 +263,7 @@ func (p *Pacer) acdPacer(retry bool) {
|
|||||||
if consecutiveRetries == 0 {
|
if consecutiveRetries == 0 {
|
||||||
if p.sleepTime != p.minSleep {
|
if p.sleepTime != p.minSleep {
|
||||||
p.sleepTime = p.minSleep
|
p.sleepTime = p.minSleep
|
||||||
fs.Debug("pacer", "Resetting sleep to minimum %v on success", p.sleepTime)
|
fs.Debugf("pacer", "Resetting sleep to minimum %v on success", p.sleepTime)
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
if consecutiveRetries > 9 {
|
if consecutiveRetries > 9 {
|
||||||
@ -277,7 +277,7 @@ func (p *Pacer) acdPacer(retry bool) {
|
|||||||
if p.sleepTime < p.minSleep {
|
if p.sleepTime < p.minSleep {
|
||||||
p.sleepTime = p.minSleep
|
p.sleepTime = p.minSleep
|
||||||
}
|
}
|
||||||
fs.Debug("pacer", "Rate limited, sleeping for %v (%d consecutive low level retries)", p.sleepTime, p.consecutiveRetries)
|
fs.Debugf("pacer", "Rate limited, sleeping for %v (%d consecutive low level retries)", p.sleepTime, p.consecutiveRetries)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -295,7 +295,7 @@ func (p *Pacer) drivePacer(retry bool) {
|
|||||||
if consecutiveRetries == 0 {
|
if consecutiveRetries == 0 {
|
||||||
if p.sleepTime != p.minSleep {
|
if p.sleepTime != p.minSleep {
|
||||||
p.sleepTime = p.minSleep
|
p.sleepTime = p.minSleep
|
||||||
fs.Debug("pacer", "Resetting sleep to minimum %v on success", p.sleepTime)
|
fs.Debugf("pacer", "Resetting sleep to minimum %v on success", p.sleepTime)
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
if consecutiveRetries > 5 {
|
if consecutiveRetries > 5 {
|
||||||
@ -304,7 +304,7 @@ func (p *Pacer) drivePacer(retry bool) {
|
|||||||
// consecutiveRetries starts at 1 so go from 1,2,3,4,5,5 => 1,2,4,8,16,16
|
// consecutiveRetries starts at 1 so go from 1,2,3,4,5,5 => 1,2,4,8,16,16
|
||||||
// maxSleep is 2**(consecutiveRetries-1) seconds + random milliseconds
|
// maxSleep is 2**(consecutiveRetries-1) seconds + random milliseconds
|
||||||
p.sleepTime = time.Second<<uint(consecutiveRetries-1) + time.Duration(rand.Int63n(int64(time.Second)))
|
p.sleepTime = time.Second<<uint(consecutiveRetries-1) + time.Duration(rand.Int63n(int64(time.Second)))
|
||||||
fs.Debug("pacer", "Rate limited, sleeping for %v (%d consecutive low level retries)", p.sleepTime, p.consecutiveRetries)
|
fs.Debugf("pacer", "Rate limited, sleeping for %v (%d consecutive low level retries)", p.sleepTime, p.consecutiveRetries)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -336,7 +336,7 @@ func (p *Pacer) call(fn Paced, retries int) (err error) {
|
|||||||
if !retry {
|
if !retry {
|
||||||
break
|
break
|
||||||
}
|
}
|
||||||
fs.Debug("pacer", "low level retry %d/%d (error %v)", i, retries, err)
|
fs.Debugf("pacer", "low level retry %d/%d (error %v)", i, retries, err)
|
||||||
}
|
}
|
||||||
if retry {
|
if retry {
|
||||||
err = fs.RetryError(err)
|
err = fs.RetryError(err)
|
||||||
|
28
s3/s3.go
28
s3/s3.go
@ -349,7 +349,7 @@ func s3Connection(name string) (*s3.S3, *session.Session, error) {
|
|||||||
ses := session.New()
|
ses := session.New()
|
||||||
c := s3.New(ses, awsConfig)
|
c := s3.New(ses, awsConfig)
|
||||||
if region == "other-v2-signature" {
|
if region == "other-v2-signature" {
|
||||||
fs.Debug(name, "Using v2 auth")
|
fs.Debugf(name, "Using v2 auth")
|
||||||
signer := func(req *request.Request) {
|
signer := func(req *request.Request) {
|
||||||
// Ignore AnonymousCredentials object
|
// Ignore AnonymousCredentials object
|
||||||
if req.Config.Credentials == credentials.AnonymousCredentials {
|
if req.Config.Credentials == credentials.AnonymousCredentials {
|
||||||
@ -426,7 +426,7 @@ func (f *Fs) newObjectWithInfo(remote string, info *s3.Object) (fs.Object, error
|
|||||||
if info != nil {
|
if info != nil {
|
||||||
// Set info but not meta
|
// Set info but not meta
|
||||||
if info.LastModified == nil {
|
if info.LastModified == nil {
|
||||||
fs.Log(o, "Failed to read last modified")
|
fs.Logf(o, "Failed to read last modified")
|
||||||
o.lastModified = time.Now()
|
o.lastModified = time.Now()
|
||||||
} else {
|
} else {
|
||||||
o.lastModified = *info.LastModified
|
o.lastModified = *info.LastModified
|
||||||
@ -488,12 +488,12 @@ func (f *Fs) list(dir string, level int, fn listFn) error {
|
|||||||
if level == 1 {
|
if level == 1 {
|
||||||
for _, commonPrefix := range resp.CommonPrefixes {
|
for _, commonPrefix := range resp.CommonPrefixes {
|
||||||
if commonPrefix.Prefix == nil {
|
if commonPrefix.Prefix == nil {
|
||||||
fs.Log(f, "Nil common prefix received")
|
fs.Logf(f, "Nil common prefix received")
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
remote := *commonPrefix.Prefix
|
remote := *commonPrefix.Prefix
|
||||||
if !strings.HasPrefix(remote, f.root) {
|
if !strings.HasPrefix(remote, f.root) {
|
||||||
fs.Log(f, "Odd name received %q", remote)
|
fs.Logf(f, "Odd name received %q", remote)
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
remote = remote[rootLength:]
|
remote = remote[rootLength:]
|
||||||
@ -509,7 +509,7 @@ func (f *Fs) list(dir string, level int, fn listFn) error {
|
|||||||
for _, object := range resp.Contents {
|
for _, object := range resp.Contents {
|
||||||
key := aws.StringValue(object.Key)
|
key := aws.StringValue(object.Key)
|
||||||
if !strings.HasPrefix(key, f.root) {
|
if !strings.HasPrefix(key, f.root) {
|
||||||
fs.Log(f, "Odd name received %q", key)
|
fs.Logf(f, "Odd name received %q", key)
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
remote := key[rootLength:]
|
remote := key[rootLength:]
|
||||||
@ -697,7 +697,7 @@ func (f *Fs) Precision() time.Duration {
|
|||||||
func (f *Fs) Copy(src fs.Object, remote string) (fs.Object, error) {
|
func (f *Fs) Copy(src fs.Object, remote string) (fs.Object, error) {
|
||||||
srcObj, ok := src.(*Object)
|
srcObj, ok := src.(*Object)
|
||||||
if !ok {
|
if !ok {
|
||||||
fs.Debug(src, "Can't copy - not same remote type")
|
fs.Debugf(src, "Can't copy - not same remote type")
|
||||||
return nil, fs.ErrorCantCopy
|
return nil, fs.ErrorCantCopy
|
||||||
}
|
}
|
||||||
srcFs := srcObj.fs
|
srcFs := srcObj.fs
|
||||||
@ -751,7 +751,7 @@ func (o *Object) Hash(t fs.HashType) (string, error) {
|
|||||||
etag := strings.Trim(strings.ToLower(o.etag), `"`)
|
etag := strings.Trim(strings.ToLower(o.etag), `"`)
|
||||||
// Check the etag is a valid md5sum
|
// Check the etag is a valid md5sum
|
||||||
if !matchMd5.MatchString(etag) {
|
if !matchMd5.MatchString(etag) {
|
||||||
// fs.Debug(o, "Invalid md5sum (probably multipart uploaded) - ignoring: %q", etag)
|
// fs.Debugf(o, "Invalid md5sum (probably multipart uploaded) - ignoring: %q", etag)
|
||||||
return "", nil
|
return "", nil
|
||||||
}
|
}
|
||||||
return etag, nil
|
return etag, nil
|
||||||
@ -793,7 +793,7 @@ func (o *Object) readMetaData() (err error) {
|
|||||||
o.bytes = size
|
o.bytes = size
|
||||||
o.meta = resp.Metadata
|
o.meta = resp.Metadata
|
||||||
if resp.LastModified == nil {
|
if resp.LastModified == nil {
|
||||||
fs.Log(o, "Failed to read last modified from HEAD: %v", err)
|
fs.Logf(o, "Failed to read last modified from HEAD: %v", err)
|
||||||
o.lastModified = time.Now()
|
o.lastModified = time.Now()
|
||||||
} else {
|
} else {
|
||||||
o.lastModified = *resp.LastModified
|
o.lastModified = *resp.LastModified
|
||||||
@ -809,18 +809,18 @@ func (o *Object) readMetaData() (err error) {
|
|||||||
func (o *Object) ModTime() time.Time {
|
func (o *Object) ModTime() time.Time {
|
||||||
err := o.readMetaData()
|
err := o.readMetaData()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
fs.Log(o, "Failed to read metadata: %v", err)
|
fs.Logf(o, "Failed to read metadata: %v", err)
|
||||||
return time.Now()
|
return time.Now()
|
||||||
}
|
}
|
||||||
// read mtime out of metadata if available
|
// read mtime out of metadata if available
|
||||||
d, ok := o.meta[metaMtime]
|
d, ok := o.meta[metaMtime]
|
||||||
if !ok || d == nil {
|
if !ok || d == nil {
|
||||||
// fs.Debug(o, "No metadata")
|
// fs.Debugf(o, "No metadata")
|
||||||
return o.lastModified
|
return o.lastModified
|
||||||
}
|
}
|
||||||
modTime, err := swift.FloatStringToTime(*d)
|
modTime, err := swift.FloatStringToTime(*d)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
fs.Log(o, "Failed to read mtime from object: %v", err)
|
fs.Logf(o, "Failed to read mtime from object: %v", err)
|
||||||
return o.lastModified
|
return o.lastModified
|
||||||
}
|
}
|
||||||
return modTime
|
return modTime
|
||||||
@ -835,7 +835,7 @@ func (o *Object) SetModTime(modTime time.Time) error {
|
|||||||
o.meta[metaMtime] = aws.String(swift.TimeToFloatString(modTime))
|
o.meta[metaMtime] = aws.String(swift.TimeToFloatString(modTime))
|
||||||
|
|
||||||
if o.bytes >= maxSizeForCopy {
|
if o.bytes >= maxSizeForCopy {
|
||||||
fs.Debug(o, "SetModTime is unsupported for objects bigger than %v bytes", fs.SizeSuffix(maxSizeForCopy))
|
fs.Debugf(o, "SetModTime is unsupported for objects bigger than %v bytes", fs.SizeSuffix(maxSizeForCopy))
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -878,7 +878,7 @@ func (o *Object) Open(options ...fs.OpenOption) (in io.ReadCloser, err error) {
|
|||||||
req.Range = &value
|
req.Range = &value
|
||||||
default:
|
default:
|
||||||
if option.Mandatory() {
|
if option.Mandatory() {
|
||||||
fs.Log(o, "Unsupported mandatory option: %v", option)
|
fs.Logf(o, "Unsupported mandatory option: %v", option)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -957,7 +957,7 @@ func (o *Object) Remove() error {
|
|||||||
func (o *Object) MimeType() string {
|
func (o *Object) MimeType() string {
|
||||||
err := o.readMetaData()
|
err := o.readMetaData()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
fs.Log(o, "Failed to read metadata: %v", err)
|
fs.Logf(o, "Failed to read metadata: %v", err)
|
||||||
return ""
|
return ""
|
||||||
}
|
}
|
||||||
return o.mimeType
|
return o.mimeType
|
||||||
|
16
sftp/sftp.go
16
sftp/sftp.go
@ -164,7 +164,7 @@ func NewFs(name, root string) (fs.Fs, error) {
|
|||||||
go func() {
|
go func() {
|
||||||
// FIXME re-open the connection here...
|
// FIXME re-open the connection here...
|
||||||
err := f.sshClient.Conn.Wait()
|
err := f.sshClient.Conn.Wait()
|
||||||
fs.ErrorLog(f, "SSH connection closed: %v", err)
|
fs.Errorf(f, "SSH connection closed: %v", err)
|
||||||
}()
|
}()
|
||||||
return f, nil
|
return f, nil
|
||||||
}
|
}
|
||||||
@ -241,7 +241,7 @@ func (f *Fs) list(out fs.ListOpts, dir string, level int, wg *sync.WaitGroup, to
|
|||||||
infos, err := f.sftpClient.ReadDir(sftpDir)
|
infos, err := f.sftpClient.ReadDir(sftpDir)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
err = errors.Wrapf(err, "error listing %q", dir)
|
err = errors.Wrapf(err, "error listing %q", dir)
|
||||||
fs.ErrorLog(f, "Listing failed: %v", err)
|
fs.Errorf(f, "Listing failed: %v", err)
|
||||||
out.SetError(err)
|
out.SetError(err)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
@ -363,7 +363,7 @@ func (f *Fs) Rmdir(dir string) error {
|
|||||||
func (f *Fs) Move(src fs.Object, remote string) (fs.Object, error) {
|
func (f *Fs) Move(src fs.Object, remote string) (fs.Object, error) {
|
||||||
srcObj, ok := src.(*Object)
|
srcObj, ok := src.(*Object)
|
||||||
if !ok {
|
if !ok {
|
||||||
fs.Debug(src, "Can't move - not same remote type")
|
fs.Debugf(src, "Can't move - not same remote type")
|
||||||
return nil, fs.ErrorCantMove
|
return nil, fs.ErrorCantMove
|
||||||
}
|
}
|
||||||
err := f.mkParentDir(remote)
|
err := f.mkParentDir(remote)
|
||||||
@ -395,7 +395,7 @@ func (f *Fs) Move(src fs.Object, remote string) (fs.Object, error) {
|
|||||||
func (f *Fs) DirMove(src fs.Fs) error {
|
func (f *Fs) DirMove(src fs.Fs) error {
|
||||||
srcFs, ok := src.(*Fs)
|
srcFs, ok := src.(*Fs)
|
||||||
if !ok {
|
if !ok {
|
||||||
fs.Debug(srcFs, "Can't move directory - not same remote type")
|
fs.Debugf(srcFs, "Can't move directory - not same remote type")
|
||||||
return fs.ErrorCantDirMove
|
return fs.ErrorCantDirMove
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -410,7 +410,7 @@ func (f *Fs) DirMove(src fs.Fs) error {
|
|||||||
|
|
||||||
// Refuse to move to or from the root
|
// Refuse to move to or from the root
|
||||||
if f.root == "" || srcFs.root == "" {
|
if f.root == "" || srcFs.root == "" {
|
||||||
fs.Debug(src, "DirMove error: Can't move root")
|
fs.Debugf(src, "DirMove error: Can't move root")
|
||||||
return errors.New("can't move root directory")
|
return errors.New("can't move root directory")
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -540,7 +540,7 @@ func (o *Object) Open(options ...fs.OpenOption) (in io.ReadCloser, err error) {
|
|||||||
offset = x.Offset
|
offset = x.Offset
|
||||||
default:
|
default:
|
||||||
if option.Mandatory() {
|
if option.Mandatory() {
|
||||||
fs.Log(o, "Unsupported mandatory option: %v", option)
|
fs.Logf(o, "Unsupported mandatory option: %v", option)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -571,9 +571,9 @@ func (o *Object) Update(in io.Reader, src fs.ObjectInfo) error {
|
|||||||
remove := func() {
|
remove := func() {
|
||||||
removeErr := o.fs.sftpClient.Remove(o.path())
|
removeErr := o.fs.sftpClient.Remove(o.path())
|
||||||
if removeErr != nil {
|
if removeErr != nil {
|
||||||
fs.Debug(src, "Failed to remove: %v", removeErr)
|
fs.Debugf(src, "Failed to remove: %v", removeErr)
|
||||||
} else {
|
} else {
|
||||||
fs.Debug(src, "Removed after failed upload: %v", err)
|
fs.Debugf(src, "Removed after failed upload: %v", err)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
_, err = file.ReadFrom(in)
|
_, err = file.ReadFrom(in)
|
||||||
|
@ -29,7 +29,7 @@ func (l *stringLock) Lock(ID string) {
|
|||||||
}
|
}
|
||||||
// Wait for the channel to be closed
|
// Wait for the channel to be closed
|
||||||
l.mu.Unlock()
|
l.mu.Unlock()
|
||||||
fs.Log(nil, "Waiting for stringLock on %q", ID)
|
fs.Logf(nil, "Waiting for stringLock on %q", ID)
|
||||||
<-ch
|
<-ch
|
||||||
l.mu.Lock()
|
l.mu.Lock()
|
||||||
}
|
}
|
||||||
|
@ -301,7 +301,7 @@ func (f *Fs) listContainerRoot(container, root string, dir string, level int, fn
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
if !strings.HasPrefix(object.Name, root) {
|
if !strings.HasPrefix(object.Name, root) {
|
||||||
fs.Log(f, "Odd name received %q", object.Name)
|
fs.Logf(f, "Odd name received %q", object.Name)
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
remote := object.Name[rootLength:]
|
remote := object.Name[rootLength:]
|
||||||
@ -484,7 +484,7 @@ func (f *Fs) Purge() error {
|
|||||||
func (f *Fs) Copy(src fs.Object, remote string) (fs.Object, error) {
|
func (f *Fs) Copy(src fs.Object, remote string) (fs.Object, error) {
|
||||||
srcObj, ok := src.(*Object)
|
srcObj, ok := src.(*Object)
|
||||||
if !ok {
|
if !ok {
|
||||||
fs.Debug(src, "Can't copy - not same remote type")
|
fs.Debugf(src, "Can't copy - not same remote type")
|
||||||
return nil, fs.ErrorCantCopy
|
return nil, fs.ErrorCantCopy
|
||||||
}
|
}
|
||||||
srcFs := srcObj.fs
|
srcFs := srcObj.fs
|
||||||
@ -534,7 +534,7 @@ func (o *Object) Hash(t fs.HashType) (string, error) {
|
|||||||
return "", err
|
return "", err
|
||||||
}
|
}
|
||||||
if isDynamicLargeObject || isStaticLargeObject {
|
if isDynamicLargeObject || isStaticLargeObject {
|
||||||
fs.Debug(o, "Returning empty Md5sum for swift large object")
|
fs.Debugf(o, "Returning empty Md5sum for swift large object")
|
||||||
return "", nil
|
return "", nil
|
||||||
}
|
}
|
||||||
return strings.ToLower(o.info.Hash), nil
|
return strings.ToLower(o.info.Hash), nil
|
||||||
@ -598,12 +598,12 @@ func (o *Object) readMetaData() (err error) {
|
|||||||
func (o *Object) ModTime() time.Time {
|
func (o *Object) ModTime() time.Time {
|
||||||
err := o.readMetaData()
|
err := o.readMetaData()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
fs.Debug(o, "Failed to read metadata: %s", err)
|
fs.Debugf(o, "Failed to read metadata: %s", err)
|
||||||
return o.info.LastModified
|
return o.info.LastModified
|
||||||
}
|
}
|
||||||
modTime, err := o.headers.ObjectMetadata().GetModTime()
|
modTime, err := o.headers.ObjectMetadata().GetModTime()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
// fs.Log(o, "Failed to read mtime from object: %v", err)
|
// fs.Logf(o, "Failed to read mtime from object: %v", err)
|
||||||
return o.info.LastModified
|
return o.info.LastModified
|
||||||
}
|
}
|
||||||
return modTime
|
return modTime
|
||||||
@ -664,11 +664,11 @@ func (o *Object) removeSegments(except string) error {
|
|||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
if except != "" && strings.HasPrefix(remote, except) {
|
if except != "" && strings.HasPrefix(remote, except) {
|
||||||
// fs.Debug(o, "Ignoring current segment file %q in container %q", segmentsRoot+remote, o.fs.segmentsContainer)
|
// fs.Debugf(o, "Ignoring current segment file %q in container %q", segmentsRoot+remote, o.fs.segmentsContainer)
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
segmentPath := segmentsRoot + remote
|
segmentPath := segmentsRoot + remote
|
||||||
fs.Debug(o, "Removing segment file %q in container %q", segmentPath, o.fs.segmentsContainer)
|
fs.Debugf(o, "Removing segment file %q in container %q", segmentPath, o.fs.segmentsContainer)
|
||||||
return o.fs.c.ObjectDelete(o.fs.segmentsContainer, segmentPath)
|
return o.fs.c.ObjectDelete(o.fs.segmentsContainer, segmentPath)
|
||||||
})
|
})
|
||||||
if err != nil {
|
if err != nil {
|
||||||
@ -677,7 +677,7 @@ func (o *Object) removeSegments(except string) error {
|
|||||||
// remove the segments container if empty, ignore errors
|
// remove the segments container if empty, ignore errors
|
||||||
err = o.fs.c.ContainerDelete(o.fs.segmentsContainer)
|
err = o.fs.c.ContainerDelete(o.fs.segmentsContainer)
|
||||||
if err == nil {
|
if err == nil {
|
||||||
fs.Debug(o, "Removed empty container %q", o.fs.segmentsContainer)
|
fs.Debugf(o, "Removed empty container %q", o.fs.segmentsContainer)
|
||||||
}
|
}
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
@ -717,7 +717,7 @@ func (o *Object) updateChunks(in io.Reader, headers swift.Headers, size int64, c
|
|||||||
headers["Content-Length"] = strconv.FormatInt(n, 10) // set Content-Length as we know it
|
headers["Content-Length"] = strconv.FormatInt(n, 10) // set Content-Length as we know it
|
||||||
segmentReader := io.LimitReader(in, n)
|
segmentReader := io.LimitReader(in, n)
|
||||||
segmentPath := fmt.Sprintf("%s/%08d", segmentsPath, i)
|
segmentPath := fmt.Sprintf("%s/%08d", segmentsPath, i)
|
||||||
fs.Debug(o, "Uploading segment file %q into %q", segmentPath, o.fs.segmentsContainer)
|
fs.Debugf(o, "Uploading segment file %q into %q", segmentPath, o.fs.segmentsContainer)
|
||||||
_, err := o.fs.c.ObjectPut(o.fs.segmentsContainer, segmentPath, segmentReader, true, "", "", headers)
|
_, err := o.fs.c.ObjectPut(o.fs.segmentsContainer, segmentPath, segmentReader, true, "", "", headers)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return "", err
|
return "", err
|
||||||
@ -770,7 +770,7 @@ func (o *Object) Update(in io.Reader, src fs.ObjectInfo) error {
|
|||||||
if isDynamicLargeObject {
|
if isDynamicLargeObject {
|
||||||
err = o.removeSegments(uniquePrefix)
|
err = o.removeSegments(uniquePrefix)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
fs.Log(o, "Failed to remove old segments - carrying on with upload: %v", err)
|
fs.Logf(o, "Failed to remove old segments - carrying on with upload: %v", err)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -193,7 +193,7 @@ func (f *Fs) listDir(fn listFn) (err error) {
|
|||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
default:
|
default:
|
||||||
fs.Debug(f, "Unknown resource type %q", element.ResourceType)
|
fs.Debugf(f, "Unknown resource type %q", element.ResourceType)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -347,7 +347,7 @@ func (o *Object) setMetaData(info *yandex.ResourceInfoResponse) {
|
|||||||
}
|
}
|
||||||
t, err := time.Parse(time.RFC3339Nano, modTimeString)
|
t, err := time.Parse(time.RFC3339Nano, modTimeString)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
fs.Log("Failed to parse modtime from %q: %v", modTimeString, err)
|
fs.Logf("Failed to parse modtime from %q: %v", modTimeString, err)
|
||||||
} else {
|
} else {
|
||||||
o.modTime = t
|
o.modTime = t
|
||||||
}
|
}
|
||||||
@ -494,7 +494,7 @@ func (o *Object) Size() int64 {
|
|||||||
func (o *Object) ModTime() time.Time {
|
func (o *Object) ModTime() time.Time {
|
||||||
err := o.readMetaData()
|
err := o.readMetaData()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
fs.Log(o, "Failed to read metadata: %v", err)
|
fs.Logf(o, "Failed to read metadata: %v", err)
|
||||||
return time.Now()
|
return time.Now()
|
||||||
}
|
}
|
||||||
return o.modTime
|
return o.modTime
|
||||||
@ -614,7 +614,7 @@ func mkDirFullPath(client *yandex.Client, path string) error {
|
|||||||
func (o *Object) MimeType() string {
|
func (o *Object) MimeType() string {
|
||||||
err := o.readMetaData()
|
err := o.readMetaData()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
fs.Log(o, "Failed to read metadata: %v", err)
|
fs.Logf(o, "Failed to read metadata: %v", err)
|
||||||
return ""
|
return ""
|
||||||
}
|
}
|
||||||
return o.mimeType
|
return o.mimeType
|
||||||
|
Loading…
Reference in New Issue
Block a user