logstash-output-neo4j 0.9.0-java

Sign up to get free protection for your applications and to get access to all the features.
@@ -0,0 +1,7 @@
1
+ ---
2
+ SHA1:
3
+ metadata.gz: e03b3eb519a8c9d1e7ddc05fedad7e490956f908
4
+ data.tar.gz: e063e60b5f48c8ea76caac9a4e302d5acb1fc78b
5
+ SHA512:
6
+ metadata.gz: 4291f9e8c6219e043af8faebc91188cad2beefa49070bc9b5317f326398565a954ffaa3df12be5232636af821d0a6e56fb8d468cf56ed559b8e5b4d5b8268a59
7
+ data.tar.gz: d18c4f58f9ae52f2af13d91e08bd4efc00f9ff816b00a72d60a6328c52467e96a8faeb4baa176bf56b37a6c02ad29da9451fb291908d32188ab0b603fc01d39b
@@ -0,0 +1,34 @@
1
+ *.gem
2
+ *.rbc
3
+ /.config
4
+ /coverage/
5
+ /InstalledFiles
6
+ /pkg/
7
+ /spec/reports/
8
+ /test/tmp/
9
+ /test/version_tmp/
10
+ /tmp/
11
+
12
+ ## Specific to RubyMotion:
13
+ .dat*
14
+ .repl_history
15
+ build/
16
+
17
+ ## Documentation cache and generated files:
18
+ /.yardoc/
19
+ /_yardoc/
20
+ /doc/
21
+ /rdoc/
22
+
23
+ ## Environment normalisation:
24
+ /.bundle/
25
+ /lib/bundler/man/
26
+
27
+ # for a library or gem, you might want to ignore these files since the code is
28
+ # intended to run in multiple environments; otherwise, check them in:
29
+ Gemfile.lock
30
+ .ruby-version
31
+ .ruby-gemset
32
+
33
+ # unless supporting rvm < 1.11.0 or doing something fancy, ignore this:
34
+ .rvmrc
data/.rspec ADDED
@@ -0,0 +1,3 @@
1
+ --format documentation
2
+ --color
3
+
data/Gemfile ADDED
@@ -0,0 +1,4 @@
1
+ source 'https://rubygems.org'
2
+ gemspec
3
+
4
+ gem "logstash", :git => "git@github.com:elasticsearch/logstash.git", :branch => "1.5"
data/LICENSE ADDED
@@ -0,0 +1,202 @@
1
+ Apache License
2
+ Version 2.0, January 2004
3
+ http://www.apache.org/licenses/
4
+
5
+ TERMS AND CONDITIONS FOR USE, REPRODUCTION, AND DISTRIBUTION
6
+
7
+ 1. Definitions.
8
+
9
+ "License" shall mean the terms and conditions for use, reproduction,
10
+ and distribution as defined by Sections 1 through 9 of this document.
11
+
12
+ "Licensor" shall mean the copyright owner or entity authorized by
13
+ the copyright owner that is granting the License.
14
+
15
+ "Legal Entity" shall mean the union of the acting entity and all
16
+ other entities that control, are controlled by, or are under common
17
+ control with that entity. For the purposes of this definition,
18
+ "control" means (i) the power, direct or indirect, to cause the
19
+ direction or management of such entity, whether by contract or
20
+ otherwise, or (ii) ownership of fifty percent (50%) or more of the
21
+ outstanding shares, or (iii) beneficial ownership of such entity.
22
+
23
+ "You" (or "Your") shall mean an individual or Legal Entity
24
+ exercising permissions granted by this License.
25
+
26
+ "Source" form shall mean the preferred form for making modifications,
27
+ including but not limited to software source code, documentation
28
+ source, and configuration files.
29
+
30
+ "Object" form shall mean any form resulting from mechanical
31
+ transformation or translation of a Source form, including but
32
+ not limited to compiled object code, generated documentation,
33
+ and conversions to other media types.
34
+
35
+ "Work" shall mean the work of authorship, whether in Source or
36
+ Object form, made available under the License, as indicated by a
37
+ copyright notice that is included in or attached to the work
38
+ (an example is provided in the Appendix below).
39
+
40
+ "Derivative Works" shall mean any work, whether in Source or Object
41
+ form, that is based on (or derived from) the Work and for which the
42
+ editorial revisions, annotations, elaborations, or other modifications
43
+ represent, as a whole, an original work of authorship. For the purposes
44
+ of this License, Derivative Works shall not include works that remain
45
+ separable from, or merely link (or bind by name) to the interfaces of,
46
+ the Work and Derivative Works thereof.
47
+
48
+ "Contribution" shall mean any work of authorship, including
49
+ the original version of the Work and any modifications or additions
50
+ to that Work or Derivative Works thereof, that is intentionally
51
+ submitted to Licensor for inclusion in the Work by the copyright owner
52
+ or by an individual or Legal Entity authorized to submit on behalf of
53
+ the copyright owner. For the purposes of this definition, "submitted"
54
+ means any form of electronic, verbal, or written communication sent
55
+ to the Licensor or its representatives, including but not limited to
56
+ communication on electronic mailing lists, source code control systems,
57
+ and issue tracking systems that are managed by, or on behalf of, the
58
+ Licensor for the purpose of discussing and improving the Work, but
59
+ excluding communication that is conspicuously marked or otherwise
60
+ designated in writing by the copyright owner as "Not a Contribution."
61
+
62
+ "Contributor" shall mean Licensor and any individual or Legal Entity
63
+ on behalf of whom a Contribution has been received by Licensor and
64
+ subsequently incorporated within the Work.
65
+
66
+ 2. Grant of Copyright License. Subject to the terms and conditions of
67
+ this License, each Contributor hereby grants to You a perpetual,
68
+ worldwide, non-exclusive, no-charge, royalty-free, irrevocable
69
+ copyright license to reproduce, prepare Derivative Works of,
70
+ publicly display, publicly perform, sublicense, and distribute the
71
+ Work and such Derivative Works in Source or Object form.
72
+
73
+ 3. Grant of Patent License. Subject to the terms and conditions of
74
+ this License, each Contributor hereby grants to You a perpetual,
75
+ worldwide, non-exclusive, no-charge, royalty-free, irrevocable
76
+ (except as stated in this section) patent license to make, have made,
77
+ use, offer to sell, sell, import, and otherwise transfer the Work,
78
+ where such license applies only to those patent claims licensable
79
+ by such Contributor that are necessarily infringed by their
80
+ Contribution(s) alone or by combination of their Contribution(s)
81
+ with the Work to which such Contribution(s) was submitted. If You
82
+ institute patent litigation against any entity (including a
83
+ cross-claim or counterclaim in a lawsuit) alleging that the Work
84
+ or a Contribution incorporated within the Work constitutes direct
85
+ or contributory patent infringement, then any patent licenses
86
+ granted to You under this License for that Work shall terminate
87
+ as of the date such litigation is filed.
88
+
89
+ 4. Redistribution. You may reproduce and distribute copies of the
90
+ Work or Derivative Works thereof in any medium, with or without
91
+ modifications, and in Source or Object form, provided that You
92
+ meet the following conditions:
93
+
94
+ (a) You must give any other recipients of the Work or
95
+ Derivative Works a copy of this License; and
96
+
97
+ (b) You must cause any modified files to carry prominent notices
98
+ stating that You changed the files; and
99
+
100
+ (c) You must retain, in the Source form of any Derivative Works
101
+ that You distribute, all copyright, patent, trademark, and
102
+ attribution notices from the Source form of the Work,
103
+ excluding those notices that do not pertain to any part of
104
+ the Derivative Works; and
105
+
106
+ (d) If the Work includes a "NOTICE" text file as part of its
107
+ distribution, then any Derivative Works that You distribute must
108
+ include a readable copy of the attribution notices contained
109
+ within such NOTICE file, excluding those notices that do not
110
+ pertain to any part of the Derivative Works, in at least one
111
+ of the following places: within a NOTICE text file distributed
112
+ as part of the Derivative Works; within the Source form or
113
+ documentation, if provided along with the Derivative Works; or,
114
+ within a display generated by the Derivative Works, if and
115
+ wherever such third-party notices normally appear. The contents
116
+ of the NOTICE file are for informational purposes only and
117
+ do not modify the License. You may add Your own attribution
118
+ notices within Derivative Works that You distribute, alongside
119
+ or as an addendum to the NOTICE text from the Work, provided
120
+ that such additional attribution notices cannot be construed
121
+ as modifying the License.
122
+
123
+ You may add Your own copyright statement to Your modifications and
124
+ may provide additional or different license terms and conditions
125
+ for use, reproduction, or distribution of Your modifications, or
126
+ for any such Derivative Works as a whole, provided Your use,
127
+ reproduction, and distribution of the Work otherwise complies with
128
+ the conditions stated in this License.
129
+
130
+ 5. Submission of Contributions. Unless You explicitly state otherwise,
131
+ any Contribution intentionally submitted for inclusion in the Work
132
+ by You to the Licensor shall be under the terms and conditions of
133
+ this License, without any additional terms or conditions.
134
+ Notwithstanding the above, nothing herein shall supersede or modify
135
+ the terms of any separate license agreement you may have executed
136
+ with Licensor regarding such Contributions.
137
+
138
+ 6. Trademarks. This License does not grant permission to use the trade
139
+ names, trademarks, service marks, or product names of the Licensor,
140
+ except as required for reasonable and customary use in describing the
141
+ origin of the Work and reproducing the content of the NOTICE file.
142
+
143
+ 7. Disclaimer of Warranty. Unless required by applicable law or
144
+ agreed to in writing, Licensor provides the Work (and each
145
+ Contributor provides its Contributions) on an "AS IS" BASIS,
146
+ WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or
147
+ implied, including, without limitation, any warranties or conditions
148
+ of TITLE, NON-INFRINGEMENT, MERCHANTABILITY, or FITNESS FOR A
149
+ PARTICULAR PURPOSE. You are solely responsible for determining the
150
+ appropriateness of using or redistributing the Work and assume any
151
+ risks associated with Your exercise of permissions under this License.
152
+
153
+ 8. Limitation of Liability. In no event and under no legal theory,
154
+ whether in tort (including negligence), contract, or otherwise,
155
+ unless required by applicable law (such as deliberate and grossly
156
+ negligent acts) or agreed to in writing, shall any Contributor be
157
+ liable to You for damages, including any direct, indirect, special,
158
+ incidental, or consequential damages of any character arising as a
159
+ result of this License or out of the use or inability to use the
160
+ Work (including but not limited to damages for loss of goodwill,
161
+ work stoppage, computer failure or malfunction, or any and all
162
+ other commercial damages or losses), even if such Contributor
163
+ has been advised of the possibility of such damages.
164
+
165
+ 9. Accepting Warranty or Additional Liability. While redistributing
166
+ the Work or Derivative Works thereof, You may choose to offer,
167
+ and charge a fee for, acceptance of support, warranty, indemnity,
168
+ or other liability obligations and/or rights consistent with this
169
+ License. However, in accepting such obligations, You may act only
170
+ on Your own behalf and on Your sole responsibility, not on behalf
171
+ of any other Contributor, and only if You agree to indemnify,
172
+ defend, and hold each Contributor harmless for any liability
173
+ incurred by, or claims asserted against, such Contributor by reason
174
+ of your accepting any such warranty or additional liability.
175
+
176
+ END OF TERMS AND CONDITIONS
177
+
178
+ APPENDIX: How to apply the Apache License to your work.
179
+
180
+ To apply the Apache License to your work, attach the following
181
+ boilerplate notice, with the fields enclosed by brackets "{}"
182
+ replaced with your own identifying information. (Don't include
183
+ the brackets!) The text should be enclosed in the appropriate
184
+ comment syntax for the file format. We also recommend that a
185
+ file or class name and description of purpose be included on the
186
+ same "printed page" as the copyright notice for easier
187
+ identification within third-party archives.
188
+
189
+ Copyright {yyyy} {name of copyright owner}
190
+
191
+ Licensed under the Apache License, Version 2.0 (the "License");
192
+ you may not use this file except in compliance with the License.
193
+ You may obtain a copy of the License at
194
+
195
+ http://www.apache.org/licenses/LICENSE-2.0
196
+
197
+ Unless required by applicable law or agreed to in writing, software
198
+ distributed under the License is distributed on an "AS IS" BASIS,
199
+ WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
200
+ See the License for the specific language governing permissions and
201
+ limitations under the License.
202
+
@@ -0,0 +1,4 @@
1
+ logstash-output-neo4j
2
+ =====================
3
+
4
+ This is the first version of a Logstash Neo4j output plugin
@@ -0,0 +1,22 @@
1
+ ROOT = File.expand_path(File.dirname(__FILE__))
2
+ $LOAD_PATH.unshift File.join(ROOT, 'lib')
3
+ Dir.glob('lib/**').each{ |d| $LOAD_PATH.unshift(File.join(ROOT, d)) }
4
+
5
+ require 'rubygems'
6
+ require 'bundler'
7
+ begin
8
+ Bundler.setup(:default, :development)
9
+ rescue Bundler::BundlerError => e
10
+ $stderr.puts e.message
11
+ $stderr.puts "Run `bundle install` to install missing gems"
12
+ exit e.status_code
13
+ end
14
+
15
+ require 'rspec'
16
+ require 'rspec/core/rake_task'
17
+
18
+ desc "Specs all at ones."
19
+ RSpec::Core::RakeTask.new(:spec) do |t|
20
+ t.fail_on_error = true
21
+ t.verbose = false
22
+ end
@@ -0,0 +1,35 @@
1
+ require 'logstash/namespace'
2
+ require 'logstash/outputs/base'
3
+ require 'logstash/outputs/timetree/timetree'
4
+
5
+ class LogStash::Outputs::Neo4j < LogStash::Outputs::Base
6
+
7
+ config_name 'neo4j'
8
+
9
+ # The path within your file system where the neo4j database is located
10
+ config :path, :validate => :string, :required => true
11
+
12
+ attr_reader :tree
13
+
14
+ def register
15
+ require 'neo4j'
16
+ if Neo4j::Session.current.nil?
17
+ @session = ::Neo4j::Session.open(:embedded_db, @path, auto_commit: true)
18
+ @session.start
19
+ end
20
+ @session = Neo4j::Session.current
21
+ @tree = Neo4jrb::TimeTree.new(@session)
22
+ end
23
+
24
+ def receive(event)
25
+ return unless output?(event)
26
+ payload = event.to_hash
27
+ timestamp = payload["@timestamp"].time
28
+ @tree.add_event(timestamp, payload)
29
+ end
30
+
31
+ def teardown
32
+ @session.shutdown
33
+ @session.close
34
+ end
35
+ end
@@ -0,0 +1,58 @@
1
+ module TimeTree
2
+
3
+ class Root
4
+ include Neo4j::ActiveNode
5
+ end
6
+
7
+ class Year
8
+ include Neo4j::ActiveNode
9
+ property :value, index: :exact
10
+ end
11
+
12
+ class Month
13
+ include Neo4j::ActiveNode
14
+ property :value, index: :exact
15
+ end
16
+
17
+ class Day
18
+ include Neo4j::ActiveNode
19
+ property :value, index: :exact
20
+ end
21
+
22
+ class Event
23
+ include Neo4j::ActiveNode
24
+
25
+ property :message
26
+ serialize :message
27
+ property :created_at
28
+ end
29
+
30
+ class Child
31
+ include Neo4j::ActiveRel
32
+ from_class :any
33
+ to_class :any
34
+ type 'child'
35
+ end
36
+
37
+ class First
38
+ include Neo4j::ActiveRel
39
+ from_class :any
40
+ to_class :any
41
+ type 'first'
42
+ end
43
+
44
+ class Last
45
+ include Neo4j::ActiveRel
46
+ from_class :any
47
+ to_class :any
48
+ type 'last'
49
+ end
50
+
51
+ class Next
52
+ include Neo4j::ActiveRel
53
+ from_class :any
54
+ to_class :any
55
+ type 'next'
56
+ end
57
+
58
+ end
@@ -0,0 +1,102 @@
1
+ require 'neo4j'
2
+ require 'logstash/outputs/timetree/model'
3
+
4
+ module Neo4jrb
5
+ class TimeTree
6
+
7
+ attr_reader :root
8
+
9
+ def initialize(session)
10
+ @session = session
11
+ @root = ::TimeTree::Root.first || create_node(::TimeTree::Root)
12
+ end
13
+
14
+ def add_event(ts, event)
15
+ Neo4j::Transaction.run do
16
+ start = refresh_tree(ts)
17
+ append(start, event)
18
+ end
19
+ end
20
+
21
+ def events_between(start_time, end_time)
22
+ raise Exception.new("Not implemented yet!")
23
+ end
24
+
25
+ def events_from(start_time)
26
+ raise Exception.new("Not implemented yet!")
27
+ end
28
+
29
+ def events_at(start_time)
30
+ query = <<-QUERY
31
+ MATCH (a:`TimeTree::Year`{value:#{start_time.year}})-[:`child`]->(b:`TimeTree::Month`{value:#{start_time.month}})-[:`child`]->(d:`TimeTree::Day`{value:#{start_time.day}})
32
+ WITH d MATCH (e:`TimeTree::Event`)<-[:`child`]-(d)
33
+ RETURN e
34
+ QUERY
35
+ resultset = Neo4j::Session.query(query)
36
+ resultset.map { |r| r.e }.to_a
37
+ end
38
+
39
+
40
+ def refresh_tree(ts)
41
+ year = append_element(@root, ::TimeTree::Year, ts.year)
42
+ month = append_element(year, ::TimeTree::Month, ts.month)
43
+ append_element(month, ::TimeTree::Day, ts.day)
44
+ end
45
+
46
+ def append(start, event)
47
+ event = create_node(::TimeTree::Event, {:message => event, :created_at => Time.now})
48
+ ::TimeTree::Child.create(:from_node => start, :to_node => event)
49
+ end
50
+
51
+ def append_element(root, clazz, value)
52
+ childs = has_child(root)
53
+ if !childs.empty?
54
+ selected = childs.select { |e| e.props[:value] == value }
55
+ return selected.first if !selected.empty?
56
+ end
57
+ node = find_or_create(clazz, root, {:value => value})
58
+ set_edge_to_first_node(root, node) if childs.empty?
59
+ ::TimeTree::Child.create(:from_node => root, :to_node => node)
60
+ set_edge_to_last_node(root, node)
61
+ node
62
+ end
63
+
64
+ def find_or_create(clazz, root, criteria)
65
+ nodes = clazz.where(criteria).map do |n|
66
+ n.nodes(:dir => :incoming, :type => :child).select do |p|
67
+ p.class == root.class && (!root.is_a?(::TimeTree::Root) && p.props[:value] == root.props[:value])
68
+ end
69
+ end.flatten
70
+ if nodes.empty? then clazz.create(criteria) else nodes.first end
71
+ end
72
+
73
+ def set_edge_to_first_node(root, node)
74
+ ::TimeTree::First.create(:from_node => root, :to_node => node)
75
+ end
76
+
77
+ def set_edge_to_last_node(root, node)
78
+ Neo4j::Transaction.run do
79
+ ::TimeTree::Last.create(:from_node => root, :to_node => node)
80
+ end
81
+ rels = root.rel(:dir => :outgoing, :type => :last)
82
+ rels.del
83
+ end
84
+
85
+ def create_node(clazz, props={})
86
+ clazz.create(props)
87
+ end
88
+
89
+ def has_child(base)
90
+ has(base, :outgoing)
91
+ end
92
+
93
+ def has_parent(base, value)
94
+ has(base, :incoming)
95
+ end
96
+
97
+ def has(base, dir)
98
+ base.nodes(:dir => dir, :type => :child)
99
+ end
100
+
101
+ end
102
+ end
@@ -0,0 +1,34 @@
1
+ Gem::Specification.new do |s|
2
+
3
+ s.name = 'logstash-output-neo4j'
4
+ s.version = '0.9.0'
5
+ s.licenses = ['Apache License (2.0)']
6
+ s.summary = "Logstash Output to Neo4j"
7
+ s.description = "Output events to Neo4j"
8
+ s.authors = ["Pere Urbon-Bayes"]
9
+ s.email = 'pere.urbon@gmail.com'
10
+ s.homepage = "http://purbon.com/"
11
+ s.require_paths = ["lib"]
12
+
13
+ # Files
14
+ s.files = `git ls-files`.split($\)
15
+
16
+ # Tests
17
+ s.test_files = s.files.grep(%r{^(test|spec|features)/})
18
+
19
+ # Special flag to let us know this is actually a logstash plugin
20
+ s.metadata = { "logstash_plugin" => "true", "logstash_group" => "output" }
21
+
22
+ s.add_runtime_dependency 'logstash-core', '>= 1.4.0', '< 2.0.0'
23
+ s.add_runtime_dependency 'jar-dependencies'
24
+
25
+ if RUBY_PLATFORM == 'java'
26
+ s.platform = RUBY_PLATFORM
27
+ s.add_runtime_dependency 'neo4j', '>= 3.0'
28
+ s.add_runtime_dependency 'neo4j-community', '~> 2.0.0'
29
+ end
30
+
31
+ s.add_development_dependency 'logstash-devutils'
32
+
33
+
34
+ end
@@ -0,0 +1,59 @@
1
+ require 'spec_helper'
2
+
3
+ describe 'Neo4jrb::Timetree' do
4
+
5
+ let(:session) { Neo4j::Session.current }
6
+ let(:tree) { Neo4jrb::TimeTree.new(session) }
7
+
8
+ it "starts without errors" do
9
+ expect(session.running?).to be_true
10
+ end
11
+
12
+ context "add an event" do
13
+
14
+ let(:timestamp) { Time.new(2015, 1, 2) }
15
+ let(:event) { { :name => 'load', :product => 'ads' } }
16
+
17
+ it "updates a new time tree" do
18
+ tree.add_event(timestamp, event)
19
+ expect(to_arr(tree.root)).to include("TimeTree::Root#NaN", "TimeTree::Year#2015", "TimeTree::Month#1", "TimeTree::Day#2")
20
+ end
21
+
22
+ it "add a second event in a different month without affecting the structure" do
23
+ time = Time.new(2015, 2, 2)
24
+ tree.add_event(time, event)
25
+ expect(to_arr(tree.root)).to include("TimeTree::Root#NaN", "TimeTree::Year#2015", "TimeTree::Month#1", "TimeTree::Day#2",
26
+ "TimeTree::Month#2", "TimeTree::Day#2")
27
+ end
28
+
29
+ it "add an event pending using an existing months and year" do
30
+ time = Time.new(2015, 1, 3)
31
+ tree.add_event(time, event)
32
+ expect(has_child(TimeTree::Month, 1, TimeTree::Day, 3)).to_not be_empty
33
+ end
34
+
35
+ end
36
+
37
+ context "fetching events by time" do
38
+
39
+ let(:event) { { :name => 'fetching_event', :product => 'logstash-output-neo4j' } }
40
+ let(:new_event) { { :name => 'another_fetching_event', :product => 'logstash-output-neo4j' } }
41
+
42
+ it "has an event added at the end of a given day" do
43
+ timestamp = Time.new(2015, 12, 26)
44
+ tree.add_event(timestamp, event)
45
+ event = tree.events_at(timestamp).first
46
+ expect(event.props).to include(:message=>{"name"=>"fetching_event", "product"=>"logstash-output-neo4j"})
47
+ end
48
+
49
+ it "can fetch more than one events together" do
50
+ timestamp = Time.new(2015, 12, 25)
51
+ [event, new_event].each do |payload|
52
+ tree.add_event(timestamp, payload)
53
+ end
54
+ events = tree.events_at(timestamp)
55
+ expect(events.count).to eq(2)
56
+ end
57
+
58
+ end
59
+ end
@@ -0,0 +1,40 @@
1
+ require 'spec_helper'
2
+ require "logstash/plugin"
3
+ require "logstash/json"
4
+
5
+ describe "outputs/neo4j" do
6
+
7
+ context "registration" do
8
+
9
+ let(:output) { LogStash::Plugin.lookup("output", "neo4j").new("path" => "/tmp/db") }
10
+
11
+ it "should register" do
12
+ expect {output.register}.to_not raise_error
13
+ end
14
+
15
+ it "should teardown" do
16
+ output.register
17
+ expect {output.teardown}.to_not raise_error
18
+ end
19
+ end
20
+
21
+ context "operation" do
22
+
23
+ let(:output) { LogStash::Outputs::Neo4j.new("path" => "/tmp/db") }
24
+ let(:props) { { :name => 'plugin.operation', :product => 'logstash-output-neo4j' } }
25
+ let(:event) { LogStash::Event.new(props) }
26
+ let(:tree) { output.tree }
27
+
28
+ before do
29
+ output.register
30
+ end
31
+
32
+ it "receive a message" do
33
+ output.receive(event)
34
+ events = tree.events_at(event.to_hash["@timestamp"].time)
35
+ expect(events.count).to eq(1)
36
+ end
37
+
38
+ end
39
+
40
+ end
@@ -0,0 +1,34 @@
1
+ require 'logstash/outputs/neo4j'
2
+
3
+ def load_fixture(name)
4
+ IO.read("spec/fixtures/#{name}")
5
+ end
6
+
7
+ RSpec.configure do |config|
8
+
9
+ config.before(:suite) do
10
+ FileUtils.rm_rf '/tmp/db'
11
+ session = Neo4j::Session.open(:embedded_db, '/tmp/db', auto_commit: true)
12
+ session.start
13
+ end
14
+
15
+ config.after(:suite) do
16
+ Neo4j::Session.current.shutdown
17
+ end
18
+
19
+ end
20
+
21
+ def to_arr(root)
22
+ Neo4j::Transaction.run do
23
+ session.graph_db.get_all_nodes.to_a.map { |m| "#{m.props[:_classname]}##{m.props[:value]||'NaN'}" }
24
+ end
25
+ end
26
+
27
+ def has_child(source_clazz, source_value, target_clazz, target_value)
28
+ source_clazz.where({:value => source_value}).map do |n|
29
+ n.nodes(dir: :outgoing, :type => :child).select do |m|
30
+ m.is_a?(target_clazz) && (!m.is_a?(::TimeTree::Root) && m.props[:value] == target_value)
31
+ end
32
+ end.flatten
33
+ end
34
+
metadata ADDED
@@ -0,0 +1,137 @@
1
+ --- !ruby/object:Gem::Specification
2
+ name: logstash-output-neo4j
3
+ version: !ruby/object:Gem::Version
4
+ version: 0.9.0
5
+ platform: java
6
+ authors:
7
+ - Pere Urbon-Bayes
8
+ autorequire:
9
+ bindir: bin
10
+ cert_chain: []
11
+ date: 2015-03-27 00:00:00.000000000 Z
12
+ dependencies:
13
+ - !ruby/object:Gem::Dependency
14
+ name: logstash-core
15
+ version_requirements: !ruby/object:Gem::Requirement
16
+ requirements:
17
+ - - '>='
18
+ - !ruby/object:Gem::Version
19
+ version: 1.4.0
20
+ - - <
21
+ - !ruby/object:Gem::Version
22
+ version: 2.0.0
23
+ requirement: !ruby/object:Gem::Requirement
24
+ requirements:
25
+ - - '>='
26
+ - !ruby/object:Gem::Version
27
+ version: 1.4.0
28
+ - - <
29
+ - !ruby/object:Gem::Version
30
+ version: 2.0.0
31
+ prerelease: false
32
+ type: :runtime
33
+ - !ruby/object:Gem::Dependency
34
+ name: jar-dependencies
35
+ version_requirements: !ruby/object:Gem::Requirement
36
+ requirements:
37
+ - - '>='
38
+ - !ruby/object:Gem::Version
39
+ version: '0'
40
+ requirement: !ruby/object:Gem::Requirement
41
+ requirements:
42
+ - - '>='
43
+ - !ruby/object:Gem::Version
44
+ version: '0'
45
+ prerelease: false
46
+ type: :runtime
47
+ - !ruby/object:Gem::Dependency
48
+ name: neo4j
49
+ version_requirements: !ruby/object:Gem::Requirement
50
+ requirements:
51
+ - - '>='
52
+ - !ruby/object:Gem::Version
53
+ version: '3.0'
54
+ requirement: !ruby/object:Gem::Requirement
55
+ requirements:
56
+ - - '>='
57
+ - !ruby/object:Gem::Version
58
+ version: '3.0'
59
+ prerelease: false
60
+ type: :runtime
61
+ - !ruby/object:Gem::Dependency
62
+ name: neo4j-community
63
+ version_requirements: !ruby/object:Gem::Requirement
64
+ requirements:
65
+ - - ~>
66
+ - !ruby/object:Gem::Version
67
+ version: 2.0.0
68
+ requirement: !ruby/object:Gem::Requirement
69
+ requirements:
70
+ - - ~>
71
+ - !ruby/object:Gem::Version
72
+ version: 2.0.0
73
+ prerelease: false
74
+ type: :runtime
75
+ - !ruby/object:Gem::Dependency
76
+ name: logstash-devutils
77
+ version_requirements: !ruby/object:Gem::Requirement
78
+ requirements:
79
+ - - '>='
80
+ - !ruby/object:Gem::Version
81
+ version: '0'
82
+ requirement: !ruby/object:Gem::Requirement
83
+ requirements:
84
+ - - '>='
85
+ - !ruby/object:Gem::Version
86
+ version: '0'
87
+ prerelease: false
88
+ type: :development
89
+ description: Output events to Neo4j
90
+ email: pere.urbon@gmail.com
91
+ executables: []
92
+ extensions: []
93
+ extra_rdoc_files: []
94
+ files:
95
+ - .gitignore
96
+ - .rspec
97
+ - Gemfile
98
+ - LICENSE
99
+ - README.md
100
+ - Rakefile
101
+ - lib/logstash/outputs/neo4j.rb
102
+ - lib/logstash/outputs/timetree/model.rb
103
+ - lib/logstash/outputs/timetree/timetree.rb
104
+ - logstash-output-neo4j.gemspec
105
+ - spec/lib/timetree_spec.rb
106
+ - spec/outputs/neo4j_spec.rb
107
+ - spec/spec_helper.rb
108
+ homepage: http://purbon.com/
109
+ licenses:
110
+ - Apache License (2.0)
111
+ metadata:
112
+ logstash_plugin: 'true'
113
+ logstash_group: output
114
+ post_install_message:
115
+ rdoc_options: []
116
+ require_paths:
117
+ - lib
118
+ required_ruby_version: !ruby/object:Gem::Requirement
119
+ requirements:
120
+ - - '>='
121
+ - !ruby/object:Gem::Version
122
+ version: '0'
123
+ required_rubygems_version: !ruby/object:Gem::Requirement
124
+ requirements:
125
+ - - '>='
126
+ - !ruby/object:Gem::Version
127
+ version: '0'
128
+ requirements: []
129
+ rubyforge_project:
130
+ rubygems_version: 2.4.5
131
+ signing_key:
132
+ specification_version: 4
133
+ summary: Logstash Output to Neo4j
134
+ test_files:
135
+ - spec/lib/timetree_spec.rb
136
+ - spec/outputs/neo4j_spec.rb
137
+ - spec/spec_helper.rb