mirror of https://github.com/jackc/pgx.git
commit
70c9a147a2
|
@ -147,6 +147,7 @@ func assertConfigsEqual(t *testing.T, expected, actual *pgxpool.Config, testName
|
||||||
assert.Equalf(t, expected.MaxConnIdleTime, actual.MaxConnIdleTime, "%s - MaxConnIdleTime", testName)
|
assert.Equalf(t, expected.MaxConnIdleTime, actual.MaxConnIdleTime, "%s - MaxConnIdleTime", testName)
|
||||||
assert.Equalf(t, expected.MaxConns, actual.MaxConns, "%s - MaxConns", testName)
|
assert.Equalf(t, expected.MaxConns, actual.MaxConns, "%s - MaxConns", testName)
|
||||||
assert.Equalf(t, expected.MinConns, actual.MinConns, "%s - MinConns", testName)
|
assert.Equalf(t, expected.MinConns, actual.MinConns, "%s - MinConns", testName)
|
||||||
|
assert.Equalf(t, expected.MinIdleConns, actual.MinIdleConns, "%s - MinIdleConns", testName)
|
||||||
assert.Equalf(t, expected.HealthCheckPeriod, actual.HealthCheckPeriod, "%s - HealthCheckPeriod", testName)
|
assert.Equalf(t, expected.HealthCheckPeriod, actual.HealthCheckPeriod, "%s - HealthCheckPeriod", testName)
|
||||||
|
|
||||||
assertConnConfigsEqual(t, expected.ConnConfig, actual.ConnConfig, testName)
|
assertConnConfigsEqual(t, expected.ConnConfig, actual.ConnConfig, testName)
|
||||||
|
|
|
@ -17,6 +17,7 @@ import (
|
||||||
|
|
||||||
var defaultMaxConns = int32(4)
|
var defaultMaxConns = int32(4)
|
||||||
var defaultMinConns = int32(0)
|
var defaultMinConns = int32(0)
|
||||||
|
var defaultMinIdleConns = int32(0)
|
||||||
var defaultMaxConnLifetime = time.Hour
|
var defaultMaxConnLifetime = time.Hour
|
||||||
var defaultMaxConnIdleTime = time.Minute * 30
|
var defaultMaxConnIdleTime = time.Minute * 30
|
||||||
var defaultHealthCheckPeriod = time.Minute
|
var defaultHealthCheckPeriod = time.Minute
|
||||||
|
@ -87,6 +88,7 @@ type Pool struct {
|
||||||
afterRelease func(*pgx.Conn) bool
|
afterRelease func(*pgx.Conn) bool
|
||||||
beforeClose func(*pgx.Conn)
|
beforeClose func(*pgx.Conn)
|
||||||
minConns int32
|
minConns int32
|
||||||
|
minIdleConns int32
|
||||||
maxConns int32
|
maxConns int32
|
||||||
maxConnLifetime time.Duration
|
maxConnLifetime time.Duration
|
||||||
maxConnLifetimeJitter time.Duration
|
maxConnLifetimeJitter time.Duration
|
||||||
|
@ -144,6 +146,13 @@ type Config struct {
|
||||||
// to create new connections.
|
// to create new connections.
|
||||||
MinConns int32
|
MinConns int32
|
||||||
|
|
||||||
|
// MinIdleConns is the minimum number of idle connections in the pool. You can increase this to ensure that
|
||||||
|
// there are always idle connections available. This can help reduce tail latencies during request processing,
|
||||||
|
// as you can avoid the latency of establishing a new connection while handling requests. It is superior
|
||||||
|
// to MinConns for this purpose.
|
||||||
|
// Similar to MinConns, the pool might temporarily dip below MinIdleConns after connection closes.
|
||||||
|
MinIdleConns int32
|
||||||
|
|
||||||
// HealthCheckPeriod is the duration between checks of the health of idle connections.
|
// HealthCheckPeriod is the duration between checks of the health of idle connections.
|
||||||
HealthCheckPeriod time.Duration
|
HealthCheckPeriod time.Duration
|
||||||
|
|
||||||
|
@ -189,6 +198,7 @@ func NewWithConfig(ctx context.Context, config *Config) (*Pool, error) {
|
||||||
afterRelease: config.AfterRelease,
|
afterRelease: config.AfterRelease,
|
||||||
beforeClose: config.BeforeClose,
|
beforeClose: config.BeforeClose,
|
||||||
minConns: config.MinConns,
|
minConns: config.MinConns,
|
||||||
|
minIdleConns: config.MinIdleConns,
|
||||||
maxConns: config.MaxConns,
|
maxConns: config.MaxConns,
|
||||||
maxConnLifetime: config.MaxConnLifetime,
|
maxConnLifetime: config.MaxConnLifetime,
|
||||||
maxConnLifetimeJitter: config.MaxConnLifetimeJitter,
|
maxConnLifetimeJitter: config.MaxConnLifetimeJitter,
|
||||||
|
@ -271,7 +281,8 @@ func NewWithConfig(ctx context.Context, config *Config) (*Pool, error) {
|
||||||
}
|
}
|
||||||
|
|
||||||
go func() {
|
go func() {
|
||||||
p.createIdleResources(ctx, int(p.minConns))
|
targetIdleResources := max(int(p.minConns), int(p.minIdleConns))
|
||||||
|
p.createIdleResources(ctx, targetIdleResources)
|
||||||
p.backgroundHealthCheck()
|
p.backgroundHealthCheck()
|
||||||
}()
|
}()
|
||||||
|
|
||||||
|
@ -334,6 +345,17 @@ func ParseConfig(connString string) (*Config, error) {
|
||||||
config.MinConns = defaultMinConns
|
config.MinConns = defaultMinConns
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if s, ok := config.ConnConfig.Config.RuntimeParams["pool_min_idle_conns"]; ok {
|
||||||
|
delete(connConfig.Config.RuntimeParams, "pool_min_idle_conns")
|
||||||
|
n, err := strconv.ParseInt(s, 10, 32)
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("cannot parse pool_min_idle_conns: %w", err)
|
||||||
|
}
|
||||||
|
config.MinIdleConns = int32(n)
|
||||||
|
} else {
|
||||||
|
config.MinIdleConns = defaultMinIdleConns
|
||||||
|
}
|
||||||
|
|
||||||
if s, ok := config.ConnConfig.Config.RuntimeParams["pool_max_conn_lifetime"]; ok {
|
if s, ok := config.ConnConfig.Config.RuntimeParams["pool_max_conn_lifetime"]; ok {
|
||||||
delete(connConfig.Config.RuntimeParams, "pool_max_conn_lifetime")
|
delete(connConfig.Config.RuntimeParams, "pool_max_conn_lifetime")
|
||||||
d, err := time.ParseDuration(s)
|
d, err := time.ParseDuration(s)
|
||||||
|
@ -472,7 +494,9 @@ func (p *Pool) checkMinConns() error {
|
||||||
// TotalConns can include ones that are being destroyed but we should have
|
// TotalConns can include ones that are being destroyed but we should have
|
||||||
// sleep(500ms) around all of the destroys to help prevent that from throwing
|
// sleep(500ms) around all of the destroys to help prevent that from throwing
|
||||||
// off this check
|
// off this check
|
||||||
toCreate := p.minConns - p.Stat().TotalConns()
|
|
||||||
|
// Create the number of connections needed to get to both minConns and minIdleConns
|
||||||
|
toCreate := max(p.minConns-p.Stat().TotalConns(), p.minIdleConns-p.Stat().IdleConns())
|
||||||
if toCreate > 0 {
|
if toCreate > 0 {
|
||||||
return p.createIdleResources(context.Background(), int(toCreate))
|
return p.createIdleResources(context.Background(), int(toCreate))
|
||||||
}
|
}
|
||||||
|
|
|
@ -43,10 +43,11 @@ func TestConnectConfig(t *testing.T) {
|
||||||
func TestParseConfigExtractsPoolArguments(t *testing.T) {
|
func TestParseConfigExtractsPoolArguments(t *testing.T) {
|
||||||
t.Parallel()
|
t.Parallel()
|
||||||
|
|
||||||
config, err := pgxpool.ParseConfig("pool_max_conns=42 pool_min_conns=1")
|
config, err := pgxpool.ParseConfig("pool_max_conns=42 pool_min_conns=1 pool_min_idle_conns=2")
|
||||||
assert.NoError(t, err)
|
assert.NoError(t, err)
|
||||||
assert.EqualValues(t, 42, config.MaxConns)
|
assert.EqualValues(t, 42, config.MaxConns)
|
||||||
assert.EqualValues(t, 1, config.MinConns)
|
assert.EqualValues(t, 1, config.MinConns)
|
||||||
|
assert.EqualValues(t, 2, config.MinIdleConns)
|
||||||
assert.NotContains(t, config.ConnConfig.Config.RuntimeParams, "pool_max_conns")
|
assert.NotContains(t, config.ConnConfig.Config.RuntimeParams, "pool_max_conns")
|
||||||
assert.NotContains(t, config.ConnConfig.Config.RuntimeParams, "pool_min_conns")
|
assert.NotContains(t, config.ConnConfig.Config.RuntimeParams, "pool_min_conns")
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in New Issue