mirror of
https://github.com/redis/go-redis.git
synced 2025-09-04 09:22:10 +03:00
This commit includes all the work on hitless upgrades with the addition of: - Pubsub Pool - Examples - Refactor of push - Refactor of pool (using atomics for most things) - Introducing of hooks in pool
150 lines
4.9 KiB
Go
150 lines
4.9 KiB
Go
package redis
|
|
|
|
import (
|
|
"context"
|
|
"errors"
|
|
"net"
|
|
"time"
|
|
|
|
"github.com/redis/go-redis/v9/internal/interfaces"
|
|
"github.com/redis/go-redis/v9/internal/pool"
|
|
"github.com/redis/go-redis/v9/push"
|
|
)
|
|
|
|
// ErrInvalidCommand is returned when an invalid command is passed to ExecuteCommand.
|
|
var ErrInvalidCommand = errors.New("invalid command type")
|
|
|
|
// ErrInvalidPool is returned when the pool type is not supported.
|
|
var ErrInvalidPool = errors.New("invalid pool type")
|
|
|
|
// newClientAdapter creates a new client adapter for regular Redis clients.
|
|
func newClientAdapter(client *baseClient) interfaces.ClientInterface {
|
|
return &clientAdapter{client: client}
|
|
}
|
|
|
|
// clientAdapter adapts a Redis client to implement interfaces.ClientInterface.
|
|
type clientAdapter struct {
|
|
client *baseClient
|
|
}
|
|
|
|
// GetOptions returns the client options.
|
|
func (ca *clientAdapter) GetOptions() interfaces.OptionsInterface {
|
|
return &optionsAdapter{options: ca.client.opt}
|
|
}
|
|
|
|
// GetPushProcessor returns the client's push notification processor.
|
|
func (ca *clientAdapter) GetPushProcessor() interfaces.NotificationProcessor {
|
|
return &pushProcessorAdapter{processor: ca.client.pushProcessor}
|
|
}
|
|
|
|
// optionsAdapter adapts Redis options to implement interfaces.OptionsInterface.
|
|
type optionsAdapter struct {
|
|
options *Options
|
|
}
|
|
|
|
// GetReadTimeout returns the read timeout.
|
|
func (oa *optionsAdapter) GetReadTimeout() time.Duration {
|
|
return oa.options.ReadTimeout
|
|
}
|
|
|
|
// GetWriteTimeout returns the write timeout.
|
|
func (oa *optionsAdapter) GetWriteTimeout() time.Duration {
|
|
return oa.options.WriteTimeout
|
|
}
|
|
|
|
// GetNetwork returns the network type.
|
|
func (oa *optionsAdapter) GetNetwork() string {
|
|
return oa.options.Network
|
|
}
|
|
|
|
// GetAddr returns the connection address.
|
|
func (oa *optionsAdapter) GetAddr() string {
|
|
return oa.options.Addr
|
|
}
|
|
|
|
// IsTLSEnabled returns true if TLS is enabled.
|
|
func (oa *optionsAdapter) IsTLSEnabled() bool {
|
|
return oa.options.TLSConfig != nil
|
|
}
|
|
|
|
// GetProtocol returns the protocol version.
|
|
func (oa *optionsAdapter) GetProtocol() int {
|
|
return oa.options.Protocol
|
|
}
|
|
|
|
// GetPoolSize returns the connection pool size.
|
|
func (oa *optionsAdapter) GetPoolSize() int {
|
|
return oa.options.PoolSize
|
|
}
|
|
|
|
// NewDialer returns a new dialer function for the connection.
|
|
func (oa *optionsAdapter) NewDialer() func(context.Context) (net.Conn, error) {
|
|
baseDialer := oa.options.NewDialer()
|
|
return func(ctx context.Context) (net.Conn, error) {
|
|
// Extract network and address from the options
|
|
network := oa.options.Network
|
|
addr := oa.options.Addr
|
|
return baseDialer(ctx, network, addr)
|
|
}
|
|
}
|
|
|
|
// connectionAdapter adapts a Redis connection to interfaces.ConnectionWithRelaxedTimeout
|
|
type connectionAdapter struct {
|
|
conn *pool.Conn
|
|
}
|
|
|
|
// Close closes the connection.
|
|
func (ca *connectionAdapter) Close() error {
|
|
return ca.conn.Close()
|
|
}
|
|
|
|
// IsUsable returns true if the connection is safe to use for new commands.
|
|
func (ca *connectionAdapter) IsUsable() bool {
|
|
return ca.conn.IsUsable()
|
|
}
|
|
|
|
// GetPoolConnection returns the underlying pool connection.
|
|
func (ca *connectionAdapter) GetPoolConnection() *pool.Conn {
|
|
return ca.conn
|
|
}
|
|
|
|
// SetRelaxedTimeout sets relaxed timeouts for this connection during hitless upgrades.
|
|
// These timeouts remain active until explicitly cleared.
|
|
func (ca *connectionAdapter) SetRelaxedTimeout(readTimeout, writeTimeout time.Duration) {
|
|
ca.conn.SetRelaxedTimeout(readTimeout, writeTimeout)
|
|
}
|
|
|
|
// SetRelaxedTimeoutWithDeadline sets relaxed timeouts with an expiration deadline.
|
|
// After the deadline, timeouts automatically revert to normal values.
|
|
func (ca *connectionAdapter) SetRelaxedTimeoutWithDeadline(readTimeout, writeTimeout time.Duration, deadline time.Time) {
|
|
ca.conn.SetRelaxedTimeoutWithDeadline(readTimeout, writeTimeout, deadline)
|
|
}
|
|
|
|
// ClearRelaxedTimeout clears relaxed timeouts for this connection.
|
|
func (ca *connectionAdapter) ClearRelaxedTimeout() {
|
|
ca.conn.ClearRelaxedTimeout()
|
|
}
|
|
|
|
// pushProcessorAdapter adapts a push.NotificationProcessor to implement interfaces.NotificationProcessor.
|
|
type pushProcessorAdapter struct {
|
|
processor push.NotificationProcessor
|
|
}
|
|
|
|
// RegisterHandler registers a handler for a specific push notification name.
|
|
func (ppa *pushProcessorAdapter) RegisterHandler(pushNotificationName string, handler interface{}, protected bool) error {
|
|
if pushHandler, ok := handler.(push.NotificationHandler); ok {
|
|
return ppa.processor.RegisterHandler(pushNotificationName, pushHandler, protected)
|
|
}
|
|
return errors.New("handler must implement push.NotificationHandler")
|
|
}
|
|
|
|
// UnregisterHandler removes a handler for a specific push notification name.
|
|
func (ppa *pushProcessorAdapter) UnregisterHandler(pushNotificationName string) error {
|
|
return ppa.processor.UnregisterHandler(pushNotificationName)
|
|
}
|
|
|
|
// GetHandler returns the handler for a specific push notification name.
|
|
func (ppa *pushProcessorAdapter) GetHandler(pushNotificationName string) interface{} {
|
|
return ppa.processor.GetHandler(pushNotificationName)
|
|
}
|