sidekiq 3.5.4 → 7.2.0

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 (228) hide show
  1. checksums.yaml +5 -5
  2. data/Changes.md +992 -6
  3. data/LICENSE.txt +9 -0
  4. data/README.md +52 -43
  5. data/bin/sidekiq +22 -4
  6. data/bin/sidekiqload +209 -115
  7. data/bin/sidekiqmon +11 -0
  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/job_spec.rb.erb +6 -0
  11. data/lib/generators/sidekiq/templates/job_test.rb.erb +8 -0
  12. data/lib/sidekiq/api.rb +633 -295
  13. data/lib/sidekiq/capsule.rb +127 -0
  14. data/lib/sidekiq/cli.rb +270 -248
  15. data/lib/sidekiq/client.rb +139 -108
  16. data/lib/sidekiq/component.rb +68 -0
  17. data/lib/sidekiq/config.rb +287 -0
  18. data/lib/sidekiq/deploy.rb +62 -0
  19. data/lib/sidekiq/embedded.rb +61 -0
  20. data/lib/sidekiq/fetch.rb +53 -121
  21. data/lib/sidekiq/job.rb +374 -0
  22. data/lib/sidekiq/job_logger.rb +51 -0
  23. data/lib/sidekiq/job_retry.rb +301 -0
  24. data/lib/sidekiq/job_util.rb +107 -0
  25. data/lib/sidekiq/launcher.rb +241 -69
  26. data/lib/sidekiq/logger.rb +131 -0
  27. data/lib/sidekiq/manager.rb +88 -190
  28. data/lib/sidekiq/metrics/query.rb +155 -0
  29. data/lib/sidekiq/metrics/shared.rb +95 -0
  30. data/lib/sidekiq/metrics/tracking.rb +136 -0
  31. data/lib/sidekiq/middleware/chain.rb +114 -56
  32. data/lib/sidekiq/middleware/current_attributes.rb +95 -0
  33. data/lib/sidekiq/middleware/i18n.rb +8 -7
  34. data/lib/sidekiq/middleware/modules.rb +21 -0
  35. data/lib/sidekiq/monitor.rb +146 -0
  36. data/lib/sidekiq/paginator.rb +29 -16
  37. data/lib/sidekiq/processor.rb +238 -118
  38. data/lib/sidekiq/rails.rb +57 -27
  39. data/lib/sidekiq/redis_client_adapter.rb +111 -0
  40. data/lib/sidekiq/redis_connection.rb +49 -50
  41. data/lib/sidekiq/ring_buffer.rb +29 -0
  42. data/lib/sidekiq/scheduled.rb +173 -52
  43. data/lib/sidekiq/sd_notify.rb +149 -0
  44. data/lib/sidekiq/systemd.rb +24 -0
  45. data/lib/sidekiq/testing/inline.rb +7 -5
  46. data/lib/sidekiq/testing.rb +197 -65
  47. data/lib/sidekiq/transaction_aware_client.rb +44 -0
  48. data/lib/sidekiq/version.rb +4 -1
  49. data/lib/sidekiq/web/action.rb +93 -0
  50. data/lib/sidekiq/web/application.rb +463 -0
  51. data/lib/sidekiq/web/csrf_protection.rb +180 -0
  52. data/lib/sidekiq/web/helpers.rb +364 -0
  53. data/lib/sidekiq/web/router.rb +104 -0
  54. data/lib/sidekiq/web.rb +113 -216
  55. data/lib/sidekiq/worker_compatibility_alias.rb +13 -0
  56. data/lib/sidekiq.rb +99 -142
  57. data/sidekiq.gemspec +26 -23
  58. data/web/assets/images/apple-touch-icon.png +0 -0
  59. data/web/assets/javascripts/application.js +163 -74
  60. data/web/assets/javascripts/base-charts.js +106 -0
  61. data/web/assets/javascripts/chart.min.js +13 -0
  62. data/web/assets/javascripts/chartjs-plugin-annotation.min.js +7 -0
  63. data/web/assets/javascripts/dashboard-charts.js +182 -0
  64. data/web/assets/javascripts/dashboard.js +37 -280
  65. data/web/assets/javascripts/metrics.js +298 -0
  66. data/web/assets/stylesheets/application-dark.css +147 -0
  67. data/web/assets/stylesheets/application-rtl.css +153 -0
  68. data/web/assets/stylesheets/application.css +181 -198
  69. data/web/assets/stylesheets/bootstrap-rtl.min.css +9 -0
  70. data/web/assets/stylesheets/bootstrap.css +4 -8
  71. data/web/locales/ar.yml +87 -0
  72. data/web/locales/cs.yml +62 -52
  73. data/web/locales/da.yml +60 -53
  74. data/web/locales/de.yml +65 -53
  75. data/web/locales/el.yml +43 -24
  76. data/web/locales/en.yml +86 -62
  77. data/web/locales/es.yml +70 -53
  78. data/web/locales/fa.yml +80 -0
  79. data/web/locales/fr.yml +86 -56
  80. data/web/locales/gd.yml +99 -0
  81. data/web/locales/he.yml +80 -0
  82. data/web/locales/hi.yml +59 -59
  83. data/web/locales/it.yml +53 -53
  84. data/web/locales/ja.yml +78 -56
  85. data/web/locales/ko.yml +52 -52
  86. data/web/locales/lt.yml +83 -0
  87. data/web/locales/nb.yml +61 -61
  88. data/web/locales/nl.yml +52 -52
  89. data/web/locales/pl.yml +45 -45
  90. data/web/locales/pt-br.yml +83 -55
  91. data/web/locales/pt.yml +51 -51
  92. data/web/locales/ru.yml +68 -60
  93. data/web/locales/sv.yml +53 -53
  94. data/web/locales/ta.yml +60 -60
  95. data/web/locales/uk.yml +62 -61
  96. data/web/locales/ur.yml +80 -0
  97. data/web/locales/vi.yml +83 -0
  98. data/web/locales/zh-cn.yml +43 -16
  99. data/web/locales/zh-tw.yml +42 -8
  100. data/web/views/_footer.erb +10 -9
  101. data/web/views/_job_info.erb +26 -5
  102. data/web/views/_metrics_period_select.erb +12 -0
  103. data/web/views/_nav.erb +6 -20
  104. data/web/views/_paging.erb +3 -1
  105. data/web/views/_poll_link.erb +3 -6
  106. data/web/views/_summary.erb +7 -7
  107. data/web/views/busy.erb +87 -28
  108. data/web/views/dashboard.erb +51 -21
  109. data/web/views/dead.erb +4 -4
  110. data/web/views/filtering.erb +7 -0
  111. data/web/views/layout.erb +15 -5
  112. data/web/views/metrics.erb +91 -0
  113. data/web/views/metrics_for_job.erb +59 -0
  114. data/web/views/morgue.erb +25 -22
  115. data/web/views/queue.erb +35 -25
  116. data/web/views/queues.erb +23 -7
  117. data/web/views/retries.erb +28 -23
  118. data/web/views/retry.erb +5 -5
  119. data/web/views/scheduled.erb +19 -17
  120. data/web/views/scheduled_job_info.erb +1 -1
  121. metadata +86 -268
  122. data/.gitignore +0 -12
  123. data/.travis.yml +0 -16
  124. data/3.0-Upgrade.md +0 -70
  125. data/COMM-LICENSE +0 -95
  126. data/Contributing.md +0 -32
  127. data/Ent-Changes.md +0 -39
  128. data/Gemfile +0 -27
  129. data/LICENSE +0 -9
  130. data/Pro-2.0-Upgrade.md +0 -138
  131. data/Pro-Changes.md +0 -454
  132. data/Rakefile +0 -9
  133. data/bin/sidekiqctl +0 -93
  134. data/lib/generators/sidekiq/templates/worker_spec.rb.erb +0 -6
  135. data/lib/generators/sidekiq/templates/worker_test.rb.erb +0 -8
  136. data/lib/generators/sidekiq/worker_generator.rb +0 -49
  137. data/lib/sidekiq/actor.rb +0 -39
  138. data/lib/sidekiq/core_ext.rb +0 -105
  139. data/lib/sidekiq/exception_handler.rb +0 -30
  140. data/lib/sidekiq/extensions/action_mailer.rb +0 -56
  141. data/lib/sidekiq/extensions/active_record.rb +0 -39
  142. data/lib/sidekiq/extensions/class_methods.rb +0 -39
  143. data/lib/sidekiq/extensions/generic_proxy.rb +0 -24
  144. data/lib/sidekiq/logging.rb +0 -104
  145. data/lib/sidekiq/middleware/server/active_record.rb +0 -13
  146. data/lib/sidekiq/middleware/server/logging.rb +0 -40
  147. data/lib/sidekiq/middleware/server/retry_jobs.rb +0 -206
  148. data/lib/sidekiq/util.rb +0 -68
  149. data/lib/sidekiq/web_helpers.rb +0 -249
  150. data/lib/sidekiq/worker.rb +0 -103
  151. data/test/config.yml +0 -9
  152. data/test/env_based_config.yml +0 -11
  153. data/test/fake_env.rb +0 -0
  154. data/test/fixtures/en.yml +0 -2
  155. data/test/helper.rb +0 -49
  156. data/test/test_api.rb +0 -493
  157. data/test/test_cli.rb +0 -335
  158. data/test/test_client.rb +0 -194
  159. data/test/test_exception_handler.rb +0 -55
  160. data/test/test_extensions.rb +0 -126
  161. data/test/test_fetch.rb +0 -104
  162. data/test/test_logging.rb +0 -34
  163. data/test/test_manager.rb +0 -168
  164. data/test/test_middleware.rb +0 -159
  165. data/test/test_processor.rb +0 -237
  166. data/test/test_rails.rb +0 -21
  167. data/test/test_redis_connection.rb +0 -126
  168. data/test/test_retry.rb +0 -325
  169. data/test/test_scheduled.rb +0 -114
  170. data/test/test_scheduling.rb +0 -49
  171. data/test/test_sidekiq.rb +0 -99
  172. data/test/test_testing.rb +0 -142
  173. data/test/test_testing_fake.rb +0 -268
  174. data/test/test_testing_inline.rb +0 -93
  175. data/test/test_util.rb +0 -16
  176. data/test/test_web.rb +0 -608
  177. data/test/test_web_helpers.rb +0 -53
  178. data/web/assets/images/bootstrap/glyphicons-halflings-white.png +0 -0
  179. data/web/assets/images/bootstrap/glyphicons-halflings.png +0 -0
  180. data/web/assets/images/status/active.png +0 -0
  181. data/web/assets/images/status/idle.png +0 -0
  182. data/web/assets/javascripts/locales/README.md +0 -27
  183. data/web/assets/javascripts/locales/jquery.timeago.ar.js +0 -96
  184. data/web/assets/javascripts/locales/jquery.timeago.bg.js +0 -18
  185. data/web/assets/javascripts/locales/jquery.timeago.bs.js +0 -49
  186. data/web/assets/javascripts/locales/jquery.timeago.ca.js +0 -18
  187. data/web/assets/javascripts/locales/jquery.timeago.cs.js +0 -18
  188. data/web/assets/javascripts/locales/jquery.timeago.cy.js +0 -20
  189. data/web/assets/javascripts/locales/jquery.timeago.da.js +0 -18
  190. data/web/assets/javascripts/locales/jquery.timeago.de.js +0 -18
  191. data/web/assets/javascripts/locales/jquery.timeago.el.js +0 -18
  192. data/web/assets/javascripts/locales/jquery.timeago.en-short.js +0 -20
  193. data/web/assets/javascripts/locales/jquery.timeago.en.js +0 -20
  194. data/web/assets/javascripts/locales/jquery.timeago.es.js +0 -18
  195. data/web/assets/javascripts/locales/jquery.timeago.et.js +0 -18
  196. data/web/assets/javascripts/locales/jquery.timeago.fa.js +0 -22
  197. data/web/assets/javascripts/locales/jquery.timeago.fi.js +0 -28
  198. data/web/assets/javascripts/locales/jquery.timeago.fr-short.js +0 -16
  199. data/web/assets/javascripts/locales/jquery.timeago.fr.js +0 -17
  200. data/web/assets/javascripts/locales/jquery.timeago.he.js +0 -18
  201. data/web/assets/javascripts/locales/jquery.timeago.hr.js +0 -49
  202. data/web/assets/javascripts/locales/jquery.timeago.hu.js +0 -18
  203. data/web/assets/javascripts/locales/jquery.timeago.hy.js +0 -18
  204. data/web/assets/javascripts/locales/jquery.timeago.id.js +0 -18
  205. data/web/assets/javascripts/locales/jquery.timeago.it.js +0 -16
  206. data/web/assets/javascripts/locales/jquery.timeago.ja.js +0 -19
  207. data/web/assets/javascripts/locales/jquery.timeago.ko.js +0 -17
  208. data/web/assets/javascripts/locales/jquery.timeago.lt.js +0 -20
  209. data/web/assets/javascripts/locales/jquery.timeago.mk.js +0 -20
  210. data/web/assets/javascripts/locales/jquery.timeago.nl.js +0 -20
  211. data/web/assets/javascripts/locales/jquery.timeago.no.js +0 -18
  212. data/web/assets/javascripts/locales/jquery.timeago.pl.js +0 -31
  213. data/web/assets/javascripts/locales/jquery.timeago.pt-br.js +0 -16
  214. data/web/assets/javascripts/locales/jquery.timeago.pt.js +0 -16
  215. data/web/assets/javascripts/locales/jquery.timeago.ro.js +0 -18
  216. data/web/assets/javascripts/locales/jquery.timeago.rs.js +0 -49
  217. data/web/assets/javascripts/locales/jquery.timeago.ru.js +0 -34
  218. data/web/assets/javascripts/locales/jquery.timeago.sk.js +0 -18
  219. data/web/assets/javascripts/locales/jquery.timeago.sl.js +0 -44
  220. data/web/assets/javascripts/locales/jquery.timeago.sv.js +0 -18
  221. data/web/assets/javascripts/locales/jquery.timeago.th.js +0 -20
  222. data/web/assets/javascripts/locales/jquery.timeago.tr.js +0 -16
  223. data/web/assets/javascripts/locales/jquery.timeago.uk.js +0 -34
  224. data/web/assets/javascripts/locales/jquery.timeago.uz.js +0 -19
  225. data/web/assets/javascripts/locales/jquery.timeago.zh-cn.js +0 -20
  226. data/web/assets/javascripts/locales/jquery.timeago.zh-tw.js +0 -20
  227. data/web/views/_poll_js.erb +0 -5
  228. /data/web/assets/images/{status-sd8051fd480.png → status.png} +0 -0
