nagare-redis 0.4.0 → 0.5.3
Sign up to get free protection for your applications and to get access to all the features.
- checksums.yaml +4 -4
- data/Gemfile +0 -1
- data/Gemfile.lock +3 -9
- data/README.md +13 -0
- data/lib/nagare/config.rb +11 -3
- data/lib/nagare/listener_pool.rb +1 -3
- data/lib/nagare/redis_streams.rb +42 -3
- data/lib/nagare/version.rb +1 -1
- data/nagare.gemspec +1 -1
- metadata +7 -7
checksums.yaml
CHANGED
@@ -1,7 +1,7 @@
|
|
1
1
|
---
|
2
2
|
SHA256:
|
3
|
-
metadata.gz:
|
4
|
-
data.tar.gz:
|
3
|
+
metadata.gz: 7378a63f1de40e7088716424831aff64be7a83e0c24aa9941d30cbf11de3a812
|
4
|
+
data.tar.gz: 26082b6b4770ebbe95815675bb237f0104fb00d8ff269e2c69ec0ffed0211108
|
5
5
|
SHA512:
|
6
|
-
metadata.gz:
|
7
|
-
data.tar.gz:
|
6
|
+
metadata.gz: dddaa8eb7e38f9bc7d521b2e88a656638a25a654436c89a3d6aec2e33c7ea16333112245ea803a44ffd90570dbc3e1f4432a50a1cf8be31f362566fbaa4f382c
|
7
|
+
data.tar.gz: 97801de9da2eefc3492b371e2c1e297876078ea6aaf227d4b5c51088d1725fae2d9dc2efc92de25b59fd6fcb4c0f5f266001d5abb50230a95bcf9b65259f645e
|
data/Gemfile
CHANGED
data/Gemfile.lock
CHANGED
@@ -1,14 +1,8 @@
|
|
1
|
-
GIT
|
2
|
-
remote: https://github.com/vavato-be/redis-rb.git
|
3
|
-
revision: 79f36a58e69beb72f3bcef2ae584b30324fa2f07
|
4
|
-
specs:
|
5
|
-
redis (6.2.0)
|
6
|
-
|
7
1
|
PATH
|
8
2
|
remote: .
|
9
3
|
specs:
|
10
|
-
nagare-redis (0.
|
11
|
-
redis (~>
|
4
|
+
nagare-redis (0.5.2)
|
5
|
+
redis (~> 4.4, >= 4.4.0)
|
12
6
|
|
13
7
|
GEM
|
14
8
|
remote: https://rubygems.org/
|
@@ -20,6 +14,7 @@ GEM
|
|
20
14
|
ast (~> 2.4.1)
|
21
15
|
rainbow (3.0.0)
|
22
16
|
rake (12.3.2)
|
17
|
+
redis (4.4.0)
|
23
18
|
regexp_parser (2.1.1)
|
24
19
|
rexml (3.2.5)
|
25
20
|
rspec (3.9.0)
|
@@ -58,7 +53,6 @@ PLATFORMS
|
|
58
53
|
DEPENDENCIES
|
59
54
|
nagare-redis!
|
60
55
|
rake (~> 12.0)
|
61
|
-
redis!
|
62
56
|
rspec (~> 3.0)
|
63
57
|
rubocop (~> 1.18.3, >= 1.18.3)
|
64
58
|
rubocop-rspec (~> 2.4.0, >= 2.4.0)
|
data/README.md
CHANGED
@@ -67,6 +67,19 @@ Nagare.configure do |config|
|
|
67
67
|
# and in the background
|
68
68
|
# Default: 3 threads
|
69
69
|
config.threads = 3
|
70
|
+
|
71
|
+
# Nagare can execute a proc for error handling. This enables you to
|
72
|
+
# use APM tools like New Relic or Appsignal with it.
|
73
|
+
# The proc takes 2 parameters, message and error.
|
74
|
+
# By default, nagare logs the error to stderr.
|
75
|
+
config.error_handler = proc do |message, error|
|
76
|
+
Appsignal.set_error(error);
|
77
|
+
end
|
78
|
+
|
79
|
+
# After exceeding the maximum number of retries, nagare moves the
|
80
|
+
# failing messages to a Dead Letter Queue stream.
|
81
|
+
# By default this stream is named 'dlq', but you can customize it.
|
82
|
+
config.dlq_stream = 'its_dead_jim'
|
70
83
|
end
|
71
84
|
```
|
72
85
|
|
data/lib/nagare/config.rb
CHANGED
@@ -5,13 +5,14 @@ module Nagare
|
|
5
5
|
# See the README for possible values and what they do
|
6
6
|
class Config
|
7
7
|
class << self
|
8
|
-
attr_accessor :group_name, :redis_url, :threads, :suffix, :min_idle_time
|
8
|
+
attr_accessor :group_name, :redis_url, :threads, :suffix, :min_idle_time,
|
9
|
+
:error_handler, :dlq_stream, :max_retries
|
9
10
|
|
10
11
|
# Runs code in the block passed in to configure Nagare and sets defaults
|
11
12
|
# when values are not set.
|
12
13
|
#
|
13
14
|
# returns [Nagare::Config] self
|
14
|
-
# rubocop:disable Metrics/CyclomaticComplexity
|
15
|
+
# rubocop:disable Metrics/CyclomaticComplexity, Metrics/PerceivedComplexity, Metrics/MethodLength, Metrics/AbcSize
|
15
16
|
def configure
|
16
17
|
yield(self)
|
17
18
|
@dead_consumer_timeout ||= 5000
|
@@ -20,9 +21,16 @@ module Nagare
|
|
20
21
|
@threads ||= 1
|
21
22
|
@suffix ||= nil
|
22
23
|
@min_idle_time ||= 600_000
|
24
|
+
@error_handler ||= proc do |message, error|
|
25
|
+
Nagare.logger.error "Failed to process message #{message}"
|
26
|
+
Nagare.logger.error error.message
|
27
|
+
Nagare.logger.error error.backtrace.join("\n")
|
28
|
+
end
|
29
|
+
@dlq_stream ||= 'dlq'
|
30
|
+
@max_retries ||= 10
|
23
31
|
self
|
24
32
|
end
|
25
|
-
# rubocop:enable Metrics/CyclomaticComplexity
|
33
|
+
# rubocop:enable Metrics/CyclomaticComplexity, Metrics/PerceivedComplexity, Metrics/MethodLength, Metrics/AbcSize
|
26
34
|
end
|
27
35
|
end
|
28
36
|
end
|
data/lib/nagare/listener_pool.rb
CHANGED
@@ -83,10 +83,8 @@ module Nagare
|
|
83
83
|
listeners.each do |listener|
|
84
84
|
invoke_listener(stream, message, listener)
|
85
85
|
rescue StandardError => e
|
86
|
-
logger.error e.message
|
87
|
-
logger.error e.backtrace.join("\n")
|
88
86
|
listener_failed = true
|
89
|
-
|
87
|
+
Nagare::Config.error_handler.call(message, e)
|
90
88
|
end
|
91
89
|
|
92
90
|
return if listener_failed
|
data/lib/nagare/redis_streams.rb
CHANGED
@@ -12,6 +12,7 @@ module Nagare
|
|
12
12
|
#
|
13
13
|
# Important: Groups are always assumed to be named `<stream>-<group>`.
|
14
14
|
# Consumers are always created using the hostname + thread id
|
15
|
+
# rubocop:disable Metrics/ClassLength
|
15
16
|
class RedisStreams
|
16
17
|
class << self
|
17
18
|
##
|
@@ -83,20 +84,57 @@ module Nagare
|
|
83
84
|
# Claums the next message of the consumer group that is stuck
|
84
85
|
# (pending and past min_idle_time since being picked up)
|
85
86
|
#
|
86
|
-
# @param
|
87
|
+
# @param stream_prefix [String] name of the stream
|
87
88
|
# @param group [String] name of the consumer group
|
88
89
|
#
|
89
90
|
# @return [Array[Hash]] array containing the 1 message or empty
|
90
|
-
|
91
|
-
|
91
|
+
# rubocop:disable Metrics/MethodLength, Metrics/AbcSize
|
92
|
+
def claim_next_stuck_message(stream_prefix, group)
|
93
|
+
stream = stream_name(stream_prefix)
|
92
94
|
result = connection.xautoclaim(stream,
|
93
95
|
"#{stream}-#{group}",
|
94
96
|
"#{hostname}-#{thread_id}",
|
95
97
|
Nagare::Config.min_idle_time,
|
96
98
|
'0-0',
|
97
99
|
count: 1)
|
100
|
+
|
101
|
+
# Move message to DLQ if retried too much and get next one
|
102
|
+
if result['entries'].any?
|
103
|
+
message_id = result['entries'].first.first
|
104
|
+
if retry_count(stream_prefix, group,
|
105
|
+
message_id) > Nagare::Config.max_retries
|
106
|
+
move_to_dlq(stream_prefix, group, result['entries'].first)
|
107
|
+
return claim_next_stuck_message(stream, group)
|
108
|
+
end
|
109
|
+
end
|
110
|
+
|
98
111
|
result['entries'] || []
|
99
112
|
end
|
113
|
+
# rubocop:enable Metrics/MethodLength, Metrics/AbcSize
|
114
|
+
|
115
|
+
##
|
116
|
+
# Uses XPENDING to verify the number of times the message was
|
117
|
+
# delivered
|
118
|
+
def retry_count(stream, group, message_id)
|
119
|
+
stream = stream_name(stream)
|
120
|
+
result = connection.xpending(stream,
|
121
|
+
"#{stream}-#{group}",
|
122
|
+
message_id,
|
123
|
+
message_id,
|
124
|
+
1)
|
125
|
+
return 0 unless result.any?
|
126
|
+
|
127
|
+
result.first['count']
|
128
|
+
end
|
129
|
+
|
130
|
+
##
|
131
|
+
# Moves a message to the dead letter queue stream
|
132
|
+
def move_to_dlq(stream, group, message)
|
133
|
+
Nagare.logger.warn "Moving message to DLQ #{message} \
|
134
|
+
from stream #{stream}"
|
135
|
+
publish(Nagare::Config.dlq_stream, stream, message.to_json)
|
136
|
+
mark_processed(stream, group, message.first)
|
137
|
+
end
|
100
138
|
|
101
139
|
##
|
102
140
|
# Reads the next messages from the consumer group in redis.
|
@@ -189,3 +227,4 @@ module Nagare
|
|
189
227
|
end
|
190
228
|
end
|
191
229
|
end
|
230
|
+
# rubocop:enable Metrics/ClassLength
|
data/lib/nagare/version.rb
CHANGED
data/nagare.gemspec
CHANGED
@@ -20,7 +20,7 @@ Gem::Specification.new do |spec|
|
|
20
20
|
spec.metadata['source_code_uri'] = 'https://github.com/vavato-be/nagare.git'
|
21
21
|
spec.metadata['changelog_uri'] = 'https://github.com/vavato-be/nagare/CHANGELOG.md'
|
22
22
|
|
23
|
-
spec.add_dependency 'redis', '~>
|
23
|
+
spec.add_dependency 'redis', '~> 4.4', '>= 4.4.0'
|
24
24
|
spec.add_development_dependency 'rubocop', '~> 1.18.3', '>= 1.18.3'
|
25
25
|
spec.add_development_dependency 'rubocop-rspec', '~> 2.4.0', '>= 2.4.0'
|
26
26
|
|
metadata
CHANGED
@@ -1,14 +1,14 @@
|
|
1
1
|
--- !ruby/object:Gem::Specification
|
2
2
|
name: nagare-redis
|
3
3
|
version: !ruby/object:Gem::Version
|
4
|
-
version: 0.
|
4
|
+
version: 0.5.3
|
5
5
|
platform: ruby
|
6
6
|
authors:
|
7
7
|
- Alex Reis
|
8
8
|
autorequire:
|
9
9
|
bindir: exe
|
10
10
|
cert_chain: []
|
11
|
-
date: 2021-
|
11
|
+
date: 2021-09-10 00:00:00.000000000 Z
|
12
12
|
dependencies:
|
13
13
|
- !ruby/object:Gem::Dependency
|
14
14
|
name: redis
|
@@ -16,20 +16,20 @@ dependencies:
|
|
16
16
|
requirements:
|
17
17
|
- - ">="
|
18
18
|
- !ruby/object:Gem::Version
|
19
|
-
version:
|
19
|
+
version: 4.4.0
|
20
20
|
- - "~>"
|
21
21
|
- !ruby/object:Gem::Version
|
22
|
-
version: '
|
22
|
+
version: '4.4'
|
23
23
|
type: :runtime
|
24
24
|
prerelease: false
|
25
25
|
version_requirements: !ruby/object:Gem::Requirement
|
26
26
|
requirements:
|
27
27
|
- - ">="
|
28
28
|
- !ruby/object:Gem::Version
|
29
|
-
version:
|
29
|
+
version: 4.4.0
|
30
30
|
- - "~>"
|
31
31
|
- !ruby/object:Gem::Version
|
32
|
-
version: '
|
32
|
+
version: '4.4'
|
33
33
|
- !ruby/object:Gem::Dependency
|
34
34
|
name: rubocop
|
35
35
|
requirement: !ruby/object:Gem::Requirement
|
@@ -124,7 +124,7 @@ required_rubygems_version: !ruby/object:Gem::Requirement
|
|
124
124
|
- !ruby/object:Gem::Version
|
125
125
|
version: '0'
|
126
126
|
requirements: []
|
127
|
-
rubygems_version: 3.0.3
|
127
|
+
rubygems_version: 3.0.3
|
128
128
|
signing_key:
|
129
129
|
specification_version: 4
|
130
130
|
summary: Persistent and resilient pub/sub using Redis Streams
|