93aa2b2e80
ErrDeadConn is returned when calling an already dead connection. But the initial failure returns the real error. So we check for IsAlive instead of ErrDeadConn. Added test for ConnPool.Begin retry logic.
250 lines
6.2 KiB
Go
250 lines
6.2 KiB
Go
package pgx
|
|
|
|
import (
|
|
"errors"
|
|
"sync"
|
|
)
|
|
|
|
type ConnPoolConfig struct {
|
|
ConnConfig
|
|
MaxConnections int // max simultaneous connections to use, default 5, must be at least 2
|
|
AfterConnect func(*Conn) error // function to call on every new connection
|
|
}
|
|
|
|
type ConnPool struct {
|
|
allConnections []*Conn
|
|
availableConnections []*Conn
|
|
cond *sync.Cond
|
|
config ConnConfig // config used when establishing connection
|
|
maxConnections int
|
|
afterConnect func(*Conn) error
|
|
logger Logger
|
|
closed bool
|
|
}
|
|
|
|
type ConnPoolStat struct {
|
|
MaxConnections int // max simultaneous connections to use
|
|
CurrentConnections int // current live connections
|
|
AvailableConnections int // unused live connections
|
|
}
|
|
|
|
// NewConnPool creates a new ConnPool. config.ConnConfig is passed through to
|
|
// Connect directly.
|
|
func NewConnPool(config ConnPoolConfig) (p *ConnPool, err error) {
|
|
p = new(ConnPool)
|
|
p.config = config.ConnConfig
|
|
p.maxConnections = config.MaxConnections
|
|
if p.maxConnections == 0 {
|
|
p.maxConnections = 5
|
|
}
|
|
if p.maxConnections < 1 {
|
|
return nil, errors.New("MaxConnections must be at least 1")
|
|
}
|
|
|
|
p.afterConnect = config.AfterConnect
|
|
if config.Logger != nil {
|
|
p.logger = config.Logger
|
|
} else {
|
|
p.logger = &discardLogger{}
|
|
}
|
|
|
|
p.allConnections = make([]*Conn, 0, p.maxConnections)
|
|
p.availableConnections = make([]*Conn, 0, p.maxConnections)
|
|
p.cond = sync.NewCond(new(sync.Mutex))
|
|
|
|
// Initially establish one connection
|
|
var c *Conn
|
|
c, err = p.createConnection()
|
|
if err != nil {
|
|
return
|
|
}
|
|
p.allConnections = append(p.allConnections, c)
|
|
p.availableConnections = append(p.availableConnections, c)
|
|
|
|
return
|
|
}
|
|
|
|
// Acquire takes exclusive use of a connection until it is released.
|
|
func (p *ConnPool) Acquire() (c *Conn, err error) {
|
|
p.cond.L.Lock()
|
|
defer p.cond.L.Unlock()
|
|
|
|
if p.closed {
|
|
return nil, errors.New("cannot acquire from closed pool")
|
|
}
|
|
|
|
// A connection is available
|
|
if len(p.availableConnections) > 0 {
|
|
c = p.availableConnections[len(p.availableConnections)-1]
|
|
p.availableConnections = p.availableConnections[:len(p.availableConnections)-1]
|
|
return
|
|
}
|
|
|
|
// No connections are available, but we can create more
|
|
if len(p.allConnections) < p.maxConnections {
|
|
c, err = p.createConnection()
|
|
if err != nil {
|
|
return
|
|
}
|
|
p.allConnections = append(p.allConnections, c)
|
|
return
|
|
}
|
|
|
|
// All connections are in use and we cannot create more
|
|
if len(p.availableConnections) == 0 {
|
|
p.logger.Warn("All connections in pool are busy - waiting...")
|
|
for len(p.availableConnections) == 0 {
|
|
p.cond.Wait()
|
|
}
|
|
}
|
|
|
|
c = p.availableConnections[len(p.availableConnections)-1]
|
|
p.availableConnections = p.availableConnections[:len(p.availableConnections)-1]
|
|
|
|
return
|
|
}
|
|
|
|
// Release gives up use of a connection.
|
|
func (p *ConnPool) Release(conn *Conn) {
|
|
if conn.TxStatus != 'I' {
|
|
conn.Exec("rollback")
|
|
}
|
|
|
|
p.cond.L.Lock()
|
|
if conn.IsAlive() {
|
|
p.availableConnections = append(p.availableConnections, conn)
|
|
} else {
|
|
ac := p.allConnections
|
|
for i, c := range ac {
|
|
if conn == c {
|
|
ac[i] = ac[len(ac)-1]
|
|
p.allConnections = ac[0 : len(ac)-1]
|
|
break
|
|
}
|
|
}
|
|
}
|
|
p.cond.L.Unlock()
|
|
p.cond.Signal()
|
|
}
|
|
|
|
// Close ends the use of a connection pool. It prevents any new connections
|
|
// from being acquired, waits until all acquired connections are released,
|
|
// then closes all underlying connections.
|
|
func (p *ConnPool) Close() {
|
|
p.cond.L.Lock()
|
|
defer p.cond.L.Unlock()
|
|
|
|
p.closed = true
|
|
|
|
// Wait until all connections are released
|
|
if len(p.availableConnections) != len(p.allConnections) {
|
|
for len(p.availableConnections) != len(p.allConnections) {
|
|
p.cond.Wait()
|
|
}
|
|
}
|
|
|
|
for _, c := range p.allConnections {
|
|
_ = c.Close()
|
|
}
|
|
}
|
|
|
|
// Stat returns connection pool statistics
|
|
func (p *ConnPool) Stat() (s ConnPoolStat) {
|
|
p.cond.L.Lock()
|
|
defer p.cond.L.Unlock()
|
|
|
|
s.MaxConnections = p.maxConnections
|
|
s.CurrentConnections = len(p.allConnections)
|
|
s.AvailableConnections = len(p.availableConnections)
|
|
return
|
|
}
|
|
|
|
func (p *ConnPool) createConnection() (c *Conn, err error) {
|
|
c, err = Connect(p.config)
|
|
if err != nil {
|
|
return
|
|
}
|
|
if p.afterConnect != nil {
|
|
err = p.afterConnect(c)
|
|
if err != nil {
|
|
return
|
|
}
|
|
}
|
|
return
|
|
}
|
|
|
|
// Exec acquires a connection, delegates the call to that connection, and releases the connection
|
|
func (p *ConnPool) Exec(sql string, arguments ...interface{}) (commandTag CommandTag, err error) {
|
|
var c *Conn
|
|
if c, err = p.Acquire(); err != nil {
|
|
return
|
|
}
|
|
defer p.Release(c)
|
|
|
|
return c.Exec(sql, arguments...)
|
|
}
|
|
|
|
// Query acquires a connection and delegates the call to that connection. When
|
|
// *Rows are closed, the connection is released automatically.
|
|
func (p *ConnPool) Query(sql string, args ...interface{}) (*Rows, error) {
|
|
c, err := p.Acquire()
|
|
if err != nil {
|
|
// Because checking for errors can be deferred to the *Rows, build one with the error
|
|
return &Rows{closed: true, err: err}, err
|
|
}
|
|
|
|
rows, err := c.Query(sql, args...)
|
|
if err != nil {
|
|
p.Release(c)
|
|
return rows, err
|
|
}
|
|
|
|
rows.pool = p
|
|
return rows, nil
|
|
}
|
|
|
|
// QueryRow acquires a connection and delegates the call to that connection. The
|
|
// connection is released automatically after Scan is called on the returned
|
|
// *Row.
|
|
func (p *ConnPool) QueryRow(sql string, args ...interface{}) *Row {
|
|
rows, _ := p.Query(sql, args...)
|
|
return (*Row)(rows)
|
|
}
|
|
|
|
// Begin acquires a connection and begins a transaction on it. When the
|
|
// transaction is closed the connection will be automatically released.
|
|
func (p *ConnPool) Begin() (*Tx, error) {
|
|
return p.BeginIso("")
|
|
}
|
|
|
|
// BeginIso acquires a connection and begins a transaction in isolation mode iso
|
|
// on it. When the transaction is closed the connection will be automatically
|
|
// released.
|
|
func (p *ConnPool) BeginIso(iso string) (*Tx, error) {
|
|
for {
|
|
c, err := p.Acquire()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
tx, err := c.BeginIso(iso)
|
|
if err != nil {
|
|
alive := c.IsAlive()
|
|
p.Release(c)
|
|
|
|
// If connection is still alive then the error is not something trying
|
|
// again on a new connection would fix, so just return the error. But
|
|
// if the connection is dead try to acquire a new connection and try
|
|
// again.
|
|
if alive {
|
|
return nil, err
|
|
} else {
|
|
continue
|
|
}
|
|
}
|
|
|
|
tx.pool = p
|
|
return tx, nil
|
|
}
|
|
}
|