mirror of https://github.com/jackc/pgx.git
Merge pull request #2225 from logicbomb/improve-error-message
Include the field name in error messages when scanning structspull/2236/head
commit
0bc29e3000
|
@ -6,6 +6,7 @@ import (
|
||||||
"database/sql/driver"
|
"database/sql/driver"
|
||||||
"encoding/json"
|
"encoding/json"
|
||||||
"errors"
|
"errors"
|
||||||
|
"fmt"
|
||||||
"reflect"
|
"reflect"
|
||||||
"testing"
|
"testing"
|
||||||
|
|
||||||
|
@ -191,11 +192,15 @@ func TestJSONCodecUnmarshalSQLNull(t *testing.T) {
|
||||||
// A string cannot scan a NULL.
|
// A string cannot scan a NULL.
|
||||||
str := "foobar"
|
str := "foobar"
|
||||||
err = conn.QueryRow(ctx, "select null::json").Scan(&str)
|
err = conn.QueryRow(ctx, "select null::json").Scan(&str)
|
||||||
require.EqualError(t, err, "can't scan into dest[0]: cannot scan NULL into *string")
|
fieldName := "json"
|
||||||
|
if conn.PgConn().ParameterStatus("crdb_version") != "" {
|
||||||
|
fieldName = "jsonb" // Seems like CockroachDB treats json as jsonb.
|
||||||
|
}
|
||||||
|
require.EqualError(t, err, fmt.Sprintf("can't scan into dest[0] (col: %s): cannot scan NULL into *string", fieldName))
|
||||||
|
|
||||||
// A non-string cannot scan a NULL.
|
// A non-string cannot scan a NULL.
|
||||||
err = conn.QueryRow(ctx, "select null::json").Scan(&n)
|
err = conn.QueryRow(ctx, "select null::json").Scan(&n)
|
||||||
require.EqualError(t, err, "can't scan into dest[0]: cannot scan NULL into *int")
|
require.EqualError(t, err, fmt.Sprintf("can't scan into dest[0] (col: %s): cannot scan NULL into *int", fieldName))
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -66,11 +66,11 @@ func TestJSONBCodecUnmarshalSQLNull(t *testing.T) {
|
||||||
// A string cannot scan a NULL.
|
// A string cannot scan a NULL.
|
||||||
str := "foobar"
|
str := "foobar"
|
||||||
err = conn.QueryRow(ctx, "select null::jsonb").Scan(&str)
|
err = conn.QueryRow(ctx, "select null::jsonb").Scan(&str)
|
||||||
require.EqualError(t, err, "can't scan into dest[0]: cannot scan NULL into *string")
|
require.EqualError(t, err, "can't scan into dest[0] (col: jsonb): cannot scan NULL into *string")
|
||||||
|
|
||||||
// A non-string cannot scan a NULL.
|
// A non-string cannot scan a NULL.
|
||||||
err = conn.QueryRow(ctx, "select null::jsonb").Scan(&n)
|
err = conn.QueryRow(ctx, "select null::jsonb").Scan(&n)
|
||||||
require.EqualError(t, err, "can't scan into dest[0]: cannot scan NULL into *int")
|
require.EqualError(t, err, "can't scan into dest[0] (col: jsonb): cannot scan NULL into *int")
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -79,7 +79,7 @@ func TestXMLCodecUnmarshalSQLNull(t *testing.T) {
|
||||||
// A string cannot scan a NULL.
|
// A string cannot scan a NULL.
|
||||||
str := "foobar"
|
str := "foobar"
|
||||||
err = conn.QueryRow(ctx, "select null::xml").Scan(&str)
|
err = conn.QueryRow(ctx, "select null::xml").Scan(&str)
|
||||||
assert.EqualError(t, err, "can't scan into dest[0]: cannot scan NULL into *string")
|
assert.EqualError(t, err, "can't scan into dest[0] (col: xml): cannot scan NULL into *string")
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -420,7 +420,7 @@ func TestConnQueryReadWrongTypeError(t *testing.T) {
|
||||||
t.Fatal("Expected Rows to have an error after an improper read but it didn't")
|
t.Fatal("Expected Rows to have an error after an improper read but it didn't")
|
||||||
}
|
}
|
||||||
|
|
||||||
if rows.Err().Error() != "can't scan into dest[0]: cannot scan int4 (OID 23) in binary format into *time.Time" {
|
if rows.Err().Error() != "can't scan into dest[0] (col: n): cannot scan int4 (OID 23) in binary format into *time.Time" {
|
||||||
t.Fatalf("Expected different Rows.Err(): %v", rows.Err())
|
t.Fatalf("Expected different Rows.Err(): %v", rows.Err())
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
11
rows.go
11
rows.go
|
@ -272,7 +272,7 @@ func (rows *baseRows) Scan(dest ...any) error {
|
||||||
|
|
||||||
err := rows.scanPlans[i].Scan(values[i], dst)
|
err := rows.scanPlans[i].Scan(values[i], dst)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
err = ScanArgError{ColumnIndex: i, Err: err}
|
err = ScanArgError{ColumnIndex: i, FieldName: fieldDescriptions[i].Name, Err: err}
|
||||||
rows.fatal(err)
|
rows.fatal(err)
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
@ -334,11 +334,16 @@ func (rows *baseRows) Conn() *Conn {
|
||||||
|
|
||||||
type ScanArgError struct {
|
type ScanArgError struct {
|
||||||
ColumnIndex int
|
ColumnIndex int
|
||||||
|
FieldName string
|
||||||
Err error
|
Err error
|
||||||
}
|
}
|
||||||
|
|
||||||
func (e ScanArgError) Error() string {
|
func (e ScanArgError) Error() string {
|
||||||
return fmt.Sprintf("can't scan into dest[%d]: %v", e.ColumnIndex, e.Err)
|
if e.FieldName == "?column?" { // Don't include the fieldname if it's unknown
|
||||||
|
return fmt.Sprintf("can't scan into dest[%d]: %v", e.ColumnIndex, e.Err)
|
||||||
|
}
|
||||||
|
|
||||||
|
return fmt.Sprintf("can't scan into dest[%d] (col: %s): %v", e.ColumnIndex, e.FieldName, e.Err)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (e ScanArgError) Unwrap() error {
|
func (e ScanArgError) Unwrap() error {
|
||||||
|
@ -366,7 +371,7 @@ func ScanRow(typeMap *pgtype.Map, fieldDescriptions []pgconn.FieldDescription, v
|
||||||
|
|
||||||
err := typeMap.Scan(fieldDescriptions[i].DataTypeOID, fieldDescriptions[i].Format, values[i], d)
|
err := typeMap.Scan(fieldDescriptions[i].DataTypeOID, fieldDescriptions[i].Format, values[i], d)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return ScanArgError{ColumnIndex: i, Err: err}
|
return ScanArgError{ColumnIndex: i, FieldName: fieldDescriptions[i].Name, Err: err}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -3,6 +3,7 @@ package pgx_test
|
||||||
import (
|
import (
|
||||||
"bytes"
|
"bytes"
|
||||||
"context"
|
"context"
|
||||||
|
"fmt"
|
||||||
"net"
|
"net"
|
||||||
"os"
|
"os"
|
||||||
"reflect"
|
"reflect"
|
||||||
|
@ -215,7 +216,12 @@ func testJSONInt16ArrayFailureDueToOverflow(t *testing.T, conn *pgx.Conn, typena
|
||||||
input := []int{1, 2, 234432}
|
input := []int{1, 2, 234432}
|
||||||
var output []int16
|
var output []int16
|
||||||
err := conn.QueryRow(context.Background(), "select $1::"+typename, input).Scan(&output)
|
err := conn.QueryRow(context.Background(), "select $1::"+typename, input).Scan(&output)
|
||||||
if err == nil || err.Error() != "can't scan into dest[0]: json: cannot unmarshal number 234432 into Go value of type int16" {
|
fieldName := typename
|
||||||
|
if conn.PgConn().ParameterStatus("crdb_version") != "" && typename == "json" {
|
||||||
|
fieldName = "jsonb" // Seems like CockroachDB treats json as jsonb.
|
||||||
|
}
|
||||||
|
expectedMessage := fmt.Sprintf("can't scan into dest[0] (col: %s): json: cannot unmarshal number 234432 into Go value of type int16", fieldName)
|
||||||
|
if err == nil || err.Error() != expectedMessage {
|
||||||
t.Errorf("%s: Expected *json.UnmarshalTypeError, but got %v", typename, err)
|
t.Errorf("%s: Expected *json.UnmarshalTypeError, but got %v", typename, err)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in New Issue