sidekiq 6.0.7 → 6.4.1

Sign up to get free protection for your applications and to get access to all the features.

Potentially problematic release.


This version of sidekiq might be problematic. Click here for more details.

Files changed (98) hide show
  1. checksums.yaml +4 -4
  2. data/Changes.md +167 -2
  3. data/LICENSE +3 -3
  4. data/README.md +10 -9
  5. data/bin/sidekiq +8 -3
  6. data/bin/sidekiqload +56 -58
  7. data/bin/sidekiqmon +1 -1
  8. data/lib/generators/sidekiq/job_generator.rb +57 -0
  9. data/lib/generators/sidekiq/templates/{worker.rb.erb → job.rb.erb} +2 -2
  10. data/lib/generators/sidekiq/templates/{worker_spec.rb.erb → job_spec.rb.erb} +1 -1
  11. data/lib/generators/sidekiq/templates/{worker_test.rb.erb → job_test.rb.erb} +1 -1
  12. data/lib/sidekiq/api.rb +145 -97
  13. data/lib/sidekiq/cli.rb +46 -12
  14. data/lib/sidekiq/client.rb +28 -46
  15. data/lib/sidekiq/delay.rb +2 -0
  16. data/lib/sidekiq/extensions/action_mailer.rb +3 -2
  17. data/lib/sidekiq/extensions/active_record.rb +4 -3
  18. data/lib/sidekiq/extensions/class_methods.rb +5 -4
  19. data/lib/sidekiq/extensions/generic_proxy.rb +3 -1
  20. data/lib/sidekiq/fetch.rb +32 -23
  21. data/lib/sidekiq/job.rb +13 -0
  22. data/lib/sidekiq/job_logger.rb +16 -28
  23. data/lib/sidekiq/job_retry.rb +10 -11
  24. data/lib/sidekiq/job_util.rb +65 -0
  25. data/lib/sidekiq/launcher.rb +104 -46
  26. data/lib/sidekiq/logger.rb +7 -2
  27. data/lib/sidekiq/manager.rb +10 -12
  28. data/lib/sidekiq/middleware/chain.rb +6 -4
  29. data/lib/sidekiq/middleware/current_attributes.rb +57 -0
  30. data/lib/sidekiq/paginator.rb +8 -8
  31. data/lib/sidekiq/processor.rb +4 -4
  32. data/lib/sidekiq/rails.rb +27 -18
  33. data/lib/sidekiq/redis_connection.rb +14 -13
  34. data/lib/sidekiq/scheduled.rb +51 -16
  35. data/lib/sidekiq/sd_notify.rb +1 -1
  36. data/lib/sidekiq/testing.rb +2 -4
  37. data/lib/sidekiq/util.rb +41 -0
  38. data/lib/sidekiq/version.rb +1 -1
  39. data/lib/sidekiq/web/action.rb +2 -2
  40. data/lib/sidekiq/web/application.rb +21 -12
  41. data/lib/sidekiq/web/csrf_protection.rb +180 -0
  42. data/lib/sidekiq/web/helpers.rb +36 -30
  43. data/lib/sidekiq/web/router.rb +5 -2
  44. data/lib/sidekiq/web.rb +36 -72
  45. data/lib/sidekiq/worker.rb +127 -12
  46. data/lib/sidekiq.rb +13 -3
  47. data/sidekiq.gemspec +11 -4
  48. data/web/assets/images/apple-touch-icon.png +0 -0
  49. data/web/assets/javascripts/application.js +82 -66
  50. data/web/assets/javascripts/dashboard.js +51 -51
  51. data/web/assets/stylesheets/application-dark.css +64 -43
  52. data/web/assets/stylesheets/application-rtl.css +0 -4
  53. data/web/assets/stylesheets/application.css +43 -239
  54. data/web/locales/ar.yml +8 -2
  55. data/web/locales/en.yml +4 -1
  56. data/web/locales/es.yml +18 -2
  57. data/web/locales/fr.yml +8 -1
  58. data/web/locales/ja.yml +3 -0
  59. data/web/locales/lt.yml +1 -1
  60. data/web/locales/pl.yml +4 -4
  61. data/web/locales/ru.yml +4 -0
  62. data/web/views/_footer.erb +1 -1
  63. data/web/views/_job_info.erb +1 -1
  64. data/web/views/_poll_link.erb +2 -5
  65. data/web/views/_summary.erb +7 -7
  66. data/web/views/busy.erb +50 -19
  67. data/web/views/dashboard.erb +22 -14
  68. data/web/views/dead.erb +1 -1
  69. data/web/views/layout.erb +2 -1
  70. data/web/views/morgue.erb +6 -6
  71. data/web/views/queue.erb +11 -11
  72. data/web/views/queues.erb +4 -4
  73. data/web/views/retries.erb +7 -7
  74. data/web/views/retry.erb +1 -1
  75. data/web/views/scheduled.erb +1 -1
  76. metadata +24 -49
  77. data/.circleci/config.yml +0 -60
  78. data/.github/contributing.md +0 -32
  79. data/.github/issue_template.md +0 -11
  80. data/.gitignore +0 -13
  81. data/.standard.yml +0 -20
  82. data/3.0-Upgrade.md +0 -70
  83. data/4.0-Upgrade.md +0 -53
  84. data/5.0-Upgrade.md +0 -56
  85. data/6.0-Upgrade.md +0 -72
  86. data/COMM-LICENSE +0 -97
  87. data/Ent-2.0-Upgrade.md +0 -37
  88. data/Ent-Changes.md +0 -256
  89. data/Gemfile +0 -24
  90. data/Gemfile.lock +0 -208
  91. data/Pro-2.0-Upgrade.md +0 -138
  92. data/Pro-3.0-Upgrade.md +0 -44
  93. data/Pro-4.0-Upgrade.md +0 -35
  94. data/Pro-5.0-Upgrade.md +0 -25
  95. data/Pro-Changes.md +0 -782
  96. data/Rakefile +0 -10
  97. data/code_of_conduct.md +0 -50
  98. data/lib/generators/sidekiq/worker_generator.rb +0 -57
