From ee2c8af8c8b0a05772813a60cc4f33e56e67b1cc Mon Sep 17 00:00:00 2001 From: Colin Surprenant Date: Thu, 21 Apr 2016 17:38:03 -0400 Subject: [PATCH] refactor Ruby Event getter and setter refactor wip gemfiles refactor Java Event getter and setter bump plugin-api to 2.0 use plugin-api 2.0 switch to core-event-java include logstash-core-event-java.jar jar file so that gem dependency using the source tree work updated core plugins to core-api 2.0 added grok for refactor branch fix rebased specs remove temp plugins github paths remove commented out tmp alias_method --- Gemfile | 6 +- Gemfile.jruby-1.9.lock | 178 ++++++++++++-- .../logstash-core-event-java.jar | Bin 0 -> 42384 bytes logstash-core-event-java/spec/event_spec.rb | 96 ++++---- .../logstash/ext/JrubyEventExtLibrary.java | 4 +- logstash-core-event/lib/logstash/event.rb | 15 +- .../lib/logstash/string_interpolation.rb | 2 +- .../spec/logstash/event_spec.rb | 194 +++++++-------- .../lib/logstash-core-plugin-api/version.rb | 2 +- .../lib/logstash/config/config_ast.rb | 2 +- logstash-core/lib/logstash/filters/base.rb | 4 +- logstash-core/lib/logstash/inputs/base.rb | 2 +- logstash-core/lib/logstash/util/decorators.rb | 12 +- logstash-core/spec/conditionals_spec.rb | 226 +++++++++--------- .../spec/logstash/filters/base_spec.rb | 32 +-- .../spec/logstash/inputs/base_spec.rb | 16 +- logstash-core/spec/logstash/pipeline_spec.rb | 30 +-- 17 files changed, 481 insertions(+), 340 deletions(-) create mode 100644 logstash-core-event-java/lib/logstash-core-event-java/logstash-core-event-java.jar diff --git a/Gemfile b/Gemfile index a6b25daefda..79c9f2af4a2 100644 --- a/Gemfile +++ b/Gemfile @@ -4,7 +4,7 @@ source "https://rubygems.org" gem "logstash-core", "5.0.0.dev", :path => "./logstash-core" gem "logstash-core-event-java", "5.0.0.dev", :path => "./logstash-core-event-java" -gem "logstash-core-plugin-api", "1.0.0", :path => "./logstash-core-plugin-api" +gem "logstash-core-plugin-api", "2.0.0", :path => "./logstash-core-plugin-api" gem "file-dependencies", "0.1.6" gem "ci_reporter_rspec", "1.0.0", :group => :development gem "simplecov", :group => :development @@ -14,10 +14,10 @@ gem "coveralls", :group => :development # 1.6 is the last supported version on jruby. gem "tins", "1.6", :group => :development gem "rspec", "~> 3.1.0", :group => :development -gem "logstash-devutils", "~> 0.0.15", :group => :development +gem "logstash-devutils", :group => :development gem "benchmark-ips", :group => :development gem "octokit", "3.8.0", :group => :build -gem "stud", "~> 0.0.21", :group => :build +gem "stud", "~> 0.0.22", :group => :build gem "fpm", "~> 1.3.3", :group => :build gem "rubyzip", "~> 1.1.7", :group => :build gem "gems", "~> 0.8.3", :group => :build diff --git a/Gemfile.jruby-1.9.lock b/Gemfile.jruby-1.9.lock index 7510eb2679d..709340a1965 100644 --- a/Gemfile.jruby-1.9.lock +++ b/Gemfile.jruby-1.9.lock @@ -1,3 +1,136 @@ +GIT + remote: git://github.com/elastic/logstash-devutils.git + revision: e9af3a24824a41d0f19b025ca359e0735e820251 + branch: feature/plugin-api-2_0 + specs: + logstash-devutils (0.0.19-java) + fivemat + gem_publisher + insist (= 1.0.0) + kramdown + minitar + rake + rspec (~> 3.1.0) + rspec-wait + stud (>= 0.0.20) + +GIT + remote: git://github.com/logstash-plugins/logstash-codec-json.git + revision: 141d488f3a73706d5ae08dd41468fed285fbbdf9 + branch: feature/plugin-api-2_0 + specs: + logstash-codec-json (2.1.4) + logstash-core-plugin-api (~> 2.0) + +GIT + remote: git://github.com/logstash-plugins/logstash-codec-json_lines.git + revision: f1a3fda4d1affaa6da4b037eec78ee5f10287eb8 + branch: feature/plugin-api-2_0 + specs: + logstash-codec-json_lines (2.1.3) + logstash-codec-line (>= 2.1.0) + logstash-core-plugin-api (~> 2.0) + +GIT + remote: git://github.com/logstash-plugins/logstash-codec-line.git + revision: f912b6ca5137bef914ea0e605cb34520b0f172d4 + branch: feature/plugin-api-2_0 + specs: + logstash-codec-line (2.1.2) + logstash-core-plugin-api (~> 2.0) + +GIT + remote: git://github.com/logstash-plugins/logstash-codec-plain.git + revision: 5c5424905d5e06d7f7adc888411057c8a5681194 + branch: feature/plugin-api-2_0 + specs: + logstash-codec-plain (2.0.4) + logstash-core-plugin-api (~> 2.0) + +GIT + remote: git://github.com/logstash-plugins/logstash-filter-clone.git + revision: 4da247817809ff68a47557f022c8049536651564 + branch: feature/plugin-api-2_0 + specs: + logstash-filter-clone (2.0.6) + logstash-core-plugin-api (~> 2.0) + +GIT + remote: git://github.com/logstash-plugins/logstash-filter-grok.git + revision: 284dc6614b0cac6770cda2c18a52f8405146f1eb + branch: feature/plugin-api-2_0 + specs: + logstash-filter-grok (2.0.5) + jls-grok (~> 0.11.1) + logstash-core-plugin-api (~> 2.0) + logstash-patterns-core + +GIT + remote: git://github.com/logstash-plugins/logstash-filter-multiline.git + revision: fd1da31b7d1e0b44319c0f2865c767b36440583d + branch: feature/plugin-api-2_0 + specs: + logstash-filter-multiline (2.0.5) + jls-grok (~> 0.11.0) + logstash-core-plugin-api (~> 2.0) + logstash-patterns-core + +GIT + remote: git://github.com/logstash-plugins/logstash-filter-mutate.git + revision: f9624cc0a05354c308b2d22a5895c737eceaa08b + branch: feature/plugin-api-2_0 + specs: + logstash-filter-mutate (2.0.6) + logstash-core-plugin-api (~> 2.0) + +GIT + remote: git://github.com/logstash-plugins/logstash-input-generator.git + revision: 2557a4e7da667d6e801cac47e27ff297b6ecc79a + branch: feature/plugin-api-2_0 + specs: + logstash-input-generator (2.0.4) + logstash-codec-plain + logstash-core-plugin-api (~> 2.0) + +GIT + remote: git://github.com/logstash-plugins/logstash-input-stdin.git + revision: cca193c505f931500bd0f6be45afada2af0578ed + branch: feature/plugin-api-2_0 + specs: + logstash-input-stdin (2.0.4) + concurrent-ruby + logstash-codec-line + logstash-core-plugin-api (~> 2.0) + +GIT + remote: git://github.com/logstash-plugins/logstash-input-tcp.git + revision: b6f2705d7fc226c08dd0905561ef9373d2801a58 + branch: feature/plugin-api-2_0 + specs: + logstash-input-tcp (3.0.5) + logstash-codec-json + logstash-codec-json_lines + logstash-codec-line + logstash-codec-plain + logstash-core-plugin-api (~> 2.0) + +GIT + remote: git://github.com/logstash-plugins/logstash-output-stdout.git + revision: c150cb4f560b047372e15887e8b6bf421af22fc9 + branch: feature/plugin-api-2_0 + specs: + logstash-output-stdout (2.0.6) + logstash-codec-line + logstash-core-plugin-api (~> 2.0) + +GIT + remote: git://github.com/logstash-plugins/logstash-patterns-core.git + revision: e039ba2d3db6a01b6501fe0cab8a82453c6dc45f + branch: feature/plugin-api-2_0 + specs: + logstash-patterns-core (2.0.5) + logstash-core-plugin-api (~> 2.0) + PATH remote: ./logstash-core specs: @@ -10,8 +143,8 @@ PATH gems (~> 0.8.3) i18n (= 0.6.9) jrjackson (~> 0.3.7) - jruby-monitoring (~> 0.1) - jruby-openssl (= 0.9.13) + jruby-monitoring (~> 0.3.1) + jruby-openssl (= 0.9.16) logstash-core-event-java (~> 5.0.0.dev) minitar (~> 0.5.4) pry (~> 0.10.1) @@ -32,7 +165,7 @@ PATH PATH remote: ./logstash-core-plugin-api specs: - logstash-core-plugin-api (1.0.0-java) + logstash-core-plugin-api (2.0.0-java) logstash-core (= 5.0.0.dev) GEM @@ -42,7 +175,7 @@ GEM arr-pm (0.0.10) cabin (> 0) backports (3.6.8) - benchmark-ips (2.5.0) + benchmark-ips (2.6.1) builder (3.2.2) cabin (0.8.1) childprocess (0.5.9) @@ -71,6 +204,7 @@ GEM file-dependencies (0.1.6) minitar filesize (0.0.4) + fivemat (1.3.2) flores (0.0.6) fpm (1.3.3) arr-pm (~> 0.0.9) @@ -85,20 +219,13 @@ GEM i18n (0.6.9) insist (1.0.0) jar-dependencies (0.3.2) - jrjackson (0.3.8) - jruby-monitoring (0.3.0) - jruby-openssl (0.9.13-java) + jls-grok (0.11.2) + cabin (>= 0.6.0) + jrjackson (0.3.9-java) + jruby-monitoring (0.3.1) + jruby-openssl (0.9.16-java) json (1.8.3-java) kramdown (1.10.0) - logstash-devutils (0.0.18-java) - gem_publisher - insist (= 1.0.0) - kramdown - minitar - rake - rspec (~> 3.1.0) - rspec-wait - stud (>= 0.0.20) method_source (0.8.2) minitar (0.5.4) multipart-post (2.0.0) @@ -173,14 +300,27 @@ DEPENDENCIES flores (~> 0.0.6) fpm (~> 1.3.3) gems (~> 0.8.3) + logstash-codec-json! + logstash-codec-json_lines! + logstash-codec-line! + logstash-codec-plain! logstash-core (= 5.0.0.dev)! logstash-core-event-java (= 5.0.0.dev)! - logstash-core-plugin-api (= 1.0.0)! - logstash-devutils (~> 0.0.15) + logstash-core-plugin-api (= 2.0.0)! + logstash-devutils! + logstash-filter-clone! + logstash-filter-grok! + logstash-filter-multiline! + logstash-filter-mutate! + logstash-input-generator! + logstash-input-stdin! + logstash-input-tcp! + logstash-output-stdout! + logstash-patterns-core! octokit (= 3.8.0) rack-test rspec (~> 3.1.0) rubyzip (~> 1.1.7) simplecov - stud (~> 0.0.21) + stud (~> 0.0.22) tins (= 1.6) diff --git a/logstash-core-event-java/lib/logstash-core-event-java/logstash-core-event-java.jar b/logstash-core-event-java/lib/logstash-core-event-java/logstash-core-event-java.jar new file mode 100644 index 0000000000000000000000000000000000000000..ec9ed306bc6a98b3217b2d78f3cc9147363b09b2 GIT binary patch literal 42384 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 Tke#`SnK1%UX5{{pvv>aovcL^1 literal 0 HcmV?d00001 diff --git a/logstash-core-event-java/spec/event_spec.rb b/logstash-core-event-java/spec/event_spec.rb index 03cbffa73c6..8ccae4f6cf9 100644 --- a/logstash-core-event-java/spec/event_spec.rb +++ b/logstash-core-event-java/spec/event_spec.rb @@ -26,10 +26,10 @@ it "should serialize deep hash from field reference assignments" do e = LogStash::Event.new({TIMESTAMP => "2015-05-28T23:02:05.350Z"}) - e["foo"] = "bar" - e["bar"] = 1 - e["baz"] = 1.0 - e["[fancy][pants][socks]"] = "shoes" + e.set("foo", "bar") + e.set("bar", 1) + e.set("baz", 1.0) + e.set("[fancy][pants][socks]", "shoes") expect(JSON.parse(e.to_json)).to eq(JSON.parse("{\"@timestamp\":\"2015-05-28T23:02:05.350Z\",\"@version\":\"1\",\"foo\":\"bar\",\"bar\":1,\"baz\":1.0,\"fancy\":{\"pants\":{\"socks\":\"shoes\"}}}")) end end @@ -37,79 +37,79 @@ context "[]" do it "should get simple values" do e = LogStash::Event.new({"foo" => "bar", "bar" => 1, "baz" => 1.0, TIMESTAMP => "2015-05-28T23:02:05.350Z"}) - expect(e["foo"]).to eq("bar") - expect(e["[foo]"]).to eq("bar") - expect(e["bar"]).to eq(1) - expect(e["[bar]"]).to eq(1) - expect(e["baz"]).to eq(1.0) - expect(e["[baz]"]).to eq(1.0) - expect(e[TIMESTAMP].to_s).to eq("2015-05-28T23:02:05.350Z") - expect(e["[#{TIMESTAMP}]"].to_s).to eq("2015-05-28T23:02:05.350Z") + expect(e.get("foo")).to eq("bar") + expect(e.get("[foo]")).to eq("bar") + expect(e.get("bar")).to eq(1) + expect(e.get("[bar]")).to eq(1) + expect(e.get("baz")).to eq(1.0) + expect(e.get("[baz]")).to eq(1.0) + expect(e.get(TIMESTAMP).to_s).to eq("2015-05-28T23:02:05.350Z") + expect(e.get("[#{TIMESTAMP}]").to_s).to eq("2015-05-28T23:02:05.350Z") end it "should get deep hash values" do e = LogStash::Event.new({"foo" => {"bar" => 1, "baz" => 1.0}}) - expect(e["[foo][bar]"]).to eq(1) - expect(e["[foo][baz]"]).to eq(1.0) + expect(e.get("[foo][bar]")).to eq(1) + expect(e.get("[foo][baz]")).to eq(1.0) end it "should get deep array values" do e = LogStash::Event.new({"foo" => ["bar", 1, 1.0]}) - expect(e["[foo][0]"]).to eq("bar") - expect(e["[foo][1]"]).to eq(1) - expect(e["[foo][2]"]).to eq(1.0) - expect(e["[foo][3]"]).to be_nil + expect(e.get("[foo][0]")).to eq("bar") + expect(e.get("[foo][1]")).to eq(1) + expect(e.get("[foo][2]")).to eq(1.0) + expect(e.get("[foo][3]")).to be_nil end end context "[]=" do it "should set simple values" do e = LogStash::Event.new() - expect(e["foo"] = "bar").to eq("bar") - expect(e["foo"]).to eq("bar") + expect(e.set("foo", "bar")).to eq("bar") + expect(e.get("foo")).to eq("bar") e = LogStash::Event.new({"foo" => "test"}) - expect(e["foo"] = "bar").to eq("bar") - expect(e["foo"]).to eq("bar") + expect(e.set("foo", "bar")).to eq("bar") + expect(e.get("foo")).to eq("bar") end it "should set deep hash values" do e = LogStash::Event.new() - expect(e["[foo][bar]"] = "baz").to eq("baz") - expect(e["[foo][bar]"]).to eq("baz") - expect(e["[foo][baz]"]).to be_nil + expect(e.set("[foo][bar]", "baz")).to eq("baz") + expect(e.get("[foo][bar]")).to eq("baz") + expect(e.get("[foo][baz]")).to be_nil end it "should set deep array values" do e = LogStash::Event.new() - expect(e["[foo][0]"] = "bar").to eq("bar") - expect(e["[foo][0]"]).to eq("bar") - expect(e["[foo][1]"] = 1).to eq(1) - expect(e["[foo][1]"]).to eq(1) - expect(e["[foo][2]"] = 1.0 ).to eq(1.0) - expect(e["[foo][2]"]).to eq(1.0) - expect(e["[foo][3]"]).to be_nil + expect(e.set("[foo][0]", "bar")).to eq("bar") + expect(e.get("[foo][0]")).to eq("bar") + expect(e.set("[foo][1]", 1)).to eq(1) + expect(e.get("[foo][1]")).to eq(1) + expect(e.set("[foo][2]", 1.0)).to eq(1.0) + expect(e.get("[foo][2]")).to eq(1.0) + expect(e.get("[foo][3]")).to be_nil end it "should add key when setting nil value" do e = LogStash::Event.new() - e["[foo]"] = nil + e.set("[foo]", nil) expect(e.to_hash).to include("foo" => nil) end # BigDecinal is now natively converted by JRuby, see https://github.com/elastic/logstash/pull/4838 it "should set BigDecimal" do e = LogStash::Event.new() - e["[foo]"] = BigDecimal.new(1) - expect(e["foo"]).to be_kind_of(BigDecimal) - expect(e["foo"]).to eq(BigDecimal.new(1)) + e.set("[foo]", BigDecimal.new(1)) + expect(e.get("foo")).to be_kind_of(BigDecimal) + expect(e.get("foo")).to eq(BigDecimal.new(1)) end it "should set RubyBignum" do e = LogStash::Event.new() - e["[foo]"] = -9223372036854776000 - expect(e["foo"]).to be_kind_of(Bignum) - expect(e["foo"]).to eq(-9223372036854776000) + e.set("[foo]", -9223372036854776000) + expect(e.get("foo")).to be_kind_of(Bignum) + expect(e.get("foo")).to eq(-9223372036854776000) end end @@ -117,7 +117,7 @@ it "getters should present a Ruby LogStash::Timestamp" do e = LogStash::Event.new() expect(e.timestamp.class).to eq(LogStash::Timestamp) - expect(e[TIMESTAMP].class).to eq(LogStash::Timestamp) + expect(e.get(TIMESTAMP).class).to eq(LogStash::Timestamp) end it "to_hash should inject a Ruby LogStash::Timestamp" do @@ -134,9 +134,9 @@ it "should set timestamp" do e = LogStash::Event.new now = Time.now - e["@timestamp"] = LogStash::Timestamp.at(now.to_i) + e.set("@timestamp", LogStash::Timestamp.at(now.to_i)) expect(e.timestamp.to_i).to eq(now.to_i) - expect(e["@timestamp"].to_i).to eq(now.to_i) + expect(e.get("@timestamp").to_i).to eq(now.to_i) end end @@ -144,16 +144,16 @@ it "should append" do event = LogStash::Event.new("message" => "hello world") event.append(LogStash::Event.new("message" => "another thing")) - expect(event["message"]).to eq(["hello world", "another thing"]) + expect(event.get("message")).to eq(["hello world", "another thing"]) end end context "tags" do it "should tag" do event = LogStash::Event.new("message" => "hello world") - expect(event["tags"]).to be_nil - event["tags"] = ["foo"] - expect(event["tags"]).to eq(["foo"]) + expect(event.get("tags")).to be_nil + event.tag("foo") + expect(event.get("tags")).to eq(["foo"]) end end @@ -258,8 +258,8 @@ def self.warn(message) expect(LogStash::Event.from_json(source_json).size).to eq(1) event = LogStash::Event.from_json(source_json)[0] - expect(event["[foo]"]).to eq(1) - expect(event["[bar]"]).to eq("baz") + expect(event.get("[foo]")).to eq(1) + expect(event.get("[bar]")).to eq("baz") end it "should ignore blank strings" do diff --git a/logstash-core-event-java/src/main/java/com/logstash/ext/JrubyEventExtLibrary.java b/logstash-core-event-java/src/main/java/com/logstash/ext/JrubyEventExtLibrary.java index 1cb630d5a75..8cf5024f339 100644 --- a/logstash-core-event-java/src/main/java/com/logstash/ext/JrubyEventExtLibrary.java +++ b/logstash-core-event-java/src/main/java/com/logstash/ext/JrubyEventExtLibrary.java @@ -140,14 +140,14 @@ public IRubyObject ruby_initialize(ThreadContext context, IRubyObject[] args) return context.nil; } - @JRubyMethod(name = "[]", required = 1) + @JRubyMethod(name = "get", required = 1) public IRubyObject ruby_get_field(ThreadContext context, RubyString reference) { Object value = this.event.getField(reference.asJavaString()); return Rubyfier.deep(context.runtime, value); } - @JRubyMethod(name = "[]=", required = 2) + @JRubyMethod(name = "set", required = 2) public IRubyObject ruby_set_field(ThreadContext context, RubyString reference, IRubyObject value) { String r = reference.asJavaString(); diff --git a/logstash-core-event/lib/logstash/event.rb b/logstash-core-event/lib/logstash/event.rb index b1eb9d46cdb..7a9b7d133c9 100644 --- a/logstash-core-event/lib/logstash/event.rb +++ b/logstash-core-event/lib/logstash/event.rb @@ -54,6 +54,7 @@ class DeprecatedMethod < StandardError; end VERSION_ONE = "1" TIMESTAMP_FAILURE_TAG = "_timestampparsefailure" TIMESTAMP_FAILURE_FIELD = "_@timestamp" + TAGS = "tags".freeze METADATA = "@metadata".freeze METADATA_BRACKETS = "[#{METADATA}]".freeze @@ -113,7 +114,7 @@ def timestamp=(val) @data[TIMESTAMP] = val end - def [](fieldref) + def get(fieldref) if fieldref.start_with?(METADATA_BRACKETS) @metadata_accessors.get(fieldref[METADATA_BRACKETS.length .. -1]) elsif fieldref == METADATA @@ -123,7 +124,7 @@ def [](fieldref) end end - def []=(fieldref, value) + def set(fieldref, value) if fieldref == TIMESTAMP && !value.is_a?(LogStash::Timestamp) raise TypeError, "The field '@timestamp' must be a (LogStash::Timestamp, not a #{value.class} (#{value})" end @@ -201,8 +202,9 @@ def sprintf(format) def tag(value) # Generalize this method for more usability - self["tags"] ||= [] - self["tags"] << value unless self["tags"].include?(value) + tags = @accessors.get(TAGS) || [] + tags << value unless tags.include?(value) + @accessors.set(TAGS, tags) end def to_hash_with_metadata @@ -269,9 +271,8 @@ def init_timestamp(o) logger.warn("Error parsing #{TIMESTAMP} string, setting current time to #{TIMESTAMP}, original in #{TIMESTAMP_FAILURE_FIELD} field", :value => o.inspect, :exception => e.message) end - @data["tags"] ||= [] - @data["tags"] << TIMESTAMP_FAILURE_TAG unless @data["tags"].include?(TIMESTAMP_FAILURE_TAG) - @data[TIMESTAMP_FAILURE_FIELD] = o + tag(TIMESTAMP_FAILURE_TAG) + @accessors.set(TIMESTAMP_FAILURE_FIELD, o) LogStash::Timestamp.now end diff --git a/logstash-core-event/lib/logstash/string_interpolation.rb b/logstash-core-event/lib/logstash/string_interpolation.rb index 13044e4c005..aaa54981165 100644 --- a/logstash-core-event/lib/logstash/string_interpolation.rb +++ b/logstash-core-event/lib/logstash/string_interpolation.rb @@ -115,7 +115,7 @@ def initialize(key) end def evaluate(event) - value = event[@key] + value = event.get(@key) case value when nil diff --git a/logstash-core-event/spec/logstash/event_spec.rb b/logstash-core-event/spec/logstash/event_spec.rb index 0389ffc6d25..8c6d60db291 100644 --- a/logstash-core-event/spec/logstash/event_spec.rb +++ b/logstash-core-event/spec/logstash/event_spec.rb @@ -8,68 +8,68 @@ shared_examples "all event tests" do context "[]=" do it "should raise an exception if you attempt to set @timestamp to a value type other than a Time object" do - expect{subject["@timestamp"] = "crash!"}.to raise_error(TypeError) + expect{subject.set("@timestamp", "crash!")}.to raise_error(TypeError) end it "should assign simple fields" do - expect(subject["foo"]).to be_nil - expect(subject["foo"] = "bar").to eq("bar") - expect(subject["foo"]).to eq("bar") + expect(subject.get("foo")).to be_nil + expect(subject.set("foo", "bar")).to eq("bar") + expect(subject.get("foo")).to eq("bar") end it "should overwrite simple fields" do - expect(subject["foo"]).to be_nil - expect(subject["foo"] = "bar").to eq("bar") - expect(subject["foo"]).to eq("bar") + expect(subject.get("foo")).to be_nil + expect(subject.set("foo", "bar")).to eq("bar") + expect(subject.get("foo")).to eq("bar") - expect(subject["foo"] = "baz").to eq("baz") - expect(subject["foo"]).to eq("baz") + expect(subject.set("foo", "baz")).to eq("baz") + expect(subject.get("foo")).to eq("baz") end it "should assign deep fields" do - expect(subject["[foo][bar]"]).to be_nil - expect(subject["[foo][bar]"] = "baz").to eq("baz") - expect(subject["[foo][bar]"]).to eq("baz") + expect(subject.get("[foo][bar]")).to be_nil + expect(subject.set("[foo][bar]", "baz")).to eq("baz") + expect(subject.get("[foo][bar]")).to eq("baz") end it "should overwrite deep fields" do - expect(subject["[foo][bar]"]).to be_nil - expect(subject["[foo][bar]"] = "baz").to eq("baz") - expect(subject["[foo][bar]"]).to eq("baz") + expect(subject.get("[foo][bar]")).to be_nil + expect(subject.set("[foo][bar]", "baz")).to eq("baz") + expect(subject.get("[foo][bar]")).to eq("baz") - expect(subject["[foo][bar]"] = "zab").to eq("zab") - expect(subject["[foo][bar]"]).to eq("zab") + expect(subject.set("[foo][bar]", "zab")).to eq("zab") + expect(subject.get("[foo][bar]")).to eq("zab") end it "allow to set the @metadata key to a hash" do - subject["@metadata"] = { "action" => "index" } - expect(subject["[@metadata][action]"]).to eq("index") + subject.set("@metadata", { "action" => "index" }) + expect(subject.get("[@metadata][action]")).to eq("index") end it "should add key when setting nil value" do - subject["[baz]"] = nil + subject.set("[baz]", nil) expect(subject.to_hash).to include("baz" => nil) end it "should set nil element within existing array value" do - subject["[foo]"] = ["bar", "baz"] + subject.set("[foo]", ["bar", "baz"]) - expect(subject["[foo][0]"] = nil).to eq(nil) - expect(subject["[foo]"]).to eq([nil, "baz"]) + expect(subject.set("[foo][0]", nil)).to eq(nil) + expect(subject.get("[foo]")).to eq([nil, "baz"]) end it "should set nil in first element within empty array" do - subject["[foo]"] = [] + subject.set("[foo]", []) - expect(subject["[foo][0]"] = nil).to eq(nil) - expect(subject["[foo]"]).to eq([nil]) + expect(subject.set("[foo][0]", nil)).to eq(nil) + expect(subject.get("[foo]")).to eq([nil]) end it "should set nil in second element within empty array" do - subject["[foo]"] = [] + subject.set("[foo]", []) - expect(subject["[foo][1]"] = nil).to eq(nil) - expect(subject["[foo]"]).to eq([nil, nil]) + expect(subject.set("[foo][1]", nil)).to eq(nil) + expect(subject.get("[foo]")).to eq([nil, nil]) end end @@ -79,7 +79,7 @@ event = LogStash::Event.new({ "reference" => data }) LogStash::Util::Decorators.add_fields({"reference_test" => "%{reference}"}, event, "dummy-plugin") data.downcase! - expect(event["reference_test"]).not_to eq(data) + expect(event.get("reference_test")).not_to eq(data) end it "should not return a Fixnum reference" do @@ -87,7 +87,7 @@ event = LogStash::Event.new({ "reference" => data }) LogStash::Util::Decorators.add_fields({"reference_test" => "%{reference}"}, event, "dummy-plugin") data += 41 - expect(event["reference_test"]).to eq("1") + expect(event.get("reference_test")).to eq("1") end it "should report a unix timestamp for %{+%s}" do @@ -124,7 +124,7 @@ it "should report fields with %{field} syntax" do expect(subject.sprintf("%{type}")).to eq("sprintf") - expect(subject.sprintf("%{message}")).to eq(subject["message"]) + expect(subject.sprintf("%{message}")).to eq(subject.get("message")) end it "should print deep fields" do @@ -153,35 +153,35 @@ end it "should render nil array values as leading empty string" do - expect(subject["foo"] = [nil, "baz"]).to eq([nil, "baz"]) + expect(subject.set("foo", [nil, "baz"])).to eq([nil, "baz"]) - expect(subject["[foo][0]"]).to be_nil - expect(subject["[foo][1]"]).to eq("baz") + expect(subject.get("[foo][0]")).to be_nil + expect(subject.get("[foo][1]")).to eq("baz") expect(subject.sprintf("%{[foo]}")).to eq(",baz") end it "should render nil array values as middle empty string" do - expect(subject["foo"] = ["bar", nil, "baz"]).to eq(["bar", nil, "baz"]) + expect(subject.set("foo", ["bar", nil, "baz"])).to eq(["bar", nil, "baz"]) - expect(subject["[foo][0]"]).to eq("bar") - expect(subject["[foo][1]"]).to be_nil - expect(subject["[foo][2]"]).to eq("baz") + expect(subject.get("[foo][0]")).to eq("bar") + expect(subject.get("[foo][1]")).to be_nil + expect(subject.get("[foo][2]")).to eq("baz") expect(subject.sprintf("%{[foo]}")).to eq("bar,,baz") end it "should render nil array values as trailing empty string" do - expect(subject["foo"] = ["bar", nil]).to eq(["bar", nil]) + expect(subject.set("foo", ["bar", nil])).to eq(["bar", nil]) - expect(subject["[foo][0]"]).to eq("bar") - expect(subject["[foo][1]"]).to be_nil + expect(subject.get("[foo][0]")).to eq("bar") + expect(subject.get("[foo][1]")).to be_nil expect(subject.sprintf("%{[foo]}")).to eq("bar,") end it "should render deep arrays with nil value" do - subject["[foo]"] = [[12, nil], 56] + subject.set("[foo]", [[12, nil], 56]) expect(subject.sprintf("%{[foo]}")).to eq("12,,56") end @@ -198,18 +198,18 @@ context "#[]" do it "should fetch data" do - expect(subject["type"]).to eq("sprintf") + expect(subject.get("type")).to eq("sprintf") end it "should fetch fields" do - expect(subject["a"]).to eq("b") - expect(subject['c']['d']).to eq("f") + expect(subject.get("a")).to eq("b") + expect(subject.get('c')['d']).to eq("f") end it "should fetch deep fields" do - expect(subject["[j][k1]"]).to eq("v") - expect(subject["[c][d]"]).to eq("f") - expect(subject['[f][g][h]']).to eq("i") - expect(subject['[j][k3][4]']).to eq("m") - expect(subject['[j][5]']).to eq(7) + expect(subject.get("[j][k1]")).to eq("v") + expect(subject.get("[c][d]")).to eq("f") + expect(subject.get('[f][g][h]')).to eq("i") + expect(subject.get('[j][k3][4]')).to eq("m") + expect(subject.get('[j][5]')).to eq(7) end @@ -217,7 +217,7 @@ count = 1000000 2.times do start = Time.now - count.times { subject["[j][k1]"] } + count.times { subject.get("[j][k1]") } duration = Time.now - start puts "event #[] rate: #{"%02.0f/sec" % (count / duration)}, elapsed: #{duration}s" end @@ -263,11 +263,11 @@ ) subject.overwrite(new_event) - expect(subject["message"]).to eq("foo bar") - expect(subject["type"]).to eq("new") + expect(subject.get("message")).to eq("foo bar") + expect(subject.get("type")).to eq("new") ["tags", "source", "a", "c", "f", "j"].each do |field| - expect(subject[field]).to be_nil + expect(subject.get(field)).to be_nil end end end @@ -275,7 +275,7 @@ context "#append" do it "should append strings to an array" do subject.append(LogStash::Event.new("message" => "another thing")) - expect(subject["message"]).to eq([ "hello world", "another thing" ]) + expect(subject.get("message")).to eq([ "hello world", "another thing" ]) end it "should concatenate tags" do @@ -283,54 +283,54 @@ # added to_a for when array is a Java Collection when produced from json input # TODO: we have to find a better way to handle this in tests. maybe override # rspec eq or == to do an explicit to_a when comparing arrays? - expect(subject["tags"].to_a).to eq([ "tag1", "tag2" ]) + expect(subject.get("tags").to_a).to eq([ "tag1", "tag2" ]) end context "when event field is nil" do it "should add single value as string" do subject.append(LogStash::Event.new({"field1" => "append1"})) - expect(subject[ "field1" ]).to eq("append1") + expect(subject.get("field1")).to eq("append1") end it "should add multi values as array" do subject.append(LogStash::Event.new({"field1" => [ "append1","append2" ]})) - expect(subject[ "field1" ]).to eq([ "append1","append2" ]) + expect(subject.get("field1")).to eq([ "append1","append2" ]) end end context "when event field is a string" do - before { subject[ "field1" ] = "original1" } + before { subject.set("field1", "original1") } it "should append string to values, if different from current" do subject.append(LogStash::Event.new({"field1" => "append1"})) - expect(subject[ "field1" ]).to eq([ "original1", "append1" ]) + expect(subject.get("field1")).to eq([ "original1", "append1" ]) end it "should not change value, if appended value is equal current" do subject.append(LogStash::Event.new({"field1" => "original1"})) - expect(subject[ "field1" ]).to eq("original1") + expect(subject.get("field1")).to eq("original1") end it "should concatenate values in an array" do subject.append(LogStash::Event.new({"field1" => [ "append1" ]})) - expect(subject[ "field1" ]).to eq([ "original1", "append1" ]) + expect(subject.get("field1")).to eq([ "original1", "append1" ]) end it "should join array, removing duplicates" do subject.append(LogStash::Event.new({"field1" => [ "append1","original1" ]})) - expect(subject[ "field1" ]).to eq([ "original1", "append1" ]) + expect(subject.get("field1")).to eq([ "original1", "append1" ]) end end context "when event field is an array" do - before { subject[ "field1" ] = [ "original1", "original2" ] } + before { subject.set("field1", [ "original1", "original2" ] )} it "should append string values to array, if not present in array" do subject.append(LogStash::Event.new({"field1" => "append1"})) - expect(subject[ "field1" ]).to eq([ "original1", "original2", "append1" ]) + expect(subject.get("field1")).to eq([ "original1", "original2", "append1" ]) end it "should not append string values, if the array already contains it" do subject.append(LogStash::Event.new({"field1" => "original1"})) - expect(subject[ "field1" ]).to eq([ "original1", "original2" ]) + expect(subject.get("field1")).to eq([ "original1", "original2" ]) end it "should join array, removing duplicates" do subject.append(LogStash::Event.new({"field1" => [ "append1","original1" ]})) - expect(subject[ "field1" ]).to eq([ "original1", "original2", "append1" ]) + expect(subject.get("field1")).to eq([ "original1", "original2", "append1" ]) end end @@ -342,7 +342,7 @@ data = { "@timestamp" => "2013-12-21T07:25:06.605Z" } event = LogStash::Event.new(data) - expect(event["@timestamp"]).to be_a(LogStash::Timestamp) + expect(event.get("@timestamp")).to be_a(LogStash::Timestamp) duration = 0 [warmup, count].each do |i| @@ -411,13 +411,13 @@ it "should tag for invalid value" do event = LogStash::Event.new("@timestamp" => "foo") expect(event.timestamp.to_i).to be_within(1).of Time.now.to_i - expect(event["tags"]).to eq([LogStash::Event::TIMESTAMP_FAILURE_TAG]) - expect(event[LogStash::Event::TIMESTAMP_FAILURE_FIELD]).to eq("foo") + expect(event.get("tags")).to eq([LogStash::Event::TIMESTAMP_FAILURE_TAG]) + expect(event.get(LogStash::Event::TIMESTAMP_FAILURE_FIELD)).to eq("foo") event = LogStash::Event.new("@timestamp" => 666) expect(event.timestamp.to_i).to be_within(1).of Time.now.to_i - expect(event["tags"]).to eq([LogStash::Event::TIMESTAMP_FAILURE_TAG]) - expect(event[LogStash::Event::TIMESTAMP_FAILURE_FIELD]).to eq(666) + expect(event.get("tags")).to eq([LogStash::Event::TIMESTAMP_FAILURE_TAG]) + expect(event.get(LogStash::Event::TIMESTAMP_FAILURE_FIELD)).to eq(666) end it "should warn for invalid value" do @@ -432,8 +432,8 @@ it "should tag for invalid string format" do event = LogStash::Event.new("@timestamp" => "foo") expect(event.timestamp.to_i).to be_within(1).of Time.now.to_i - expect(event["tags"]).to eq([LogStash::Event::TIMESTAMP_FAILURE_TAG]) - expect(event[LogStash::Event::TIMESTAMP_FAILURE_FIELD]).to eq("foo") + expect(event.get("tags")).to eq([LogStash::Event::TIMESTAMP_FAILURE_TAG]) + expect(event.get(LogStash::Event::TIMESTAMP_FAILURE_FIELD)).to eq("foo") end it "should warn for invalid string format" do @@ -478,7 +478,7 @@ end it "should still allow normal field access" do - expect(subject["hello"]).to eq("world") + expect(subject.get("hello")).to eq("world") end end @@ -491,15 +491,15 @@ expect(fieldref).to start_with("[@metadata]") # Set it. - subject[fieldref] = value + subject.set(fieldref, value) end it "should still allow normal field access" do - expect(subject["normal"]).to eq("normal") + expect(subject.get("normal")).to eq("normal") end it "should allow getting" do - expect(subject[fieldref]).to eq(value) + expect(subject.get(fieldref)).to eq(value) end it "should be hidden from .to_json" do @@ -522,10 +522,10 @@ context "with no metadata" do subject { LogStash::Event.new("foo" => "bar") } it "should have no metadata" do - expect(subject["@metadata"]).to be_empty + expect(subject.get("@metadata")).to be_empty end it "should still allow normal field access" do - expect(subject["foo"]).to eq("bar") + expect(subject.get("foo")).to eq("bar") end it "should not include the @metadata key" do @@ -599,7 +599,7 @@ end it "return the string containing the timestamp, the host and the message" do - expect(event1.to_s).to eq("#{timestamp.to_iso8601} #{event1["host"]} #{event1["message"]}") + expect(event1.to_s).to eq("#{timestamp.to_iso8601} #{event1.get("host")} #{event1.get("message")}") end end @@ -607,27 +607,27 @@ let(:event) { LogStash::Event.new({ "message" => "foo" }) } it "should invalidate target caching" do - expect(event["[a][0]"]).to be_nil + expect(event.get("[a][0]")).to be_nil - expect(event["[a][0]"] = 42).to eq(42) - expect(event["[a][0]"]).to eq(42) - expect(event["[a]"]).to eq({"0" => 42}) + expect(event.set("[a][0]", 42)).to eq(42) + expect(event.get("[a][0]")).to eq(42) + expect(event.get("[a]")).to eq({"0" => 42}) - expect(event["[a]"] = [42, 24]).to eq([42, 24]) - expect(event["[a]"]).to eq([42, 24]) + expect(event.set("[a]", [42, 24])).to eq([42, 24]) + expect(event.get("[a]")).to eq([42, 24]) - expect(event["[a][0]"]).to eq(42) + expect(event.get("[a][0]")).to eq(42) - expect(event["[a]"] = [24, 42]).to eq([24, 42]) - expect(event["[a][0]"]).to eq(24) + expect(event.set("[a]", [24, 42])).to eq([24, 42]) + expect(event.get("[a][0]")).to eq(24) - expect(event["[a][0]"] = {"a "=> 99, "b" => 98}).to eq({"a "=> 99, "b" => 98}) - expect(event["[a][0]"]).to eq({"a "=> 99, "b" => 98}) + expect(event.set("[a][0]", {"a "=> 99, "b" => 98})).to eq({"a "=> 99, "b" => 98}) + expect(event.get("[a][0]")).to eq({"a "=> 99, "b" => 98}) - expect(event["[a]"]).to eq([{"a "=> 99, "b" => 98}, 42]) - expect(event["[a][0]"]).to eq({"a "=> 99, "b" => 98}) - expect(event["[a][1]"]).to eq(42) - expect(event["[a][0][b]"]).to eq(98) + expect(event.get("[a]")).to eq([{"a "=> 99, "b" => 98}, 42]) + expect(event.get("[a][0]")).to eq({"a "=> 99, "b" => 98}) + expect(event.get("[a][1]")).to eq(42) + expect(event.get("[a][0][b]")).to eq(98) end end end diff --git a/logstash-core-plugin-api/lib/logstash-core-plugin-api/version.rb b/logstash-core-plugin-api/lib/logstash-core-plugin-api/version.rb index e83d1586c2e..ced5939392b 100644 --- a/logstash-core-plugin-api/lib/logstash-core-plugin-api/version.rb +++ b/logstash-core-plugin-api/lib/logstash-core-plugin-api/version.rb @@ -1,2 +1,2 @@ # encoding: utf-8 -LOGSTASH_CORE_PLUGIN_API = "1.0.0" +LOGSTASH_CORE_PLUGIN_API = "2.0.0" diff --git a/logstash-core/lib/logstash/config/config_ast.rb b/logstash-core/lib/logstash/config/config_ast.rb index 7a583d578df..96a7ee46a85 100644 --- a/logstash-core/lib/logstash/config/config_ast.rb +++ b/logstash-core/lib/logstash/config/config_ast.rb @@ -512,7 +512,7 @@ def compile end class Selector < RValue def compile - return "event[#{text_value.inspect}]" + return "event.get(#{text_value.inspect})" end end class SelectorElement < Node; end diff --git a/logstash-core/lib/logstash/filters/base.rb b/logstash-core/lib/logstash/filters/base.rb index c666a225bf7..d97c2342d63 100644 --- a/logstash-core/lib/logstash/filters/base.rb +++ b/logstash-core/lib/logstash/filters/base.rb @@ -187,12 +187,12 @@ def filter_matched(event) # this is important because a construct like event["tags"].delete(tag) will not work # in the current Java event implementation. see https://github.com/elastic/logstash/issues/4140 @remove_tag.each do |tag| - tags = event["tags"] + tags = event.get("tags") break if tags.nil? || tags.empty? tag = event.sprintf(tag) @logger.debug? and @logger.debug("filters/#{self.class.name}: removing tag", :tag => tag) tags.delete(tag) - event["tags"] = tags + event.set("tags", tags) end end # def filter_matched diff --git a/logstash-core/lib/logstash/inputs/base.rb b/logstash-core/lib/logstash/inputs/base.rb index d02bbd567e5..cb7a25d720b 100644 --- a/logstash-core/lib/logstash/inputs/base.rb +++ b/logstash-core/lib/logstash/inputs/base.rb @@ -97,7 +97,7 @@ def stop? protected def decorate(event) # Only set 'type' if not already set. This is backwards-compatible behavior - event["type"] = @type if @type && !event.include?("type") + event.set("type", @type) if @type && !event.include?("type") LogStash::Util::Decorators.add_fields(@add_field,event,"inputs/#{self.class.name}") LogStash::Util::Decorators.add_tags(@tags,event,"inputs/#{self.class.name}") diff --git a/logstash-core/lib/logstash/util/decorators.rb b/logstash-core/lib/logstash/util/decorators.rb index 265656e5ce9..c7807e883b1 100644 --- a/logstash-core/lib/logstash/util/decorators.rb +++ b/logstash-core/lib/logstash/util/decorators.rb @@ -7,7 +7,7 @@ module LogStash::Util # Decorators provides common manipulation on the event data. module Decorators extend self - + @logger = Cabin::Channel.get(LogStash) # fields is a hash of field => value @@ -22,11 +22,11 @@ def add_fields(fields,event, pluginname) # note below that the array field needs to be updated then reassigned to the event. # this is important because a construct like event[field] << v will not work # in the current Java event implementation. see https://github.com/elastic/logstash/issues/4140 - a = Array(event[field]) + a = Array(event.get(field)) a << v - event[field] = a + event.set(field, a) else - event[field] = v + event.set(field, v) end @logger.debug? and @logger.debug("#{pluginname}: adding value to field", :field => field, :value => value) end @@ -41,9 +41,9 @@ def add_tags(tags, event, pluginname) # note below that the tags array field needs to be updated then reassigned to the event. # this is important because a construct like event["tags"] << tag will not work # in the current Java event implementation. see https://github.com/elastic/logstash/issues/4140 - tags = event["tags"] || [] + tags = event.get("tags") || [] tags << tag - event["tags"] = tags + event.set("tags", tags) end end diff --git a/logstash-core/spec/conditionals_spec.rb b/logstash-core/spec/conditionals_spec.rb index dab6fc901e3..c4c68ec4c58 100644 --- a/logstash-core/spec/conditionals_spec.rb +++ b/logstash-core/spec/conditionals_spec.rb @@ -64,24 +64,24 @@ def conditional(expression, &block) CONFIG sample({"foo" => "bar"}) do - expect(subject["always"]).to eq("awesome") - expect(subject["hello"]).to eq("world") - expect(subject["fancy"]).to be_nil - expect(subject["free"]).to be_nil + expect(subject.get("always")).to eq("awesome") + expect(subject.get("hello")).to eq("world") + expect(subject.get("fancy")).to be_nil + expect(subject.get("free")).to be_nil end sample({"notfoo" => "bar"}) do - expect(subject["always"]).to eq("awesome") - expect(subject["hello"]).to be_nil - expect(subject["fancy"]).to be_nil - expect(subject["free"]).to eq("hugs") + expect(subject.get("always")).to eq("awesome") + expect(subject.get("hello")).to be_nil + expect(subject.get("fancy")).to be_nil + expect(subject.get("free")).to eq("hugs") end sample({"bar" => "baz"}) do - expect(subject["always"]).to eq("awesome") - expect(subject["hello"]).to be_nil - expect(subject["fancy"]).to eq("pants") - expect(subject["free"]).to be_nil + expect(subject.get("always")).to eq("awesome") + expect(subject.get("hello")).to be_nil + expect(subject.get("fancy")).to eq("pants") + expect(subject.get("free")).to be_nil end end @@ -102,31 +102,31 @@ def conditional(expression, &block) CONFIG sample("foo" => "bar", "nest" => 124) do - expect(subject["always"]).to be_nil - expect(subject["hello"]).to be_nil - expect(subject["fancy"]).to be_nil - expect(subject["free"]).to be_nil + expect(subject.get("always")).to be_nil + expect(subject.get("hello")).to be_nil + expect(subject.get("fancy")).to be_nil + expect(subject.get("free")).to be_nil end sample("foo" => "bar", "nest" => 123) do - expect(subject["always"]).to eq("awesome") - expect(subject["hello"]).to eq("world") - expect(subject["fancy"]).to be_nil - expect(subject["free"]).to be_nil + expect(subject.get("always")).to eq("awesome") + expect(subject.get("hello")).to eq("world") + expect(subject.get("fancy")).to be_nil + expect(subject.get("free")).to be_nil end sample("notfoo" => "bar", "nest" => 123) do - expect(subject["always"]).to eq("awesome") - expect(subject["hello"]).to be_nil - expect(subject["fancy"]).to be_nil - expect(subject["free"]).to eq("hugs") + expect(subject.get("always")).to eq("awesome") + expect(subject.get("hello")).to be_nil + expect(subject.get("fancy")).to be_nil + expect(subject.get("free")).to eq("hugs") end sample("bar" => "baz", "nest" => 123) do - expect(subject["always"]).to eq("awesome") - expect(subject["hello"]).to be_nil - expect(subject["fancy"]).to eq("pants") - expect(subject["free"]).to be_nil + expect(subject.get("always")).to eq("awesome") + expect(subject.get("hello")).to be_nil + expect(subject.get("fancy")).to eq("pants") + expect(subject.get("free")).to be_nil end end @@ -140,7 +140,7 @@ def conditional(expression, &block) CONFIG sample("foo" => 123, "bar" => 123) do - expect(subject["tags"] ).to include("woot") + expect(subject.get("tags") ).to include("woot") end end @@ -169,12 +169,12 @@ def conditional(expression, &block) CONFIG sample("foo" => "foo", "foobar" => "foobar", "greeting" => "hello world") do - expect(subject["tags"]).to include("field in field") - expect(subject["tags"]).to include("field in string") - expect(subject["tags"]).to include("string in field") - expect(subject["tags"]).to include("field in list") - expect(subject["tags"]).not_to include("shouldnotexist") - expect(subject["tags"]).to include("shouldexist") + expect(subject.get("tags")).to include("field in field") + expect(subject.get("tags")).to include("field in string") + expect(subject.get("tags")).to include("string in field") + expect(subject.get("tags")).to include("field in list") + expect(subject.get("tags")).not_to include("shouldnotexist") + expect(subject.get("tags")).to include("shouldexist") end end @@ -192,107 +192,107 @@ def conditional(expression, &block) sample("foo" => "foo", "somelist" => [ "one", "two" ], "foobar" => "foobar", "greeting" => "hello world", "tags" => [ "fancypantsy" ]) do # verify the original exists - expect(subject["tags"]).to include("fancypantsy") + expect(subject.get("tags")).to include("fancypantsy") - expect(subject["tags"]).to include("baz") - expect(subject["tags"]).not_to include("foo") - expect(subject["tags"]).to include("notfoo") - expect(subject["tags"]).to include("notsomelist") - expect(subject["tags"]).not_to include("somelist") - expect(subject["tags"]).to include("no string in missing field") + expect(subject.get("tags")).to include("baz") + expect(subject.get("tags")).not_to include("foo") + expect(subject.get("tags")).to include("notfoo") + expect(subject.get("tags")).to include("notsomelist") + expect(subject.get("tags")).not_to include("somelist") + expect(subject.get("tags")).to include("no string in missing field") end end describe "operators" do conditional "[message] == 'sample'" do - sample("sample") { expect(subject["tags"] ).to include("success") } - sample("different") { expect(subject["tags"] ).to include("failure") } + sample("sample") { expect(subject.get("tags") ).to include("success") } + sample("different") { expect(subject.get("tags") ).to include("failure") } end conditional "[message] != 'sample'" do - sample("sample") { expect(subject["tags"] ).to include("failure") } - sample("different") { expect(subject["tags"] ).to include("success") } + sample("sample") { expect(subject.get("tags") ).to include("failure") } + sample("different") { expect(subject.get("tags") ).to include("success") } end conditional "[message] < 'sample'" do - sample("apple") { expect(subject["tags"] ).to include("success") } - sample("zebra") { expect(subject["tags"] ).to include("failure") } + sample("apple") { expect(subject.get("tags") ).to include("success") } + sample("zebra") { expect(subject.get("tags") ).to include("failure") } end conditional "[message] > 'sample'" do - sample("zebra") { expect(subject["tags"] ).to include("success") } - sample("apple") { expect(subject["tags"] ).to include("failure") } + sample("zebra") { expect(subject.get("tags") ).to include("success") } + sample("apple") { expect(subject.get("tags") ).to include("failure") } end conditional "[message] <= 'sample'" do - sample("apple") { expect(subject["tags"] ).to include("success") } - sample("zebra") { expect(subject["tags"] ).to include("failure") } - sample("sample") { expect(subject["tags"] ).to include("success") } + sample("apple") { expect(subject.get("tags") ).to include("success") } + sample("zebra") { expect(subject.get("tags") ).to include("failure") } + sample("sample") { expect(subject.get("tags") ).to include("success") } end conditional "[message] >= 'sample'" do - sample("zebra") { expect(subject["tags"] ).to include("success") } - sample("sample") { expect(subject["tags"] ).to include("success") } - sample("apple") { expect(subject["tags"] ).to include("failure") } + sample("zebra") { expect(subject.get("tags") ).to include("success") } + sample("sample") { expect(subject.get("tags") ).to include("success") } + sample("apple") { expect(subject.get("tags") ).to include("failure") } end conditional "[message] =~ /sample/" do - sample("apple") { expect(subject["tags"] ).to include("failure") } - sample("sample") { expect(subject["tags"] ).to include("success") } - sample("some sample") { expect(subject["tags"] ).to include("success") } + sample("apple") { expect(subject.get("tags") ).to include("failure") } + sample("sample") { expect(subject.get("tags") ).to include("success") } + sample("some sample") { expect(subject.get("tags") ).to include("success") } end conditional "[message] !~ /sample/" do - sample("apple") { expect(subject["tags"]).to include("success") } - sample("sample") { expect(subject["tags"]).to include("failure") } - sample("some sample") { expect(subject["tags"]).to include("failure") } + sample("apple") { expect(subject.get("tags")).to include("success") } + sample("sample") { expect(subject.get("tags")).to include("failure") } + sample("some sample") { expect(subject.get("tags")).to include("failure") } end end describe "negated expressions" do conditional "!([message] == 'sample')" do - sample("sample") { expect(subject["tags"]).not_to include("success") } - sample("different") { expect(subject["tags"]).not_to include("failure") } + sample("sample") { expect(subject.get("tags")).not_to include("success") } + sample("different") { expect(subject.get("tags")).not_to include("failure") } end conditional "!([message] != 'sample')" do - sample("sample") { expect(subject["tags"]).not_to include("failure") } - sample("different") { expect(subject["tags"]).not_to include("success") } + sample("sample") { expect(subject.get("tags")).not_to include("failure") } + sample("different") { expect(subject.get("tags")).not_to include("success") } end conditional "!([message] < 'sample')" do - sample("apple") { expect(subject["tags"]).not_to include("success") } - sample("zebra") { expect(subject["tags"]).not_to include("failure") } + sample("apple") { expect(subject.get("tags")).not_to include("success") } + sample("zebra") { expect(subject.get("tags")).not_to include("failure") } end conditional "!([message] > 'sample')" do - sample("zebra") { expect(subject["tags"]).not_to include("success") } - sample("apple") { expect(subject["tags"]).not_to include("failure") } + sample("zebra") { expect(subject.get("tags")).not_to include("success") } + sample("apple") { expect(subject.get("tags")).not_to include("failure") } end conditional "!([message] <= 'sample')" do - sample("apple") { expect(subject["tags"]).not_to include("success") } - sample("zebra") { expect(subject["tags"]).not_to include("failure") } - sample("sample") { expect(subject["tags"]).not_to include("success") } + sample("apple") { expect(subject.get("tags")).not_to include("success") } + sample("zebra") { expect(subject.get("tags")).not_to include("failure") } + sample("sample") { expect(subject.get("tags")).not_to include("success") } end conditional "!([message] >= 'sample')" do - sample("zebra") { expect(subject["tags"]).not_to include("success") } - sample("sample") { expect(subject["tags"]).not_to include("success") } - sample("apple") { expect(subject["tags"]).not_to include("failure") } + sample("zebra") { expect(subject.get("tags")).not_to include("success") } + sample("sample") { expect(subject.get("tags")).not_to include("success") } + sample("apple") { expect(subject.get("tags")).not_to include("failure") } end conditional "!([message] =~ /sample/)" do - sample("apple") { expect(subject["tags"]).not_to include("failure") } - sample("sample") { expect(subject["tags"]).not_to include("success") } - sample("some sample") { expect(subject["tags"]).not_to include("success") } + sample("apple") { expect(subject.get("tags")).not_to include("failure") } + sample("sample") { expect(subject.get("tags")).not_to include("success") } + sample("some sample") { expect(subject.get("tags")).not_to include("success") } end conditional "!([message] !~ /sample/)" do - sample("apple") { expect(subject["tags"]).not_to include("success") } - sample("sample") { expect(subject["tags"]).not_to include("failure") } - sample("some sample") { expect(subject["tags"]).not_to include("failure") } + sample("apple") { expect(subject.get("tags")).not_to include("success") } + sample("sample") { expect(subject.get("tags")).not_to include("failure") } + sample("some sample") { expect(subject.get("tags")).not_to include("failure") } end end @@ -300,47 +300,47 @@ def conditional(expression, &block) describe "value as an expression" do # testing that a field has a value should be true. conditional "[message]" do - sample("apple") { expect(subject["tags"]).to include("success") } - sample("sample") { expect(subject["tags"]).to include("success") } - sample("some sample") { expect(subject["tags"]).to include("success") } + sample("apple") { expect(subject.get("tags")).to include("success") } + sample("sample") { expect(subject.get("tags")).to include("success") } + sample("some sample") { expect(subject.get("tags")).to include("success") } end # testing that a missing field has a value should be false. conditional "[missing]" do - sample("apple") { expect(subject["tags"]).to include("failure") } - sample("sample") { expect(subject["tags"]).to include("failure") } - sample("some sample") { expect(subject["tags"]).to include("failure") } + sample("apple") { expect(subject.get("tags")).to include("failure") } + sample("sample") { expect(subject.get("tags")).to include("failure") } + sample("some sample") { expect(subject.get("tags")).to include("failure") } end end describe "logic operators" do describe "and" do conditional "[message] and [message]" do - sample("whatever") { expect(subject["tags"]).to include("success") } + sample("whatever") { expect(subject.get("tags")).to include("success") } end conditional "[message] and ![message]" do - sample("whatever") { expect(subject["tags"]).to include("failure") } + sample("whatever") { expect(subject.get("tags")).to include("failure") } end conditional "![message] and [message]" do - sample("whatever") { expect(subject["tags"]).to include("failure") } + sample("whatever") { expect(subject.get("tags")).to include("failure") } end conditional "![message] and ![message]" do - sample("whatever") { expect(subject["tags"]).to include("failure") } + sample("whatever") { expect(subject.get("tags")).to include("failure") } end end describe "or" do conditional "[message] or [message]" do - sample("whatever") { expect(subject["tags"]).to include("success") } + sample("whatever") { expect(subject.get("tags")).to include("success") } end conditional "[message] or ![message]" do - sample("whatever") { expect(subject["tags"]).to include("success") } + sample("whatever") { expect(subject.get("tags")).to include("success") } end conditional "![message] or [message]" do - sample("whatever") { expect(subject["tags"]).to include("success") } + sample("whatever") { expect(subject.get("tags")).to include("success") } end conditional "![message] or ![message]" do - sample("whatever") { expect(subject["tags"]).to include("failure") } + sample("whatever") { expect(subject.get("tags")).to include("failure") } end end end @@ -348,19 +348,19 @@ def conditional(expression, &block) describe "field references" do conditional "[field with space]" do sample("field with space" => "hurray") do - expect(subject["tags"]).to include("success") + expect(subject.get("tags")).to include("success") end end conditional "[field with space] == 'hurray'" do sample("field with space" => "hurray") do - expect(subject["tags"]).to include("success") + expect(subject.get("tags")).to include("success") end end conditional "[nested field][reference with][some spaces] == 'hurray'" do sample({"nested field" => { "reference with" => { "some spaces" => "hurray" } } }) do - expect(subject["tags"]).to include("success") + expect(subject.get("tags")).to include("success") end end end @@ -385,13 +385,13 @@ def conditional(expression, &block) expect(subject).to be_an(Array) expect(subject.length).to eq(2) - expect(subject[0]["type"]).to eq("original") - expect(subject[0]["cond1"]).to eq("true") - expect(subject[0]["cond2"]).to eq(nil) + expect(subject[0].get("type")).to eq("original") + expect(subject[0].get("cond1")).to eq("true") + expect(subject[0].get("cond2")).to eq(nil) - expect(subject[1]["type"]).to eq("clone") - # expect(subject[1]["cond1"]).to eq(nil) - # expect(subject[1]["cond2"]).to eq("true") + expect(subject[1].get("type")).to eq("clone") + # expect(subject[1].get("cond1")).to eq(nil) + # expect(subject[1].get("cond2")).to eq("true") end end @@ -413,16 +413,16 @@ def conditional(expression, &block) sample({"type" => "original"}) do # puts subject.inspect - expect(subject[0]["cond1"]).to eq(nil) - expect(subject[0]["cond2"]).to eq(nil) + expect(subject[0].get("cond1")).to eq(nil) + expect(subject[0].get("cond2")).to eq(nil) - expect(subject[1]["type"]).to eq("clone1") - expect(subject[1]["cond1"]).to eq("true") - expect(subject[1]["cond2"]).to eq(nil) + expect(subject[1].get("type")).to eq("clone1") + expect(subject[1].get("cond1")).to eq("true") + expect(subject[1].get("cond2")).to eq(nil) - expect(subject[2]["type"]).to eq("clone2") - expect(subject[2]["cond1"]).to eq(nil) - expect(subject[2]["cond2"]).to eq("true") + expect(subject[2].get("type")).to eq("clone2") + expect(subject[2].get("cond1")).to eq(nil) + expect(subject[2].get("cond2")).to eq("true") end end diff --git a/logstash-core/spec/logstash/filters/base_spec.rb b/logstash-core/spec/logstash/filters/base_spec.rb index 177c44dcb8c..26c5d6c5438 100644 --- a/logstash-core/spec/logstash/filters/base_spec.rb +++ b/logstash-core/spec/logstash/filters/base_spec.rb @@ -62,7 +62,7 @@ def filter(event) CONFIG sample "example" do - insist { subject["new_field"] } == ["new_value", "new_value_2"] + insist { subject.get("new_field") } == ["new_value", "new_value_2"] end end @@ -76,7 +76,7 @@ def filter(event) CONFIG sample("type" => "noop") do - insist { subject["tags"] } == ["test"] + insist { subject.get("tags") } == ["test"] end end @@ -90,11 +90,11 @@ def filter(event) CONFIG sample("type" => "noop") do - insist { subject["tags"] } == ["test"] + insist { subject.get("tags") } == ["test"] end sample("type" => "noop", "tags" => ["t1", "t2"]) do - insist { subject["tags"] } == ["t1", "t2", "test"] + insist { subject.get("tags") } == ["t1", "t2", "test"] end end @@ -108,19 +108,19 @@ def filter(event) CONFIG sample("type" => "noop") do - insist { subject["tags"] } == ["test"] + insist { subject.get("tags") } == ["test"] end sample("type" => "noop", "tags" => ["t1"]) do - insist { subject["tags"] } == ["t1", "test"] + insist { subject.get("tags") } == ["t1", "test"] end sample("type" => "noop", "tags" => ["t1", "t2"]) do - insist { subject["tags"] } == ["t1", "t2", "test"] + insist { subject.get("tags") } == ["t1", "t2", "test"] end sample("type" => "noop", "tags" => ["t1", "t2", "t3"]) do - insist { subject["tags"] } == ["t1", "t2", "t3", "test"] + insist { subject.get("tags") } == ["t1", "t2", "t3", "test"] end end @@ -134,27 +134,27 @@ def filter(event) CONFIG sample("type" => "noop", "tags" => ["t4"]) do - insist { subject["tags"] } == ["t4"] + insist { subject.get("tags") } == ["t4"] end sample("type" => "noop", "tags" => ["t1", "t2", "t3"]) do - insist { subject["tags"] } == ["t1"] + insist { subject.get("tags") } == ["t1"] end # also test from Json deserialized data to test the handling of native Java collections by JrJackson # see https://github.com/elastic/logstash/issues/2261 sample(LogStash::Json.load("{\"type\":\"noop\", \"tags\":[\"t1\", \"t2\", \"t3\"]}")) do - insist { subject["tags"] } == ["t1"] + insist { subject.get("tags") } == ["t1"] end sample("type" => "noop", "tags" => ["t1", "t2"]) do - insist { subject["tags"] } == ["t1"] + insist { subject.get("tags") } == ["t1"] end # also test from Json deserialized data to test the handling of native Java collections by JrJackson # see https://github.com/elastic/logstash/issues/2261 sample(LogStash::Json.load("{\"type\":\"noop\", \"tags\":[\"t1\", \"t2\"]}")) do - insist { subject["tags"] } == ["t1"] + insist { subject.get("tags") } == ["t1"] end end @@ -168,13 +168,13 @@ def filter(event) CONFIG sample("type" => "noop", "tags" => ["t1", "goaway", "t3"], "blackhole" => "goaway") do - insist { subject["tags"] } == ["t1", "t3"] + insist { subject.get("tags") } == ["t1", "t3"] end # also test from Json deserialized data to test the handling of native Java collections by JrJackson # see https://github.com/elastic/logstash/issues/2261 sample(LogStash::Json.load("{\"type\":\"noop\", \"tags\":[\"t1\", \"goaway\", \"t3\"], \"blackhole\":\"goaway\"}")) do - insist { subject["tags"] } == ["t1", "t3"] + insist { subject.get("tags") } == ["t1", "t3"] end end @@ -230,7 +230,7 @@ def filter(event) sample("type" => "noop", "t1" => ["t2", "t3"]) do insist { subject }.include?("t1") - insist { subject["[t1][0]"] } == "t3" + insist { subject.get("[t1][0]") } == "t3" end end diff --git a/logstash-core/spec/logstash/inputs/base_spec.rb b/logstash-core/spec/logstash/inputs/base_spec.rb index d87f07b49f6..a3f01fa89e1 100644 --- a/logstash-core/spec/logstash/inputs/base_spec.rb +++ b/logstash-core/spec/logstash/inputs/base_spec.rb @@ -15,50 +15,50 @@ def register; end input = LogStash::Inputs::NOOP.new("tags" => "value") evt = LogStash::Event.new({"type" => "noop"}) input.instance_eval {decorate(evt)} - expect(evt["tags"]).to eq(["value"]) + expect(evt.get("tags")).to eq(["value"]) end it "should add multiple tag" do input = LogStash::Inputs::NOOP.new("tags" => ["value1","value2"]) evt = LogStash::Event.new({"type" => "noop"}) input.instance_eval {decorate(evt)} - expect(evt["tags"]).to eq(["value1","value2"]) + expect(evt.get("tags")).to eq(["value1","value2"]) end it "should allow duplicates tag" do input = LogStash::Inputs::NOOP.new("tags" => ["value","value"]) evt = LogStash::Event.new({"type" => "noop"}) input.instance_eval {decorate(evt)} - expect(evt["tags"]).to eq(["value","value"]) + expect(evt.get("tags")).to eq(["value","value"]) end it "should add tag with sprintf" do input = LogStash::Inputs::NOOP.new("tags" => "%{type}") evt = LogStash::Event.new({"type" => "noop"}) input.instance_eval {decorate(evt)} - expect(evt["tags"]).to eq(["noop"]) + expect(evt.get("tags")).to eq(["noop"]) end it "should add single field" do input = LogStash::Inputs::NOOP.new("add_field" => {"field" => "value"}) evt = LogStash::Event.new({"type" => "noop"}) input.instance_eval {decorate(evt)} - expect(evt["field"]).to eq("value") + expect(evt.get("field")).to eq("value") end it "should add single field with sprintf" do input = LogStash::Inputs::NOOP.new("add_field" => {"%{type}" => "%{type}"}) evt = LogStash::Event.new({"type" => "noop"}) input.instance_eval {decorate(evt)} - expect(evt["noop"]).to eq("noop") + expect(evt.get("noop")).to eq("noop") end it "should add multiple field" do input = LogStash::Inputs::NOOP.new("add_field" => {"field" => ["value1", "value2"], "field2" => "value"}) evt = LogStash::Event.new({"type" => "noop"}) input.instance_eval {decorate(evt)} - expect(evt["field"]).to eq(["value1","value2"]) - expect(evt["field2"]).to eq("value") + expect(evt.get("field")).to eq(["value1","value2"]) + expect(evt.get("field2")).to eq("value") end end diff --git a/logstash-core/spec/logstash/pipeline_spec.rb b/logstash-core/spec/logstash/pipeline_spec.rb index f7ef5c36221..e17ddff719c 100644 --- a/logstash-core/spec/logstash/pipeline_spec.rb +++ b/logstash-core/spec/logstash/pipeline_spec.rb @@ -279,7 +279,7 @@ class TestPipeline < LogStash::Pipeline CONFIG sample("hello") do - expect(subject["message"]).to eq("hello") + expect(subject.get("message")).to eq("hello") end end @@ -299,10 +299,10 @@ class TestPipeline < LogStash::Pipeline sample(["foo", "bar"]) do expect(subject.size).to eq(2) - expect(subject[0]["message"]).to eq("foo\nbar") - expect(subject[0]["type"]).to be_nil - expect(subject[1]["message"]).to eq("foo\nbar") - expect(subject[1]["type"]).to eq("clone1") + expect(subject[0].get("message")).to eq("foo\nbar") + expect(subject[0].get("type")).to be_nil + expect(subject[1].get("message")).to eq("foo\nbar") + expect(subject[1].get("type")).to eq("clone1") end end end @@ -354,17 +354,17 @@ class TestPipeline < LogStash::Pipeline sample("hello") do expect(subject.size).to eq(3) - expect(subject[0]["message"]).to eq("hello") - expect(subject[0]["type"]).to be_nil - expect(subject[0]["foo"]).to eq("bar") + expect(subject[0].get("message")).to eq("hello") + expect(subject[0].get("type")).to be_nil + expect(subject[0].get("foo")).to eq("bar") - expect(subject[1]["message"]).to eq("hello") - expect(subject[1]["type"]).to eq("clone1") - expect(subject[1]["foo"]).to eq("bar") + expect(subject[1].get("message")).to eq("hello") + expect(subject[1].get("type")).to eq("clone1") + expect(subject[1].get("foo")).to eq("bar") - expect(subject[2]["message"]).to eq("hello") - expect(subject[2]["type"]).to eq("clone2") - expect(subject[2]["foo"]).to eq("bar") + expect(subject[2].get("message")).to eq("hello") + expect(subject[2].get("type")).to eq("clone2") + expect(subject[2].get("foo")).to eq("bar") end end end @@ -441,7 +441,7 @@ class TestPipeline < LogStash::Pipeline # give us a bit of time to flush the events wait(5).for do next unless output && output.events && output.events.first - output.events.first["message"].split("\n").count + output.events.first.get("message").split("\n").count end.to eq(number_of_events) pipeline.shutdown end