Skip to content

Commit 659270e

Browse files
authored
Merge pull request #214 from laushunyu/master
fix: failed valuer convert return unexpected true
2 parents 60a1fd2 + 71552f3 commit 659270e

File tree

2 files changed

+83
-9
lines changed

2 files changed

+83
-9
lines changed

copier.go

+22-9
Original file line numberDiff line numberDiff line change
@@ -610,9 +610,14 @@ func set(to, from reflect.Value, deepCopy bool, converters map[converterPair]Typ
610610
}
611611
}
612612

613+
// try convert directly
613614
if from.Type().ConvertibleTo(to.Type()) {
614615
to.Set(from.Convert(to.Type()))
615-
} else if toScanner, ok := to.Addr().Interface().(sql.Scanner); ok {
616+
return true, nil
617+
}
618+
619+
// try Scanner
620+
if toScanner, ok := to.Addr().Interface().(sql.Scanner); ok {
616621
// `from` -> `to`
617622
// *string -> sql.NullString
618623
if from.Kind() == reflect.Ptr {
@@ -627,10 +632,13 @@ func set(to, from reflect.Value, deepCopy bool, converters map[converterPair]Typ
627632
// string -> sql.NullString
628633
// set `to` by invoking method Scan(`from`)
629634
err := toScanner.Scan(from.Interface())
630-
if err != nil {
631-
return false, nil
635+
if err == nil {
636+
return true, nil
632637
}
633-
} else if fromValuer, ok := driverValuer(from); ok {
638+
}
639+
640+
// try Valuer
641+
if fromValuer, ok := driverValuer(from); ok {
634642
// `from` -> `to`
635643
// sql.NullString -> string
636644
v, err := fromValuer.Value()
@@ -644,16 +652,21 @@ func set(to, from reflect.Value, deepCopy bool, converters map[converterPair]Typ
644652
rv := reflect.ValueOf(v)
645653
if rv.Type().AssignableTo(to.Type()) {
646654
to.Set(rv)
647-
} else if to.CanSet() && rv.Type().ConvertibleTo(to.Type()) {
655+
return true, nil
656+
}
657+
if to.CanSet() && rv.Type().ConvertibleTo(to.Type()) {
648658
to.Set(rv.Convert(to.Type()))
659+
return true, nil
649660
}
650-
} else if from.Kind() == reflect.Ptr {
651-
return set(to, from.Elem(), deepCopy, converters)
652-
} else {
653661
return false, nil
654662
}
655663

656-
return true, nil
664+
// from is ptr
665+
if from.Kind() == reflect.Ptr {
666+
return set(to, from.Elem(), deepCopy, converters)
667+
}
668+
669+
return false, nil
657670
}
658671

659672
// lookupAndCopyWithConverter looks up the type pair, on success the TypeConverter Fn func is called to copy src to dst field.

copier_converter_test.go

+61
Original file line numberDiff line numberDiff line change
@@ -1,7 +1,11 @@
11
package copier_test
22

33
import (
4+
"bytes"
5+
"database/sql/driver"
6+
"encoding/json"
47
"errors"
8+
"reflect"
59
"strconv"
610
"testing"
711
"time"
@@ -218,3 +222,60 @@ func TestCopyWithConverterRaisingError(t *testing.T) {
218222
return
219223
}
220224
}
225+
226+
type IntArray []int
227+
228+
func (a IntArray) Value() (driver.Value, error) {
229+
return json.Marshal(a)
230+
}
231+
232+
type Int int
233+
234+
type From struct {
235+
Data IntArray
236+
}
237+
238+
type To struct {
239+
Data []byte
240+
}
241+
242+
type FailedTo struct {
243+
Data []Int
244+
}
245+
246+
func TestValuerConv(t *testing.T) {
247+
// when the field of struct implement driver.Valuer and cannot convert to dest type directly,
248+
// copier.set() will return a unexpected (true, nil)
249+
250+
typ1 := reflect.TypeOf(IntArray{})
251+
typ2 := reflect.TypeOf([]Int{})
252+
253+
if typ1 == typ2 || typ1.ConvertibleTo(typ2) || typ1.AssignableTo(typ2) {
254+
// in 1.22 and older, u can not convert typ1 to typ2
255+
t.Errorf("can not convert %v to %v direct", typ1, typ2)
256+
}
257+
258+
var (
259+
from = From{
260+
Data: IntArray{1, 2, 3},
261+
}
262+
to To
263+
failedTo FailedTo
264+
)
265+
if err := copier.Copy(&to, from); err != nil {
266+
t.Fatal(err)
267+
}
268+
if err := copier.Copy(&failedTo, from); err != nil {
269+
t.Fatal(err)
270+
}
271+
272+
// Testcase1: valuer conv case
273+
if !bytes.Equal(to.Data, []byte(`[1,2,3]`)) {
274+
t.Errorf("can not convert %v to %v using valuer", typ1, typ2)
275+
}
276+
277+
// Testcase2: fallback case when valuer conv failed
278+
if len(failedTo.Data) != 3 || failedTo.Data[0] != 1 || failedTo.Data[1] != 2 || failedTo.Data[2] != 3 {
279+
t.Errorf("copier failed from %#v to %#v", from, failedTo)
280+
}
281+
}

0 commit comments

Comments
 (0)