mirror of https://github.com/jackc/pgx.git
rework JSONCodec.PlanScan
parent
0bc29e3000
commit
a5353af354
|
@ -71,6 +71,27 @@ func (c *JSONCodec) PlanEncode(m *Map, oid uint32, format int16, value any) Enco
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// JSON needs its on scan plan for pointers to handle 'null'::json(b).
|
||||||
|
// Consider making pointerPointerScanPlan more flexible in the future.
|
||||||
|
type jsonPointerScanPlan struct {
|
||||||
|
next ScanPlan
|
||||||
|
}
|
||||||
|
|
||||||
|
func (p jsonPointerScanPlan) Scan(src []byte, dst any) error {
|
||||||
|
el := reflect.ValueOf(dst).Elem()
|
||||||
|
if src == nil || string(src) == "null" {
|
||||||
|
el.SetZero()
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
el.Set(reflect.New(el.Type().Elem()))
|
||||||
|
if p.next != nil {
|
||||||
|
return p.next.Scan(src, el.Interface())
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
type encodePlanJSONCodecEitherFormatString struct{}
|
type encodePlanJSONCodecEitherFormatString struct{}
|
||||||
|
|
||||||
func (encodePlanJSONCodecEitherFormatString) Encode(value any, buf []byte) (newBuf []byte, err error) {
|
func (encodePlanJSONCodecEitherFormatString) Encode(value any, buf []byte) (newBuf []byte, err error) {
|
||||||
|
@ -117,62 +138,36 @@ func (e *encodePlanJSONCodecEitherFormatMarshal) Encode(value any, buf []byte) (
|
||||||
return buf, nil
|
return buf, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (c *JSONCodec) PlanScan(m *Map, oid uint32, format int16, target any) ScanPlan {
|
func (c *JSONCodec) PlanScan(m *Map, oid uint32, formatCode int16, target any) ScanPlan {
|
||||||
switch target.(type) {
|
return c.planScan(m, oid, formatCode, target, 0)
|
||||||
case *string:
|
|
||||||
return scanPlanAnyToString{}
|
|
||||||
|
|
||||||
case **string:
|
|
||||||
// This is to fix **string scanning. It seems wrong to special case **string, but it's not clear what a better
|
|
||||||
// solution would be.
|
|
||||||
//
|
|
||||||
// https://github.com/jackc/pgx/issues/1470 -- **string
|
|
||||||
// https://github.com/jackc/pgx/issues/1691 -- ** anything else
|
|
||||||
|
|
||||||
if wrapperPlan, nextDst, ok := TryPointerPointerScanPlan(target); ok {
|
|
||||||
if nextPlan := m.planScan(oid, format, nextDst, 0); nextPlan != nil {
|
|
||||||
if _, failed := nextPlan.(*scanPlanFail); !failed {
|
|
||||||
wrapperPlan.SetNext(nextPlan)
|
|
||||||
return wrapperPlan
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
case *[]byte:
|
|
||||||
return scanPlanJSONToByteSlice{}
|
|
||||||
case BytesScanner:
|
|
||||||
return scanPlanBinaryBytesToBytesScanner{}
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
// Cannot rely on sql.Scanner being handled later because scanPlanJSONToJSONUnmarshal will take precedence.
|
|
||||||
//
|
|
||||||
// https://github.com/jackc/pgx/issues/1418
|
|
||||||
if isSQLScanner(target) {
|
|
||||||
return &scanPlanSQLScanner{formatCode: format}
|
|
||||||
}
|
|
||||||
|
|
||||||
return &scanPlanJSONToJSONUnmarshal{
|
|
||||||
unmarshal: c.Unmarshal,
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// we need to check if the target is a pointer to a sql.Scanner (or any of the pointer ref tree implements a sql.Scanner).
|
// JSON cannot fallback to pointerPointerScanPlan because of 'null'::json(b),
|
||||||
//
|
// so we need to duplicate the logic here.
|
||||||
// https://github.com/jackc/pgx/issues/2146
|
func (c *JSONCodec) planScan(m *Map, oid uint32, formatCode int16, target any, depth int) ScanPlan {
|
||||||
func isSQLScanner(v any) bool {
|
if depth > 8 {
|
||||||
if _, is := v.(sql.Scanner); is {
|
return &scanPlanFail{m: m, oid: oid, formatCode: formatCode}
|
||||||
return true
|
|
||||||
}
|
}
|
||||||
|
|
||||||
val := reflect.ValueOf(v)
|
switch target.(type) {
|
||||||
for val.Kind() == reflect.Ptr {
|
case *string:
|
||||||
if _, ok := val.Interface().(sql.Scanner); ok {
|
return &scanPlanAnyToString{}
|
||||||
return true
|
case *[]byte:
|
||||||
}
|
return &scanPlanJSONToByteSlice{}
|
||||||
val = val.Elem()
|
case BytesScanner:
|
||||||
|
return &scanPlanBinaryBytesToBytesScanner{}
|
||||||
|
case sql.Scanner:
|
||||||
|
return &scanPlanSQLScanner{formatCode: formatCode}
|
||||||
|
}
|
||||||
|
|
||||||
|
rv := reflect.ValueOf(target)
|
||||||
|
if rv.Kind() == reflect.Pointer && rv.Elem().Kind() == reflect.Pointer {
|
||||||
|
var plan jsonPointerScanPlan
|
||||||
|
plan.next = c.planScan(m, oid, formatCode, rv.Elem().Interface(), depth+1)
|
||||||
|
return plan
|
||||||
|
} else {
|
||||||
|
return &scanPlanJSONToJSONUnmarshal{unmarshal: c.Unmarshal}
|
||||||
}
|
}
|
||||||
return false
|
|
||||||
}
|
}
|
||||||
|
|
||||||
type scanPlanAnyToString struct{}
|
type scanPlanAnyToString struct{}
|
||||||
|
@ -202,7 +197,7 @@ type scanPlanJSONToJSONUnmarshal struct {
|
||||||
}
|
}
|
||||||
|
|
||||||
func (s *scanPlanJSONToJSONUnmarshal) Scan(src []byte, dst any) error {
|
func (s *scanPlanJSONToJSONUnmarshal) Scan(src []byte, dst any) error {
|
||||||
if src == nil {
|
if src == nil || string(src) == "null" {
|
||||||
dstValue := reflect.ValueOf(dst)
|
dstValue := reflect.ValueOf(dst)
|
||||||
if dstValue.Kind() == reflect.Ptr {
|
if dstValue.Kind() == reflect.Ptr {
|
||||||
el := dstValue.Elem()
|
el := dstValue.Elem()
|
||||||
|
|
|
@ -326,3 +326,34 @@ func TestJSONCodecScanToNonPointerValues(t *testing.T) {
|
||||||
require.Equal(t, 42, m)
|
require.Equal(t, 42, m)
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func TestJSONCodecScanNull(t *testing.T) {
|
||||||
|
defaultConnTestRunner.RunTest(context.Background(), t, func(ctx context.Context, t testing.TB, conn *pgx.Conn) {
|
||||||
|
var dest struct{}
|
||||||
|
err := conn.QueryRow(ctx, "select null::jsonb").Scan(&dest)
|
||||||
|
require.Error(t, err)
|
||||||
|
require.Contains(t, err.Error(), "cannot scan NULL into *struct {}")
|
||||||
|
|
||||||
|
err = conn.QueryRow(ctx, "select 'null'::jsonb").Scan(&dest)
|
||||||
|
require.Error(t, err)
|
||||||
|
require.Contains(t, err.Error(), "cannot scan NULL into *struct {}")
|
||||||
|
|
||||||
|
var destPointer *struct{}
|
||||||
|
err = conn.QueryRow(ctx, "select null::jsonb").Scan(&destPointer)
|
||||||
|
require.NoError(t, err)
|
||||||
|
require.Nil(t, destPointer)
|
||||||
|
|
||||||
|
err = conn.QueryRow(ctx, "select 'null'::jsonb").Scan(&destPointer)
|
||||||
|
require.NoError(t, err)
|
||||||
|
require.Nil(t, destPointer)
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestJSONCodecScanNullToPointerToSQLScanner(t *testing.T) {
|
||||||
|
defaultConnTestRunner.RunTest(context.Background(), t, func(ctx context.Context, t testing.TB, conn *pgx.Conn) {
|
||||||
|
var dest *Issue2146
|
||||||
|
err := conn.QueryRow(ctx, "select null::jsonb").Scan(&dest)
|
||||||
|
require.NoError(t, err)
|
||||||
|
require.Nil(t, dest)
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
|
@ -396,11 +396,7 @@ type scanPlanSQLScanner struct {
|
||||||
}
|
}
|
||||||
|
|
||||||
func (plan *scanPlanSQLScanner) Scan(src []byte, dst any) error {
|
func (plan *scanPlanSQLScanner) Scan(src []byte, dst any) error {
|
||||||
scanner := getSQLScanner(dst)
|
scanner := dst.(sql.Scanner)
|
||||||
|
|
||||||
if scanner == nil {
|
|
||||||
return fmt.Errorf("cannot scan into %T", dst)
|
|
||||||
}
|
|
||||||
|
|
||||||
if src == nil {
|
if src == nil {
|
||||||
// This is necessary because interface value []byte:nil does not equal nil:nil for the binary format path and the
|
// This is necessary because interface value []byte:nil does not equal nil:nil for the binary format path and the
|
||||||
|
@ -413,25 +409,6 @@ func (plan *scanPlanSQLScanner) Scan(src []byte, dst any) error {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// we don't know if the target is a sql.Scanner or a pointer on a sql.Scanner, so we need to check recursively
|
|
||||||
func getSQLScanner(target any) sql.Scanner {
|
|
||||||
if sc, is := target.(sql.Scanner); is {
|
|
||||||
return sc
|
|
||||||
}
|
|
||||||
|
|
||||||
val := reflect.ValueOf(target)
|
|
||||||
for val.Kind() == reflect.Ptr {
|
|
||||||
if _, ok := val.Interface().(sql.Scanner); ok {
|
|
||||||
if val.IsNil() {
|
|
||||||
val.Set(reflect.New(val.Type().Elem()))
|
|
||||||
}
|
|
||||||
return val.Interface().(sql.Scanner)
|
|
||||||
}
|
|
||||||
val = val.Elem()
|
|
||||||
}
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
type scanPlanString struct{}
|
type scanPlanString struct{}
|
||||||
|
|
||||||
func (scanPlanString) Scan(src []byte, dst any) error {
|
func (scanPlanString) Scan(src []byte, dst any) error {
|
||||||
|
|
Loading…
Reference in New Issue