b2: make error handling compliant

This commit is contained in:
Nick Craig-Wood 2016-07-01 16:23:23 +01:00
parent 9290004bb8
commit c8e2531c8b
2 changed files with 44 additions and 24 deletions

View file

@ -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

View file

@ -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)