surfliner-metadata_consumer 0.1.0.pre.alpha.3 → 0.1.0.pre.alpha.5

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -0,0 +1,62 @@
1
+ module Surfliner
2
+ module Mq
3
+ # An object encapsulating RabbitMQ configuration.
4
+ class ConnectionConfig
5
+ # @return [String] The RabbitMQ hostname
6
+ attr_reader :host
7
+
8
+ # @return [String] The RabbitMQ AMQP port
9
+ attr_reader :port
10
+
11
+ # @return [String] The RabbitMQ username
12
+ attr_reader :username
13
+
14
+ # @return [String] The RabbitMQ passsword
15
+ attr_reader :password
16
+
17
+ # Initializes a new `MqConfig` object.
18
+ # @param host [String] RabbitMQ hostname
19
+ # @param port [String] RabbitMQ AMQP port
20
+ # @param username [String] RabbitMQ username
21
+ # @param password [String] RabbitMQ passsword
22
+ def initialize(host:, port:, username:, password:)
23
+ @host = host
24
+ @port = port
25
+ @username = username
26
+ @password = password
27
+ end
28
+
29
+ class << self
30
+ # Reads RabbitMQ configuration from environment variables and
31
+ # returns it as a new `ConnectionConfig` object.
32
+ #
33
+ # | Variable | Sample value | Description |
34
+ # |-----------------------------|--------------|------------------------------|
35
+ # | `RABBITMQ_HOST` | `rabbitmq` | Hostname of RabbitMQ server |
36
+ # | `RABBITMQ_NODE_PORT_NUMBER` | `5672` | Port name of RabbitMQ server |
37
+ # | `RABBITMQ_USERNAME` | `user` | RabbitMQ username |
38
+ # | `RABBITMQ_PASSWORD` | `bitnami` | RabbitMQ password |
39
+ #
40
+ # @return [ConnectionConfig] The configuration.
41
+ def from_env
42
+ ConnectionConfig.new(
43
+ host: ENV.fetch("RABBITMQ_HOST"),
44
+ port: ENV.fetch("RABBITMQ_NODE_PORT_NUMBER"),
45
+ username: ENV.fetch("RABBITMQ_USERNAME"),
46
+ password: ENV.fetch("RABBITMQ_PASSWORD")
47
+ )
48
+ end
49
+ end
50
+
51
+ # @return [String] the connection URL as a string
52
+ def session_url
53
+ @session_url ||= "amqp://#{username}:#{password}@#{host}:#{port}"
54
+ end
55
+
56
+ # @return [String] the connection URL as a string, without the password
57
+ def redacted_url
58
+ @redacted_url ||= session_url.sub(password, "REDACTED")
59
+ end
60
+ end
61
+ end
62
+ end
@@ -0,0 +1,49 @@
1
+ module Surfliner
2
+ module Mq
3
+ # Encapsulates queue configuration
4
+ class QueueConfig
5
+ # @return [String] The queue to listen to
6
+ attr_reader :name
7
+
8
+ # @return [String] platform routing key to listen to
9
+ attr_reader :routing_key
10
+
11
+ # @return [Hash] RabbitMQ queue options. (See Bunny::Channel#queue)
12
+ attr_reader :options
13
+
14
+ # @param name [String] queue exchange to listen to
15
+ # @param routing_key [String] platform routing key to listen to
16
+ # @param options [Hash] RabbitMQ queue options. (See Bunny::Channel#queue)
17
+ def initialize(name:, routing_key:, options: {})
18
+ @name = name
19
+ @routing_key = routing_key
20
+ @options = options
21
+ end
22
+
23
+ class << self
24
+ # Returns a default (environment-variable-based) configuration with the
25
+ # specified options.
26
+ #
27
+ # | Variable | Sample value | Description |
28
+ # |---------------------------------|-------------------------------|----------------------|
29
+ # | `RABBITMQ_QUEUE` | `surfliner.metadata` | RabbitMQ queue name |
30
+ # | `RABBITMQ_PLATFORM_ROUTING_KEY` | `surfliner.metadata.daylight` | RabbitMQ routing key |
31
+ #
32
+ # @param options [Hash] RabbitMQ queue options. (See Bunny::Channel#queue)
33
+ # @return [QueueConfig] The configuration.
34
+ def from_env(**options)
35
+ QueueConfig.new(
36
+ name: ENV.fetch("RABBITMQ_QUEUE"),
37
+ routing_key: default_routing_key,
38
+ options:
39
+ )
40
+ end
41
+
42
+ # @return [String] The default routing key from `ENV["RABBITMQ_PLATFORM_ROUTING_KEY"]`
43
+ def default_routing_key
44
+ ENV.fetch("RABBITMQ_PLATFORM_ROUTING_KEY")
45
+ end
46
+ end
47
+ end
48
+ end
49
+ end
@@ -0,0 +1,65 @@
1
+ require "surfliner/util/assert"
2
+
3
+ module Surfliner
4
+ module Mq
5
+ # Encapsulates a RabbitMQ topic
6
+ class Topic
7
+ include Surfliner::Util::Assert
8
+
9
+ # @return [String] The name of the topic
10
+ attr_reader :name
11
+
12
+ # @return [Bunny::Channel] The channel being used
13
+ attr_reader :channel
14
+
15
+ # @return [Logger] The logger
16
+ attr_reader :logger
17
+
18
+ # @return [Hash] RabbitMQ topic options. (See Bunny::Channel#topic)
19
+ attr_reader :options
20
+
21
+ # @param name [String] The name of the topic
22
+ # @param channel [Bunny::Channel] The channel to use
23
+ # @param logger [Logger] the logger
24
+ # @param options [Hash] RabbitMQ topic options. (See Bunny::Channel#topic)
25
+ def initialize(name, channel:, logger:, options: {})
26
+ @name = not_nil!(name, "topic name")
27
+ @channel = not_nil!(channel, "channel")
28
+ @logger = not_nil!(logger, "logger")
29
+ @options = not_nil!(options, "options")
30
+ end
31
+
32
+ # @return [Bunny::Exchange] The exchange for the topic
33
+ def exchange
34
+ @exchange ||= channel.topic(name, options)
35
+ end
36
+
37
+ # Publishes the specified payload
38
+ # @param payload [String] the payload to publish
39
+ # @param routing_key [String] platform routing key to publish to
40
+ # @return [Bunny::Exchange] see #exchange
41
+ def publish(payload, routing_key: QueueConfig.default_routing_key)
42
+ logger.info "Publishing to #{routing_key} with payload: #{payload}"
43
+ exchange.publish(payload, routing_key:)
44
+ end
45
+
46
+ # @param config [QueueConfig] queue configuration
47
+ # @return [Bunny::Queue] the queue
48
+ def bind_queue(config = QueueConfig.from_env)
49
+ channel.queue(config.name, config.options).tap do |q|
50
+ q.bind(exchange, routing_key: config.routing_key)
51
+ end
52
+ end
53
+
54
+ class << self
55
+ # Creates a new `MqTopic` from the specified configuration
56
+ # @param config [TopicConfig] the configuration
57
+ # @param channel [Bunny::Channel] The channel to use
58
+ # @param logger [Logger] the logger
59
+ def from_config(config, channel:, logger:)
60
+ new(config.name, channel:, logger:, options: config.options)
61
+ end
62
+ end
63
+ end
64
+ end
65
+ end
@@ -0,0 +1,37 @@
1
+ module Surfliner
2
+ module Mq
3
+ # Encapsulates topic configuration
4
+ class TopicConfig
5
+ # @return [String] The topic exchange to listen to
6
+ attr_reader :name
7
+
8
+ # @return [Hash] RabbitMQ topic options. (See Bunny::Channel#topic)
9
+ attr_reader :options
10
+
11
+ # @param name [String] topic exchange to listen to
12
+ # @param options [Hash] RabbitMQ topic options. (See Bunny::Channel#topic)
13
+ def initialize(name:, options: {})
14
+ @name = name
15
+ @options = options
16
+ end
17
+
18
+ class << self
19
+ # Returns a default (environment-variable-based) configuration with the
20
+ # specified options.
21
+ #
22
+ # | Variable | Sample value | Description |
23
+ # |------------------|----------------------|---------------------|
24
+ # | `RABBITMQ_TOPIC` | `surfliner.metadata` | RabbitMQ topic name |
25
+ #
26
+ # @param options [Hash] RabbitMQ topic options. (See Bunny::Channel#topic)
27
+ # @return [TopicConfig] The configuration.
28
+ def from_env(**options)
29
+ TopicConfig.new(
30
+ name: ENV.fetch("RABBITMQ_TOPIC"),
31
+ options:
32
+ )
33
+ end
34
+ end
35
+ end
36
+ end
37
+ end
@@ -0,0 +1,4 @@
1
+ Dir.glob(File.expand_path("mq/*.rb", __dir__)).each(&method(:require))
2
+
3
+ # General RabbitMQ functionality
4
+ module Surfliner::Mq; end
@@ -0,0 +1,21 @@
1
+ module Surfliner
2
+ # Miscellaneous utility module
3
+ module Util
4
+ # Helper methods for argument validation
5
+ module Assert
6
+ class << self
7
+ include Assert
8
+ end
9
+
10
+ # @param arg_value [Object, nil] the argument value
11
+ # @param arg_name [String] the name of the argument, for error messages
12
+ # @return arg_value
13
+ # @raise [ArgumentError] if `arg_value` is `nil`
14
+ def not_nil!(arg_value, arg_name = "argument")
15
+ return arg_value unless arg_value.nil?
16
+
17
+ raise ArgumentError, "#{arg_name} must not be nil"
18
+ end
19
+ end
20
+ end
21
+ end
@@ -0,0 +1,4 @@
1
+ Dir.glob(File.expand_path("util/*.rb", __dir__)).each(&method(:require))
2
+
3
+ # Miscellaneous utility module
4
+ module Surfliner::Util; end
metadata CHANGED
@@ -1,13 +1,13 @@
1
1
  --- !ruby/object:Gem::Specification
