mirror of https://github.com/jackc/pgx.git
Add *Tx.AfterClose hook
ConnPool now implements its connection with Tx via this hook instead of manipulating the internals of Tx.pull/120/head
parent
0f7bf19387
commit
e8dcf5b3ac
|
@ -287,7 +287,11 @@ func (p *ConnPool) BeginIso(iso string) (*Tx, error) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
tx.pool = p
|
tx.AfterClose(p.txAfterClose)
|
||||||
return tx, nil
|
return tx, nil
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (p *ConnPool) txAfterClose(tx *Tx) {
|
||||||
|
p.Release(tx.Conn())
|
||||||
|
}
|
||||||
|
|
25
tx.go
25
tx.go
|
@ -54,9 +54,9 @@ func (c *Conn) begin(isoLevel string) (*Tx, error) {
|
||||||
// All Tx methods return ErrTxClosed if Commit or Rollback has already been
|
// All Tx methods return ErrTxClosed if Commit or Rollback has already been
|
||||||
// called on the Tx.
|
// called on the Tx.
|
||||||
type Tx struct {
|
type Tx struct {
|
||||||
pool *ConnPool
|
conn *Conn
|
||||||
conn *Conn
|
afterClose func(*Tx)
|
||||||
closed bool
|
closed bool
|
||||||
}
|
}
|
||||||
|
|
||||||
// Commit commits the transaction
|
// Commit commits the transaction
|
||||||
|
@ -85,9 +85,8 @@ func (tx *Tx) Rollback() error {
|
||||||
}
|
}
|
||||||
|
|
||||||
func (tx *Tx) close() {
|
func (tx *Tx) close() {
|
||||||
if tx.pool != nil {
|
if tx.afterClose != nil {
|
||||||
tx.pool.Release(tx.conn)
|
tx.afterClose(tx)
|
||||||
tx.pool = nil
|
|
||||||
}
|
}
|
||||||
tx.closed = true
|
tx.closed = true
|
||||||
}
|
}
|
||||||
|
@ -117,3 +116,17 @@ func (tx *Tx) QueryRow(sql string, args ...interface{}) *Row {
|
||||||
rows, _ := tx.Query(sql, args...)
|
rows, _ := tx.Query(sql, args...)
|
||||||
return (*Row)(rows)
|
return (*Row)(rows)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// AfterClose adds f to a LILO queue of functions that will be called when
|
||||||
|
// the transaction is closed (either Commit or Rollback).
|
||||||
|
func (tx *Tx) AfterClose(f func(*Tx)) {
|
||||||
|
if tx.afterClose == nil {
|
||||||
|
tx.afterClose = f
|
||||||
|
} else {
|
||||||
|
prevFn := tx.afterClose
|
||||||
|
tx.afterClose = func(tx *Tx) {
|
||||||
|
f(tx)
|
||||||
|
prevFn(tx)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
36
tx_test.go
36
tx_test.go
|
@ -3,6 +3,7 @@ package pgx_test
|
||||||
import (
|
import (
|
||||||
"github.com/jackc/pgx"
|
"github.com/jackc/pgx"
|
||||||
"testing"
|
"testing"
|
||||||
|
"time"
|
||||||
)
|
)
|
||||||
|
|
||||||
func TestTransactionSuccessfulCommit(t *testing.T) {
|
func TestTransactionSuccessfulCommit(t *testing.T) {
|
||||||
|
@ -114,3 +115,38 @@ func TestBeginIso(t *testing.T) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func TestTxAfterClose(t *testing.T) {
|
||||||
|
t.Parallel()
|
||||||
|
|
||||||
|
conn := mustConnect(t, *defaultConnConfig)
|
||||||
|
defer closeConn(t, conn)
|
||||||
|
|
||||||
|
tx, err := conn.Begin()
|
||||||
|
if err != nil {
|
||||||
|
t.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
var zeroTime, t1, t2 time.Time
|
||||||
|
tx.AfterClose(func(tx *pgx.Tx) {
|
||||||
|
t1 = time.Now()
|
||||||
|
})
|
||||||
|
|
||||||
|
tx.AfterClose(func(tx *pgx.Tx) {
|
||||||
|
t2 = time.Now()
|
||||||
|
})
|
||||||
|
|
||||||
|
tx.Rollback()
|
||||||
|
|
||||||
|
if t1 == zeroTime {
|
||||||
|
t.Error("First Tx.AfterClose callback not called")
|
||||||
|
}
|
||||||
|
|
||||||
|
if t2 == zeroTime {
|
||||||
|
t.Error("Second Tx.AfterClose callback not called")
|
||||||
|
}
|
||||||
|
|
||||||
|
if t1.After(t2) {
|
||||||
|
t.Error("AfterClose callbacks called out of order: %v, %v", t1, t2)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
Loading…
Reference in New Issue