Eager initialize minpoolsize on connect
This commit is contained in:
@@ -225,6 +225,10 @@ func ConnectConfig(ctx context.Context, config *Config) (*Pool, error) {
|
|||||||
go p.backgroundHealthCheck()
|
go p.backgroundHealthCheck()
|
||||||
|
|
||||||
if !config.LazyConnect {
|
if !config.LazyConnect {
|
||||||
|
if err := p.createIdleResources(ctx, int(p.minConns)); err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
// Initially establish one connection
|
// Initially establish one connection
|
||||||
res, err := p.p.Acquire(ctx)
|
res, err := p.p.Acquire(ctx)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
@@ -377,6 +381,29 @@ func (p *Pool) checkMinConns() {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (p *Pool) createIdleResources(parentCtx context.Context, targetResources int) error {
|
||||||
|
ctx, cancel := context.WithCancel(parentCtx)
|
||||||
|
defer cancel()
|
||||||
|
|
||||||
|
errs := make(chan error)
|
||||||
|
|
||||||
|
for i := 0; i < targetResources; i++ {
|
||||||
|
go func() {
|
||||||
|
err := p.p.CreateResource(ctx)
|
||||||
|
errs <- err
|
||||||
|
}()
|
||||||
|
}
|
||||||
|
|
||||||
|
for i := 0; i < targetResources; i++ {
|
||||||
|
if err := <-errs; err != nil {
|
||||||
|
cancel()
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
// Acquire returns a connection (*Conn) from the Pool
|
// Acquire returns a connection (*Conn) from the Pool
|
||||||
func (p *Pool) Acquire(ctx context.Context) (*Conn, error) {
|
func (p *Pool) Acquire(ctx context.Context) (*Conn, error) {
|
||||||
for {
|
for {
|
||||||
|
|||||||
@@ -5,6 +5,7 @@ import (
|
|||||||
"errors"
|
"errors"
|
||||||
"fmt"
|
"fmt"
|
||||||
"os"
|
"os"
|
||||||
|
"sync/atomic"
|
||||||
"testing"
|
"testing"
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
@@ -800,3 +801,104 @@ func TestIdempotentPoolClose(t *testing.T) {
|
|||||||
// Close the already closed pool.
|
// Close the already closed pool.
|
||||||
require.NotPanics(t, func() { pool.Close() })
|
require.NotPanics(t, func() { pool.Close() })
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func TestConnectCreatesMinPool(t *testing.T) {
|
||||||
|
t.Parallel()
|
||||||
|
|
||||||
|
config, err := pgxpool.ParseConfig(os.Getenv("PGX_TEST_DATABASE"))
|
||||||
|
require.NoError(t, err)
|
||||||
|
|
||||||
|
config.MinConns = int32(12)
|
||||||
|
config.MaxConns = int32(15)
|
||||||
|
config.LazyConnect = false
|
||||||
|
|
||||||
|
acquireAttempts := int64(0)
|
||||||
|
connectAttempts := int64(0)
|
||||||
|
|
||||||
|
config.BeforeAcquire = func(ctx context.Context, conn *pgx.Conn) bool {
|
||||||
|
atomic.AddInt64(&acquireAttempts, 1)
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
config.BeforeConnect = func(ctx context.Context, cfg *pgx.ConnConfig) error {
|
||||||
|
atomic.AddInt64(&connectAttempts, 1)
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
pool, err := pgxpool.ConnectConfig(context.Background(), config)
|
||||||
|
require.NoError(t, err)
|
||||||
|
defer pool.Close()
|
||||||
|
|
||||||
|
stat := pool.Stat()
|
||||||
|
require.Equal(t, int32(12), stat.IdleConns())
|
||||||
|
require.Equal(t, int64(1), stat.AcquireCount())
|
||||||
|
require.Equal(t, int32(12), stat.TotalConns())
|
||||||
|
require.Equal(t, int64(0), acquireAttempts)
|
||||||
|
require.Equal(t, int64(12), connectAttempts)
|
||||||
|
}
|
||||||
|
func TestConnectSkipMinPoolWithLazy(t *testing.T) {
|
||||||
|
t.Parallel()
|
||||||
|
|
||||||
|
config, err := pgxpool.ParseConfig(os.Getenv("PGX_TEST_DATABASE"))
|
||||||
|
require.NoError(t, err)
|
||||||
|
|
||||||
|
config.MinConns = int32(12)
|
||||||
|
config.MaxConns = int32(15)
|
||||||
|
config.LazyConnect = true
|
||||||
|
|
||||||
|
acquireAttempts := int64(0)
|
||||||
|
connectAttempts := int64(0)
|
||||||
|
|
||||||
|
config.BeforeAcquire = func(ctx context.Context, conn *pgx.Conn) bool {
|
||||||
|
atomic.AddInt64(&acquireAttempts, 1)
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
config.BeforeConnect = func(ctx context.Context, cfg *pgx.ConnConfig) error {
|
||||||
|
atomic.AddInt64(&connectAttempts, 1)
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
pool, err := pgxpool.ConnectConfig(context.Background(), config)
|
||||||
|
require.NoError(t, err)
|
||||||
|
defer pool.Close()
|
||||||
|
|
||||||
|
stat := pool.Stat()
|
||||||
|
require.Equal(t, int32(0), stat.IdleConns())
|
||||||
|
require.Equal(t, int64(0), stat.AcquireCount())
|
||||||
|
require.Equal(t, int32(0), stat.TotalConns())
|
||||||
|
require.Equal(t, int64(0), acquireAttempts)
|
||||||
|
require.Equal(t, int64(0), connectAttempts)
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestConnectMinPoolZero(t *testing.T) {
|
||||||
|
t.Parallel()
|
||||||
|
|
||||||
|
config, err := pgxpool.ParseConfig(os.Getenv("PGX_TEST_DATABASE"))
|
||||||
|
require.NoError(t, err)
|
||||||
|
|
||||||
|
config.MinConns = int32(0)
|
||||||
|
config.MaxConns = int32(15)
|
||||||
|
config.LazyConnect = false
|
||||||
|
|
||||||
|
acquireAttempts := int64(0)
|
||||||
|
connectAttempts := int64(0)
|
||||||
|
|
||||||
|
config.BeforeAcquire = func(ctx context.Context, conn *pgx.Conn) bool {
|
||||||
|
atomic.AddInt64(&acquireAttempts, 1)
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
config.BeforeConnect = func(ctx context.Context, cfg *pgx.ConnConfig) error {
|
||||||
|
atomic.AddInt64(&connectAttempts, 1)
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
pool, err := pgxpool.ConnectConfig(context.Background(), config)
|
||||||
|
require.NoError(t, err)
|
||||||
|
defer pool.Close()
|
||||||
|
|
||||||
|
stat := pool.Stat()
|
||||||
|
require.Equal(t, int32(1), stat.IdleConns())
|
||||||
|
require.Equal(t, int64(1), stat.AcquireCount())
|
||||||
|
require.Equal(t, int32(1), stat.TotalConns())
|
||||||
|
require.Equal(t, int64(0), acquireAttempts)
|
||||||
|
require.Equal(t, int64(1), connectAttempts)
|
||||||
|
}
|
||||||
|
|||||||
Reference in New Issue
Block a user