fuse: Add bandwidth accounting and buffering

This fixes rclone mount ignoring bwlimit and increases buffering which
should speed up transfers greatly.

Fixes #796
Fixes #690
This commit is contained in:
Nick Craig-Wood 2016-11-30 20:18:14 +00:00
parent 2cbdb95ce5
commit be4fd51289
4 changed files with 34 additions and 16 deletions

View file

@ -27,10 +27,11 @@ func newReadFileHandle(o fs.Object) (*ReadFileHandle, error) {
if err != nil { if err != nil {
return nil, err return nil, err
} }
return &ReadFileHandle{ fh := &ReadFileHandle{
r: r,
o: o, o: o,
}, nil r: fs.NewAccount(r, o), // account and buffer the transfer
}
return fh, nil
} }
// Check interface satisfied // Check interface satisfied
@ -63,7 +64,7 @@ func (fh *ReadFileHandle) seek(offset int64) error {
if err != nil { if err != nil {
fs.Debug(fh.o, "ReadFileHandle.Read seek close old failed: %v", err) fs.Debug(fh.o, "ReadFileHandle.Read seek close old failed: %v", err)
} }
fh.r = r fh.r = fs.NewAccount(r, fh.o) // account and buffer the transfer
} }
fh.offset = offset fh.offset = offset
return nil return nil

View file

@ -38,8 +38,9 @@ func newWriteFileHandle(d *Dir, f *File, src fs.ObjectInfo) (*WriteFileHandle, e
file: f, file: f,
} }
fh.pipeReader, fh.pipeWriter = io.Pipe() fh.pipeReader, fh.pipeWriter = io.Pipe()
r := fs.NewAccountSizeName(fh.pipeReader, 0, src.Remote()) // account and buffer the transfer
go func() { go func() {
o, err := d.f.Put(fh.pipeReader, src) o, err := d.f.Put(r, src)
fh.o = o fh.o = o
fh.result <- err fh.result <- err
}() }()

View file

@ -276,12 +276,28 @@ type Account struct {
wholeFileDisabled bool // disables the whole file when doing parts wholeFileDisabled bool // disables the whole file when doing parts
} }
// NewAccount makes a Account reader for an object // NewAccountSizeName makes a Account reader for an io.ReadCloser of
func NewAccount(in io.ReadCloser, obj Object) *Account { // the given size and name
//
// If the file is above a certain size it adds an Async reader
func NewAccountSizeName(in io.ReadCloser, size int64, name string) *Account {
// On big files add a buffer
if size > 10<<20 {
const memUsed = 16 * 1024 * 1024
const bufSize = 128 * 1024
const buffers = memUsed / bufSize
newIn, err := newAsyncReader(in, buffers, bufSize)
if err != nil {
ErrorLog(name, "Failed to make buffer: %v", err)
} else {
in = newIn
}
}
acc := &Account{ acc := &Account{
in: in, in: in,
size: obj.Size(), size: size,
name: obj.Remote(), name: name,
exit: make(chan struct{}), exit: make(chan struct{}),
avg: ewma.NewMovingAverage(), avg: ewma.NewMovingAverage(),
lpTime: time.Now(), lpTime: time.Now(),
@ -291,6 +307,11 @@ func NewAccount(in io.ReadCloser, obj Object) *Account {
return acc return acc
} }
// NewAccount makes a Account reader for an object
func NewAccount(in io.ReadCloser, obj Object) *Account {
return NewAccountSizeName(in, obj.Size(), obj.Remote())
}
// disableWholeFileAccounting turns off the whole file accounting // disableWholeFileAccounting turns off the whole file accounting
func (acc *Account) disableWholeFileAccounting() { func (acc *Account) disableWholeFileAccounting() {
acc.mu.Lock() acc.mu.Lock()

View file

@ -264,12 +264,7 @@ func Copy(f Fs, dst Object, remote string, src Object) (err error) {
if err != nil { if err != nil {
err = errors.Wrap(err, "failed to open source object") err = errors.Wrap(err, "failed to open source object")
} else { } else {
// On big files add a buffer in := NewAccount(in0, src) // account and buffer the transfer
if src.Size() > 10<<20 {
in0, _ = newAsyncReader(in0, 4, 4<<20)
}
in := NewAccount(in0, src) // account the transfer
wrappedSrc := &overrideRemoteObject{Object: src, remote: remote} wrappedSrc := &overrideRemoteObject{Object: src, remote: remote}
if doUpdate { if doUpdate {
@ -1131,7 +1126,7 @@ func Cat(f Fs, w io.Writer) error {
ErrorLog(o, "Failed to close: %v", err) ErrorLog(o, "Failed to close: %v", err)
} }
}() }()
inAccounted := NewAccount(in, o) // account the transfer inAccounted := NewAccount(in, o) // account and buffer the transfer
_, err = io.Copy(w, inAccounted) _, err = io.Copy(w, inAccounted)
if err != nil { if err != nil {
Stats.Error() Stats.Error()