2
2
  name: surfliner-metadata_consumer
3
3
  version: !ruby/object:Gem::Version
4
- version: 0.1.0.pre.alpha.3
4
+ version: 0.1.0.pre.alpha.5
5
5
  platform: ruby
6
6
  authors:
7
7
  - Project Surfliner
8
8
  bindir: bin
9
9
  cert_chain: []
10
- date: 2025-03-07 00:00:00.000000000 Z
10
+ date: 2025-03-25 00:00:00.000000000 Z
11
11
  dependencies:
12
12
  - !ruby/object:Gem::Dependency
13
13
  name: bunny
@@ -86,103 +86,117 @@ dependencies:
86
86
  - !ruby/object:Gem::Version
87
87
  version: '3'
88
88
  - !ruby/object:Gem::Dependency
89
- name: debug
89
+ name: ci_reporter_rspec
90
90
  requirement: !ruby/object:Gem::Requirement
91
91
  requirements:
92
92
  - - "~>"
93
93
  - !ruby/object:Gem::Version
94
- version: 1.9.2
94
+ version: '1.0'
95
95
  type: :development
96
96
  prerelease: false
97
97
  version_requirements: !ruby/object:Gem::Requirement
98
98
  requirements:
99
99
  - - "~>"
100
100
  - !ruby/object:Gem::Version