data/lib/sidekiq/web.rb CHANGED
@@ -10,12 +10,11 @@ require "sidekiq/web/helpers"
10
10
  require "sidekiq/web/router"
11
11
  require "sidekiq/web/action"
12
12
  require "sidekiq/web/application"
13
+ require "sidekiq/web/csrf_protection"
13
14
 
14
- require "rack/protection"
15
-
15
+ require "rack/content_length"
16
16
  require "rack/builder"
17
- require "rack/file"
18
- require "rack/session/cookie"
17
+ require "rack/static"
19
18
 
20
19
  module Sidekiq
21
20
  class Web
@@ -39,14 +38,6 @@ module Sidekiq
39
38
  self
40
39
  end
41
40
 
42
- def middlewares
43
- @middlewares ||= []
44
- end
45
-
46
- def use(*middleware_args, &block)
47
- middlewares << [middleware_args, block]
48
- end
49
-
50
41
  def default_tabs
51
42
  DEFAULT_TABS
52
43
  end
@@ -72,32 +63,45 @@ module Sidekiq
72
63
  opts.each { |key| set(key, false) }
73
64
  end
74
65
 
75
- # Helper for the Sinatra syntax: Sidekiq::Web.set(:session_secret, Rails.application.secrets...)
66
+ def middlewares
67
+ @middlewares ||= []
68
+ end
69
+
70
+ def use(*args, &block)
71
+ middlewares << [args, block]
72
+ end
73
+
76
74
  def set(attribute, value)
77
75
  send(:"#{attribute}=", value)
78
76
  end
79
77
 
80
- attr_accessor :app_url, :session_secret, :redis_pool, :sessions
78
+ def sessions=(val)
79
+ puts "WARNING: Sidekiq::Web.sessions= is no longer relevant and will be removed in Sidekiq 7.0. #{caller(1..1).first}"
80
+ end
81
+
82
+ def session_secret=(val)
83
+ puts "WARNING: Sidekiq::Web.session_secret= is no longer relevant and will be removed in Sidekiq 7.0. #{caller(1..1).first}"
84
+ end
85
+
86
+ attr_accessor :app_url, :redis_pool
81
87
  attr_writer :locales, :views
