Add nil comparison to where tpl function
authorTatsushi Demachi <tdemachi@gmail.com>
Sat, 27 Jun 2015 17:15:42 +0000 (02:15 +0900)
committerBjørn Erik Pedersen <bjorn.erik.pedersen@gmail.com>
Mon, 29 Jun 2015 17:50:45 +0000 (19:50 +0200)
`where` template function's internal condition check function always
returns `false` when a target value doesn't exist or it's nil value but
this behavior makes it difficult to filter values which doesn't have a
particular parameter.

To solve it, this adds nil value comparison to the function.
`where Values ".Param.key" nil` like clause can be used for the case
above.

Only "=", "==", "eq", "!=", "<>", "ne" operators are allowed to be used
with `nil`. If an other operator is passed with `nil`, the condition
check function returns `false` like before.

Fix #1232

tpl/template_funcs.go
tpl/template_funcs_test.go

index 7849aaae168a1941770b5af2673bbdf99f0cd5d4..258591314d4f6e09bf9e1039b37a66514b5364cb 100644 (file)
@@ -536,17 +536,21 @@ func evaluateSubElem(obj reflect.Value, elemName string) (reflect.Value, error)
 }
 
 func checkCondition(v, mv reflect.Value, op string) (bool, error) {
-       if !v.IsValid() || !mv.IsValid() {
-               return false, nil
-       }
-
-       var isNil bool
-       v, isNil = indirect(v)
-       if isNil {
-               return false, nil
-       }
-       mv, isNil = indirect(mv)
-       if isNil {
+       v, vIsNil := indirect(v)
+       if !v.IsValid() {
+               vIsNil = true
+       }
+       mv, mvIsNil := indirect(mv)
+       if !mv.IsValid() {
+               mvIsNil = true
+       }
+       if vIsNil || mvIsNil {
+               switch op {
+               case "", "=", "==", "eq":
+                       return vIsNil == mvIsNil, nil
+               case "!=", "<>", "ne":
+                       return vIsNil != mvIsNil, nil
+               }
                return false, nil
        }
 
index 573b0fe23ab40d81a263e00a646cb63fefc09e68..81018626a2e90ea7f0274ffa5d3418d6d991dffd 100644 (file)
@@ -673,6 +673,7 @@ func TestCheckCondition(t *testing.T) {
                        "",
                        expect{true, false},
                },
+               {reflect.ValueOf(nil), reflect.ValueOf(nil), "", expect{true, false}},
                {reflect.ValueOf(123), reflect.ValueOf(456), "!=", expect{true, false}},
                {reflect.ValueOf("foo"), reflect.ValueOf("bar"), "!=", expect{true, false}},
                {
@@ -681,6 +682,7 @@ func TestCheckCondition(t *testing.T) {
                        "!=",
                        expect{true, false},
                },
+               {reflect.ValueOf(123), reflect.ValueOf(nil), "!=", expect{true, false}},
                {reflect.ValueOf(456), reflect.ValueOf(123), ">=", expect{true, false}},
                {reflect.ValueOf("foo"), reflect.ValueOf("bar"), ">=", expect{true, false}},
                {
@@ -943,6 +945,31 @@ func TestWhere(t *testing.T) {
                                {A: "a", B: "b"}, {A: "e", B: "f"},
                        },
                },
+               {
+                       sequence: []map[string]int{
+                               {"a": 1, "b": 2}, {"a": 3}, {"a": 5, "b": 6},
+                       },
+                       key: "b", op: "", match: nil,
+                       expect: []map[string]int{
+                               {"a": 3},
+                       },
+               },
+               {
+                       sequence: []map[string]int{
+                               {"a": 1, "b": 2}, {"a": 3}, {"a": 5, "b": 6},
+                       },
+                       key: "b", op: "!=", match: nil,
+                       expect: []map[string]int{
+                               {"a": 1, "b": 2}, {"a": 5, "b": 6},
+                       },
+               },
+               {
+                       sequence: []map[string]int{
+                               {"a": 1, "b": 2}, {"a": 3}, {"a": 5, "b": 6},
+                       },
+                       key: "b", op: ">", match: nil,
+                       expect: []map[string]int{},
+               },
                {sequence: (*[]TstX)(nil), key: "A", match: "a", expect: false},
                {sequence: TstX{A: "a", B: "b"}, key: "A", match: "a", expect: false},
                {sequence: []map[string]*TstX{{"foo": nil}}, key: "foo.B", match: "d", expect: false},