101
- version: 1.9.2
101
+ version: '1.0'
102
102
  - !ruby/object:Gem::Dependency
103
- name: rspec
103
+ name: colorize
104
104
  requirement: !ruby/object:Gem::Requirement
105
105
  requirements:
106
106
  - - "~>"
107
107
  - !ruby/object:Gem::Version
108
- version: '3.13'
108
+ version: '0.8'
109
109
  type: :development
110
110
  prerelease: false
111
111
  version_requirements: !ruby/object:Gem::Requirement
112
112
  requirements:
113
113
  - - "~>"
114
114
  - !ruby/object:Gem::Version
115
- version: '3.13'
115
+ version: '0.8'
116
116
  - !ruby/object:Gem::Dependency
117
- name: standard
117
+ name: debug
118
118
  requirement: !ruby/object:Gem::Requirement
119
119
  requirements:
120
120
  - - "~>"
121
121
  - !ruby/object:Gem::Version
122
- version: '1.31'
122
+ version: 1.9.2
123
123
  type: :development
124
124
  prerelease: false
125
125
  version_requirements: !ruby/object:Gem::Requirement
126
126
  requirements:
127
127
  - - "~>"
128
128
  - !ruby/object:Gem::Version
129
- version: '1.31'
129
+ version: 1.9.2
130
130
  - !ruby/object:Gem::Dependency
