forked from TrueCloudLab/rclone
filefabric: make backoff exponential for error_background to fix errors
Before this change the backoff for the error_background error was 6 seconds. This means that if it wasn't resolved in 60 seconds (with the default 10 low level retries) then an error was reported. This error was being reported frequently in the integration tests, so is likely affecting real users too. This patch changes the backoff into an exponential backoff 1,2,4,8...1024 seconds to make sure we wait long enough for the background operation to complete. See #5734
This commit is contained in:
parent
6cda4c2c3c
commit
cc2f6f722c
1 changed files with 13 additions and 7 deletions
|
@ -222,13 +222,14 @@ var retryStatusCodes = []struct {
|
||||||
// delete in that folder. Please try again later or use
|
// delete in that folder. Please try again later or use
|
||||||
// another name. (error_background)
|
// another name. (error_background)
|
||||||
code: "error_background",
|
code: "error_background",
|
||||||
sleep: 6 * time.Second,
|
sleep: 1 * time.Second,
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
|
|
||||||
// shouldRetry returns a boolean as to whether this resp and err
|
// shouldRetry returns a boolean as to whether this resp and err
|
||||||
// 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(ctx context.Context, resp *http.Response, err error, status api.OKError) (bool, error) {
|
// try should be the number of the tries so far, counting up from 1
|
||||||
|
func (f *Fs) shouldRetry(ctx context.Context, resp *http.Response, err error, status api.OKError, try int) (bool, error) {
|
||||||
if fserrors.ContextError(ctx, &err) {
|
if fserrors.ContextError(ctx, &err) {
|
||||||
return false, err
|
return false, err
|
||||||
}
|
}
|
||||||
|
@ -244,9 +245,10 @@ func (f *Fs) shouldRetry(ctx context.Context, resp *http.Response, err error, st
|
||||||
for _, retryCode := range retryStatusCodes {
|
for _, retryCode := range retryStatusCodes {
|
||||||
if code == retryCode.code {
|
if code == retryCode.code {
|
||||||
if retryCode.sleep > 0 {
|
if retryCode.sleep > 0 {
|
||||||
// make this thread only sleep extra time
|
// make this thread only sleep exponentially increasing extra time
|
||||||
fs.Debugf(f, "Sleeping for %v to wait for %q error to clear", retryCode.sleep, retryCode.code)
|
sleepTime := retryCode.sleep << (try - 1)
|
||||||
time.Sleep(retryCode.sleep)
|
fs.Debugf(f, "Sleeping for %v to wait for %q error to clear", sleepTime, retryCode.code)
|
||||||
|
time.Sleep(sleepTime)
|
||||||
}
|
}
|
||||||
return true, err
|
return true, err
|
||||||
}
|
}
|
||||||
|
@ -400,11 +402,13 @@ func (f *Fs) rpc(ctx context.Context, function string, p params, result api.OKEr
|
||||||
ContentType: "application/x-www-form-urlencoded",
|
ContentType: "application/x-www-form-urlencoded",
|
||||||
Options: options,
|
Options: options,
|
||||||
}
|
}
|
||||||
|
try := 0
|
||||||
err = f.pacer.Call(func() (bool, error) {
|
err = f.pacer.Call(func() (bool, error) {
|
||||||
|
try++
|
||||||
// Refresh the body each retry
|
// Refresh the body each retry
|
||||||
opts.Body = strings.NewReader(data.Encode())
|
opts.Body = strings.NewReader(data.Encode())
|
||||||
resp, err = f.srv.CallJSON(ctx, &opts, nil, result)
|
resp, err = f.srv.CallJSON(ctx, &opts, nil, result)
|
||||||
return f.shouldRetry(ctx, resp, err, result)
|
return f.shouldRetry(ctx, resp, err, result, try)
|
||||||
})
|
})
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return resp, err
|
return resp, err
|
||||||
|
@ -1278,9 +1282,11 @@ func (o *Object) Update(ctx context.Context, in io.Reader, src fs.ObjectInfo, op
|
||||||
var contentLength = size
|
var contentLength = size
|
||||||
opts.ContentLength = &contentLength // NB CallJSON scribbles on this which is naughty
|
opts.ContentLength = &contentLength // NB CallJSON scribbles on this which is naughty
|
||||||
}
|
}
|
||||||
|
try := 0
|
||||||
err = o.fs.pacer.CallNoRetry(func() (bool, error) {
|
err = o.fs.pacer.CallNoRetry(func() (bool, error) {
|
||||||
|
try++
|
||||||
resp, err := o.fs.srv.CallJSON(ctx, &opts, nil, &uploader)
|
resp, err := o.fs.srv.CallJSON(ctx, &opts, nil, &uploader)
|
||||||
return o.fs.shouldRetry(ctx, resp, err, nil)
|
return o.fs.shouldRetry(ctx, resp, err, nil, try)
|
||||||
})
|
})
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return errors.Wrap(err, "failed to upload")
|
return errors.Wrap(err, "failed to upload")
|
||||||
|
|
Loading…
Reference in a new issue