82
88
  end
83
89
 
84
90
  def self.inherited(child)
85
91
  child.app_url = app_url
86
- child.session_secret = session_secret
87
92
  child.redis_pool = redis_pool
88
- child.sessions = sessions
89
93
  end
90
94
 
91
95
  def settings
92
96
  self.class.settings
93
97
  end
94
98
 
95
- def use(*middleware_args, &block)
96
- middlewares << [middleware_args, block]
99
+ def middlewares
100
+ @middlewares ||= self.class.middlewares
97
101
  end
98
102
 
99
- def middlewares
100
- @middlewares ||= Web.middlewares.dup
103
+ def use(*args, &block)
104
+ middlewares << [args, block]
101
105
  end
102
106
 
103
107
  def call(env)
@@ -125,18 +129,8 @@ module Sidekiq
125
129
  send(:"#{attribute}=", value)
126
130
  end
127
131
 
128
- # Default values
129
- set :sessions, true
130
-
131
- attr_writer :sessions
132
-
133
- def sessions
134
- unless instance_variable_defined?("@sessions")
135
- @sessions = self.class.sessions
136
- @sessions = @sessions.to_hash.dup if @sessions.respond_to?(:to_hash)
137
- end
138
-
139
- @sessions
132
+ def sessions=(val)
133
+ puts "Sidekiq::Web#sessions= is no longer relevant and will be removed in Sidekiq 7.0. #{caller[2..2].first}"
140
134
  end
141
135
 
142
136
  def self.register(extension)
@@ -145,50 +139,20 @@ module Sidekiq
145
139
 
146
140
  private
147
141
 
148
- def using?(middleware)
149
- middlewares.any? do |(m, _)|
150
- m.is_a?(Array) && (m[0] == middleware || m[0].is_a?(middleware))
151
- end
152
- end
153
-
154
- def build_sessions
155
- middlewares = self.middlewares
156
-
157
- unless using?(::Rack::Protection) || ENV["RACK_ENV"] == "test"
158
- middlewares.unshift [[::Rack::Protection, {use: :authenticity_token}], nil]
159
- end
160
-
161
- s = sessions
162
- return unless s
163
-
164
- unless using? ::Rack::Session::Cookie
165
- unless (secret = Web.session_secret)
166
- require "securerandom"
167
- secret = SecureRandom.hex(64)
168
- end
169
-
170
- options = {secret: secret}
171
- options = options.merge(s.to_hash) if s.respond_to? :to_hash
172
-
173
- middlewares.unshift [[::Rack::Session::Cookie, options], nil]
174
- end
175
- end
176
-
177
142
  def build
178
- build_sessions
179
-
180
- middlewares = self.middlewares
181
143
  klass = self.class
144
+ m = middlewares
182
145
 
183
- ::Rack::Builder.new do
184
- %w[stylesheets javascripts images].each do |asset_dir|
185
- map "/#{asset_dir}" do
186
- run ::Rack::File.new("#{ASSETS}/#{asset_dir}", {"Cache-Control" => "public, max-age=86400"})
187
- end
188
- end
189
-
190
- middlewares.each { |middleware, block| use(*middleware, &block) }
146
+ rules = []
147
+ rules = [[:all, {"Cache-Control" => "public, max-age=86400"}]] unless ENV["SIDEKIQ_WEB_TESTING"]
191
148
 
149
+ ::Rack::Builder.new do
150
+ use Rack::Static, urls: ["/stylesheets", "/images", "/javascripts"],
151
+ root: ASSETS,
152
+ cascade: true,
153
+ header_rules: rules
154
+ m.each { |middleware, block| use(*middleware, &block) }
155
+ use Sidekiq::Web::CsrfProtection unless $TESTING
192
156
  run WebApplication.new(klass)
193
157
  end
194
158
  end
@@ -9,6 +9,7 @@ module Sidekiq
9
9
  #
10
10
  # class HardWorker
11
11
  # include Sidekiq::Worker