131
- name: ci_reporter_rspec
131
+ name: dotenv
132
132
  requirement: !ruby/object:Gem::Requirement
133
133
  requirements:
134
134
  - - "~>"
135
135
  - !ruby/object:Gem::Version
136
- version: '1.0'
136
+ version: '2.7'
137
137
  type: :development
138
138
  prerelease: false
139
139
  version_requirements: !ruby/object:Gem::Requirement
140
140
  requirements:
141
141
  - - "~>"
142
142
  - !ruby/object:Gem::Version
143
- version: '1.0'
143
+ version: '2.7'
144
144
  - !ruby/object:Gem::Dependency
145
- name: colorize
145
+ name: github-markup
146
146
  requirement: !ruby/object:Gem::Requirement
147
147
  requirements:
148
148
  - - "~>"
149
149
  - !ruby/object:Gem::Version
150
- version: '0.8'
150
+ version: '5.0'
151
151
  type: :development
152
152
  prerelease: false
153
153
  version_requirements: !ruby/object:Gem::Requirement
154
154
  requirements:
155
155
  - - "~>"
156
156
  - !ruby/object:Gem::Version
157
- version: '0.8'
157
+ version: '5.0'
158
158
  - !ruby/object:Gem::Dependency
159
- name: dotenv
159
+ name: rake
160
160
  requirement: !ruby/object:Gem::Requirement
161
161
  requirements:
162
162
  - - "~>"
163
163
  - !ruby/object:Gem::Version
164
- version: '2.7'
164
+ version: '13.0'
165
165
  type: :development
166
166
  prerelease: false
167
167
  version_requirements: !ruby/object:Gem::Requirement
168
168
  requirements:
169
169
  - - "~>"
170
170
  - !ruby/object:Gem::Version
171
- version: '2.7'
171
+ version: '13.0'
172
172
  - !ruby/object:Gem::Dependency
173
- name: rake
173
+ name: redcarpet
174
174
  requirement: !ruby/object:Gem::Requirement
175
175
  requirements:
176
176
  - - "~>"
177
177
  - !ruby/object:Gem::Version
178
- version: '13.0'
178
+ version: '3.6'
179
179
  type: :development
180
180
  prerelease: false
181
181
  version_requirements: !ruby/object:Gem::Requirement
182
182
  requirements:
183
183
  - - "~>"
184
184
  - !ruby/object:Gem::Version
185
- version: '13.0'
185
+ version: '3.6'
186
+ - !ruby/object:Gem::Dependency
187
+ name: rspec
188
+ requirement: !ruby/object:Gem::Requirement
189
+ requirements:
190
+ - - "~>"
191
+ - !ruby/object:Gem::Version
192
+ version: '3.13'
193
+ type: :development
194
+ prerelease: false
195
+ version_requirements: !ruby/object:Gem::Requirement
196
+ requirements:
197
+ - - "~>"
198
+ - !ruby/object:Gem::Version
199
+ version: '3.13'
186
200
  - !ruby/object:Gem::Dependency
