Платформа ЦРНП "Мирокод" для разработки проектов
https://git.mirocod.ru
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
231 lines
4.1 KiB
231 lines
4.1 KiB
package redis |
|
|
|
import ( |
|
"log" |
|
"net" |
|
"time" |
|
) |
|
|
|
type baseClient struct { |
|
connPool pool |
|
opt *options |
|
cmds []Cmder |
|
} |
|
|
|
func (c *baseClient) writeCmd(cn *conn, cmds ...Cmder) error { |
|
buf := cn.buf[:0] |
|
for _, cmd := range cmds { |
|
buf = appendArgs(buf, cmd.args()) |
|
} |
|
|
|
_, err := cn.Write(buf) |
|
return err |
|
} |
|
|
|
func (c *baseClient) conn() (*conn, error) { |
|
cn, isNew, err := c.connPool.Get() |
|
if err != nil { |
|
return nil, err |
|
} |
|
|
|
if isNew { |
|
if err := c.initConn(cn); err != nil { |
|
c.removeConn(cn) |
|
return nil, err |
|
} |
|
} |
|
|
|
return cn, nil |
|
} |
|
|
|
func (c *baseClient) initConn(cn *conn) error { |
|
if c.opt.Password == "" && c.opt.DB == 0 { |
|
return nil |
|
} |
|
|
|
pool := newSingleConnPool(c.connPool, false) |
|
pool.SetConn(cn) |
|
|
|
// Client is not closed because we want to reuse underlying connection. |
|
client := &Client{ |
|
baseClient: &baseClient{ |
|
opt: c.opt, |
|
connPool: pool, |
|
}, |
|
} |
|
|
|
if c.opt.Password != "" { |
|
if err := client.Auth(c.opt.Password).Err(); err != nil { |
|
return err |
|
} |
|
} |
|
|
|
if c.opt.DB > 0 { |
|
if err := client.Select(c.opt.DB).Err(); err != nil { |
|
return err |
|
} |
|
} |
|
|
|
return nil |
|
} |
|
|
|
func (c *baseClient) freeConn(cn *conn, ei error) error { |
|
if cn.rd.Buffered() > 0 { |
|
return c.connPool.Remove(cn) |
|
} |
|
if _, ok := ei.(redisError); ok { |
|
return c.connPool.Put(cn) |
|
} |
|
return c.connPool.Remove(cn) |
|
} |
|
|
|
func (c *baseClient) removeConn(cn *conn) { |
|
if err := c.connPool.Remove(cn); err != nil { |
|
log.Printf("pool.Remove failed: %s", err) |
|
} |
|
} |
|
|
|
func (c *baseClient) putConn(cn *conn) { |
|
if err := c.connPool.Put(cn); err != nil { |
|
log.Printf("pool.Put failed: %s", err) |
|
} |
|
} |
|
|
|
func (c *baseClient) Process(cmd Cmder) { |
|
if c.cmds == nil { |
|
c.run(cmd) |
|
} else { |
|
c.cmds = append(c.cmds, cmd) |
|
} |
|
} |
|
|
|
func (c *baseClient) run(cmd Cmder) { |
|
cn, err := c.conn() |
|
if err != nil { |
|
cmd.setErr(err) |
|
return |
|
} |
|
|
|
if timeout := cmd.writeTimeout(); timeout != nil { |
|
cn.writeTimeout = *timeout |
|
} else { |
|
cn.writeTimeout = c.opt.WriteTimeout |
|
} |
|
|
|
if timeout := cmd.readTimeout(); timeout != nil { |
|
cn.readTimeout = *timeout |
|
} else { |
|
cn.readTimeout = c.opt.ReadTimeout |
|
} |
|
|
|
if err := c.writeCmd(cn, cmd); err != nil { |
|
c.freeConn(cn, err) |
|
cmd.setErr(err) |
|
return |
|
} |
|
|
|
if err := cmd.parseReply(cn.rd); err != nil { |
|
c.freeConn(cn, err) |
|
return |
|
} |
|
|
|
c.putConn(cn) |
|
} |
|
|
|
// Close closes the client, releasing any open resources. |
|
func (c *baseClient) Close() error { |
|
return c.connPool.Close() |
|
} |
|
|
|
//------------------------------------------------------------------------------ |
|
|
|
type options struct { |
|
Password string |
|
DB int64 |
|
|
|
DialTimeout time.Duration |
|
ReadTimeout time.Duration |
|
WriteTimeout time.Duration |
|
|
|
PoolSize int |
|
IdleTimeout time.Duration |
|
} |
|
|
|
type Options struct { |
|
Network string |
|
Addr string |
|
|
|
// Dialer creates new network connection and has priority over |
|
// Network and Addr options. |
|
Dialer func() (net.Conn, error) |
|
|
|
Password string |
|
DB int64 |
|
|
|
DialTimeout time.Duration |
|
ReadTimeout time.Duration |
|
WriteTimeout time.Duration |
|
|
|
PoolSize int |
|
IdleTimeout time.Duration |
|
} |
|
|
|
func (opt *Options) getPoolSize() int { |
|
if opt.PoolSize == 0 { |
|
return 10 |
|
} |
|
return opt.PoolSize |
|
} |
|
|
|
func (opt *Options) getDialTimeout() time.Duration { |
|
if opt.DialTimeout == 0 { |
|
return 5 * time.Second |
|
} |
|
return opt.DialTimeout |
|
} |
|
|
|
func (opt *Options) options() *options { |
|
return &options{ |
|
DB: opt.DB, |
|
Password: opt.Password, |
|
|
|
DialTimeout: opt.getDialTimeout(), |
|
ReadTimeout: opt.ReadTimeout, |
|
WriteTimeout: opt.WriteTimeout, |
|
|
|
PoolSize: opt.getPoolSize(), |
|
IdleTimeout: opt.IdleTimeout, |
|
} |
|
} |
|
|
|
type Client struct { |
|
*baseClient |
|
} |
|
|
|
func NewClient(clOpt *Options) *Client { |
|
opt := clOpt.options() |
|
dialer := clOpt.Dialer |
|
if dialer == nil { |
|
dialer = func() (net.Conn, error) { |
|
return net.DialTimeout(clOpt.Network, clOpt.Addr, opt.DialTimeout) |
|
} |
|
} |
|
return &Client{ |
|
baseClient: &baseClient{ |
|
opt: opt, |
|
connPool: newConnPool(newConnFunc(dialer), opt), |
|
}, |
|
} |
|
} |
|
|
|
// Deprecated. Use NewClient instead. |
|
func NewTCPClient(opt *Options) *Client { |
|
opt.Network = "tcp" |
|
return NewClient(opt) |
|
} |
|
|
|
// Deprecated. Use NewClient instead. |
|
func NewUnixClient(opt *Options) *Client { |
|
opt.Network = "unix" |
|
return NewClient(opt) |
|
}
|
|
|