sidekiq 6.2.2 → 8.1.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.
Files changed (181) hide show
  1. checksums.yaml +4 -4
  2. data/Changes.md +726 -11
  3. data/LICENSE.txt +9 -0
  4. data/README.md +70 -39
  5. data/bin/kiq +17 -0
  6. data/bin/lint-herb +13 -0
  7. data/bin/multi_queue_bench +271 -0
  8. data/bin/sidekiq +4 -9
  9. data/bin/sidekiqload +214 -115
  10. data/bin/sidekiqmon +4 -1
  11. data/bin/webload +69 -0
  12. data/lib/active_job/queue_adapters/sidekiq_adapter.rb +124 -0
  13. data/lib/generators/sidekiq/job_generator.rb +71 -0
  14. data/lib/generators/sidekiq/templates/{worker.rb.erb → job.rb.erb} +3 -3
  15. data/lib/generators/sidekiq/templates/{worker_spec.rb.erb → job_spec.rb.erb} +1 -1
  16. data/lib/generators/sidekiq/templates/{worker_test.rb.erb → job_test.rb.erb} +1 -1
  17. data/lib/sidekiq/api.rb +729 -264
  18. data/lib/sidekiq/capsule.rb +135 -0
  19. data/lib/sidekiq/cli.rb +124 -100
  20. data/lib/sidekiq/client.rb +153 -106
  21. data/lib/sidekiq/component.rb +132 -0
  22. data/lib/sidekiq/config.rb +320 -0
  23. data/lib/sidekiq/deploy.rb +64 -0
  24. data/lib/sidekiq/embedded.rb +64 -0
  25. data/lib/sidekiq/fetch.rb +27 -26
  26. data/lib/sidekiq/iterable_job.rb +56 -0
  27. data/lib/sidekiq/job/interrupt_handler.rb +24 -0
  28. data/lib/sidekiq/job/iterable/active_record_enumerator.rb +53 -0
  29. data/lib/sidekiq/job/iterable/csv_enumerator.rb +47 -0
  30. data/lib/sidekiq/job/iterable/enumerators.rb +135 -0
  31. data/lib/sidekiq/job/iterable.rb +322 -0
  32. data/lib/sidekiq/job.rb +397 -5
  33. data/lib/sidekiq/job_logger.rb +23 -32
  34. data/lib/sidekiq/job_retry.rb +141 -68
  35. data/lib/sidekiq/job_util.rb +113 -0
  36. data/lib/sidekiq/launcher.rb +122 -98
  37. data/lib/sidekiq/loader.rb +57 -0
  38. data/lib/sidekiq/logger.rb +27 -106
  39. data/lib/sidekiq/manager.rb +41 -43
  40. data/lib/sidekiq/metrics/query.rb +184 -0
  41. data/lib/sidekiq/metrics/shared.rb +109 -0
  42. data/lib/sidekiq/metrics/tracking.rb +153 -0
  43. data/lib/sidekiq/middleware/chain.rb +96 -51
  44. data/lib/sidekiq/middleware/current_attributes.rb +120 -0
  45. data/lib/sidekiq/middleware/i18n.rb +8 -4
  46. data/lib/sidekiq/middleware/modules.rb +23 -0
  47. data/lib/sidekiq/monitor.rb +16 -6
  48. data/lib/sidekiq/paginator.rb +37 -10
  49. data/lib/sidekiq/processor.rb +105 -87
  50. data/lib/sidekiq/profiler.rb +73 -0
  51. data/lib/sidekiq/rails.rb +49 -36
  52. data/lib/sidekiq/redis_client_adapter.rb +117 -0
  53. data/lib/sidekiq/redis_connection.rb +55 -86
  54. data/lib/sidekiq/ring_buffer.rb +32 -0
  55. data/lib/sidekiq/scheduled.rb +106 -50
  56. data/lib/sidekiq/systemd.rb +2 -0
  57. data/lib/sidekiq/test_api.rb +331 -0
  58. data/lib/sidekiq/testing/inline.rb +2 -30
  59. data/lib/sidekiq/testing.rb +2 -342
  60. data/lib/sidekiq/transaction_aware_client.rb +59 -0
  61. data/lib/sidekiq/tui/controls.rb +53 -0
  62. data/lib/sidekiq/tui/filtering.rb +53 -0
  63. data/lib/sidekiq/tui/tabs/base_tab.rb +204 -0
  64. data/lib/sidekiq/tui/tabs/busy.rb +118 -0
  65. data/lib/sidekiq/tui/tabs/dead.rb +19 -0
  66. data/lib/sidekiq/tui/tabs/home.rb +144 -0
  67. data/lib/sidekiq/tui/tabs/metrics.rb +131 -0
  68. data/lib/sidekiq/tui/tabs/queues.rb +95 -0
  69. data/lib/sidekiq/tui/tabs/retries.rb +19 -0
  70. data/lib/sidekiq/tui/tabs/scheduled.rb +19 -0
  71. data/lib/sidekiq/tui/tabs/set_tab.rb +96 -0
  72. data/lib/sidekiq/tui/tabs.rb +15 -0
  73. data/lib/sidekiq/tui.rb +382 -0
  74. data/lib/sidekiq/version.rb +6 -1
  75. data/lib/sidekiq/web/action.rb +149 -64
  76. data/lib/sidekiq/web/application.rb +376 -268
  77. data/lib/sidekiq/web/config.rb +117 -0
  78. data/lib/sidekiq/web/helpers.rb +213 -87
  79. data/lib/sidekiq/web/router.rb +61 -74
  80. data/lib/sidekiq/web.rb +71 -100
  81. data/lib/sidekiq/worker_compatibility_alias.rb +13 -0
  82. data/lib/sidekiq.rb +95 -196
  83. data/sidekiq.gemspec +14 -11
  84. data/web/assets/images/logo.png +0 -0
  85. data/web/assets/images/status.png +0 -0
  86. data/web/assets/javascripts/application.js +171 -57
  87. data/web/assets/javascripts/base-charts.js +120 -0
  88. data/web/assets/javascripts/chart.min.js +13 -0
  89. data/web/assets/javascripts/chartjs-adapter-date-fns.min.js +7 -0
  90. data/web/assets/javascripts/chartjs-plugin-annotation.min.js +7 -0
  91. data/web/assets/javascripts/dashboard-charts.js +194 -0
  92. data/web/assets/javascripts/dashboard.js +41 -274
  93. data/web/assets/javascripts/metrics.js +280 -0
  94. data/web/assets/stylesheets/style.css +776 -0
  95. data/web/locales/ar.yml +72 -70
  96. data/web/locales/cs.yml +64 -62
  97. data/web/locales/da.yml +62 -53
  98. data/web/locales/de.yml +67 -65
  99. data/web/locales/el.yml +45 -24
  100. data/web/locales/en.yml +93 -69
  101. data/web/locales/es.yml +91 -68
  102. data/web/locales/fa.yml +67 -65
  103. data/web/locales/fr.yml +82 -67
  104. data/web/locales/gd.yml +110 -0
  105. data/web/locales/he.yml +67 -64
  106. data/web/locales/hi.yml +61 -59
  107. data/web/locales/it.yml +94 -54
  108. data/web/locales/ja.yml +74 -68
  109. data/web/locales/ko.yml +54 -52
  110. data/web/locales/lt.yml +68 -66
  111. data/web/locales/nb.yml +63 -61
  112. data/web/locales/nl.yml +54 -52
  113. data/web/locales/pl.yml +47 -45
  114. data/web/locales/{pt-br.yml → pt-BR.yml} +85 -56
  115. data/web/locales/pt.yml +53 -51
  116. data/web/locales/ru.yml +69 -66
  117. data/web/locales/sv.yml +55 -53
  118. data/web/locales/ta.yml +62 -60
  119. data/web/locales/tr.yml +102 -0
  120. data/web/locales/uk.yml +87 -61
  121. data/web/locales/ur.yml +66 -64
  122. data/web/locales/vi.yml +69 -67
  123. data/web/locales/zh-CN.yml +107 -0
  124. data/web/locales/{zh-tw.yml → zh-TW.yml} +44 -9
  125. data/web/views/_footer.html.erb +32 -0
  126. data/web/views/_job_info.html.erb +115 -0
  127. data/web/views/_metrics_period_select.html.erb +15 -0
  128. data/web/views/_nav.html.erb +45 -0
  129. data/web/views/_paging.html.erb +26 -0
  130. data/web/views/_poll_link.html.erb +4 -0
  131. data/web/views/_summary.html.erb +40 -0
  132. data/web/views/busy.html.erb +151 -0
  133. data/web/views/dashboard.html.erb +104 -0
  134. data/web/views/dead.html.erb +38 -0
  135. data/web/views/filtering.html.erb +6 -0
  136. data/web/views/layout.html.erb +26 -0
  137. data/web/views/metrics.html.erb +85 -0
  138. data/web/views/metrics_for_job.html.erb +58 -0
  139. data/web/views/morgue.html.erb +69 -0
  140. data/web/views/profiles.html.erb +43 -0
  141. data/web/views/queue.html.erb +57 -0
  142. data/web/views/queues.html.erb +46 -0
  143. data/web/views/retries.html.erb +77 -0
  144. data/web/views/retry.html.erb +39 -0
  145. data/web/views/scheduled.html.erb +64 -0
  146. data/web/views/{scheduled_job_info.erb → scheduled_job_info.html.erb} +3 -3
  147. metadata +130 -61
  148. data/LICENSE +0 -9
  149. data/lib/generators/sidekiq/worker_generator.rb +0 -57
  150. data/lib/sidekiq/delay.rb +0 -41
  151. data/lib/sidekiq/exception_handler.rb +0 -27
  152. data/lib/sidekiq/extensions/action_mailer.rb +0 -48
  153. data/lib/sidekiq/extensions/active_record.rb +0 -43
  154. data/lib/sidekiq/extensions/class_methods.rb +0 -43
  155. data/lib/sidekiq/extensions/generic_proxy.rb +0 -33
  156. data/lib/sidekiq/util.rb +0 -95
  157. data/lib/sidekiq/web/csrf_protection.rb +0 -180
  158. data/lib/sidekiq/worker.rb +0 -244
  159. data/web/assets/stylesheets/application-dark.css +0 -147
  160. data/web/assets/stylesheets/application-rtl.css +0 -246
  161. data/web/assets/stylesheets/application.css +0 -1053
  162. data/web/assets/stylesheets/bootstrap-rtl.min.css +0 -9
  163. data/web/assets/stylesheets/bootstrap.css +0 -5
  164. data/web/locales/zh-cn.yml +0 -68
  165. data/web/views/_footer.erb +0 -20
  166. data/web/views/_job_info.erb +0 -89
  167. data/web/views/_nav.erb +0 -52
  168. data/web/views/_paging.erb +0 -23
  169. data/web/views/_poll_link.erb +0 -7
  170. data/web/views/_status.erb +0 -4
  171. data/web/views/_summary.erb +0 -40
  172. data/web/views/busy.erb +0 -132
  173. data/web/views/dashboard.erb +0 -83
  174. data/web/views/dead.erb +0 -34
  175. data/web/views/layout.erb +0 -42
  176. data/web/views/morgue.erb +0 -78
  177. data/web/views/queue.erb +0 -55
  178. data/web/views/queues.erb +0 -38
  179. data/web/views/retries.erb +0 -83
  180. data/web/views/retry.erb +0 -34
  181. data/web/views/scheduled.erb +0 -57