187
201
  name: simplecov
188
202
  requirement: !ruby/object:Gem::Requirement
@@ -211,6 +225,20 @@ dependencies:
211
225
  - - "~>"
212
226
  - !ruby/object:Gem::Version
213
227
  version: '2.1'
228
+ - !ruby/object:Gem::Dependency
229
+ name: standard
230
+ requirement: !ruby/object:Gem::Requirement
231
+ requirements:
232
+ - - "~>"
233
+ - !ruby/object:Gem::Version
234
+ version: '1.31'
235
+ type: :development
236
+ prerelease: false
237
+ version_requirements: !ruby/object:Gem::Requirement
238
+ requirements:
239
+ - - "~>"
240
+ - !ruby/object:Gem::Version
241
+ version: '1.31'
214
242
  - !ruby/object:Gem::Dependency
215
243
  name: webmock
216
244
  requirement: !ruby/object:Gem::Requirement
@@ -240,7 +268,7 @@ dependencies:
240
268
  - !ruby/object:Gem::Version
241
269
  version: 0.9.37
242
270
  executables:
243
- - daylight-index-listen
271
+ - index-on-publish
244
272
  - simulate-publish-event
245
273
  extensions: []
246
274
  extra_rdoc_files: []
@@ -252,12 +280,10 @@ files:
252
280
  - LICENSE
253
281
  - README.md
254
282
  - Rakefile
255
- - bin/daylight-index-listen
283
+ - bin/index-on-publish
256
284
  - bin/simulate-publish-event
257
285
  - lib/surfliner/metadata_consumer.rb
258
286
  - lib/surfliner/metadata_consumer/consumer.rb
259
- - lib/surfliner/metadata_consumer/mq_config.rb
260
- - lib/surfliner/metadata_consumer/mq_connection.rb
261
287
  - lib/surfliner/metadata_consumer/payload.rb
262
288
  - lib/surfliner/metadata_consumer/solr.rb
263
289
  - lib/surfliner/metadata_consumer/solr/delete_handler.rb
@@ -265,6 +291,14 @@ files:
265
291
  - lib/surfliner/metadata_consumer/solr/message_handler.rb
266
292
  - lib/surfliner/metadata_consumer/superskunk_client.rb
267
293
  - lib/surfliner/metadata_consumer/version.rb
294
+ - lib/surfliner/mq.rb
295
+ - lib/surfliner/mq/connection.rb
296
+ - lib/surfliner/mq/connection_config.rb
297
+ - lib/surfliner/mq/queue_config.rb
298
+ - lib/surfliner/mq/topic.rb
299
+ - lib/surfliner/mq/topic_config.rb
300
+ - lib/surfliner/util.rb
301
+ - lib/surfliner/util/assert.rb
268
302
  homepage: https://gitlab.com/surfliner/metadata_consumer
269
303
  licenses:
270
304
  - MIT
@@ -279,7 +313,7 @@ required_ruby_version: !ruby/object:Gem::Requirement
279
313
  requirements:
280
314
  - - ">="
281
315
  - !ruby/object:Gem::Version
282
- version: 3.3.1
316
+ version: '3.2'
283
317
  required_rubygems_version: !ruby/object:Gem::Requirement
284
318
  requirements:
285
319
  - - ">="
