active_record_streams 0.1.0

Sign up to get free protection for your applications and to get access to all the features.
Files changed (40) hide show
  1. checksums.yaml +7 -0
  2. data/.github/PULL_REQUEST_TEMPLATE.md +35 -0
  3. data/.gitignore +12 -0
  4. data/.rspec +5 -0
  5. data/.rubocop.yml +10 -0
  6. data/.travis.yml +12 -0
  7. data/CHANGELOG.md +13 -0
  8. data/Gemfile +7 -0
  9. data/Gemfile.lock +96 -0
  10. data/MIT-LICENSE +20 -0
  11. data/README.md +168 -0
  12. data/Rakefile +8 -0
  13. data/active_record_streams.gemspec +55 -0
  14. data/lib/active_record_streams.rb +26 -0
  15. data/lib/active_record_streams/config.rb +15 -0
  16. data/lib/active_record_streams/config_spec.rb +32 -0
  17. data/lib/active_record_streams/credentials.rb +31 -0
  18. data/lib/active_record_streams/credentials_spec.rb +79 -0
  19. data/lib/active_record_streams/extensions/active_record/base.rb +41 -0
  20. data/lib/active_record_streams/extensions/active_record/base_spec.rb +45 -0
  21. data/lib/active_record_streams/extensions/active_record/persistence.rb +17 -0
  22. data/lib/active_record_streams/extensions/active_record/persistence_spec.rb +36 -0
  23. data/lib/active_record_streams/extensions/active_record/relation.rb +60 -0
  24. data/lib/active_record_streams/extensions/active_record/relation_spec.rb +41 -0
  25. data/lib/active_record_streams/message.rb +21 -0
  26. data/lib/active_record_streams/message_spec.rb +29 -0
  27. data/lib/active_record_streams/publishers/http_stream.rb +58 -0
  28. data/lib/active_record_streams/publishers/http_stream_spec.rb +73 -0
  29. data/lib/active_record_streams/publishers/kinesis_client.rb +26 -0
  30. data/lib/active_record_streams/publishers/kinesis_client_spec.rb +49 -0
  31. data/lib/active_record_streams/publishers/kinesis_stream.rb +64 -0
  32. data/lib/active_record_streams/publishers/kinesis_stream_spec.rb +93 -0
  33. data/lib/active_record_streams/publishers/sns_client.rb +25 -0
  34. data/lib/active_record_streams/publishers/sns_client_spec.rb +46 -0
  35. data/lib/active_record_streams/publishers/sns_stream.rb +52 -0
  36. data/lib/active_record_streams/publishers/sns_stream_spec.rb +89 -0
  37. data/lib/active_record_streams/version.rb +5 -0
  38. data/lib/active_record_streams/version_spec.rb +9 -0
  39. data/lib/active_record_streams_spec.rb +21 -0
  40. metadata +199 -0
