vfs: Use operations.Copy instead of CopyFile for efficiency

This commit is contained in:
Nick Craig-Wood 2018-02-24 11:08:09 +00:00
parent 54deb01f00
commit 354f1ad722

View file

@ -8,6 +8,7 @@ import (
"sync" "sync"
"github.com/ncw/rclone/fs" "github.com/ncw/rclone/fs"
"github.com/ncw/rclone/fs/accounting"
"github.com/ncw/rclone/fs/log" "github.com/ncw/rclone/fs/log"
"github.com/ncw/rclone/fs/operations" "github.com/ncw/rclone/fs/operations"
"github.com/pkg/errors" "github.com/pkg/errors"
@ -81,6 +82,18 @@ func newRWFileHandle(d *Dir, f *File, remote string, flags int) (fh *RWFileHandl
return fh, nil return fh, nil
} }
// copy an object to or from the remote while accounting for it
func copyObj(f fs.Fs, dst fs.Object, remote string, src fs.Object) (newDst fs.Object, err error) {
if operations.NeedTransfer(dst, src) {
accounting.Stats.Transferring(src.Remote())
newDst, err = operations.Copy(f, dst, remote, src)
accounting.Stats.DoneTransferring(src.Remote(), err == nil)
} else {
newDst = dst
}
return newDst, err
}
// openPending opens the file if there is a pending open // openPending opens the file if there is a pending open
// //
// call with the lock held // call with the lock held
@ -99,22 +112,26 @@ func (fh *RWFileHandle) openPending(truncate bool) (err error) {
// try to open a exising cache file // try to open a exising cache file
fd, err = os.OpenFile(fh.osPath, cacheFileOpenFlags&^os.O_CREATE, 0600) fd, err = os.OpenFile(fh.osPath, cacheFileOpenFlags&^os.O_CREATE, 0600)
if os.IsNotExist(err) { if os.IsNotExist(err) {
// Fetch the file if it hasn't changed // cache file does not exist, so need to fetch it if we have an object to fetch
// FIXME retries // it from
err = operations.CopyFile(fh.d.vfs.cache.f, fh.d.vfs.f, fh.remote, fh.remote) if fh.o != nil {
if err != nil { _, err = copyObj(fh.d.vfs.cache.f, nil, fh.remote, fh.o)
// if the object wasn't found AND O_CREATE is set then... if err != nil {
cause := errors.Cause(err) cause := errors.Cause(err)
notFound := cause == fs.ErrorObjectNotFound || cause == fs.ErrorDirNotFound if cause != fs.ErrorObjectNotFound && cause != fs.ErrorDirNotFound {
if notFound { // return any non NotFound errors
// Remove cached item if there is one return errors.Wrap(err, "open RW handle failed to cache file")
rmErr := os.Remove(fh.osPath)
if rmErr != nil && !os.IsNotExist(rmErr) {
return errors.Wrap(rmErr, "open RW handle failed to delete stale cache file")
} }
// continue here with err=fs.Error{Object,Dir}NotFound
} }
if notFound && fh.flags&os.O_CREATE != 0 { }
// ...ignore error as we are about to create the file // if err == nil, then we have cached the file successfully, otherwise err is
// indicating some kind of non existent file/directory either
// os.IsNotExist(err) or fs.Error{Object,Dir}NotFound
if err != nil {
if fh.flags&os.O_CREATE != 0 {
// if the object wasn't found AND O_CREATE is set then
// ignore error as we are about to create the file
fh.file.setSize(0) fh.file.setSize(0)
fh.writeCalled = true fh.writeCalled = true
} else { } else {
@ -241,17 +258,16 @@ func (fh *RWFileHandle) close() (err error) {
if copy { if copy {
// Transfer the temp file to the remote // Transfer the temp file to the remote
// FIXME retries cacheObj, err := fh.d.vfs.cache.f.NewObject(fh.remote)
err = operations.CopyFile(fh.d.vfs.f, fh.d.vfs.cache.f, fh.remote, fh.remote)
if err != nil { if err != nil {
err = errors.Wrap(err, "failed to transfer file from cache to remote") err = errors.Wrap(err, "failed to find cache file")
fs.Errorf(fh.logPrefix(), "%v", err) fs.Errorf(fh.logPrefix(), "%v", err)
return err return err
} }
o, err := fh.d.vfs.f.NewObject(fh.remote) o, err := copyObj(fh.d.vfs.f, fh.o, fh.remote, cacheObj)
if err != nil { if err != nil {
err = errors.Wrap(err, "failed to find object after transfer to remote") err = errors.Wrap(err, "failed to transfer file from cache to remote")
fs.Errorf(fh.logPrefix(), "%v", err) fs.Errorf(fh.logPrefix(), "%v", err)
return err return err
} }