nagare-redis 0.1.5 → 0.5.2

Sign up to get free protection for your applications and to get access to all the features.
checksums.yaml CHANGED
@@ -1,7 +1,7 @@
1
1
  ---
2
2
  SHA256:
3
- metadata.gz: f7c85971f2e4fda73c0c03700cc54c07e2b2b40dcf508b6f6f6f95f9e595aba4
4
- data.tar.gz: d0b1617d063bd24878e6a59abac5d5a64f2e8a6311992844d01ec4646bc67adb
3
+ metadata.gz: 47ac2065d72b1abf37d0855641328d767503d4ff5cde242aa046132102870271
4
+ data.tar.gz: '0974078409071d939db89e48f22831e7b9c495ad40a7ff1e780a0e075b608f92'
5
5
  SHA512:
6
- metadata.gz: 37907d60c4a97ae9d16f2bc2aa5f4054b769fe33ce91145c63aba78b5d09d3c2efe21240154de6f69056667f11c0f9482c4fdc68df7dda1abeea1facaf3e5238
7
- data.tar.gz: cfcace1bce2ab24afaad1568616b23ad91aac10e5780bd33f2c82d938844720234e4a31acb12d3f12f4fcc157c812f06ae99e31d6510b552fd85ff502f8a8d37
6
+ metadata.gz: cbd3d90cc9f37efd7300cd4cb155271c90bc20417e4c27712b31fb21ae8a7204bfcf49f261f7e28e9ae00f233ada2516ac5fd9b7236fc1848dfa51d8914b6201
7
+ data.tar.gz: 85f6cecb070588aa4ea9d19a6886378099c87faf8e5ad13c4121b2026176ca54cf089372635810d8dfc581c1537f9453e8143f80c30cbc31abf1f4a3de27f27b
data/CHANGELOG.md CHANGED
@@ -1,5 +1,12 @@
1
1
  # Changelog
2
2
 
3
+ ## [0.4.0](https://www.github.com/vavato-be/nagare/compare/v0.3.0...v0.4.0) (2021-07-13)
4
+
5
+
6
+ ### Features
7
+
8
+ * Increment version number automaticaly on release ([fe5f181](https://www.github.com/vavato-be/nagare/commit/fe5f1816f65869d1ecba04ee34c4eb290e2823d4))
9
+
3
10
  ## [0.3.0](https://www.github.com/vavato-be/nagare/compare/v0.2.0...v0.3.0) (2021-07-13)
4
11
 
5
12
 
data/Gemfile CHANGED
@@ -4,5 +4,4 @@ source 'https://rubygems.org'
4
4
  gemspec
5
5
 
6
6
  gem 'rake', '~> 12.0'
7
- gem 'redis', github: 'vavato-be/redis-rb'
8
7
  gem 'rspec', '~> 3.0'
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.1.5)
11
- redis (~> 6.2, >= 6.2.0)
4
+ nagare-redis (0.5.1)
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
@@ -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
- # TODO: Notify Appsignal
87
+ Nagare::Config.error_handler.call(message, e)
90
88
  end
91
89
 
92
90
  return if listener_failed
@@ -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 stream [String] name of the stream
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
- def claim_next_stuck_message(stream, group)
91
- stream = stream_name(stream)
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)
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
@@ -1,5 +1,11 @@
1
1
  # frozen_string_literal: true
2
2
 
3
+ # rubocop:disable
3
4
  module Nagare
4
- VERSION = '0.1.5'
5
+ # Needs to be double quotes for release-please ruby updater
6
+ # see https://github.com/googleapis/release-please/blob/master/src/updaters/version-rb.ts
7
+ #
8
+ # rubocop:disable Style/StringLiterals
9
+ VERSION = "0.5.2"
10
+ # rubocop:enable Style/StringLiterals
5
11
  end
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', '~> 6.2', '>= 6.2.0'
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.1.5
4
+ version: 0.5.2
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-07-13 00:00:00.000000000 Z
11
+ date: 2021-09-09 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: 6.2.0
19
+ version: 4.4.0
20
20
  - - "~>"
21
21
  - !ruby/object:Gem::Version
22
- version: '6.2'
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: 6.2.0
29
+ version: 4.4.0
30
30
  - - "~>"
31
31
  - !ruby/object:Gem::Version
32
- version: '6.2'
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.1
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