@@ -0,0 +1,52 @@
1
+ # frozen_string_literal: true
2
+
3
+ module ActiveRecordStreams
4
+ module Publishers
5
+ class SnsStream
6
+ ANY_TABLE = '*'
7
+
8
+ ##
9
+ # @param [String] topic_arn
10
+ # @param [String] table_name
11
+ # @param [Enumerable<String>] ignored_tables
12
+ # @param [Hash] overrides
13
+
14
+ def initialize(
15
+ topic_arn:,
16
+ table_name: ANY_TABLE,
17
+ ignored_tables: [],
18
+ overrides: {}
19
+ )
20
+ @topic_arn = topic_arn
21
+ @table_name = table_name
22
+ @ignored_tables = ignored_tables
23
+ @overrides = overrides
24
+ end
25
+
26
+ ##
27
+ # @param [String] table_name
28
+ # @param [ActiveRecordStreams::Message] message
29
+
30
+ def publish(table_name, message)
31
+ return unless (any_table? && allowed_table?(table_name)) ||
32
+ table_name == @table_name
33
+
34
+ client.publish(@topic_arn, message.json, @overrides)
35
+ end
36
+
37
+ private
38
+
39
+ def any_table?
40
+ @table_name == ANY_TABLE
41
+ end
42
+
43
+ def allowed_table?(table_name)
44
+ !@ignored_tables.include?(table_name)
45
+ end
46
+
47
+ def client
48
+ @client ||= SnsClient.new
49
+ end
50
+ end
51
+ end
52
+ end
@@ -0,0 +1,89 @@
1
+ # frozen_string_literal: true
2
+
3
+ require_relative 'sns_stream'
4
+
5
+ RSpec.describe ActiveRecordStreams::Publishers::SnsStream do
6
+ let(:topic_arn) { 'arn::some-topic-arn' }
7
+ let(:desired_table_name) { '*' }
8
+ let(:actual_table_name) { 'lovely_records' }
9
+ let(:ignored_tables) { [] }
10
+ let(:overrides) { {} }
11
+
12
+ let(:message) { double(json: '{}') }
13
+ let(:sns_client) { double(publish: nil) }
14
+
15
+ subject do
16
+ described_class.new(
17
+ topic_arn: topic_arn,
18
+ table_name: desired_table_name,
19
+ ignored_tables: ignored_tables,
20
+ overrides: overrides
21
+ )
22
+ end
23
+
24
+ before do
25
+ allow(ActiveRecordStreams::Publishers::SnsClient)
26
+ .to receive(:new)
27
+ .and_return(sns_client)
28
+ end
29
+
30
+ describe '#publish' do
31
+ context 'any table' do
32
+ it 'sends event to sns' do
33
+ expect(sns_client).to receive(:publish) do |topic_arn, data, _overrides|
34
+ expect(topic_arn).to eq(topic_arn)
35
+ expect(data).to eq(message.json)
36
+ end
37
+
38
+ subject.publish(actual_table_name, message)
39
+ end
40
+ end
41
+
42
+ context 'specific table' do
43
+ let(:desired_table_name) { actual_table_name }
44
+
45
+ it 'sends event to sns' do
46
+ expect(sns_client).to receive(:publish) do |topic_arn, data, _overrides|
47
+ expect(topic_arn).to eq(topic_arn)
48
+ expect(data).to eq(message.json)
49
+ end
50
+
51
+ subject.publish(actual_table_name, message)
52
+ end
53
+ end
54
+
55
+ context 'non-matching table' do
56
+ let(:desired_table_name) { 'another_table' }
57
+
58
+ it 'does not send event to sns' do
59
+ expect(sns_client).not_to receive(:publish)
60
+
61
+ subject.publish(actual_table_name, message)
62
+ end
63
+ end
64
+
65
+ context 'blacklisted table' do
66
+ let(:ignored_tables) { [actual_table_name] }
67
+
68
+ it 'does not send event to sns' do
69
+ expect(sns_client).not_to receive(:publish)
70
+
71
+ subject.publish(actual_table_name, message)
72
+ end
73
+ end
74
+
75
+ context 'with overrides' do
76
+ let(:overrides) { { 'hello' => 'world' } }
77
+
78
+ it 'sends event to a sns' do
79
+ expect(sns_client).to receive(:publish) do |topic_arn, data, overrides|
80
+ expect(topic_arn).to eq(topic_arn)
81
+ expect(data).to eq(message.json)
82
+ expect(overrides).to eq(overrides)
83
+ end
84
+
85
+ subject.publish(actual_table_name, message)
86
+ end
87
+ end
88
+ end
89
+ end
@@ -0,0 +1,5 @@
1
+ # frozen_string_literal: true
2
+
3
+ module ActiveRecordStreams
4
+ VERSION = '0.1.0'
5
+ end
@@ -0,0 +1,9 @@
1
+ # frozen_string_literal: true
2
+
3
+ require_relative 'version'
4
+
5
+ RSpec.describe ActiveRecordStreams::VERSION do
6
+ it 'has a version number' do
7
+ expect(subject).to eq('0.1.0')
8
+ end
9
+ end
@@ -0,0 +1,21 @@
1
+ # frozen_string_literal: true
2
+
3
+ require_relative '../spec/mocks/active_record/persistence'
4
+ require_relative '../spec/mocks/active_record/relation'
5
+ require_relative '../spec/mocks/active_record/base'
6
+
7
+ require_relative 'active_record_streams'
8
+
9
+ RSpec.describe ActiveRecordStreams do
10
+ describe '.configure' do
11
+ it 'yields configuration' do
12
+ expect { |b| subject.configure(&b) }.to yield_control
13
+ end
14
+ end
15
+
16
+ describe '.config' do
17
+ it 'returns configuration' do
18
+ expect(subject.config).to be_a(ActiveRecordStreams::Config)
19
+ end
20
+ end
21
+ end
metadata ADDED
@@ -0,0 +1,199 @@
1
+ --- !ruby/object:Gem::Specification
2
+ name: active_record_streams
3
+ version: !ruby/object:Gem::Version
4
+ version: 0.1.0
5
+ platform: ruby
6
+ authors:
7
+ - Advanon Team
8
+ autorequire:
9
+ bindir: bin
10
+ cert_chain: []
11
+ date: 2019-04-19 00:00:00.000000000 Z
12
+ dependencies:
13
+ - !ruby/object:Gem::Dependency
14
+ name: activerecord
15
+ requirement: !ruby/object:Gem::Requirement
16
+ requirements:
17
+ - - "~>"
18
+ - !ruby/object:Gem::Version
19
+ version: 4.2.10
20
+ type: :runtime
21
+ prerelease: false
22
+ version_requirements: !ruby/object:Gem::Requirement
23
+ requirements:
24
+ - - "~>"
25
+ - !ruby/object:Gem::Version
26
+ version: 4.2.10
27
+ - !ruby/object:Gem::Dependency
28
+ name: aws-sdk
29
+ requirement: !ruby/object:Gem::Requirement
30
+ requirements:
31
+ - - "~>"
32
+ - !ruby/object:Gem::Version
33
+ version: 2.11.9
34
+ type: :runtime
35
+ prerelease: false
36
+ version_requirements: !ruby/object:Gem::Requirement
37
+ requirements:
38
+ - - "~>"
39
+ - !ruby/object:Gem::Version
40
+ version: 2.11.9
41
+ - !ruby/object:Gem::Dependency
42
+ name: bundler
43
+ requirement: !ruby/object:Gem::Requirement
44
+ requirements:
45
+ - - "~>"
46
+ - !ruby/object:Gem::Version
47
+ version: '1.17'
48
+ type: :development
49
+ prerelease: false
50
+ version_requirements: !ruby/object:Gem::Requirement
51
+ requirements:
52
+ - - "~>"
53
+ - !ruby/object:Gem::Version
54
+ version: '1.17'
55
+ - !ruby/object:Gem::Dependency
56
+ name: pry
57
+ requirement: !ruby/object:Gem::Requirement
58
+ requirements:
59
+ - - "~>"
60
+ - !ruby/object:Gem::Version
61
+ version: 0.12.2
62
+ type: :development
63
+ prerelease: false
64
+ version_requirements: !ruby/object:Gem::Requirement
65
+ requirements:
66
+ - - "~>"
67
+ - !ruby/object:Gem::Version
68
+ version: 0.12.2
69
+ - !ruby/object:Gem::Dependency
70
+ name: rake
71
+ requirement: !ruby/object:Gem::Requirement
72
+ requirements:
73
+ - - "~>"
74
+ - !ruby/object:Gem::Version
75
+ version: '10.0'
76
+ type: :development
77
+ prerelease: false
78
+ version_requirements: !ruby/object:Gem::Requirement
79
+ requirements:
80
+ - - "~>"
81
+ - !ruby/object:Gem::Version
82
+ version: '10.0'
83
+ - !ruby/object:Gem::Dependency
84
+ name: rspec
85
+ requirement: !ruby/object:Gem::Requirement
86
+ requirements:
87
+ - - "~>"
88
+ - !ruby/object:Gem::Version
89
+ version: '3.0'
90
+ type: :development
91
+ prerelease: false
92
+ version_requirements: !ruby/object:Gem::Requirement
93
+ requirements:
94
+ - - "~>"
95
+ - !ruby/object:Gem::Version
96
+ version: '3.0'
97
+ - !ruby/object:Gem::Dependency
98
+ name: rubocop
99
+ requirement: !ruby/object:Gem::Requirement
100
+ requirements:
101
+ - - "~>"
102
+ - !ruby/object:Gem::Version
103
+ version: 0.67.2
104
+ type: :development
105
+ prerelease: false
106
+ version_requirements: !ruby/object:Gem::Requirement
107
+ requirements:
108
+ - - "~>"
109
+ - !ruby/object:Gem::Version
110
+ version: 0.67.2
111
+ - !ruby/object:Gem::Dependency
112
+ name: rubocop-performance
113
+ requirement: !ruby/object:Gem::Requirement
114
+ requirements:
115
+ - - "~>"
116
+ - !ruby/object:Gem::Version
117
+ version: 1.1.0
118
+ type: :development
119
+ prerelease: false
120
+ version_requirements: !ruby/object:Gem::Requirement
121
+ requirements:
122
+ - - "~>"
123
+ - !ruby/object:Gem::Version
124
+ version: 1.1.0
125
+ description: |2
126
+ Publish events about changes to your ActiveRecord models to different
127
+ targets, like HTTP endpoints, AWS SNS topics or Kinesis streams
128
+ email:
129
+ - team@advanon.com
130
+ executables: []
131
+ extensions: []
132
+ extra_rdoc_files: []
133
+ files:
134
+ - ".github/PULL_REQUEST_TEMPLATE.md"
135
+ - ".gitignore"
136
+ - ".rspec"
137
+ - ".rubocop.yml"
138
+ - ".travis.yml"
139
+ - CHANGELOG.md
140
+ - Gemfile
141
+ - Gemfile.lock
142
+ - MIT-LICENSE
143
+ - README.md
144
+ - Rakefile
145
+ - active_record_streams.gemspec
146
+ - lib/active_record_streams.rb
147
+ - lib/active_record_streams/config.rb
148
+ - lib/active_record_streams/config_spec.rb
149
+ - lib/active_record_streams/credentials.rb
150
+ - lib/active_record_streams/credentials_spec.rb
151
+ - lib/active_record_streams/extensions/active_record/base.rb
152
+ - lib/active_record_streams/extensions/active_record/base_spec.rb
153
+ - lib/active_record_streams/extensions/active_record/persistence.rb
154
+ - lib/active_record_streams/extensions/active_record/persistence_spec.rb
155
+ - lib/active_record_streams/extensions/active_record/relation.rb
156
+ - lib/active_record_streams/extensions/active_record/relation_spec.rb
157
+ - lib/active_record_streams/message.rb
158
+ - lib/active_record_streams/message_spec.rb
159
+ - lib/active_record_streams/publishers/http_stream.rb
160
+ - lib/active_record_streams/publishers/http_stream_spec.rb
161
+ - lib/active_record_streams/publishers/kinesis_client.rb
162
+ - lib/active_record_streams/publishers/kinesis_client_spec.rb
163
+ - lib/active_record_streams/publishers/kinesis_stream.rb
164
+ - lib/active_record_streams/publishers/kinesis_stream_spec.rb
165
+ - lib/active_record_streams/publishers/sns_client.rb
166
+ - lib/active_record_streams/publishers/sns_client_spec.rb
167
+ - lib/active_record_streams/publishers/sns_stream.rb
168
+ - lib/active_record_streams/publishers/sns_stream_spec.rb
169
+ - lib/active_record_streams/version.rb
170
+ - lib/active_record_streams/version_spec.rb
171
+ - lib/active_record_streams_spec.rb
172
+ homepage: https://advanon.com
173
+ licenses: []
174
+ metadata:
175
+ allowed_push_host: https://rubygems.org
176
+ homepage_uri: https://advanon.com
177
+ source_code_uri: https://github.com/Advanon/active_record_streams
178
+ changelog_uri: https://github.com/Advanon/active_record_streams/blob/master/CHANGELOG.md
179
+ post_install_message:
180
+ rdoc_options: []
181
+ require_paths:
182
+ - lib
183
+ required_ruby_version: !ruby/object:Gem::Requirement
184
+ requirements:
185
+ - - ">="
186
+ - !ruby/object:Gem::Version
187
+ version: 2.3.3
188
+ required_rubygems_version: !ruby/object:Gem::Requirement
189
+ requirements:
190
+ - - ">="
191
+ - !ruby/object:Gem::Version
192
+ version: '0'
193
+ requirements: []
194
+ rubyforge_project:
195
+ rubygems_version: 2.6.14
196
+ signing_key:
197
+ specification_version: 4
198
+ summary: Stream ActiveRecord events via HTTP, AWS SNS or Kinesis streams
199
+ test_files: []