12
+ # sidekiq_options queue: 'critical', retry: 5
12
13
  #
13
14
  # def perform(*args)
14
15
  # # do some work
@@ -20,6 +21,26 @@ module Sidekiq
20
21
  # HardWorker.perform_async(1, 2, 3)
21
22
  #
22
23
  # Note that perform_async is a class method, perform is an instance method.
24
+ #
25
+ # Sidekiq::Worker also includes several APIs to provide compatibility with
26
+ # ActiveJob.
27
+ #
28
+ # class SomeWorker
29
+ # include Sidekiq::Worker
30
+ # queue_as :critical
31
+ #
32
+ # def perform(...)
33
+ # end
34
+ # end
35
+ #
36
+ # SomeWorker.set(wait_until: 1.hour).perform_async(123)
37
+ #
38
+ # Note that arguments passed to the job must still obey Sidekiq's
39
+ # best practice for simple, JSON-native data types. Sidekiq will not
40
+ # implement ActiveJob's more complex argument serialization. For
41
+ # this reason, we don't implement `perform_later` as our call semantics
42
+ # are very different.
43
+ #
23
44
  module Worker
24
45
  ##
25
46
  # The Options module is extracted so we can include it in ActiveJob::Base
@@ -150,33 +171,97 @@ module Sidekiq
150
171
  # SomeWorker.set(queue: 'foo').perform_async(....)
151
172
  #
152
173
  class Setter
174
+ include Sidekiq::JobUtil
175
+
153
176
  def initialize(klass, opts)
154
177
  @klass = klass
155
178
  @opts = opts
179
+
180
+ # ActiveJob compatibility
181
+ interval = @opts.delete(:wait_until) || @opts.delete(:wait)
182
+ at(interval) if interval
156
183
  end
157
184
 
158
185
  def set(options)
186
+ interval = options.delete(:wait_until) || options.delete(:wait)
159
187
  @opts.merge!(options)
188
+ at(interval) if interval
160
189
  self
161
190
  end
162
191
 
163
192
  def perform_async(*args)
164
- @klass.client_push(@opts.merge("args" => args, "class" => @klass))
193
+ if @opts["sync"] == true
194
+ perform_inline(*args)
195
+ else
196
+ @klass.client_push(@opts.merge("args" => args, "class" => @klass))
197
+ end
198
+ end
199
+
200
+ # Explicit inline execution of a job. Returns nil if the job did not
201
+ # execute, true otherwise.
202
+ def perform_inline(*args)
203
+ raw = @opts.merge("args" => args, "class" => @klass).transform_keys(&:to_s)
204
+
205
+ # validate and normalize payload
206
+ item = normalize_item(raw)
207
+ queue = item["queue"]
208
+
209
+ # run client-side middleware
210
+ result = Sidekiq.client_middleware.invoke(item["class"], item, queue, Sidekiq.redis_pool) do
211
+ item
212
+ end
213
+ return nil unless result
214
+
215
+ # round-trip the payload via JSON
216
+ msg = Sidekiq.load_json(Sidekiq.dump_json(item))
217
+
218
+ # prepare the job instance
219
+ klass = msg["class"].constantize
220
+ job = klass.new
221
+ job.jid = msg["jid"]
222
+ job.bid = msg["bid"] if job.respond_to?(:bid)
223
+
224
+ # run the job through server-side middleware
225
+ result = Sidekiq.server_middleware.invoke(job, msg, msg["queue"]) do
226
+ # perform it
227
+ job.perform(*msg["args"])
228
+ true
229
+ end
230
+ return nil unless result
231
+ # jobs do not return a result. they should store any
232
+ # modified state.
233
+ true
234
+ end
235
+ alias_method :perform_sync, :perform_inline
236
+
237
+ def perform_bulk(args, batch_size: 1_000)
238
+ hash = @opts.transform_keys(&:to_s)
239
+ pool = Thread.current[:sidekiq_via_pool] || @klass.get_sidekiq_options["pool"] || Sidekiq.redis_pool
240
+ client = Sidekiq::Client.new(pool)
241
+ result = args.each_slice(batch_size).flat_map do |slice|
242
+ client.push_bulk(hash.merge("class" => @klass, "args" => slice))
243
+ end
244
+
245
+ result.is_a?(Enumerator::Lazy) ? result.force : result
165
246
  end
