Compare commits

..

No commits in common. "a0bad4d6e955f824624a16984d545fdae7c455c5" and "b3320886380a17e782dd7e4e5454469356e1a181" have entirely different histories.

2 changed files with 14 additions and 175 deletions

View File

@ -5,48 +5,7 @@ import (
)
func Walk(x interface{}, fn func(string)) {
val := getValue(x)
walkValue := func(value reflect.Value) {
Walk(value.Interface(), fn)
}
switch val.Kind() {
case reflect.String:
fn(val.String())
case reflect.Struct:
for i := 0; i < val.NumField(); i++ {
// XXX: Interface() to get interface
walkValue(val.Field(i))
}
case reflect.Slice, reflect.Array:
for i := 0; i < val.Len(); i++ {
walkValue(val.Index(i))
}
case reflect.Map:
for _, key := range val.MapKeys() {
walkValue(val.MapIndex(key))
}
case reflect.Chan:
for {
if v, ok := val.Recv(); ok {
walkValue(v)
} else {
break
}
}
case reflect.Func:
valFnResult := val.Call(nil)
for _, res := range valFnResult {
walkValue(res)
}
}
}
func getValue(x interface{}) reflect.Value {
val := reflect.ValueOf(x)
if val.Kind() == reflect.Pointer {
return val.Elem()
}
return val
field := val.Field(0)
fn(field.String())
}

View File

@ -1,147 +1,27 @@
package walk
import (
"reflect"
"testing"
)
type Profile struct {
Age int
City string
}
type Person struct {
Name string
Profile Profile
}
func TestWalk(t *testing.T) {
cases := []struct {
Name string
Input interface{}
ExpectedCalls []string
}{
{
"struct with one string field",
struct {
Name string
}{"Chris"},
[]string{"Chris"},
},
{
"struct with two string fields",
struct {
Name string
City string
}{"Chris", "London"},
[]string{"Chris", "London"},
},
{
"struct with non string field",
struct {
Name string
Age int
}{"Chris", 29},
[]string{"Chris"},
},
{
"nested fields",
Person{"Chris", Profile{29, "London"}},
[]string{"Chris", "London"},
},
{
"pointer to things",
&Person{"Chris", Profile{29, "London"}},
[]string{"Chris", "London"},
},
{
"slices",
[]Profile{
{29, "London"},
{33, "Paris"},
},
[]string{"London", "Paris"},
},
{
"arrays",
[2]Profile{
{29, "London"},
{33, "Paris"},
},
[]string{"London", "Paris"},
},
}
for _, test := range cases {
t.Run(test.Name, func(t *testing.T) {
var got []string
Walk(test.Input, func(input string) {
got = append(got, input)
})
if !reflect.DeepEqual(got, test.ExpectedCalls) {
t.Errorf("got %v want %v", got, test.ExpectedCalls)
}
})
}
t.Run("with maps", func(t *testing.T) {
aMap := map[string]string{
"Cow": "meuh",
"Sheep": "meh",
}
t.Run("walk function test", func(t *testing.T) {
expected := "Chris"
var got []string
Walk(aMap, func(input string) {
x := struct {
Name string
}{expected}
Walk(x, func(input string) {
got = append(got, input)
})
assertContains(t, got, "meuh")
assertContains(t, got, "meh")
})
t.Run("with channels", func(t *testing.T) {
aChannel := make(chan Profile)
go func() {
aChannel <- Profile{21, "Berlin"}
aChannel <- Profile{28, "Beijing"}
close(aChannel)
}()
var got []string
want := []string{"Berlin", "Beijing"}
Walk(aChannel, func(input string) {
got = append(got, input)
})
if !reflect.DeepEqual(got, want) {
t.Errorf("got %v want %v", got, want)
if len(got) != 1 {
t.Errorf("wrong number of function calls, got %d want %d", len(got), 1)
}
})
t.Run("with functions", func(t *testing.T) {
aFunction := func() (Profile, Profile) {
return Profile{21, "Berlin"}, Profile{28, "Beijing"}
}
var got []string
want := []string{"Berlin", "Beijing"}
Walk(aFunction, func(input string) {
got = append(got, input)
})
if !reflect.DeepEqual(got, want) {
t.Errorf("got %v want %v", got, want)
if got[0] != expected {
t.Errorf("got %q want %q", got[0], expected)
}
})
}
func assertContains(t testing.TB, haystack []string, needle string) {
t.Helper()
contains := false
for _, x := range haystack {
if x == needle {
contains = true
}
}
if !contains {
t.Errorf("expected %v to contain %q but it didn't", haystack, needle)
}
}