tplimpl: return an error on unsupported type in isSet
authorCameron Moore <moorereason@gmail.com>
Fri, 24 Feb 2017 04:35:20 +0000 (22:35 -0600)
committerBjørn Erik Pedersen <bjorn.erik.pedersen@gmail.com>
Wed, 1 Mar 2017 23:11:10 +0000 (00:11 +0100)
Fixes #3092

tpl/tplimpl/template_funcs.go
tpl/tplimpl/template_funcs_test.go

index 20f46a4e2a9028e7b0ecf8f77efe4871a238f9a8..d4eec5b2f4471ebd76777c91d83f3ac997817d3f 100644 (file)
@@ -1276,22 +1276,24 @@ func (p pairList) sort() interface{} {
 
 // isSet returns whether a given array, channel, slice, or map has a key
 // defined.
-func isSet(a interface{}, key interface{}) bool {
+func isSet(a interface{}, key interface{}) (bool, error) {
        av := reflect.ValueOf(a)
        kv := reflect.ValueOf(key)
 
        switch av.Kind() {
        case reflect.Array, reflect.Chan, reflect.Slice:
                if int64(av.Len()) > kv.Int() {
-                       return true
+                       return true, nil
                }
        case reflect.Map:
                if kv.Type() == av.Type().Key() {
-                       return av.MapIndex(kv).IsValid()
+                       return av.MapIndex(kv).IsValid(), nil
                }
+       default:
+               return false, fmt.Errorf("unsupported type %q", av.Kind())
        }
 
-       return false
+       return false, nil
 }
 
 // returnWhenSet returns a given value if it set.  Otherwise, it returns an
index db7533dd35f3632b276f00d80b0adca84cea01d2..942e4e4094d773eb85bc3c9191f86ef69441c4d0 100644 (file)
@@ -1017,13 +1017,31 @@ func TestIntersect(t *testing.T) {
 
 func TestIsSet(t *testing.T) {
        t.Parallel()
-       aSlice := []interface{}{1, 2, 3, 5}
-       aMap := map[string]interface{}{"a": 1, "b": 2}
 
-       assert.True(t, isSet(aSlice, 2))
-       assert.True(t, isSet(aMap, "b"))
-       assert.False(t, isSet(aSlice, 22))
-       assert.False(t, isSet(aMap, "bc"))
+       for _, test := range []struct {
+               src    interface{}
+               key    interface{}
+               res    bool
+               isErr  bool
+               errStr string
+       }{
+               {[]interface{}{1, 2, 3, 5}, 2, true, false, ""},
+               {[]interface{}{1, 2, 3, 5}, 22, false, false, ""},
+
+               {map[string]interface{}{"a": 1, "b": 2}, "b", true, false, ""},
+               {map[string]interface{}{"a": 1, "b": 2}, "bc", false, false, ""},
+
+               {time.Now(), 1, false, true, `unsupported type "struct"`},
+       } {
+               res, err := isSet(test.src, test.key)
+               if test.isErr {
+                       assert.EqualError(t, err, test.errStr)
+                       continue
+               }
+
+               assert.NoError(t, err)
+               assert.Equal(t, test.res, res)
+       }
 }
 
 func (x *TstX) TstRp() string {