[fix #1512] add `not in` comparison into Trigger Agent (#1545)

Andrey Yantsen 8 年之前
父节点
当前提交
3b4b4f3761
共有 2 个文件被更改,包括 48 次插入22 次删除
  1. 26 22
      app/models/agents/trigger_agent.rb
  2. 22 0
      spec/models/agents/trigger_agent_spec.rb

+ 26 - 22
app/models/agents/trigger_agent.rb

@@ -2,7 +2,7 @@ module Agents
2 2
   class TriggerAgent < Agent
3 3
     cannot_be_scheduled!
4 4
 
5
-    VALID_COMPARISON_TYPES = %w[regex !regex field<value field<=value field==value field!=value field>=value field>value]
5
+    VALID_COMPARISON_TYPES = %w[regex !regex field<value field<=value field==value field!=value field>=value field>value not\ in]
6 6
 
7 7
     description <<-MD
8 8
       The Trigger Agent will watch for a specific value in an Event payload.
@@ -11,7 +11,7 @@ module Agents
11 11
 
12 12
       The `type` can be one of #{VALID_COMPARISON_TYPES.map { |t| "`#{t}`" }.to_sentence} and compares with the `value`.  Note that regex patterns are matched case insensitively.  If you want case sensitive matching, prefix your pattern with `(?-i)`.
13 13
 
14
-      The `value` can be a single value or an array of values. In the case of an array, if one or more values match then the rule matches.
14
+      The `value` can be a single value or an array of values. In the case of an array, all items must be strings, and if one or more values match, then the rule matches. Note: avoid using `field!=value` with arrays, you should use `not in` instead.
15 15
 
16 16
       By default, all rules must match for the Agent to trigger. You can switch this so that only one rule must match by
17 17
       setting `must_match` to `1`.
@@ -75,26 +75,30 @@ module Agents
75 75
           rule_values = rule['value']
76 76
           rule_values = [rule_values] unless rule_values.is_a?(Array)
77 77
 
78
-          rule_values.any? do |rule_value|
79
-            case rule['type']
80
-            when "regex"
81
-              value_at_path.to_s =~ Regexp.new(rule_value, Regexp::IGNORECASE)
82
-            when "!regex"
83
-              value_at_path.to_s !~ Regexp.new(rule_value, Regexp::IGNORECASE)
84
-            when "field>value"
85
-              value_at_path.to_f > rule_value.to_f
86
-            when "field>=value"
87
-              value_at_path.to_f >= rule_value.to_f
88
-            when "field<value"
89
-              value_at_path.to_f < rule_value.to_f
90
-            when "field<=value"
91
-              value_at_path.to_f <= rule_value.to_f
92
-            when "field==value"
93
-              value_at_path.to_s == rule_value.to_s
94
-            when "field!=value"
95
-              value_at_path.to_s != rule_value.to_s
96
-            else
97
-              raise "Invalid type of #{rule['type']} in TriggerAgent##{id}"
78
+          if rule['type'] == 'not in'
79
+            !rule_values.include?(value_at_path.to_s)
80
+          elsif rule['type'] == 'field==value'
81
+            rule_values.include?(value_at_path.to_s)
82
+          else
83
+            rule_values.any? do |rule_value|
84
+              case rule['type']
85
+              when "regex"
86
+                value_at_path.to_s =~ Regexp.new(rule_value, Regexp::IGNORECASE)
87
+              when "!regex"
88
+                value_at_path.to_s !~ Regexp.new(rule_value, Regexp::IGNORECASE)
89
+              when "field>value"
90
+                value_at_path.to_f > rule_value.to_f
91
+              when "field>=value"
92
+                value_at_path.to_f >= rule_value.to_f
93
+              when "field<value"
94
+                value_at_path.to_f < rule_value.to_f
95
+              when "field<=value"
96
+                value_at_path.to_f <= rule_value.to_f
97
+              when "field!=value"
98
+                value_at_path.to_s != rule_value.to_s
99
+              else
100
+                raise "Invalid type of #{rule['type']} in TriggerAgent##{id}"
101
+              end
98 102
             end
99 103
           end
100 104
         end

+ 22 - 0
spec/models/agents/trigger_agent_spec.rb

@@ -270,6 +270,28 @@ describe Agents::TriggerAgent do
270 270
       }.to change { Event.count }.by(1)
271 271
     end
272 272
 
273
+    it "handles array of `not in` comparisons" do
274
+      @event.payload['foo']['bar']['baz'] = "hello world"
275
+      @checker.options['rules'].first['type'] = "not in"
276
+      @checker.options['rules'].first['value'] = ["hello world", "hello world"]
277
+
278
+      expect {
279
+        @checker.receive([@event])
280
+      }.not_to change { Event.count }
281
+
282
+      @checker.options['rules'].first['value'] = ["hello there", "hello world"]
283
+
284
+      expect {
285
+        @checker.receive([@event])
286
+      }.not_to change { Event.count }
287
+
288
+      @checker.options['rules'].first['value'] = ["hello there", "hello here"]
289
+
290
+      expect {
291
+        @checker.receive([@event])
292
+      }.to change { Event.count }.by(1)
293
+    end
294
+
273 295
     it "does fine without dots in the path" do
274 296
       @event.payload = { 'hello' => "world" }
275 297
       @checker.options['rules'].first['type'] = "field==value"