2019-10-12 12:29:02 +00:00
package putio
import (
2021-03-11 14:44:01 +00:00
"context"
2019-10-12 12:29:02 +00:00
"fmt"
"net/http"
2022-03-22 08:40:49 +00:00
"strconv"
"time"
2019-10-12 12:29:02 +00:00
"github.com/putdotio/go-putio/putio"
"github.com/rclone/rclone/fs/fserrors"
2022-03-22 08:40:49 +00:00
"github.com/rclone/rclone/lib/pacer"
2019-10-12 12:29:02 +00:00
)
2022-04-03 20:14:46 +00:00
func checkStatusCode ( resp * http . Response , expected ... int ) error {
for _ , code := range expected {
if resp . StatusCode == code {
return nil
}
2019-10-12 12:29:02 +00:00
}
2022-04-03 20:14:46 +00:00
return & statusCodeError { response : resp }
2019-10-12 12:29:02 +00:00
}
type statusCodeError struct {
response * http . Response
}
func ( e * statusCodeError ) Error ( ) string {
return fmt . Sprintf ( "unexpected status code (%d) response while doing %s to %s" , e . response . StatusCode , e . response . Request . Method , e . response . Request . URL . String ( ) )
}
2022-03-22 08:40:49 +00:00
// This method is called from fserrors.ShouldRetry() to determine if an error should be retried.
// Some errors (e.g. 429 Too Many Requests) are handled before this step, so they are not included here.
2019-10-12 12:29:02 +00:00
func ( e * statusCodeError ) Temporary ( ) bool {
2022-03-22 08:40:49 +00:00
return e . response . StatusCode >= 500
2019-10-12 12:29:02 +00:00
}
// shouldRetry returns a boolean as to whether this err deserves to be
// retried. It returns the err as a convenience
2021-03-11 14:44:01 +00:00
func shouldRetry ( ctx context . Context , err error ) ( bool , error ) {
if fserrors . ContextError ( ctx , & err ) {
return false , err
}
2019-10-12 12:29:02 +00:00
if err == nil {
return false , nil
}
if perr , ok := err . ( * putio . ErrorResponse ) ; ok {
err = & statusCodeError { response : perr . Response }
}
2022-03-22 08:40:49 +00:00
if scerr , ok := err . ( * statusCodeError ) ; ok && scerr . response . StatusCode == 429 {
delay := defaultRateLimitSleep
header := scerr . response . Header . Get ( "x-ratelimit-reset" )
if header != "" {
if resetTime , cerr := strconv . ParseInt ( header , 10 , 64 ) ; cerr == nil {
delay = time . Until ( time . Unix ( resetTime + 1 , 0 ) )
}
}
return true , pacer . RetryAfterError ( scerr , delay )
}
2019-10-12 12:29:02 +00:00
if fserrors . ShouldRetry ( err ) {
return true , err
}
return false , err
}