|
|
|
package redis
|
|
|
|
|
|
|
|
import (
|
|
|
|
"crypto/tls"
|
|
|
|
"io"
|
|
|
|
"runtime"
|
|
|
|
"strings"
|
|
|
|
|
|
|
|
red "github.com/redis/go-redis/v9"
|
|
|
|
"github.com/zeromicro/go-zero/core/syncx"
|
|
|
|
)
|
|
|
|
|
|
|
|
const addrSep = ","
|
|
|
|
|
|
|
|
var (
|
|
|
|
clusterManager = syncx.NewResourceManager()
|
|
|
|
// clusterPoolSize is default pool size for cluster type of redis.
|
|
|
|
clusterPoolSize = 5 * runtime.GOMAXPROCS(0)
|
|
|
|
)
|
|
|
|
|
|
|
|
func getCluster(r *Redis) (*red.ClusterClient, error) {
|
|
|
|
val, err := clusterManager.GetResource(r.Addr, func() (io.Closer, error) {
|
|
|
|
var tlsConfig *tls.Config
|
|
|
|
if r.tls {
|
|
|
|
tlsConfig = &tls.Config{
|
|
|
|
InsecureSkipVerify: true,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
store := red.NewClusterClient(&red.ClusterOptions{
|
|
|
|
Addrs: splitClusterAddrs(r.Addr),
|
|
|
|
Password: r.Pass,
|
|
|
|
MaxRetries: maxRetries,
|
|
|
|
MinIdleConns: idleConns,
|
|
|
|
TLSConfig: tlsConfig,
|
|
|
|
})
|
|
|
|
store.AddHook(durationHook)
|
|
|
|
for _, hook := range r.hooks {
|
|
|
|
store.AddHook(hook)
|
|
|
|
}
|
|
|
|
|
|
|
|
connCollector.registerClient(&statGetter{
|
|
|
|
clientType: ClusterType,
|
|
|
|
key: r.Addr,
|
|
|
|
poolSize: clusterPoolSize,
|
|
|
|
poolStats: func() *red.PoolStats {
|
|
|
|
return store.PoolStats()
|
|
|
|
},
|
|
|
|
})
|
|
|
|
|
|
|
|
return store, nil
|
|
|
|
})
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return val.(*red.ClusterClient), nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func splitClusterAddrs(addr string) []string {
|
|
|
|
addrs := strings.Split(addr, addrSep)
|
|
|
|
unique := make(map[string]struct{})
|
|
|
|
for _, each := range addrs {
|
|
|
|
unique[strings.TrimSpace(each)] = struct{}{}
|
|
|
|
}
|
|
|
|
|
|
|
|
addrs = addrs[:0]
|
|
|
|
for k := range unique {
|
|
|
|
addrs = append(addrs, k)
|
|
|
|
}
|
|
|
|
|
|
|
|
return addrs
|
|
|
|
}
|