2026-01-12 18:47:45 +08:00
|
|
|
|
package operation_setting
|
|
|
|
|
|
|
|
|
|
|
|
import (
|
|
|
|
|
|
"fmt"
|
|
|
|
|
|
"sort"
|
|
|
|
|
|
"strconv"
|
|
|
|
|
|
"strings"
|
2026-03-06 18:22:25 +08:00
|
|
|
|
|
|
|
|
|
|
"github.com/QuantumNous/new-api/types"
|
2026-01-12 18:47:45 +08:00
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
|
|
type StatusCodeRange struct {
|
|
|
|
|
|
Start int
|
|
|
|
|
|
End int
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
var AutomaticDisableStatusCodeRanges = []StatusCodeRange{{Start: 401, End: 401}}
|
|
|
|
|
|
|
2026-01-14 14:34:12 +08:00
|
|
|
|
// Default behavior matches legacy hardcoded retry rules in controller/relay.go shouldRetry:
|
|
|
|
|
|
// retry for 1xx, 3xx, 4xx(except 400/408), 5xx(except 504/524), and no retry for 2xx.
|
|
|
|
|
|
var AutomaticRetryStatusCodeRanges = []StatusCodeRange{
|
|
|
|
|
|
{Start: 100, End: 199},
|
|
|
|
|
|
{Start: 300, End: 399},
|
|
|
|
|
|
{Start: 401, End: 407},
|
|
|
|
|
|
{Start: 409, End: 499},
|
|
|
|
|
|
{Start: 500, End: 503},
|
|
|
|
|
|
{Start: 505, End: 523},
|
|
|
|
|
|
{Start: 525, End: 599},
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2026-02-22 20:03:46 +08:00
|
|
|
|
var alwaysSkipRetryStatusCodes = map[int]struct{}{
|
|
|
|
|
|
504: {},
|
|
|
|
|
|
524: {},
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2026-03-06 18:22:25 +08:00
|
|
|
|
var alwaysSkipRetryCodes = map[types.ErrorCode]struct{}{
|
|
|
|
|
|
types.ErrorCodeBadResponseBody: {},
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2026-01-12 18:47:45 +08:00
|
|
|
|
func AutomaticDisableStatusCodesToString() string {
|
2026-01-14 14:34:12 +08:00
|
|
|
|
return statusCodeRangesToString(AutomaticDisableStatusCodeRanges)
|
2026-01-12 18:47:45 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
func AutomaticDisableStatusCodesFromString(s string) error {
|
|
|
|
|
|
ranges, err := ParseHTTPStatusCodeRanges(s)
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
|
return err
|
|
|
|
|
|
}
|
|
|
|
|
|
AutomaticDisableStatusCodeRanges = ranges
|
|
|
|
|
|
return nil
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
func ShouldDisableByStatusCode(code int) bool {
|
2026-01-14 14:34:12 +08:00
|
|
|
|
return shouldMatchStatusCodeRanges(AutomaticDisableStatusCodeRanges, code)
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
func AutomaticRetryStatusCodesToString() string {
|
|
|
|
|
|
return statusCodeRangesToString(AutomaticRetryStatusCodeRanges)
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
func AutomaticRetryStatusCodesFromString(s string) error {
|
|
|
|
|
|
ranges, err := ParseHTTPStatusCodeRanges(s)
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
|
return err
|
|
|
|
|
|
}
|
|
|
|
|
|
AutomaticRetryStatusCodeRanges = ranges
|
|
|
|
|
|
return nil
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2026-02-22 20:03:46 +08:00
|
|
|
|
func IsAlwaysSkipRetryStatusCode(code int) bool {
|
|
|
|
|
|
_, exists := alwaysSkipRetryStatusCodes[code]
|
|
|
|
|
|
return exists
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2026-03-06 18:22:25 +08:00
|
|
|
|
func IsAlwaysSkipRetryCode(errorCode types.ErrorCode) bool {
|
|
|
|
|
|
_, exists := alwaysSkipRetryCodes[errorCode]
|
|
|
|
|
|
return exists
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2026-01-14 14:34:12 +08:00
|
|
|
|
func ShouldRetryByStatusCode(code int) bool {
|
2026-02-22 20:03:46 +08:00
|
|
|
|
if IsAlwaysSkipRetryStatusCode(code) {
|
|
|
|
|
|
return false
|
|
|
|
|
|
}
|
2026-01-14 14:34:12 +08:00
|
|
|
|
return shouldMatchStatusCodeRanges(AutomaticRetryStatusCodeRanges, code)
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
func statusCodeRangesToString(ranges []StatusCodeRange) string {
|
|
|
|
|
|
if len(ranges) == 0 {
|
|
|
|
|
|
return ""
|
|
|
|
|
|
}
|
|
|
|
|
|
parts := make([]string, 0, len(ranges))
|
|
|
|
|
|
for _, r := range ranges {
|
|
|
|
|
|
if r.Start == r.End {
|
|
|
|
|
|
parts = append(parts, strconv.Itoa(r.Start))
|
|
|
|
|
|
continue
|
|
|
|
|
|
}
|
|
|
|
|
|
parts = append(parts, fmt.Sprintf("%d-%d", r.Start, r.End))
|
|
|
|
|
|
}
|
|
|
|
|
|
return strings.Join(parts, ",")
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
func shouldMatchStatusCodeRanges(ranges []StatusCodeRange, code int) bool {
|
2026-01-12 18:47:45 +08:00
|
|
|
|
if code < 100 || code > 599 {
|
|
|
|
|
|
return false
|
|
|
|
|
|
}
|
2026-01-14 14:34:12 +08:00
|
|
|
|
for _, r := range ranges {
|
2026-01-12 18:47:45 +08:00
|
|
|
|
if code < r.Start {
|
|
|
|
|
|
return false
|
|
|
|
|
|
}
|
|
|
|
|
|
if code <= r.End {
|
|
|
|
|
|
return true
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
return false
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
func ParseHTTPStatusCodeRanges(input string) ([]StatusCodeRange, error) {
|
|
|
|
|
|
input = strings.TrimSpace(input)
|
|
|
|
|
|
if input == "" {
|
|
|
|
|
|
return nil, nil
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
input = strings.NewReplacer(",", ",").Replace(input)
|
|
|
|
|
|
segments := strings.Split(input, ",")
|
|
|
|
|
|
|
|
|
|
|
|
var ranges []StatusCodeRange
|
|
|
|
|
|
var invalid []string
|
|
|
|
|
|
|
|
|
|
|
|
for _, seg := range segments {
|
|
|
|
|
|
seg = strings.TrimSpace(seg)
|
|
|
|
|
|
if seg == "" {
|
|
|
|
|
|
continue
|
|
|
|
|
|
}
|
|
|
|
|
|
r, err := parseHTTPStatusCodeToken(seg)
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
|
invalid = append(invalid, seg)
|
|
|
|
|
|
continue
|
|
|
|
|
|
}
|
|
|
|
|
|
ranges = append(ranges, r)
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
if len(invalid) > 0 {
|
|
|
|
|
|
return nil, fmt.Errorf("invalid http status code rules: %s", strings.Join(invalid, ", "))
|
|
|
|
|
|
}
|
|
|
|
|
|
if len(ranges) == 0 {
|
|
|
|
|
|
return nil, nil
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
sort.Slice(ranges, func(i, j int) bool {
|
|
|
|
|
|
if ranges[i].Start == ranges[j].Start {
|
|
|
|
|
|
return ranges[i].End < ranges[j].End
|
|
|
|
|
|
}
|
|
|
|
|
|
return ranges[i].Start < ranges[j].Start
|
|
|
|
|
|
})
|
|
|
|
|
|
|
|
|
|
|
|
merged := []StatusCodeRange{ranges[0]}
|
|
|
|
|
|
for _, r := range ranges[1:] {
|
|
|
|
|
|
last := &merged[len(merged)-1]
|
|
|
|
|
|
if r.Start <= last.End+1 {
|
|
|
|
|
|
if r.End > last.End {
|
|
|
|
|
|
last.End = r.End
|
|
|
|
|
|
}
|
|
|
|
|
|
continue
|
|
|
|
|
|
}
|
|
|
|
|
|
merged = append(merged, r)
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
return merged, nil
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
func parseHTTPStatusCodeToken(token string) (StatusCodeRange, error) {
|
|
|
|
|
|
token = strings.TrimSpace(token)
|
|
|
|
|
|
token = strings.ReplaceAll(token, " ", "")
|
|
|
|
|
|
if token == "" {
|
|
|
|
|
|
return StatusCodeRange{}, fmt.Errorf("empty token")
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
if strings.Contains(token, "-") {
|
|
|
|
|
|
parts := strings.Split(token, "-")
|
|
|
|
|
|
if len(parts) != 2 || parts[0] == "" || parts[1] == "" {
|
|
|
|
|
|
return StatusCodeRange{}, fmt.Errorf("invalid range token: %s", token)
|
|
|
|
|
|
}
|
|
|
|
|
|
start, err := strconv.Atoi(parts[0])
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
|
return StatusCodeRange{}, fmt.Errorf("invalid range start: %s", token)
|
|
|
|
|
|
}
|
|
|
|
|
|
end, err := strconv.Atoi(parts[1])
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
|
return StatusCodeRange{}, fmt.Errorf("invalid range end: %s", token)
|
|
|
|
|
|
}
|
|
|
|
|
|
if start > end {
|
|
|
|
|
|
return StatusCodeRange{}, fmt.Errorf("range start > end: %s", token)
|
|
|
|
|
|
}
|
|
|
|
|
|
if start < 100 || end > 599 {
|
|
|
|
|
|
return StatusCodeRange{}, fmt.Errorf("range out of bounds: %s", token)
|
|
|
|
|
|
}
|
|
|
|
|
|
return StatusCodeRange{Start: start, End: end}, nil
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
code, err := strconv.Atoi(token)
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
|
return StatusCodeRange{}, fmt.Errorf("invalid status code: %s", token)
|
|
|
|
|
|
}
|
|
|
|
|
|
if code < 100 || code > 599 {
|
|
|
|
|
|
return StatusCodeRange{}, fmt.Errorf("status code out of bounds: %s", token)
|
|
|
|
|
|
}
|
|
|
|
|
|
return StatusCodeRange{Start: code, End: code}, nil
|
|
|
|
|
|
}
|