Ensure pgxpool.Pool.QueryRow.Scan releases connection on panic

Otherwise a connection would be leaked and closing the pool would block.

https://github.com/jackc/pgx/issues/1628
pull/1631/head
Jack Christensen 2023-06-03 07:39:56 -05:00
parent 229d2aaa49
commit 608f39f426
2 changed files with 26 additions and 0 deletions

View File

@ -698,6 +698,25 @@ func TestPoolQueryRowErrNoRows(t *testing.T) {
require.Equal(t, pgx.ErrNoRows, err)
}
// https://github.com/jackc/pgx/issues/1628
func TestPoolQueryRowScanPanicReleasesConnection(t *testing.T) {
t.Parallel()
ctx, cancel := context.WithTimeout(context.Background(), 10*time.Second)
defer cancel()
pool, err := pgxpool.New(ctx, os.Getenv("PGX_TEST_DATABASE"))
require.NoError(t, err)
defer pool.Close()
require.Panics(t, func() {
var greeting *string
pool.QueryRow(ctx, "select 'Hello, world!'").Scan(greeting) // Note lack of &. This means that a typed nil is passed to Scan.
})
// If the connection is not released this will block forever in the defer pool.Close().
}
func TestPoolSendBatch(t *testing.T) {
t.Parallel()

View File

@ -101,7 +101,14 @@ func (row *poolRow) Scan(dest ...any) error {
return row.err
}
panicked := true
defer func() {
if panicked && row.c != nil {
row.c.Release()
}
}()
err := row.r.Scan(dest...)
panicked = false
if row.c != nil {
row.c.Release()
}