166
247
 
167
248
  # +interval+ must be a timestamp, numeric or something that acts
168
249
  # numeric (like an activesupport time interval).
169
250
  def perform_in(interval, *args)
251
+ at(interval).perform_async(*args)
252
+ end
253
+ alias_method :perform_at, :perform_in
254
+
255
+ private
256
+
257
+ def at(interval)
170
258
  int = interval.to_f
171
259
  now = Time.now.to_f
172
260
  ts = (int < 1_000_000_000 ? now + int : int)
173
-
174
- payload = @opts.merge("class" => @klass, "args" => args)
175
261
  # Optimization to enqueue something now that is scheduled to go out now or in the past
176
- payload["at"] = ts if ts > now
177
- @klass.client_push(payload)
262
+ @opts["at"] = ts if ts > now
263
+ self
178
264
  end
179
- alias_method :perform_at, :perform_in
180
265
  end
181
266
 
182
267
  module ClassMethods
@@ -192,12 +277,45 @@ module Sidekiq
192
277
  raise ArgumentError, "Do not call .delay_until on a Sidekiq::Worker class, call .perform_at"
193
278
  end
194
279
 
280
+ def queue_as(q)
281
+ sidekiq_options("queue" => q.to_s)
282
+ end
283
+
195
284
  def set(options)
196
285
  Setter.new(self, options)
197
286
  end
198
287
 
199
288
  def perform_async(*args)
200
- client_push("class" => self, "args" => args)
289
+ Setter.new(self, {}).perform_async(*args)
290
+ end
291
+
292
+ # Inline execution of job's perform method after passing through Sidekiq.client_middleware and Sidekiq.server_middleware
293
+ def perform_inline(*args)
294
+ Setter.new(self, {}).perform_inline(*args)
295
+ end
296
+
297
+ ##
298
+ # Push a large number of jobs to Redis, while limiting the batch of
299
+ # each job payload to 1,000. This method helps cut down on the number
300
+ # of round trips to Redis, which can increase the performance of enqueueing
301
+ # large numbers of jobs.
302
+ #
303
+ # +items+ must be an Array of Arrays.
304
+ #
305
+ # For finer-grained control, use `Sidekiq::Client.push_bulk` directly.
306
+ #
307
+ # Example (3 Redis round trips):
308
+ #
309
+ # SomeWorker.perform_async(1)
310
+ # SomeWorker.perform_async(2)
311
+ # SomeWorker.perform_async(3)
312
+ #
313
+ # Would instead become (1 Redis round trip):
314
+ #
315
+ # SomeWorker.perform_bulk([[1], [2], [3]])
316
+ #
317
+ def perform_bulk(*args, **kwargs)
318
+ Setter.new(self, {}).perform_bulk(*args, **kwargs)
201
319
  end
202
320
 
203
321
  # +interval+ must be a timestamp, numeric or something that acts
@@ -235,12 +353,9 @@ module Sidekiq
235
353
 
236
354
  def client_push(item) # :nodoc:
237
355
  pool = Thread.current[:sidekiq_via_pool] || get_sidekiq_options["pool"] || Sidekiq.redis_pool
238
- # stringify
239
- item.keys.each do |key|
240
- item[key.to_s] = item.delete(key)
241
- end
356
+ stringified_item = item.transform_keys(&:to_s)
242
357
 
243
- Sidekiq::Client.new(pool).push(item)
358
+ Sidekiq::Client.new(pool).push(stringified_item)
244
359
  end
245
360
  end
246
361
  end
data/lib/sidekiq.rb CHANGED
@@ -6,6 +6,7 @@ fail "Sidekiq #{Sidekiq::VERSION} does not support Ruby versions below 2.5.0." i
6
6
  require "sidekiq/logger"
7
7
  require "sidekiq/client"
8
8
  require "sidekiq/worker"
9
+ require "sidekiq/job"
9
10
  require "sidekiq/redis_connection"
