package putio

import (
	"context"
	"fmt"
	"net/http"
	"strconv"
	"time"

	"github.com/putdotio/go-putio/putio"
	"github.com/rclone/rclone/fs/fserrors"
	"github.com/rclone/rclone/lib/pacer"
)

func checkStatusCode(resp *http.Response, expected ...int) error {
	for _, code := range expected {
		if resp.StatusCode == code {
			return nil
		}
	}
	return &statusCodeError{response: resp}
}

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())
}

// 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.
func (e *statusCodeError) Temporary() bool {
	return e.response.StatusCode >= 500
}

// shouldRetry returns a boolean as to whether this err deserves to be
// retried.  It returns the err as a convenience
func shouldRetry(ctx context.Context, err error) (bool, error) {
	if fserrors.ContextError(ctx, &err) {
		return false, err
	}
	if err == nil {
		return false, nil
	}
	if perr, ok := err.(*putio.ErrorResponse); ok {
		err = &statusCodeError{response: perr.Response}
	}
	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)
	}
	if fserrors.ShouldRetry(err) {
		return true, err
	}
	return false, err
}