Fix `echoParam` template function to accept string key name
authorTatsushi Demachi <tdemachi@gmail.com>
Wed, 7 Jan 2015 12:26:46 +0000 (21:26 +0900)
committerbep <bjorn.erik.pedersen@gmail.com>
Thu, 8 Jan 2015 01:32:17 +0000 (02:32 +0100)
This changes `echoParam` template function behavior to accept not only
an array or a slice and its index pair but also a map and its key pair.

This also changes the function that float and uint values are treated as
a valid result type of it.

Fix #771

tpl/template.go
tpl/template_test.go

index e8cdd405049ba5316af7e1caed2cbc13027d31fe..a051eba0b2b9ec95a906b960c61e02d2b93525b0 100644 (file)
@@ -785,20 +785,36 @@ func IsSet(a interface{}, key interface{}) bool {
        return false
 }
 
-func ReturnWhenSet(a interface{}, index int) interface{} {
-       av := reflect.ValueOf(a)
+func ReturnWhenSet(a, k interface{}) interface{} {
+       av, isNil := indirect(reflect.ValueOf(a))
+       if isNil {
+               return ""
+       }
 
+       var avv reflect.Value
        switch av.Kind() {
        case reflect.Array, reflect.Slice:
-               if av.Len() > index {
+               index, ok := k.(int)
+               if ok && av.Len() > index {
+                       avv = av.Index(index)
+               }
+       case reflect.Map:
+               kv := reflect.ValueOf(k)
+               if kv.Type().AssignableTo(av.Type().Key()) {
+                       avv = av.MapIndex(kv)
+               }
+       }
 
-                       avv := av.Index(index)
-                       switch avv.Kind() {
-                       case reflect.Int, reflect.Int8, reflect.Int16, reflect.Int32, reflect.Int64:
-                               return avv.Int()
-                       case reflect.String:
-                               return avv.String()
-                       }
+       if avv.IsValid() {
+               switch avv.Kind() {
+               case reflect.Int, reflect.Int8, reflect.Int16, reflect.Int32, reflect.Int64:
+                       return avv.Int()
+               case reflect.Uint, reflect.Uint8, reflect.Uint16, reflect.Uint32, reflect.Uint64:
+                       return avv.Uint()
+               case reflect.Float32, reflect.Float64:
+                       return avv.Float()
+               case reflect.String:
+                       return avv.String()
                }
        }
 
index 578d1d884c67d24e63c9e9e3e4e9dedbc0b6869f..98cf2d061dc119045b938f02ba663fc2aaf170fc 100644 (file)
@@ -791,6 +791,31 @@ func TestSort(t *testing.T) {
        }
 }
 
+func TestReturnWhenSet(t *testing.T) {
+       for i, this := range []struct {
+               data   interface{}
+               key    interface{}
+               expect interface{}
+       }{
+               {[]int{1, 2, 3}, 1, int64(2)},
+               {[]uint{1, 2, 3}, 1, uint64(2)},
+               {[]float64{1.1, 2.2, 3.3}, 1, float64(2.2)},
+               {[]string{"foo", "bar", "baz"}, 1, "bar"},
+               {[]TstX{TstX{A: "a", B: "b"}, TstX{A: "c", B: "d"}, TstX{A: "e", B: "f"}}, 1, ""},
+               {map[string]int{"foo": 1, "bar": 2, "baz": 3}, "bar", int64(2)},
+               {map[string]uint{"foo": 1, "bar": 2, "baz": 3}, "bar", uint64(2)},
+               {map[string]float64{"foo": 1.1, "bar": 2.2, "baz": 3.3}, "bar", float64(2.2)},
+               {map[string]string{"foo": "FOO", "bar": "BAR", "baz": "BAZ"}, "bar", "BAR"},
+               {map[string]TstX{"foo": TstX{A: "a", B: "b"}, "bar": TstX{A: "c", B: "d"}, "baz": TstX{A: "e", B: "f"}}, "bar", ""},
+               {(*[]string)(nil), "bar", ""},
+       } {
+               result := ReturnWhenSet(this.data, this.key)
+               if !reflect.DeepEqual(result, this.expect) {
+                       t.Errorf("[%d] ReturnWhenSet got %v (type %v) but expected %v (type %v)", i, result, reflect.TypeOf(result), this.expect, reflect.TypeOf(this.expect))
+               }
+       }
+}
+
 func TestMarkdownify(t *testing.T) {
 
        result := Markdownify("Hello **World!**")