sidekiq 3.1.4 → 3.2.0

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.

checksums.yaml CHANGED
@@ -1,7 +1,7 @@
1
1
  ---
2
2
  SHA1:
3
- metadata.gz: 842995b5a36bd3651ba6ed63d3f0c2c5f99a2c1b
4
- data.tar.gz: 4f362b15d3748d3df41dad8ba1364d967a5e37cd
3
+ metadata.gz: fdef31a4d431682169e826c2f7e1817f4392ca5b
4
+ data.tar.gz: 93a6c396de18207948950ee7d8e1a52a7c81b18f
5
5
  SHA512:
6
- metadata.gz: dc001192b00a22ad94893693543234b2542a6f91af74abc0a9963bf19bfd96c33babe084efcad2ce454f26bd78bdbbb356364fc86ba7d63bc5e0fb441a514a09
7
- data.tar.gz: 7ea2ab5908163f70dc386b9ab89ce84321f3e93bf5bc4d20f0152ecc3a1ae67dba100abf088bcea4c9c163cf904623208877c16b993127d3e927038a0dafbfde
6
+ metadata.gz: 681bee80c219bf1bdac44797c4353ae1bead8495cec2dc0f4055b150e751007e1805eda4aabcd83fe3b273df4d7f403d0b244b4f37b374895c9d42ed24cb6c5e
7
+ data.tar.gz: c85781ce1ce3549d57640b50fe54c5ca0e5433122dfcb1793a8a31db5c06baa6dd10b31bebf2d355156c574b1481937cf68f826c30af6e13eba741d4161dd489
data/Changes.md CHANGED
@@ -1,3 +1,13 @@
1
+ 3.2.0
2
+ -----------
3
+
4
+ - **Fix issue which caused duplicate job execution in Rails 3.x**
5
+ This issue is caused by [improper exception handling in ActiveRecord](https://github.com/rails/rails/blob/3-2-stable/activerecord/lib/active_record/connection_adapters/abstract_adapter.rb#L281) which changes Sidekiq's Shutdown exception into a database
6
+ error, making Sidekiq think the job needs to be retried. **The fix requires Ruby 2.1**. [#1805]
7
+ - Update how Sidekiq eager loads Rails application code [#1791, jonleighton]
8
+ - Change logging timestamp to show milliseconds.
9
+ - Reverse sorting of Dead tab so newer jobs are listed first [#1802]
10
+
1
11
  3.1.4
2
12
  -----------
3
13
 
@@ -3,6 +3,12 @@ Sidekiq Pro Changelog
3
3
 
4
4
  Please see [http://sidekiq.org/pro](http://sidekiq.org/pro) for more details and how to buy.
5
5
 
6
+ 1.7.4
7
+ -----------
8
+
9
+ - Awesome ANSI art startup banner!
10
+
11
+
6
12
  1.7.3
7
13
  -----------
8
14
 
@@ -221,10 +221,13 @@ module Sidekiq
221
221
  raise ArgumentError, "#{options[:require]} does not exist" unless File.exist?(options[:require])
222
222
 
223
223
  if File.directory?(options[:require])
224
- require 'rails'
224
+ # Painful contortions, see 1791 for discussion
225
+ require File.expand_path("#{options[:require]}/config/application.rb")
226
+ ::Rails::Application.initializer "sidekiq.eager_load" do
227
+ ::Rails.application.config.eager_load = true
228
+ end
225
229
  require 'sidekiq/rails'
226
230
  require File.expand_path("#{options[:require]}/config/environment.rb")
227
- ::Rails.application.eager_load!
228
231
  options[:tag] ||= default_tag
229
232
  else
230
233
  require options[:require]
@@ -36,4 +36,4 @@ module Sidekiq
36
36
  end
37
37
  end
38
38
 
39
- Module.__send__(:include, Sidekiq::Extensions::Klass)
39
+ Module.__send__(:include, Sidekiq::Extensions::Klass) unless defined?(::Rails)
@@ -7,7 +7,7 @@ module Sidekiq
7
7
  class Pretty < Logger::Formatter
8
8
  # Provide a call() method that returns the formatted message.
9
9
  def call(severity, time, program_name, message)
10
- "#{time.utc.iso8601} #{::Process.pid} TID-#{Thread.current.object_id.to_s(36)}#{context} #{severity}: #{message}\n"
10
+ "#{time.utc.iso8601(3)} #{::Process.pid} TID-#{Thread.current.object_id.to_s(36)}#{context} #{severity}: #{message}\n"
11
11
  end
12
12
 
13
13
  def context
@@ -64,6 +64,9 @@ module Sidekiq
64
64
  # ignore, will be pushed back onto queue during hard_shutdown
65
65
  raise
66
66
  rescue Exception => e
67
+ # ignore, will be pushed back onto queue during hard_shutdown
68
+ raise Sidekiq::Shutdown if exception_caused_by_shutdown?(e)
69
+
67
70
  raise e unless msg['retry']
68
71
  max_retry_attempts = retry_attempts_from(msg['retry'], @max_retries)
69
72
 
@@ -163,6 +166,14 @@ module Sidekiq
163
166
  end
164
167
  end
165
168
 
169
+ def exception_caused_by_shutdown?(e)
170
+ # In Ruby 2.1.0 only, check if exception is a result of shutdown.
171
+ return false unless defined?(e.cause)
172
+
173
+ e.cause.instance_of?(Sidekiq::Shutdown) ||
174
+ exception_caused_by_shutdown?(e.cause)
175
+ end
176
+
166
177
  end
167
178
  end
168
179
  end
@@ -1,5 +1,7 @@
1
1
  module Sidekiq
2
2
  def self.hook_rails!
3
+ return if defined?(@delay_removed)
4
+
3
5
  ActiveSupport.on_load(:active_record) do
4
6
  include Sidekiq::Extensions::ActiveRecord
5
7
  end
@@ -7,12 +9,16 @@ module Sidekiq
7
9
  ActiveSupport.on_load(:action_mailer) do
8
10
  extend Sidekiq::Extensions::ActionMailer
9
11
  end
12
+
13
+ Module.__send__(:include, Sidekiq::Extensions::Klass)
10
14
  end
11
15
 
12
16
  # Removes the generic aliases which MAY clash with names of already
13
17
  # created methods by other applications. The methods `sidekiq_delay`,
14
18
  # `sidekiq_delay_for` and `sidekiq_delay_until` can be used instead.
15
19
  def self.remove_delay!
20
+ @delay_removed = true
21
+
16
22
  [Extensions::ActiveRecord,
17
23
  Extensions::ActionMailer,
18
24
  Extensions::Klass].each do |mod|
@@ -55,15 +55,19 @@ module Sidekiq
55
55
 
56
56
  def log_info(options)
57
57
  # Don't log Redis AUTH password
58
+ redacted = "REDACTED"
58
59
  scrubbed_options = options.dup
59
60
  if scrubbed_options[:url] && (uri = URI.parse(scrubbed_options[:url])) && uri.password
60
- uri.password = "REDACTED"
61
+ uri.password = redacted
61
62
  scrubbed_options[:url] = uri.to_s
62
63
  end
64
+ if scrubbed_options[:password]
65
+ scrubbed_options[:password] = redacted
66
+ end
63
67
  if Sidekiq.server?
64
68
  Sidekiq.logger.info("Booting Sidekiq #{Sidekiq::VERSION} with redis options #{scrubbed_options}")
65
69
  else
66
- Sidekiq.logger.info("#{Sidekiq::NAME} client with redis options #{scrubbed_options}")
70
+ Sidekiq.logger.debug("#{Sidekiq::NAME} client with redis options #{scrubbed_options}")
67
71
  end
68
72
  end
69
73
 
@@ -1,3 +1,3 @@
1
1
  module Sidekiq
2
- VERSION = "3.1.4"
2
+ VERSION = "3.2.0"
3
3
  end
@@ -44,9 +44,9 @@ module Sidekiq
44
44
 
45
45
  post "/busy" do
46
46
  if params['hostname']
47
- pro = Sidekiq::Process.new('hostname' => params["hostname"], 'pid' => params['pid'])
48
- pro.quiet! if params[:quiet]
49
- pro.stop! if params[:stop]
47
+ p = Sidekiq::Process.new('hostname' => params["hostname"], 'pid' => params['pid'])
48
+ p.quiet! if params[:quiet]
49
+ p.stop! if params[:stop]
50
50
  else
51
51
  Sidekiq::ProcessSet.new.each do |pro|
52
52
  pro.quiet! if params[:quiet]
@@ -83,7 +83,7 @@ module Sidekiq
83
83
 
84
84
  get '/morgue' do
85
85
  @count = (params[:count] || 25).to_i
86
- (@current_page, @total_size, @dead) = page("dead", params[:page], @count)
86
+ (@current_page, @total_size, @dead) = page("dead", params[:page], @count, :reverse => true)
87
87
  @dead = @dead.map {|msg, score| Sidekiq::SortedEntry.new(nil, score, msg) }
88
88
  erb :morgue
89
89
  end
@@ -100,6 +100,21 @@ class TestExtensions < Sidekiq::Test
100
100
  def queue_size(name='default')
101
101
  Sidekiq::Queue.new(name).size
102
102
  end
103
+
104
+ it 'allows removing of the #delay methods' do
105
+ Sidekiq.remove_delay!
106
+ assert_equal 0, queue_size
107
+ assert_raises NoMethodError do
108
+ SomeModule.delay.doit(Date.today)
109
+ end
110
+
111
+ Sidekiq.instance_eval { remove_instance_variable :@delay_removed }
112
+ # Reload modified modules
113
+ load 'sidekiq/extensions/action_mailer.rb'
114
+ load 'sidekiq/extensions/active_record.rb'
115
+ load 'sidekiq/extensions/generic_proxy.rb'
116
+ load 'sidekiq/extensions/class_methods.rb'
117
+ end
103
118
  end
104
119
 
105
120
  end
@@ -117,6 +117,64 @@ class TestRetry < Sidekiq::Test
117
117
  @redis.verify
118
118
  end
119
119
 
120
+ it 'shuts down without retrying work-in-progress, which will resume' do
121
+ @redis.expect :zadd, 1, ['retry', String, String]
122
+ msg = { 'class' => 'Bob', 'args' => [1,2,'foo'], 'retry' => true }
123
+ handler = Sidekiq::Middleware::Server::RetryJobs.new
124
+ assert_raises Sidekiq::Shutdown do
125
+ handler.call(worker, msg, 'default') do
126
+ raise Sidekiq::Shutdown
127
+ end
128
+ end
129
+ assert_raises(MockExpectationError, "zadd should not be called") do
130
+ @redis.verify
131
+ end
132
+ end
133
+
134
+ it 'shuts down cleanly when shutdown causes exception' do
135
+ skip('Not supported in Ruby < 2.1.0') if RUBY_VERSION < '2.1.0'
136
+
137
+ @redis.expect :zadd, 1, ['retry', String, String]
138
+ msg = { 'class' => 'Bob', 'args' => [1,2,'foo'], 'retry' => true }
139
+ handler = Sidekiq::Middleware::Server::RetryJobs.new
140
+ assert_raises Sidekiq::Shutdown do
141
+ handler.call(worker, msg, 'default') do
142
+ begin
143
+ raise Sidekiq::Shutdown
144
+ rescue Interrupt
145
+ raise "kerblammo!"
146
+ end
147
+ end
148
+ end
149
+ assert_raises(MockExpectationError, "zadd should not be called") do
150
+ @redis.verify
151
+ end
152
+ end
153
+
154
+ it 'shuts down cleanly when shutdown causes chained exceptions' do
155
+ skip('Not supported in Ruby < 2.1.0') if RUBY_VERSION < '2.1.0'
156
+
157
+ @redis.expect :zadd, 1, ['retry', String, String]
158
+ msg = { 'class' => 'Bob', 'args' => [1,2,'foo'], 'retry' => true }
159
+ handler = Sidekiq::Middleware::Server::RetryJobs.new
160
+ assert_raises Sidekiq::Shutdown do
161
+ handler.call(worker, msg, 'default') do
162
+ begin
163
+ raise Sidekiq::Shutdown
164
+ rescue Interrupt
165
+ begin
166
+ raise "kerblammo!"
167
+ rescue
168
+ raise "kablooie!"
169
+ end
170
+ end
171
+ end
172
+ end
173
+ assert_raises(MockExpectationError, "zadd should not be called") do
174
+ @redis.verify
175
+ end
176
+ end
177
+
120
178
  it 'allows a retry queue' do
121
179
  @redis.expect :zadd, 1, ['retry', String, String]
122
180
  msg = { 'class' => 'Bob', 'args' => [1,2,'foo'], 'retry' => true, 'retry_queue' => 'retry' }
@@ -66,7 +66,7 @@ var realtimeGraph = function(updatePath) {
66
66
 
67
67
  var label = document.createElement('div');
68
68
  label.className = 'tag';
69
- label.innerHTML = d.name + ": " + Math.floor(d.formattedYValue);
69
+ label.innerHTML = d.name + ": " + Math.floor(d.formattedYValue).numberWithDelimiter();
70
70
 
71
71
  line.appendChild(swatch);
72
72
  line.appendChild(label);
@@ -173,7 +173,7 @@ var historyGraph = function() {
173
173
 
174
174
  var label = document.createElement('div');
175
175
  label.className = 'tag';
176
- label.innerHTML = d.name + ": " + Math.floor(d.formattedYValue);
176
+ label.innerHTML = d.name + ": " + Math.floor(d.formattedYValue).numberWithDelimiter();
177
177
 
178
178
  line.appendChild(swatch);
179
179
  line.appendChild(label);
@@ -11,8 +11,7 @@ body {
11
11
  text-rendering: optimizeLegibility;
12
12
  -webkit-font-smoothing: antialiased;
13
13
  font-family: "Helvetica Neue", Helvetica, Arial, sans-serif;
14
- background-color: #f3f3f3;
15
- background-image: url();
14
+ background: #f3f3f3 url();
16
15
  }
17
16
 
18
17
  a {
@@ -67,7 +66,7 @@ footer .edits {
67
66
  }
68
67
 
69
68
  body {
70
- padding: 0px 20px;
69
+ padding: 0 20px;
71
70
  }
72
71
 
73
72
  h3 {
@@ -92,21 +91,18 @@ header.row .pagination {
92
91
  .summary_bar .summary {
93
92
  margin-top: 12px;
94
93
  background-color: #fff;
95
- -webkit-box-shadow: 0px 0px 5px rgba(50, 50, 50, 0.25);
96
- -moz-box-shadow: 0px 0px 5px rgba(50, 50, 50, 0.25);
97
- box-shadow: 0px 0px 5px rgba(50, 50, 50, 0.25);
94
+ -webkit-box-shadow: 0 0 5px rgba(50, 50, 50, 0.25);
95
+ -moz-box-shadow: 0 0 5px rgba(50, 50, 50, 0.25);
96
+ box-shadow: 0 0 5px rgba(50, 50, 50, 0.25);
98
97
  -webkit-border-radius: 4px;
99
- -webkit-border-radius: 4px;
100
- -moz-border-radius: 4px;
101
98
  -moz-border-radius: 4px;
102
99
  border-radius: 4px;
103
- border-radius: 4px;
104
100
  padding: 8px;
105
101
  margin-bottom: 10px;
106
- border-width: 0px;
102
+ border-width: 0;
107
103
  }
108
104
  .poll-wrapper {
109
- margin: 9px 0px;
105
+ margin: 9px 0;
110
106
  }
111
107
  .nav #live-poll {
112
108
  height: 25px;
@@ -147,7 +143,7 @@ header.row .pagination {
147
143
  }
148
144
  @media (max-width: 979px) and (min-width: 768px) {
149
145
  .summary_bar ul li.col-sm-2 {
150
- margin: 0px 10px;
146
+ margin: 0 10px;
151
147
  width: 96px !important;
152
148
  }
153
149
  }
@@ -173,7 +169,7 @@ header.row .pagination {
173
169
  font-size: 1em;
174
170
  font-weight: bold;
175
171
  float: right;
176
- padding: 0px 0px 2px 0px;
172
+ padding: 0 0 2px 0;
177
173
  width: 100%;
178
174
  }
179
175
 
@@ -194,13 +190,13 @@ td form {
194
190
  }
195
191
 
196
192
  .table tr > td.table-checkbox, .table tr > th.table-checkbox {
197
- padding: 0px;
193
+ padding: 0;
198
194
  }
199
195
 
200
196
  table .table-checkbox label {
201
197
  height: 100%;
202
198
  width: 100%;
203
- padding: 0px 16px;
199
+ padding: 0 16px;
204
200
  margin-bottom: 0;
205
201
  line-height: 32px;
206
202
  }
@@ -242,7 +238,7 @@ table .table-checkbox label {
242
238
 
243
239
  img.smallogo {
244
240
  width: 30px;
245
- margin: 0px 0px 6px 0px;
241
+ margin: 0 0 6px 0;
246
242
  }
247
243
 
248
244
  .navbar-fixed-bottom li {
@@ -272,9 +268,9 @@ img.smallogo {
272
268
  -ms-border-radius: 3px;
273
269
  -o-border-radius: 3px;
274
270
  border-radius: 3px;
275
- -webkit-box-shadow: 0px 0px 2px rgba(0, 0, 0, 0.4);
276
- -moz-box-shadow: 0px 0px 2px rgba(0, 0, 0, 0.4);
277
- box-shadow: 0px 0px 2px rgba(0, 0, 0, 0.4);
271
+ -webkit-box-shadow: 0 0 2px rgba(0, 0, 0, 0.4);
272
+ -moz-box-shadow: 0 0 2px rgba(0, 0, 0, 0.4);
273
+ box-shadow: 0 0 2px rgba(0, 0, 0, 0.4);
278
274
  background-image: -webkit-gradient(linear, 50% 0%, 50% 100%, color-stop(0%, #fafafa), color-stop(100%, #ededed));
279
275
  background-image: -webkit-linear-gradient(#fafafa, #ededed);
280
276
  background-image: -moz-linear-gradient(#fafafa, #ededed);
@@ -353,7 +349,7 @@ img.smallogo {
353
349
  margin: 5px 10px 5px 5px;
354
350
  }
355
351
  .stat p{
356
- font-size: normal;
352
+ font-size: 1em;
357
353
  margin: 5px 5px 5px 10px;
358
354
  }
359
355
  }
@@ -384,7 +380,7 @@ div.interval-slider input {
384
380
 
385
381
  #realtime-legend,
386
382
  #history-legend {
387
- width: 490px;
383
+ width: 580px;
388
384
  text-align: left;
389
385
  margin-top: 5px;
390
386
  float: right;
@@ -633,7 +629,7 @@ div.interval-slider input {
633
629
  }
634
630
 
635
631
  .poll-wrapper {
636
- margin: 0px 9px 9px;
632
+ margin: 0 9px 9px;
637
633
  }
638
634
 
639
635
  .navbar.navbar-fixed-bottom ul {
@@ -0,0 +1,68 @@
1
+ # elements like %{queue} are variables and should not be translated
2
+ zh-cn: # <---- change this to your locale code
3
+ Dashboard: 信息板
4
+ Status: 状态
5
+ Time: 时间
6
+ Namespace: 命名空间
7
+ Realtime: 实时
8
+ History: 历史记录
9
+ Busy: 执行中
10
+ Processed: 已处理
11
+ Failed: 已失败
12
+ Scheduled: 已计划
13
+ Retries: 重试
14
+ Enqueued: 已进入队列
15
+ Worker: 工人
16
+ LivePoll: 实时轮询
17
+ StopPolling: 停止轮询
18
+ Queue: 队列
19
+ Class: 类别
20
+ Job: 作业
21
+ Arguments: 参数
22
+ Extras: 额外的
23
+ Started: 已开始
24
+ ShowAll: 显示全部
25
+ CurrentMessagesInQueue: 目前在<span class='title'>%{queue}</span>的作业
26
+ Delete: 删除
27
+ AddToQueue: 添加至队列
28
+ AreYouSureDeleteJob: 你确定要删除这个作业么?
29
+ AreYouSureDeleteQueue: 你确定要删除%{queue}这个队列?
30
+ Queues: 队列
31
+ Size: 容量
32
+ Actions: 动作
33
+ NextRetry: 下次重试
34
+ RetryCount: 重试次數
35
+ RetryNow: 现在重试
36
+ LastRetry: 最后一次重试
37
+ OriginallyFailed: 原本已失败
38
+ AreYouSure: 你确定?
39
+ DeleteAll: 删除全部
40
+ RetryAll: 重试全部
41
+ NoRetriesFound: 沒有发现可重试
42
+ Error: 错误
43
+ ErrorClass: 错误类别
44
+ ErrorMessage: 错误消息
45
+ ErrorBacktrace: 错误的回调追踪
46
+ GoBack: ← 返回
47
+ NoScheduledFound: 沒有发现计划作业
48
+ When: 当
49
+ ScheduledJobs: 计划作业
50
+ idle: 闲置
51
+ active: 活动中
52
+ Version: 版本
53
+ Connections: 连接
54
+ MemoryUsage: 内存占用
55
+ PeakMemoryUsage: 内存占用峰值
56
+ Uptime: 上线时间 (天数)
57
+ OneWeek: 一周
58
+ OneMonth: 一个月
59
+ ThreeMonths: 三个月
60
+ SixMonths: 六个月
61
+ Failures: 失败
62
+ DeadJobs: 已停滞作业
63
+ NoDeadJobsFound: 沒有发现任何已停滞的作业
64
+ Dead: 已停滞
65
+ Processes: 处理中
66
+ Thread: 线程
67
+ Threads: 线程
68
+ Jobs: 作业
metadata CHANGED
@@ -1,14 +1,14 @@
1
1
  --- !ruby/object:Gem::Specification
2
2
  name: sidekiq
3
3
  version: !ruby/object:Gem::Version
4
- version: 3.1.4
4
+ version: 3.2.0
5
5
  platform: ruby
6
6
  authors:
7
7
  - Mike Perham
8
8
  autorequire:
9
9
  bindir: bin
10
10
  cert_chain: []
11
- date: 2014-06-13 00:00:00.000000000 Z
11
+ date: 2014-07-03 00:00:00.000000000 Z
12
12
  dependencies:
13
13
  - !ruby/object:Gem::Dependency
14
14
  name: redis
@@ -300,6 +300,7 @@ files:
300
300
  - web/locales/pt.yml
301
301
  - web/locales/ru.yml
302
302
  - web/locales/sv.yml
303
+ - web/locales/zh-cn.yml
303
304
  - web/locales/zh-tw.yml
304
305
  - web/views/_job_info.erb
305
306
  - web/views/_nav.erb