Skip to content

Commit 57ce6c2

Browse files
authored
Add a WithMapVisitor option. (#28)
1 parent 35e0dbd commit 57ce6c2

File tree

2 files changed

+88
-0
lines changed

2 files changed

+88
-0
lines changed

copy.go

+21
Original file line numberDiff line numberDiff line change
@@ -240,8 +240,19 @@ type options struct {
240240

241241
// CopyListSize can control the number of elements copied from src depending on src's Value
242242
CopyListSize func(src *reflect.Value) int
243+
244+
// MapVisitor is called for every filtered field in structToMap.
245+
//
246+
// It is called before copying the data from source to destination allowing custom processing.
247+
// If the visitor function returns true the visited field is skipped.
248+
MapVisitor mapVisitor
243249
}
244250

251+
// mapVisitor is called for every filtered field in structToMap.
252+
type mapVisitor func(
253+
filter FieldFilter, src interface{}, dst map[string]interface{},
254+
srcFieldName, dstFieldName string, srcFieldValue reflect.Value) (skipToNext bool)
255+
245256
// Option function modifies the given options.
246257
type Option func(*options)
247258

@@ -259,6 +270,13 @@ func WithCopyListSize(f func(src *reflect.Value) int) Option {
259270
}
260271
}
261272

273+
// WithMapVisitor sets the fields visitor function for StructToMap.
274+
func WithMapVisitor(visitor mapVisitor) Option {
275+
return func(o *options) {
276+
o.MapVisitor = visitor
277+
}
278+
}
279+
262280
func newDefaultOptions() *options {
263281
// set default CopyListSize is func which return src.Len()
264282
return &options{CopyListSize: func(src *reflect.Value) int { return src.Len() }}
@@ -306,6 +324,9 @@ func structToMap(filter FieldFilter, src interface{}, dst map[string]interface{}
306324
}
307325

308326
dstName := dstKey(userOptions.DstTag, srcType.Field(i))
327+
if userOptions.MapVisitor != nil && userOptions.MapVisitor(filter, src, dst, fieldName, dstName, srcField) {
328+
continue
329+
}
309330

310331
switch srcField.Kind() {
311332
case reflect.Ptr, reflect.Interface:

copy_test.go

+67
Original file line numberDiff line numberDiff line change
@@ -4,6 +4,7 @@ import (
44
"fmt"
55
"reflect"
66
"testing"
7+
"time"
78

89
"github.com/stretchr/testify/assert"
910
"github.com/stretchr/testify/require"
@@ -1790,6 +1791,72 @@ func TestStructToMap_CopyIntArray_WithMaxCopyListSize(t *testing.T) {
17901791
}, dst)
17911792
}
17921793

1794+
func TestStructToMap_CopyStructWithPrivateFields_WithMapVisitor(t *testing.T) {
1795+
type A struct {
1796+
Time time.Time
1797+
Other int
1798+
}
1799+
unixTime := time.Unix(10, 10)
1800+
src := &A{Time: unixTime}
1801+
dst := map[string]interface{}{}
1802+
mask := fieldmask_utils.MaskFromString("Time")
1803+
err := fieldmask_utils.StructToMap(mask, src, dst, fieldmask_utils.WithMapVisitor(
1804+
func(_ fieldmask_utils.FieldFilter, _ interface{}, dst map[string]interface{},
1805+
srcFieldName, dstFieldName string, srcFieldValue reflect.Value) (skipToNext bool) {
1806+
if srcFieldName == "Time" {
1807+
dst[dstFieldName] = srcFieldValue.Interface()
1808+
skipToNext = true
1809+
}
1810+
return
1811+
}))
1812+
require.NoError(t, err)
1813+
assert.Equal(t, map[string]interface{}{
1814+
"Time": unixTime,
1815+
}, dst)
1816+
}
1817+
1818+
func TestStructToMap_MapVisitorVisitsOnlyFilteredFields(t *testing.T) {
1819+
type A struct {
1820+
Field1 int
1821+
Field2 string
1822+
Field3 int
1823+
}
1824+
src := &A{Field1: 42, Field2: "hello", Field3: 44}
1825+
dst := map[string]interface{}{}
1826+
mask := fieldmask_utils.MaskFromString("Field1, Field2")
1827+
var visitedFields []string
1828+
err := fieldmask_utils.StructToMap(mask, src, dst, fieldmask_utils.WithMapVisitor(
1829+
func(_ fieldmask_utils.FieldFilter, _ interface{}, _ map[string]interface{},
1830+
srcFieldName, _ string, _ reflect.Value) (skipToNext bool) {
1831+
visitedFields = append(visitedFields, srcFieldName)
1832+
return
1833+
}))
1834+
require.NoError(t, err)
1835+
assert.Equal(t, visitedFields, []string{"Field1", "Field2"})
1836+
}
1837+
1838+
func TestStructToMap_WithMapVisitor_SkipsToNextField(t *testing.T) {
1839+
type A struct {
1840+
Field1 int
1841+
Field2 string
1842+
Field3 int
1843+
}
1844+
src := &A{Field1: 42, Field2: "hello", Field3: 44}
1845+
dst := map[string]interface{}{}
1846+
mask := fieldmask_utils.MaskFromString("Field1, Field2")
1847+
err := fieldmask_utils.StructToMap(mask, src, dst, fieldmask_utils.WithMapVisitor(
1848+
func(_ fieldmask_utils.FieldFilter, _ interface{}, _ map[string]interface{},
1849+
srcFieldName, dstFieldName string, _ reflect.Value) (skipToNext bool) {
1850+
if srcFieldName == "Field1" {
1851+
dst[dstFieldName] = 33
1852+
skipToNext = true
1853+
}
1854+
return
1855+
}))
1856+
require.NoError(t, err)
1857+
assert.Equal(t, map[string]interface{}{"Field1": 33, "Field2": "hello"}, dst)
1858+
}
1859+
17931860
func TestStructToStruct_CopySlice_WithDiffentItemKind(t *testing.T) {
17941861
type A struct {
17951862
Field1 []int

0 commit comments

Comments
 (0)