@@ -0,0 +1,301 @@
1
+ # frozen_string_literal: true
2
+
3
+ require "zlib"
4
+ require "base64"
5
+ require "sidekiq/component"
6
+
7
+ module Sidekiq
8
+ ##
9
+ # Automatically retry jobs that fail in Sidekiq.
10
+ # Sidekiq's retry support assumes a typical development lifecycle:
11
+ #
12
+ # 0. Push some code changes with a bug in it.
13
+ # 1. Bug causes job processing to fail, Sidekiq's middleware captures
14
+ # the job and pushes it onto a retry queue.
15
+ # 2. Sidekiq retries jobs in the retry queue multiple times with
16
+ # an exponential delay, the job continues to fail.
17
+ # 3. After a few days, a developer deploys a fix. The job is
18
+ # reprocessed successfully.
19
+ # 4. Once retries are exhausted, Sidekiq will give up and move the
20
+ # job to the Dead Job Queue (aka morgue) where it must be dealt with
21
+ # manually in the Web UI.
22
+ # 5. After 6 months on the DJQ, Sidekiq will discard the job.
23
+ #
24
+ # A job looks like:
25
+ #
26
+ # { 'class' => 'HardJob', 'args' => [1, 2, 'foo'], 'retry' => true }
27
+ #
28
+ # The 'retry' option also accepts a number (in place of 'true'):
29
+ #
30
+ # { 'class' => 'HardJob', 'args' => [1, 2, 'foo'], 'retry' => 5 }
31
+ #
32
+ # The job will be retried this number of times before giving up. (If simply
33
+ # 'true', Sidekiq retries 25 times)
34
+ #
35
+ # Relevant options for job retries:
36
+ #
37
+ # * 'queue' - the queue for the initial job
38
+ # * 'retry_queue' - if job retries should be pushed to a different (e.g. lower priority) queue
39
+ # * 'retry_count' - number of times we've retried so far.
40
+ # * 'error_message' - the message from the exception
41
+ # * 'error_class' - the exception class
42
+ # * 'failed_at' - the first time it failed
43
+ # * 'retried_at' - the last time it was retried
44
+ # * 'backtrace' - the number of lines of error backtrace to store
45
+ #
46
+ # We don't store the backtrace by default as that can add a lot of overhead
47
+ # to the job and everyone is using an error service, right?
48
+ #
49
+ # The default number of retries is 25 which works out to about 3 weeks
50
+ # You can change the default maximum number of retries in your initializer:
51
+ #
52
+ # Sidekiq.default_configuration[:max_retries] = 7
53
+ #
54
+ # or limit the number of retries for a particular job and send retries to
55
+ # a low priority queue with:
56
+ #
57
+ # class MyJob
58
+ # include Sidekiq::Job
59
+ # sidekiq_options retry: 10, retry_queue: 'low'
60
+ # end
61
+ #
62
+ class JobRetry
63
+ class Handled < ::RuntimeError; end
64
+
65
+ class Skip < Handled; end
66
+
67
+ include Sidekiq::Component
68
+
69
+ DEFAULT_MAX_RETRY_ATTEMPTS = 25
70
+
71
+ def initialize(capsule)
72
+ @config = @capsule = capsule
73
+ @max_retries = Sidekiq.default_configuration[:max_retries] || DEFAULT_MAX_RETRY_ATTEMPTS
74
+ @backtrace_cleaner = Sidekiq.default_configuration[:backtrace_cleaner]
75
+ end
76
+
77
+ # The global retry handler requires only the barest of data.
78
+ # We want to be able to retry as much as possible so we don't
79
+ # require the job to be instantiated.
80
+ def global(jobstr, queue)
81
+ yield
82
+ rescue Handled => ex
83
+ raise ex
84
+ rescue Sidekiq::Shutdown => ey
85
+ # ignore, will be pushed back onto queue during hard_shutdown
86
+ raise ey
87
+ rescue Exception => e
88
+ # ignore, will be pushed back onto queue during hard_shutdown
89
+ raise Sidekiq::Shutdown if exception_caused_by_shutdown?(e)
90
+
91
+ msg = Sidekiq.load_json(jobstr)
92
+ if msg["retry"]
93
+ process_retry(nil, msg, queue, e)
94
+ else
95
+ @capsule.config.death_handlers.each do |handler|
96
+ handler.call(msg, e)
97
+ rescue => handler_ex
98
+ handle_exception(handler_ex, {context: "Error calling death handler", job: msg})
99
+ end
100
+ end
101
+
102
+ raise Handled
103
+ end
104
+
105
+ # The local retry support means that any errors that occur within
106
+ # this block can be associated with the given job instance.
107
+ # This is required to support the `sidekiq_retries_exhausted` block.
108
+ #
109
+ # Note that any exception from the block is wrapped in the Skip
110
+ # exception so the global block does not reprocess the error. The
111
+ # Skip exception is unwrapped within Sidekiq::Processor#process before
112
+ # calling the handle_exception handlers.
113
+ def local(jobinst, jobstr, queue)
114
+ yield
115
+ rescue Handled => ex
116
+ raise ex
117
+ rescue Sidekiq::Shutdown => ey
118
+ # ignore, will be pushed back onto queue during hard_shutdown
119
+ raise ey
120
+ rescue Exception => e
121
+ # ignore, will be pushed back onto queue during hard_shutdown
122
+ raise Sidekiq::Shutdown if exception_caused_by_shutdown?(e)
123
+
124
+ msg = Sidekiq.load_json(jobstr)
125
+ if msg["retry"].nil?
126
+ msg["retry"] = jobinst.class.get_sidekiq_options["retry"]
127
+ end
128
+
129
+ raise e unless msg["retry"]
130
+ process_retry(jobinst, msg, queue, e)
131
+ # We've handled this error associated with this job, don't
132
+ # need to handle it at the global level
133
+ raise Skip
134
+ end
135
+
136
+ private
137
+
138
+ # Note that +jobinst+ can be nil here if an error is raised before we can
139
+ # instantiate the job instance. All access must be guarded and
140
+ # best effort.
141
+ def process_retry(jobinst, msg, queue, exception)
142
+ max_retry_attempts = retry_attempts_from(msg["retry"], @max_retries)
143
+
144
+ msg["queue"] = (msg["retry_queue"] || queue)
145
+
146
+ m = exception_message(exception)
147
+ if m.respond_to?(:scrub!)
148
+ m.force_encoding("utf-8")
149
+ m.scrub!
150
+ end
151
+
152
+ msg["error_message"] = m
153
+ msg["error_class"] = exception.class.name
154
+ count = if msg["retry_count"]
155
+ msg["retried_at"] = Time.now.to_f
156
+ msg["retry_count"] += 1
157
+ else
158
+ msg["failed_at"] = Time.now.to_f
159
+ msg["retry_count"] = 0
160
+ end
161
+
162
+ if msg["backtrace"]
163
+ backtrace = @backtrace_cleaner.call(exception.backtrace)
164
+ lines = if msg["backtrace"] == true
165
+ backtrace
166
+ else
167
+ backtrace[0...msg["backtrace"].to_i]
168
+ end
169
+
170
+ msg["error_backtrace"] = compress_backtrace(lines)
171
+ end
172
+
173
+ return retries_exhausted(jobinst, msg, exception) if count >= max_retry_attempts
174
+
175
+ rf = msg["retry_for"]
176
+ return retries_exhausted(jobinst, msg, exception) if rf && ((msg["failed_at"] + rf) < Time.now.to_f)
177
+
178
+ strategy, delay = delay_for(jobinst, count, exception, msg)
179
+ case strategy
180
+ when :discard
181
+ return # poof!
182
+ when :kill
183
+ return retries_exhausted(jobinst, msg, exception)
184
+ end
185
+
186
+ # Logging here can break retries if the logging device raises ENOSPC #3979
187
+ # logger.debug { "Failure! Retry #{count} in #{delay} seconds" }
188
+ jitter = rand(10) * (count + 1)
189
+ retry_at = Time.now.to_f + delay + jitter
190
+ payload = Sidekiq.dump_json(msg)
191
+ redis do |conn|
192
+ conn.zadd("retry", retry_at.to_s, payload)
193
+ end
194
+ end
195
+
196
+ # returns (strategy, seconds)
197
+ def delay_for(jobinst, count, exception, msg)
198
+ rv = begin
199
+ # sidekiq_retry_in can return two different things:
200
+ # 1. When to retry next, as an integer of seconds
201
+ # 2. A symbol which re-routes the job elsewhere, e.g. :discard, :kill, :default
202
+ block = jobinst&.sidekiq_retry_in_block
203
+
204
+ # the sidekiq_retry_in_block can be defined in a wrapped class (ActiveJob for instance)
205
+ unless msg["wrapped"].nil?
206
+ wrapped = Object.const_get(msg["wrapped"])
207
+ block = wrapped.respond_to?(:sidekiq_retry_in_block) ? wrapped.sidekiq_retry_in_block : nil
208
+ end
209
+ block&.call(count, exception, msg)
210
+ rescue Exception => e
211
+ handle_exception(e, {context: "Failure scheduling retry using the defined `sidekiq_retry_in` in #{jobinst.class.name}, falling back to default"})
212
+ nil
213
+ end
214
+
215
+ rv = rv.to_i if rv.respond_to?(:to_i)
216
+ delay = (count**4) + 15
217
+ if Integer === rv && rv > 0
218
+ delay = rv
219
+ elsif rv == :discard
220
+ return [:discard, nil] # do nothing, job goes poof
221
+ elsif rv == :kill
222
+ return [:kill, nil]
223
+ end
224
+
225
+ [:default, delay]
226
+ end
227
+
228
+ def retries_exhausted(jobinst, msg, exception)
229
+ rv = begin
230
+ block = jobinst&.sidekiq_retries_exhausted_block
231
+
232
+ # the sidekiq_retries_exhausted_block can be defined in a wrapped class (ActiveJob for instance)
233
+ unless msg["wrapped"].nil?
234
+ wrapped = Object.const_get(msg["wrapped"])
235
+ block = wrapped.respond_to?(:sidekiq_retries_exhausted_block) ? wrapped.sidekiq_retries_exhausted_block : nil
236
+ end
237
+ block&.call(msg, exception)
238
+ rescue => e
239
+ handle_exception(e, {context: "Error calling retries_exhausted", job: msg})
240
+ end
241
+
242
+ return if rv == :discard # poof!
243
+ send_to_morgue(msg) unless msg["dead"] == false
244
+
245
+ @capsule.config.death_handlers.each do |handler|
246
+ handler.call(msg, exception)
247
+ rescue => e
248
+ handle_exception(e, {context: "Error calling death handler", job: msg})
249
+ end
250
+ end
251
+
252
+ def send_to_morgue(msg)
253
+ logger.info { "Adding dead #{msg["class"]} job #{msg["jid"]}" }
254
+ payload = Sidekiq.dump_json(msg)
255
+ now = Time.now.to_f
256
+
257
+ redis do |conn|
258
+ conn.multi do |xa|
259
+ xa.zadd("dead", now.to_s, payload)
260
+ xa.zremrangebyscore("dead", "-inf", now - @capsule.config[:dead_timeout_in_seconds])
261
+ xa.zremrangebyrank("dead", 0, - @capsule.config[:dead_max_jobs])
262
+ end
263
+ end
264
+ end
265
+
266
+ def retry_attempts_from(msg_retry, default)
267
+ if msg_retry.is_a?(Integer)
268
+ msg_retry
269
+ else
270
+ default
271
+ end
272
+ end
273
+
274
+ def exception_caused_by_shutdown?(e, checked_causes = [])
275
+ return false unless e.cause
276
+
277
+ # Handle circular causes
278
+ checked_causes << e.object_id
279
+ return false if checked_causes.include?(e.cause.object_id)
280
+
281
+ e.cause.instance_of?(Sidekiq::Shutdown) ||
282
+ exception_caused_by_shutdown?(e.cause, checked_causes)
283
+ end
284
+
285
+ # Extract message from exception.
286
+ # Set a default if the message raises an error
287
+ def exception_message(exception)
288
+ # App code can stuff all sorts of crazy binary data into the error message
289
+ # that won't convert to JSON.
290
+ exception.message.to_s[0, 10_000]
291
+ rescue
292
+ +"!!! ERROR MESSAGE THREW AN ERROR !!!"
293
+ end
294
+
295
+ def compress_backtrace(backtrace)
296
+ serialized = Sidekiq.dump_json(backtrace)
297
+ compressed = Zlib::Deflate.deflate(serialized)
298
+ Base64.encode64(compressed)
299
+ end
300
+ end
301
+ end
@@ -0,0 +1,107 @@
1
+ require "securerandom"
2
+ require "time"
3
+
4
+ module Sidekiq
5
+ module JobUtil
6
+ # These functions encapsulate various job utilities.
7
+
8
+ TRANSIENT_ATTRIBUTES = %w[]
9
+
10
+ def validate(item)
11
+ raise(ArgumentError, "Job must be a Hash with 'class' and 'args' keys: `#{item}`") unless item.is_a?(Hash) && item.key?("class") && item.key?("args")
12
+ raise(ArgumentError, "Job args must be an Array: `#{item}`") unless item["args"].is_a?(Array) || item["args"].is_a?(Enumerator::Lazy)
13
+ raise(ArgumentError, "Job class must be either a Class or String representation of the class name: `#{item}`") unless item["class"].is_a?(Class) || item["class"].is_a?(String)
14
+ raise(ArgumentError, "Job 'at' must be a Numeric timestamp: `#{item}`") if item.key?("at") && !item["at"].is_a?(Numeric)
15
+ raise(ArgumentError, "Job tags must be an Array: `#{item}`") if item["tags"] && !item["tags"].is_a?(Array)
16
+ raise(ArgumentError, "retry_for must be a relative amount of time, e.g. 48.hours `#{item}`") if item["retry_for"] && item["retry_for"] > 1_000_000_000
17
+ end
18
+
19
+ def verify_json(item)
20
+ job_class = item["wrapped"] || item["class"]
21
+ args = item["args"]
22
+ mode = Sidekiq::Config::DEFAULTS[:on_complex_arguments]
23
+
24
+ if mode == :raise || mode == :warn
25
+ if (unsafe_item = json_unsafe?(args))
26
+ msg = <<~EOM
27
+ Job arguments to #{job_class} must be native JSON types, but #{unsafe_item.inspect} is a #{unsafe_item.class}.
28
+ See https://github.com/sidekiq/sidekiq/wiki/Best-Practices
29
+ To disable this error, add `Sidekiq.strict_args!(false)` to your initializer.
30
+ EOM
31
+
32
+ if mode == :raise
33
+ raise(ArgumentError, msg)
34
+ else
35
+ warn(msg)
36
+ end
37
+ end
38
+ end
39
+ end
40
+
41
+ def normalize_item(item)
42
+ validate(item)
43
+
44
+ # merge in the default sidekiq_options for the item's class and/or wrapped element
45
+ # this allows ActiveJobs to control sidekiq_options too.
46
+ defaults = normalized_hash(item["class"])
47
+ defaults = defaults.merge(item["wrapped"].get_sidekiq_options) if item["wrapped"].respond_to?(:get_sidekiq_options)
48
+ item = defaults.merge(item)
49
+
50
+ raise(ArgumentError, "Job must include a valid queue name") if item["queue"].nil? || item["queue"] == ""
51
+
52
+ # remove job attributes which aren't necessary to persist into Redis
53
+ TRANSIENT_ATTRIBUTES.each { |key| item.delete(key) }
54
+
55
+ item["jid"] ||= SecureRandom.hex(12)
56
+ item["class"] = item["class"].to_s
57
+ item["queue"] = item["queue"].to_s
58
+ item["retry_for"] = item["retry_for"].to_i if item["retry_for"]
59
+ item["created_at"] ||= Time.now.to_f
60
+ item
61
+ end
62
+
63
+ def normalized_hash(item_class)
64
+ if item_class.is_a?(Class)
65
+ raise(ArgumentError, "Message must include a Sidekiq::Job class, not class name: #{item_class.ancestors.inspect}") unless item_class.respond_to?(:get_sidekiq_options)
66
+ item_class.get_sidekiq_options
67
+ else
68
+ Sidekiq.default_job_options
69
+ end
70
+ end
71
+
72
+ private
73
+
74
+ RECURSIVE_JSON_UNSAFE = {
75
+ Integer => ->(val) {},
76
+ Float => ->(val) {},
77
+ TrueClass => ->(val) {},
78
+ FalseClass => ->(val) {},
79
+ NilClass => ->(val) {},
80
+ String => ->(val) {},
81
+ Array => ->(val) {
82
+ val.each do |e|
83
+ unsafe_item = RECURSIVE_JSON_UNSAFE[e.class].call(e)
84
+ return unsafe_item unless unsafe_item.nil?
85
+ end
86
+ nil
87
+ },
88
+ Hash => ->(val) {
89
+ val.each do |k, v|
90
+ return k unless String === k
91
+
92
+ unsafe_item = RECURSIVE_JSON_UNSAFE[v.class].call(v)
93
+ return unsafe_item unless unsafe_item.nil?
94
+ end
95
+ nil
96
+ }
97
+ }
98
+
99
+ RECURSIVE_JSON_UNSAFE.default = ->(val) { val }
100
+ RECURSIVE_JSON_UNSAFE.compare_by_identity
101
+ private_constant :RECURSIVE_JSON_UNSAFE
102
+
103
+ def json_unsafe?(item)
104
+ RECURSIVE_JSON_UNSAFE[item.class].call(item)
105
+ end
106
+ end
107
+ end