mirror of
https://github.com/rclone/rclone.git
synced 2025-01-25 15:49:33 +01:00
b2: make error handling compliant
This commit is contained in:
parent
9290004bb8
commit
c8e2531c8b
56
b2/b2.go
56
b2/b2.go
@ -29,17 +29,18 @@ import (
|
|||||||
)
|
)
|
||||||
|
|
||||||
const (
|
const (
|
||||||
defaultEndpoint = "https://api.backblaze.com"
|
defaultEndpoint = "https://api.backblaze.com"
|
||||||
headerPrefix = "x-bz-info-" // lower case as that is what the server returns
|
headerPrefix = "x-bz-info-" // lower case as that is what the server returns
|
||||||
timeKey = "src_last_modified_millis"
|
timeKey = "src_last_modified_millis"
|
||||||
timeHeader = headerPrefix + timeKey
|
timeHeader = headerPrefix + timeKey
|
||||||
sha1Key = "large_file_sha1"
|
sha1Key = "large_file_sha1"
|
||||||
sha1Header = "X-Bz-Content-Sha1"
|
sha1Header = "X-Bz-Content-Sha1"
|
||||||
minSleep = 10 * time.Millisecond
|
testModeHeader = "X-Bz-Test-Mode"
|
||||||
maxSleep = 2 * time.Second
|
retryAfterHeader = "Retry-After"
|
||||||
decayConstant = 2 // bigger for slower decay, exponential
|
minSleep = 10 * time.Millisecond
|
||||||
maxParts = 10000
|
maxSleep = 2 * time.Second
|
||||||
testModeHeader = "X-Bz-Test-Mode"
|
decayConstant = 1 // bigger for slower decay, exponential
|
||||||
|
maxParts = 10000
|
||||||
)
|
)
|
||||||
|
|
||||||
// Globals
|
// Globals
|
||||||
@ -152,6 +153,27 @@ var retryErrorCodes = []int{
|
|||||||
// shouldRetryNoAuth returns a boolean as to whether this resp and err
|
// shouldRetryNoAuth 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) shouldRetryNoReauth(resp *http.Response, err error) (bool, error) {
|
func (f *Fs) shouldRetryNoReauth(resp *http.Response, err error) (bool, error) {
|
||||||
|
// For 429 or 503 errors look at the Retry-After: header and
|
||||||
|
// set the retry appropriately, starting with a minimum of 1
|
||||||
|
// second if it isn't set.
|
||||||
|
if resp != nil && (resp.StatusCode == 429 || resp.StatusCode == 503) {
|
||||||
|
var retryAfter = 1
|
||||||
|
retryAfterString := resp.Header.Get(retryAfterHeader)
|
||||||
|
if retryAfterString != "" {
|
||||||
|
var err error
|
||||||
|
retryAfter, err = strconv.Atoi(retryAfterString)
|
||||||
|
if err != nil {
|
||||||
|
fs.ErrorLog(f, "Malformed %s header %q: %v", retryAfterHeader, retryAfterString, err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
retryAfterDuration := time.Duration(retryAfter) * time.Second
|
||||||
|
if f.pacer.GetSleep() < retryAfterDuration {
|
||||||
|
fs.Debug(f, "Setting sleep to %v after error: %v", retryAfterDuration, err)
|
||||||
|
// We set 1/2 the value here because the pacer will double it immediately
|
||||||
|
f.pacer.SetSleep(retryAfterDuration / 2)
|
||||||
|
}
|
||||||
|
return true, err
|
||||||
|
}
|
||||||
return fs.ShouldRetry(err) || fs.ShouldRetryHTTP(resp, retryErrorCodes), err
|
return fs.ShouldRetry(err) || fs.ShouldRetryHTTP(resp, retryErrorCodes), err
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1194,15 +1216,13 @@ func (o *Object) Update(in io.Reader, src fs.ObjectInfo) (err error) {
|
|||||||
// Don't retry, return a retry error instead
|
// Don't retry, return a retry error instead
|
||||||
err = o.fs.pacer.CallNoRetry(func() (bool, error) {
|
err = o.fs.pacer.CallNoRetry(func() (bool, error) {
|
||||||
resp, err := o.fs.srv.CallJSON(&opts, nil, &response)
|
resp, err := o.fs.srv.CallJSON(&opts, nil, &response)
|
||||||
if resp != nil && resp.StatusCode == 401 {
|
retry, err := o.fs.shouldRetryNoReauth(resp, err)
|
||||||
fs.Debug(o, "Unauthorized: %v", err)
|
// On retryable error clear UploadURL
|
||||||
// Invalidate this Upload URL
|
if retry {
|
||||||
|
fs.Debug(o, "Clearing upload URL because of error: %v", err)
|
||||||
upload = nil
|
upload = nil
|
||||||
// Refetch upload URLs
|
|
||||||
o.fs.clearUploadURL()
|
|
||||||
return true, err
|
|
||||||
}
|
}
|
||||||
return o.fs.shouldRetryNoReauth(resp, err)
|
return retry, err
|
||||||
})
|
})
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
|
12
b2/upload.go
12
b2/upload.go
@ -178,14 +178,14 @@ func (up *largeUpload) transferChunk(part int64, body []byte) error {
|
|||||||
var response api.UploadPartResponse
|
var response api.UploadPartResponse
|
||||||
|
|
||||||
resp, err := up.f.srv.CallJSON(&opts, nil, &response)
|
resp, err := up.f.srv.CallJSON(&opts, nil, &response)
|
||||||
if resp != nil && resp.StatusCode == 401 {
|
retry, err := up.f.shouldRetryNoReauth(resp, err)
|
||||||
fs.Debug(up.o, "Unauthorized: %v", err)
|
// On retryable error clear PartUploadURL
|
||||||
// Refetch upload part URLs and ditch this current one
|
if retry {
|
||||||
up.clearUploadURL()
|
fs.Debug(up.o, "Clearing part upload URL because of error: %v", err)
|
||||||
return true, err
|
upload = nil
|
||||||
}
|
}
|
||||||
up.returnUploadURL(upload)
|
up.returnUploadURL(upload)
|
||||||
return up.f.shouldRetryNoReauth(resp, err)
|
return retry, err
|
||||||
})
|
})
|
||||||
if err != nil {
|
if err != nil {
|
||||||
fs.Debug(up.o, "Error sending chunk %d: %v", part, err)
|
fs.Debug(up.o, "Error sending chunk %d: %v", part, err)
|
||||||
|
Loading…
Reference in New Issue
Block a user