data/lib/sidekiq/job.rb CHANGED
@@ -1,8 +1,400 @@
1
- require "sidekiq/worker"
1
+ # frozen_string_literal: true
2
+
3
+ require "sidekiq/client"
2
4
 
3
5
  module Sidekiq
4
- # Sidekiq::Job is a new alias for Sidekiq::Worker, coming in 6.3.0.
5
- # You can opt into this by requiring 'sidekiq/job' in your initializer
6
- # and then using `include Sidekiq::Job` rather than `Sidekiq::Worker`.
7
- Job = Worker
6
+ ##
7
+ # Include this module in your job class and you can easily create
8
+ # asynchronous jobs:
9
+ #
10
+ # class HardJob
11
+ # include Sidekiq::Job
12
+ # sidekiq_options queue: 'critical', retry: 5
13
+ #
14
+ # def perform(*args)
15
+ # # do some work
16
+ # end
17
+ # end
18
+ #
19
+ # Then in your Rails app, you can do this:
20
+ #
21
+ # HardJob.perform_async(1, 2, 3)
22
+ #
23
+ # Note that perform_async is a class method, perform is an instance method.
24
+ #
25
+ # Sidekiq::Job also includes several APIs to provide compatibility with
26
+ # ActiveJob.
27
+ #
28
+ # class SomeJob
29
+ # include Sidekiq::Job
30
+ # queue_as :critical
31
+ #
32
+ # def perform(...)
33
+ # end
34
+ # end
35
+ #
36
+ # SomeJob.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
+ #
44
+ module Job
45
+ ##
46
+ # The Options module is extracted so we can include it in ActiveJob::Base
47
+ # and allow native AJs to configure Sidekiq features/internals.
48
+ module Options
49
+ def self.included(base)
50
+ base.extend(ClassMethods)
51
+ base.sidekiq_class_attribute :sidekiq_options_hash
52
+ base.sidekiq_class_attribute :sidekiq_retry_in_block
53
+ base.sidekiq_class_attribute :sidekiq_retries_exhausted_block
54
+ end
55
+
56
+ module ClassMethods
57
+ ACCESSOR_MUTEX = Mutex.new
58
+
59
+ ##
60
+ # Allows customization for this type of Job.
61
+ # Legal options:
62
+ #
63
+ # queue - name of queue to use for this job type, default *default*
64
+ # retry - enable retries for this Job in case of error during execution,
65
+ # *true* to use the default or *Integer* count
66
+ # backtrace - whether to save any error backtrace in the retry payload to display in web UI,
67
+ # can be true, false or an integer number of lines to save, default *false*
68
+ #
69
+ # In practice, any option is allowed. This is the main mechanism to configure the
70
+ # options for a specific job.
71
+ def sidekiq_options(opts = {})
72
+ # stringify 2 levels of keys
73
+ opts = opts.to_h do |k, v|
74
+ [k.to_s, (Hash === v) ? v.transform_keys(&:to_s) : v]
75
+ end
76
+
77
+ self.sidekiq_options_hash = get_sidekiq_options.merge(opts)
78
+ end
79
+
80
+ def sidekiq_retry_in(&block)
81
+ self.sidekiq_retry_in_block = block
82
+ end
83
+
84
+ def sidekiq_retries_exhausted(&block)
85
+ self.sidekiq_retries_exhausted_block = block
86
+ end
87
+
88
+ def get_sidekiq_options # :nodoc:
89
+ self.sidekiq_options_hash ||= Sidekiq.default_job_options
90
+ end
91
+
92
+ def sidekiq_class_attribute(*attrs)
93
+ instance_reader = true
94
+ instance_writer = true
95
+
96
+ attrs.each do |name|
97
+ synchronized_getter = "__synchronized_#{name}"
98
+
99
+ singleton_class.instance_eval do
100
+ undef_method(name) if method_defined?(name) || private_method_defined?(name)
101
+ end
102
+
103
+ define_singleton_method(synchronized_getter) { nil }
104
+ singleton_class.class_eval do
105
+ private(synchronized_getter)
106
+ end
107
+
108
+ define_singleton_method(name) { ACCESSOR_MUTEX.synchronize { send synchronized_getter } }
109
+
110
+ ivar = "@#{name}"
111
+
112
+ singleton_class.instance_eval do
113
+ m = "#{name}="
114
+ undef_method(m) if method_defined?(m) || private_method_defined?(m)
115
+ end
116
+ define_singleton_method(:"#{name}=") do |val|
117
+ singleton_class.class_eval do
118
+ ACCESSOR_MUTEX.synchronize do
119
+ undef_method(synchronized_getter) if method_defined?(synchronized_getter) || private_method_defined?(synchronized_getter)
120
+ define_method(synchronized_getter) { val }
121
+ end
122
+ end
123
+
124
+ if singleton_class?
125
+ class_eval do
126
+ undef_method(name) if method_defined?(name) || private_method_defined?(name)
127
+ define_method(name) do
128
+ if instance_variable_defined? ivar
129
+ instance_variable_get ivar
130
+ else
131
+ singleton_class.send name
132
+ end
133
+ end
134
+ end
135
+ end
136
+ val
137
+ end
138
+
139
+ if instance_reader
140
+ undef_method(name) if method_defined?(name) || private_method_defined?(name)
141
+ define_method(name) do
142
+ if instance_variable_defined?(ivar)
143
+ instance_variable_get ivar
144
+ else
145
+ self.class.public_send name
146
+ end
147
+ end
148
+ end
149
+
150
+ if instance_writer
151
+ m = "#{name}="
152
+ undef_method(m) if method_defined?(m) || private_method_defined?(m)
153
+ attr_writer name
154
+ end
155
+ end
156
+ end
157
+ end
158
+ end
159
+
160
+ attr_accessor :jid
161
+
162
+ # This attribute is implementation-specific and not a public API
163
+ attr_accessor :_context
164
+
165
+ def self.included(base)
166
+ raise ArgumentError, "Sidekiq::Job cannot be included in an ActiveJob: #{base.name}" if base.ancestors.any? { |c| c.name == "ActiveJob::Base" }
167
+
168
+ base.include(Options)
169
+ base.extend(ClassMethods)
170
+ end
171
+
172
+ def logger
173
+ Sidekiq.logger
174
+ end
175
+
176
+ def interrupted?
177
+ @_context&.stopping?
178
+ end
179
+
180
+ # This helper class encapsulates the set options for `set`, e.g.
181
+ #
182
+ # SomeJob.set(queue: 'foo').perform_async(....)
183
+ #
184
+ class Setter
185
+ include Sidekiq::JobUtil
186
+
187
+ def initialize(klass, opts)
188
+ @klass = klass
189
+ # NB: the internal hash always has stringified keys
190
+ @opts = opts.transform_keys(&:to_s)
191
+
192
+ # ActiveJob compatibility
193
+ interval = @opts.delete("wait_until") || @opts.delete("wait")
194
+ at(interval) if interval
195
+ end
196
+
197
+ def set(options)
198
+ hash = options.transform_keys(&:to_s)
199
+ interval = hash.delete("wait_until") || @opts.delete("wait")
200
+ @opts.merge!(hash)
201
+ at(interval) if interval
202
+ self
203
+ end
204
+
205
+ def perform_async(*args)
206
+ if @opts["sync"] == true
207
+ perform_inline(*args)
208
+ else
209
+ @klass.client_push(@opts.merge("args" => args, "class" => @klass))
210
+ end
211
+ end
212
+
213
+ # Explicit inline execution of a job. Returns nil if the job did not
214
+ # execute, true otherwise.
215
+ def perform_inline(*args)
216
+ raw = @opts.merge("args" => args, "class" => @klass)
217
+
218
+ # validate and normalize payload
219
+ item = normalize_item(raw)
220
+ queue = item["queue"]
221
+
222
+ # run client-side middleware
223
+ cfg = Sidekiq.default_configuration
224
+ result = cfg.client_middleware.invoke(item["class"], item, queue, cfg.redis_pool) do
225
+ item
226
+ end
227
+ return nil unless result
228
+
229
+ verify_json(item)
230
+
231
+ # round-trip the payload via JSON
232
+ msg = Sidekiq.load_json(Sidekiq.dump_json(item))
233
+
234
+ # prepare the job instance
235
+ klass = Object.const_get(msg["class"])
236
+ job = klass.new
237
+ job.jid = msg["jid"]
238
+ job.bid = msg["bid"] if job.respond_to?(:bid)
239
+
240
+ # run the job through server-side middleware
241
+ result = cfg.server_middleware.invoke(job, msg, msg["queue"]) do
242
+ # perform it
243
+ job.perform(*msg["args"])
244
+ true
245
+ end
246
+ return nil unless result
247
+ # jobs do not return a result. they should store any
248
+ # modified state.
249
+ true
250
+ end
251
+ alias_method :perform_sync, :perform_inline
252
+
253
+ def perform_bulk(args, **options)
254
+ client = @klass.build_client
255
+ client.push_bulk(@opts.merge({"class" => @klass, "args" => args}, options))
256
+ end
257
+
258
+ # +interval+ must be a timestamp, numeric or something that acts
259
+ # numeric (like an activesupport time interval).
260
+ def perform_in(interval, *args)
261
+ at(interval).perform_async(*args)
262
+ end
263
+ alias_method :perform_at, :perform_in
264
+
265
+ private
266
+
267
+ def at(interval)
268
+ int = interval.to_f
269
+ now = Time.now.to_f
270
+ ts = ((int < 1_000_000_000) ? now + int : int)
271
+ # Optimization to enqueue something now that is scheduled to go out now or in the past
272
+ @opts["at"] = ts if ts > now
273
+ self
274
+ end
275
+ end
276
+
277
+ module ClassMethods
278
+ def delay(*args)
279
+ raise ArgumentError, "Do not call .delay on a Sidekiq::Job class, call .perform_async"
280
+ end
281
+
282
+ def delay_for(*args)
283
+ raise ArgumentError, "Do not call .delay_for on a Sidekiq::Job class, call .perform_in"
284
+ end
285
+
286
+ def delay_until(*args)
287
+ raise ArgumentError, "Do not call .delay_until on a Sidekiq::Job class, call .perform_at"
288
+ end
289
+
290
+ def queue_as(q)
291
+ sidekiq_options("queue" => q.to_s)
292
+ end
293
+
294
+ def set(options)
295
+ Setter.new(self, options)
296
+ end
297
+
298
+ def perform_async(*args)
299
+ Setter.new(self, {}).perform_async(*args)
300
+ end
301
+
302
+ # Inline execution of job's perform method after passing through Sidekiq.client_middleware and Sidekiq.server_middleware
303
+ def perform_inline(*args)
304
+ Setter.new(self, {}).perform_inline(*args)
305
+ end
306
+ alias_method :perform_sync, :perform_inline
307
+
308
+ ##
309
+ # Push a large number of jobs to Redis, while limiting the batch of
310
+ # each job payload to 1,000. This method helps cut down on the number
311
+ # of round trips to Redis, which can increase the performance of enqueueing
312
+ # large numbers of jobs.
313
+ #
314
+ # +items+ must be an Array of Arrays.
315
+ #
316
+ # The +:at+ option schedules the jobs for future execution. It accepts
317
+ # either a single Numeric timestamp (or seconds-from-now) applied to every
318
+ # job, or an Array of Numeric values with the same size as +items+ to
319
+ # schedule each job at its corresponding time.
320
+ #
321
+ # For finer-grained control, use `Sidekiq::Client.push_bulk` directly.
322
+ #
323
+ # Example (3 Redis round trips):
324
+ #
325
+ # SomeJob.perform_async(1)
326
+ # SomeJob.perform_async(2)
327
+ # SomeJob.perform_async(3)
328
+ #
329
+ # Would instead become (1 Redis round trip):
330
+ #
331
+ # SomeJob.perform_bulk([[1], [2], [3]])
332
+ #
333
+ # Scheduling every job 60 seconds from now (single Numeric +:at+):
334
+ #
335
+ # SomeJob.perform_bulk([[1], [2], [3]], at: 60)
336
+ #
337
+ # Scheduling each job at its own time (Array +:at+):
338
+ #
339
+ # SomeJob.perform_bulk([[1], [2]], at: [Time.now.to_f + 30, Time.now.to_f + 60])
340
+ #
341
+ def perform_bulk(*args, **kwargs)
342
+ Setter.new(self, {}).perform_bulk(*args, **kwargs)
343
+ end
344
+
345
+ # +interval+ must be a timestamp, numeric or something that acts
346
+ # numeric (like an activesupport time interval).
347
+ def perform_in(interval, *args)
348
+ int = interval.to_f
349
+ now = Time.now.to_f
350
+ ts = ((int < 1_000_000_000) ? now + int : int)
351
+
352
+ item = {"class" => self, "args" => args}
353
+
354
+ # Optimization to enqueue something now that is scheduled to go out now or in the past
355
+ item["at"] = ts if ts > now
356
+
357
+ client_push(item)
358
+ end
359
+ alias_method :perform_at, :perform_in
360
+
361
+ ##
362
+ # Allows customization for this type of Job.
363
+ # Legal options:
364
+ #
365
+ # queue - use a named queue for this Job, default 'default'
366
+ # retry - enable the RetryJobs middleware for this Job, *true* to use the default
367
+ # or *Integer* count
368
+ # backtrace - whether to save any error backtrace in the retry payload to display in web UI,
369
+ # can be true, false or an integer number of lines to save, default *false*
370
+ # pool - use the given Redis connection pool to push this type of job to a given shard.
371
+ #
372
+ # In practice, any option is allowed. This is the main mechanism to configure the
373
+ # options for a specific job.
374
+ def sidekiq_options(opts = {})
375
+ super
376
+ end
377
+
378
+ def client_push(item) # :nodoc:
379
+ raise ArgumentError, "Job payloads should contain no Symbols: #{item}" if item.any? { |k, v| k.is_a?(::Symbol) }
380
+
381
+ # allow the user to dynamically re-target jobs to another shard using the "pool" attribute
382
+ # FooJob.set(pool: SOME_POOL).perform_async
383
+ old = Thread.current[:sidekiq_redis_pool]
384
+ pool = item.delete("pool")
385
+ Thread.current[:sidekiq_redis_pool] = pool if pool
386
+ begin
387
+ build_client.push(item)
388
+ ensure
389
+ Thread.current[:sidekiq_redis_pool] = old
390
+ end
391
+ end
392
+
393
+ def build_client # :nodoc:
394
+ pool = Thread.current[:sidekiq_redis_pool] || get_sidekiq_options["pool"] || Sidekiq.default_configuration.redis_pool
395
+ client_class = Thread.current[:sidekiq_client_class] || get_sidekiq_options["client_class"] || Sidekiq::Client
396
+ client_class.new(pool: pool)
397
+ end
398
+ end
399
+ end
8
400
  end