10
11
  require "sidekiq/delay"
11
12
 
@@ -20,10 +21,12 @@ module Sidekiq
20
21
  labels: [],
21
22
  concurrency: 10,
22
23
  require: ".",
24
+ strict: true,
23
25
  environment: nil,
24
26
  timeout: 25,
25
27
  poll_interval_average: nil,
26
28
  average_scheduled_poll_interval: 5,
29
+ on_complex_arguments: :warn,
27
30
  error_handlers: [],
28
31
  death_handlers: [],
29
32
  lifecycle_events: {
@@ -95,10 +98,13 @@ module Sidekiq
95
98
  retryable = true
96
99
  begin
97
100
  yield conn
98
- rescue Redis::CommandError => ex
101
+ rescue Redis::BaseError => ex
99
102
  # 2550 Failover can cause the server to become a replica, need
100
103
  # to disconnect and reopen the socket to get back to the primary.
101
- if retryable && ex.message =~ /READONLY/
104
+ # 4495 Use the same logic if we have a "Not enough replicas" error from the primary
105
+ # 4985 Use the same logic when a blocking command is force-unblocked
106
+ # The same retry logic is also used in client.rb
107
+ if retryable && ex.message =~ /READONLY|NOREPLICAS|UNBLOCKED/
102
108
  conn.disconnect!
103
109
  retryable = false
104
110
  retry
@@ -196,7 +202,7 @@ module Sidekiq
196
202
  end
197
203
 
198
204
  def self.logger
199
- @logger ||= Sidekiq::Logger.new(STDOUT, level: Logger::INFO)
205
+ @logger ||= Sidekiq::Logger.new($stdout, level: Logger::INFO)
200
206
  end
201
207
 
202
208
  def self.logger=(logger)
@@ -248,6 +254,10 @@ module Sidekiq
248
254
  options[:lifecycle_events][event] << block
249
255
  end
250
256
 
257
+ def self.strict_args!(mode = :raise)
258
+ options[:on_complex_arguments] = mode
259
+ end
260
+
251
261
  # We are shutting down Sidekiq but what about workers that
252
262
  # are working on some long job? This error is
253
263
  # raised in workers that have not finished within the hard
data/sidekiq.gemspec CHANGED
@@ -5,17 +5,24 @@ Gem::Specification.new do |gem|
5
5
  gem.email = ["mperham@gmail.com"]
6
6
  gem.summary = "Simple, efficient background processing for Ruby"
7
7
  gem.description = "Simple, efficient background processing for Ruby."
8
- gem.homepage = "http://sidekiq.org"
8
+ gem.homepage = "https://sidekiq.org"
9
9
  gem.license = "LGPL-3.0"
10
10
 
11
11
  gem.executables = ["sidekiq", "sidekiqmon"]
12
- gem.files = `git ls-files | grep -Ev '^(test|myapp|examples)'`.split("\n")
12
+ gem.files = ["sidekiq.gemspec", "README.md", "Changes.md", "LICENSE"] + `git ls-files | grep -E '^(bin|lib|web)'`.split("\n")
13
13
  gem.name = "sidekiq"
14
14
  gem.version = Sidekiq::VERSION
15
15
  gem.required_ruby_version = ">= 2.5.0"
16
16
 
17
- gem.add_dependency "redis", ">= 4.1.0"
17
+ gem.metadata = {
18
+ "homepage_uri" => "https://sidekiq.org",
19
+ "bug_tracker_uri" => "https://github.com/mperham/sidekiq/issues",
20
+ "documentation_uri" => "https://github.com/mperham/sidekiq/wiki",
21
+ "changelog_uri" => "https://github.com/mperham/sidekiq/blob/main/Changes.md",
22
+ "source_code_uri" => "https://github.com/mperham/sidekiq"
23
+ }
24
+
25
+ gem.add_dependency "redis", ">= 4.2.0"
18
26
  gem.add_dependency "connection_pool", ">= 2.2.2"
19
27
  gem.add_dependency "rack", "~> 2.0"
20
- gem.add_dependency "rack-protection", ">= 2.0.0"
21
28
  end