@@ -1,23 +0,0 @@
1
- #!/usr/bin/env ruby
2
- require "bundler/setup"
3
- require "logger"
4
- require "opentelemetry/sdk"
5
-
6
- require "surfliner/metadata_consumer"
7
-
8
- unless ENV["OTEL_SDK_DISABLED"] == "true"
9
- OpenTelemetry::SDK.configure do |c|
10
- c.service_name = "surfliner-daylight-consumer"
11
- c.use_all # enables auto instrumentation for Bunny, Net::HTTP, etc...
12
- end
13
- end
14
- tracer = OpenTelemetry.tracer_provider.tracer("DaylightConsumerTracer")
15
-
16
- $stdout.sync = true # don't buffer log output
17
- logger = Logger.new($stdout).tap do |logger|
18
- logger.level = ENV.fetch("LOG_LEVEL", Logger::INFO)
19
- end
20
-
21
- handler = Surfliner::MetadataConsumer::Solr::MessageHandler
22
- consumer = Surfliner::MetadataConsumer::Consumer.new(tracer:, logger:, handler:)
23
- consumer.run
@@ -1,79 +0,0 @@
1
- module Surfliner
2
- module MetadataConsumer
3
- # An object encapsulating RabbitMQ configuration.
4
- class MqConfig
5
- # @return [String] The RabbitMQ hostname
6
- attr_reader :host
7
-
8
- # @return [String] The RabbitMQ AMQP port
9
- attr_reader :port
10
-
11
- # @return [String] The RabbitMQ username
12
- attr_reader :username
13
-
14
- # @return [String] The RabbitMQ passsword
15
- attr_reader :password
16
-
17
- # @return [String] The topic exchange to listen to
18
- attr_reader :topic
19
-
20
- # @return [String] The name of the queue to listen to
21
- attr_reader :queue_name
22
-
23
- # @return [String] The platform routing key to listen to
24
- attr_reader :routing_key
25
-
26
- # Initializes a new `MqConfig` object.
27
- # @param host [The] RabbitMQ hostname
28
- # @param port [The] RabbitMQ AMQP port
29
- # @param username [The] RabbitMQ username
30
- # @param password [The] RabbitMQ passsword
31
- # @param topic [The] topic exchange to listen to
32
- # @param queue_name [The] name of the queue to listen to
33
- # @param routing_key [The] platform routing key to listen to
34
- def initialize(host:, port:, username:, password:, topic:, queue_name:, routing_key:)
35
- @host = host
36
- @port = port
37
- @username = username
38
- @password = password
39
- @topic = topic
40
- @queue_name = queue_name
41
- @routing_key = routing_key
42
- end
43
-
44
- class << self
45
- # Reads RabbitMQ configuration from environment variables and
46
- # returns it as a new `MqConfig` object.
47
- #
48
- # - `RABBITMQ_HOST` → `host`
49
- # - `RABBITMQ_NODE_PORT_NUMBER` → `port`
50
- # - `RABBITMQ_USERNAME` → `username`
51
- # - `RABBITMQ_PASSWORD` → `password`
52
- # - `RABBITMQ_TOPIC` → `topic`
53
- # - `RABBITMQ_QUEUE` → `queue_name`
54
- # - `RABBITMQ_PLATFORM_ROUTING_KEY` → `routing_key`
55
- def from_env
56
- MqConfig.new(
57
- host: ENV.fetch("RABBITMQ_HOST"),
58
- port: ENV.fetch("RABBITMQ_NODE_PORT_NUMBER"),
59
- username: ENV.fetch("RABBITMQ_USERNAME"),
60
- password: ENV.fetch("RABBITMQ_PASSWORD"),
61
- topic: ENV.fetch("RABBITMQ_TOPIC"),
62
- queue_name: ENV.fetch("RABBITMQ_QUEUE"),
63
- routing_key: ENV.fetch("RABBITMQ_PLATFORM_ROUTING_KEY")
64
- )
65
- end
66
- end
67
-
68
- # @return [String] the connection URL as a string
69
- def connection_url
70
- @connection_url ||= "amqp://#{username}:#{password}@#{host}:#{port}"
71
- end
72
-
73
- # @return [String] the connection URL as a string, without the password
74
- def redacted_url
75
- @redacted_url ||= connection_url.sub(password, "REDACTED")
76
- end
77
- end
78
- end
79
- end