mirror of
https://github.com/redis/go-redis.git
synced 2025-04-17 20:17:02 +03:00
fix: connection pool timeout, increase retries (#3298)
* fix: connection pool timeout, increase retries Signed-off-by: monkey <golang@88.com> * fix: add shouldRetry test Signed-off-by: monkey <golang@88.com> --------- Signed-off-by: monkey <golang@88.com> Co-authored-by: Nedyalko Dyakov <nedyalko.dyakov@gmail.com>
This commit is contained in:
parent
1c9309fdc2
commit
4f09082f6b
3
error.go
3
error.go
@ -53,6 +53,9 @@ func shouldRetry(err error, retryTimeout bool) bool {
|
|||||||
return true
|
return true
|
||||||
case nil, context.Canceled, context.DeadlineExceeded:
|
case nil, context.Canceled, context.DeadlineExceeded:
|
||||||
return false
|
return false
|
||||||
|
case pool.ErrPoolTimeout:
|
||||||
|
// connection pool timeout, increase retries. #3289
|
||||||
|
return true
|
||||||
}
|
}
|
||||||
|
|
||||||
if v, ok := err.(timeoutError); ok {
|
if v, ok := err.(timeoutError); ok {
|
||||||
|
65
error_test.go
Normal file
65
error_test.go
Normal file
@ -0,0 +1,65 @@
|
|||||||
|
package redis_test
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
"errors"
|
||||||
|
"io"
|
||||||
|
|
||||||
|
. "github.com/bsm/ginkgo/v2"
|
||||||
|
. "github.com/bsm/gomega"
|
||||||
|
"github.com/redis/go-redis/v9"
|
||||||
|
)
|
||||||
|
|
||||||
|
type testTimeout struct {
|
||||||
|
timeout bool
|
||||||
|
}
|
||||||
|
|
||||||
|
func (t testTimeout) Timeout() bool {
|
||||||
|
return t.timeout
|
||||||
|
}
|
||||||
|
|
||||||
|
func (t testTimeout) Error() string {
|
||||||
|
return "test timeout"
|
||||||
|
}
|
||||||
|
|
||||||
|
var _ = Describe("error", func() {
|
||||||
|
BeforeEach(func() {
|
||||||
|
|
||||||
|
})
|
||||||
|
|
||||||
|
AfterEach(func() {
|
||||||
|
|
||||||
|
})
|
||||||
|
|
||||||
|
It("should retry", func() {
|
||||||
|
data := map[error]bool{
|
||||||
|
io.EOF: true,
|
||||||
|
io.ErrUnexpectedEOF: true,
|
||||||
|
nil: false,
|
||||||
|
context.Canceled: false,
|
||||||
|
context.DeadlineExceeded: false,
|
||||||
|
redis.ErrPoolTimeout: true,
|
||||||
|
errors.New("ERR max number of clients reached"): true,
|
||||||
|
errors.New("LOADING Redis is loading the dataset in memory"): true,
|
||||||
|
errors.New("READONLY You can't write against a read only replica"): true,
|
||||||
|
errors.New("CLUSTERDOWN The cluster is down"): true,
|
||||||
|
errors.New("TRYAGAIN Command cannot be processed, please try again"): true,
|
||||||
|
errors.New("other"): false,
|
||||||
|
}
|
||||||
|
|
||||||
|
for err, expected := range data {
|
||||||
|
Expect(redis.ShouldRetry(err, false)).To(Equal(expected))
|
||||||
|
Expect(redis.ShouldRetry(err, true)).To(Equal(expected))
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
|
It("should retry timeout", func() {
|
||||||
|
t1 := testTimeout{timeout: true}
|
||||||
|
Expect(redis.ShouldRetry(t1, true)).To(Equal(true))
|
||||||
|
Expect(redis.ShouldRetry(t1, false)).To(Equal(false))
|
||||||
|
|
||||||
|
t2 := testTimeout{timeout: false}
|
||||||
|
Expect(redis.ShouldRetry(t2, true)).To(Equal(true))
|
||||||
|
Expect(redis.ShouldRetry(t2, false)).To(Equal(true))
|
||||||
|
})
|
||||||
|
})
|
@ -11,6 +11,8 @@ import (
|
|||||||
"github.com/redis/go-redis/v9/internal/pool"
|
"github.com/redis/go-redis/v9/internal/pool"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
var ErrPoolTimeout = pool.ErrPoolTimeout
|
||||||
|
|
||||||
func (c *baseClient) Pool() pool.Pooler {
|
func (c *baseClient) Pool() pool.Pooler {
|
||||||
return c.connPool
|
return c.connPool
|
||||||
}
|
}
|
||||||
@ -102,3 +104,7 @@ func (c *Ring) ShardByName(name string) *ringShard {
|
|||||||
func (c *ModuleLoadexConfig) ToArgs() []interface{} {
|
func (c *ModuleLoadexConfig) ToArgs() []interface{} {
|
||||||
return c.toArgs()
|
return c.toArgs()
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func ShouldRetry(err error, retryTimeout bool) bool {
|
||||||
|
return shouldRetry(err, retryTimeout)
|
||||||
|
}
|
||||||
|
Loading…
x
Reference in New Issue
Block a user