Added LastStmtSent and use it to retry on errors if statement was not sent
Previously, a failed connection could be put back in a pool and when the next query was attempted it would fail immediately trying to prepare the query or reset the deadline. It wasn't clear if the Query or Exec call could safely be retried since there was no way to know where it failed. You can now call LastQuerySent and if it returns false then you're guaranteed that the last call to Query(Ex)/Exec(Ex) didn't get far enough to attempt to send the query. The call can be retried with a new connection. This is used in the stdlib to return a ErrBadConn if a network error occurred and the statement was not attempted. Fixes #427
This commit is contained in:
@@ -4,6 +4,7 @@ import (
|
||||
"bytes"
|
||||
"context"
|
||||
"database/sql"
|
||||
"database/sql/driver"
|
||||
"encoding/json"
|
||||
"fmt"
|
||||
"math"
|
||||
@@ -989,6 +990,28 @@ func TestConnExecContextCancel(t *testing.T) {
|
||||
}
|
||||
}
|
||||
|
||||
func TestConnExecContextFailureRetry(t *testing.T) {
|
||||
db := openDB(t)
|
||||
defer closeDB(t, db)
|
||||
|
||||
// we get a connection, immediately close it, and then get it back
|
||||
{
|
||||
conn, err := stdlib.AcquireConn(db)
|
||||
if err != nil {
|
||||
t.Fatalf("stdlib.AcquireConn unexpectedly failed: %v", err)
|
||||
}
|
||||
conn.Close()
|
||||
stdlib.ReleaseConn(db, conn)
|
||||
}
|
||||
conn, err := db.Conn(context.Background())
|
||||
if err != nil {
|
||||
t.Fatalf("db.Conn unexpectedly failed: %v", err)
|
||||
}
|
||||
if _, err := conn.ExecContext(context.Background(), "select 1"); err != driver.ErrBadConn {
|
||||
t.Fatalf("Expected conn.ExecContext to return driver.ErrBadConn, but instead received: %v", err)
|
||||
}
|
||||
}
|
||||
|
||||
func TestConnQueryContextSuccess(t *testing.T) {
|
||||
db := openDB(t)
|
||||
defer closeDB(t, db)
|
||||
@@ -1083,6 +1106,28 @@ func TestConnQueryContextCancel(t *testing.T) {
|
||||
}
|
||||
}
|
||||
|
||||
func TestConnQueryContextFailureRetry(t *testing.T) {
|
||||
db := openDB(t)
|
||||
defer closeDB(t, db)
|
||||
|
||||
// we get a connection, immediately close it, and then get it back
|
||||
{
|
||||
conn, err := stdlib.AcquireConn(db)
|
||||
if err != nil {
|
||||
t.Fatalf("stdlib.AcquireConn unexpectedly failed: %v", err)
|
||||
}
|
||||
conn.Close()
|
||||
stdlib.ReleaseConn(db, conn)
|
||||
}
|
||||
conn, err := db.Conn(context.Background())
|
||||
if err != nil {
|
||||
t.Fatalf("db.Conn unexpectedly failed: %v", err)
|
||||
}
|
||||
if _, err := conn.QueryContext(context.Background(), "select 1"); err != driver.ErrBadConn {
|
||||
t.Fatalf("Expected conn.QueryContext to return driver.ErrBadConn, but instead received: %v", err)
|
||||
}
|
||||
}
|
||||
|
||||
func TestRowsColumnTypeDatabaseTypeName(t *testing.T) {
|
||||
db := openDB(t)
|
||||
defer closeDB(t, db)
|
||||
|
||||
Reference in New Issue
Block a user