mirror of
https://codeberg.org/forgejo/forgejo.git
synced 2025-01-24 17:15:12 -05:00
109 lines
2 KiB
Go
109 lines
2 KiB
Go
|
package redis
|
||
|
|
||
|
import (
|
||
|
"context"
|
||
|
"io"
|
||
|
"net"
|
||
|
"strings"
|
||
|
|
||
|
"github.com/go-redis/redis/v7/internal/pool"
|
||
|
"github.com/go-redis/redis/v7/internal/proto"
|
||
|
)
|
||
|
|
||
|
var ErrClosed = pool.ErrClosed
|
||
|
|
||
|
type Error interface {
|
||
|
error
|
||
|
|
||
|
// RedisError is a no-op function but
|
||
|
// serves to distinguish types that are Redis
|
||
|
// errors from ordinary errors: a type is a
|
||
|
// Redis error if it has a RedisError method.
|
||
|
RedisError()
|
||
|
}
|
||
|
|
||
|
var _ Error = proto.RedisError("")
|
||
|
|
||
|
func isRetryableError(err error, retryTimeout bool) bool {
|
||
|
switch err {
|
||
|
case nil, context.Canceled, context.DeadlineExceeded:
|
||
|
return false
|
||
|
case io.EOF:
|
||
|
return true
|
||
|
}
|
||
|
if netErr, ok := err.(net.Error); ok {
|
||
|
if netErr.Timeout() {
|
||
|
return retryTimeout
|
||
|
}
|
||
|
return true
|
||
|
}
|
||
|
|
||
|
s := err.Error()
|
||
|
if s == "ERR max number of clients reached" {
|
||
|
return true
|
||
|
}
|
||
|
if strings.HasPrefix(s, "LOADING ") {
|
||
|
return true
|
||
|
}
|
||
|
if strings.HasPrefix(s, "READONLY ") {
|
||
|
return true
|
||
|
}
|
||
|
if strings.HasPrefix(s, "CLUSTERDOWN ") {
|
||
|
return true
|
||
|
}
|
||
|
return false
|
||
|
}
|
||
|
|
||
|
func isRedisError(err error) bool {
|
||
|
_, ok := err.(proto.RedisError)
|
||
|
return ok
|
||
|
}
|
||
|
|
||
|
func isBadConn(err error, allowTimeout bool) bool {
|
||
|
if err == nil {
|
||
|
return false
|
||
|
}
|
||
|
if isRedisError(err) {
|
||
|
// Close connections in read only state in case domain addr is used
|
||
|
// and domain resolves to a different Redis Server. See #790.
|
||
|
return isReadOnlyError(err)
|
||
|
}
|
||
|
if allowTimeout {
|
||
|
if netErr, ok := err.(net.Error); ok && netErr.Timeout() {
|
||
|
return false
|
||
|
}
|
||
|
}
|
||
|
return true
|
||
|
}
|
||
|
|
||
|
func isMovedError(err error) (moved bool, ask bool, addr string) {
|
||
|
if !isRedisError(err) {
|
||
|
return
|
||
|
}
|
||
|
|
||
|
s := err.Error()
|
||
|
switch {
|
||
|
case strings.HasPrefix(s, "MOVED "):
|
||
|
moved = true
|
||
|
case strings.HasPrefix(s, "ASK "):
|
||
|
ask = true
|
||
|
default:
|
||
|
return
|
||
|
}
|
||
|
|
||
|
ind := strings.LastIndex(s, " ")
|
||
|
if ind == -1 {
|
||
|
return false, false, ""
|
||
|
}
|
||
|
addr = s[ind+1:]
|
||
|
return
|
||
|
}
|
||
|
|
||
|
func isLoadingError(err error) bool {
|
||
|
return strings.HasPrefix(err.Error(), "LOADING ")
|
||
|
}
|
||
|
|
||
|
func isReadOnlyError(err error) bool {
|
||
|
return strings.HasPrefix(err.Error(), "READONLY ")
|
||
|
}
|