@@ -2,56 +2,47 @@
2
2
 
3
3
  module Sidekiq
4
4
  class JobLogger
5
- def initialize(logger = Sidekiq.logger)
6
- @logger = logger
5
+ def initialize(config)
6
+ @config = config
7
+ @logger = @config.logger
8
+ @skip = !!@config[:skip_default_job_logging]
7
9
  end
8
10
 
9
11
  def call(item, queue)
10
12
  start = ::Process.clock_gettime(::Process::CLOCK_MONOTONIC)
11
- @logger.info("start")
13
+ @logger.info { "start" } unless @skip
12
14
 
13
15
  yield
14
16
 
15
- with_elapsed_time_context(start) do
16
- @logger.info("done")
17
- end
17
+ Sidekiq::Context.add(:elapsed, elapsed(start))
18
+ @logger.info { "done" } unless @skip
18
19
  rescue Exception
19
- with_elapsed_time_context(start) do
20
- @logger.info("fail")
21
- end
22
-
20
+ Sidekiq::Context.add(:elapsed, elapsed(start))
21
+ @logger.info { "fail" } unless @skip
23
22
  raise
24
23
  end
25
24
 
26
25
  def prepare(job_hash, &block)
27
- level = job_hash["log_level"]
28
- if level
29
- @logger.log_at(level) do
30
- Sidekiq::Context.with(job_hash_context(job_hash), &block)
31
- end
32
- else
33
- Sidekiq::Context.with(job_hash_context(job_hash), &block)
34
- end
35
- end
36
-
37
- def job_hash_context(job_hash)
38
26
  # If we're using a wrapper class, like ActiveJob, use the "wrapped"
39
27
  # attribute to expose the underlying thing.
40
28
  h = {
41
- class: job_hash["display_class"] || job_hash["wrapped"] || job_hash["class"],
42
- jid: job_hash["jid"]
29
+ jid: job_hash["jid"],
30
+ class: job_hash["wrapped"] || job_hash["class"]
43
31
  }
44
- h[:bid] = job_hash["bid"] if job_hash["bid"]
45
- h[:tags] = job_hash["tags"] if job_hash["tags"]
46
- h
47
- end
48
32
 
49
- def with_elapsed_time_context(start, &block)
50
- Sidekiq::Context.with(elapsed_time_context(start), &block)
51
- end
33
+ @config[:logged_job_attributes].each do |attr|
34
+ h[attr.to_sym] = job_hash[attr] if job_hash.has_key?(attr)
35
+ end
52
36
 
53
- def elapsed_time_context(start)
54
- {elapsed: elapsed(start).to_s}
37
+ Thread.current[:sidekiq_context] = h
38
+ level = job_hash["log_level"]
39
+ if level
40
+ @logger.with_level(level, &block)
41
+ else
42
+ yield
43
+ end
44
+ ensure
45
+ Thread.current[:sidekiq_context] = nil
55
46
  end
56
47
 
57
48
  private