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 +4 -4
- data/Changes.md +10 -0
- data/Pro-Changes.md +6 -0
- data/lib/sidekiq/cli.rb +5 -2
- data/lib/sidekiq/extensions/class_methods.rb +1 -1
- data/lib/sidekiq/logging.rb +1 -1
- data/lib/sidekiq/middleware/server/retry_jobs.rb +11 -0
- data/lib/sidekiq/rails.rb +6 -0
- data/lib/sidekiq/redis_connection.rb +6 -2
- data/lib/sidekiq/version.rb +1 -1
- data/lib/sidekiq/web.rb +4 -4
- data/test/test_extensions.rb +15 -0
- data/test/test_retry.rb +58 -0
- data/web/assets/javascripts/dashboard.js +2 -2
- data/web/assets/stylesheets/application.css +18 -22
- data/web/locales/zh-cn.yml +68 -0
- metadata +3 -2
checksums.yaml
CHANGED
@@ -1,7 +1,7 @@
|
|
1
1
|
---
|
2
2
|
SHA1:
|
3
|
-
metadata.gz:
|
4
|
-
data.tar.gz:
|
3
|
+
metadata.gz: fdef31a4d431682169e826c2f7e1817f4392ca5b
|
4
|
+
data.tar.gz: 93a6c396de18207948950ee7d8e1a52a7c81b18f
|
5
5
|
SHA512:
|
6
|
-
metadata.gz:
|
7
|
-
data.tar.gz:
|
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
|
|
data/Pro-Changes.md
CHANGED
data/lib/sidekiq/cli.rb
CHANGED
@@ -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
|
-
|
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]
|
data/lib/sidekiq/logging.rb
CHANGED
@@ -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
|
data/lib/sidekiq/rails.rb
CHANGED
@@ -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 =
|
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.
|
70
|
+
Sidekiq.logger.debug("#{Sidekiq::NAME} client with redis options #{scrubbed_options}")
|
67
71
|
end
|
68
72
|
end
|
69
73
|
|
data/lib/sidekiq/version.rb
CHANGED
data/lib/sidekiq/web.rb
CHANGED
@@ -44,9 +44,9 @@ module Sidekiq
|
|
44
44
|
|
45
45
|
post "/busy" do
|
46
46
|
if params['hostname']
|
47
|
-
|
48
|
-
|
49
|
-
|
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
|
data/test/test_extensions.rb
CHANGED
@@ -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
|
data/test/test_retry.rb
CHANGED
@@ -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:
|
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:
|
96
|
-
-moz-box-shadow:
|
97
|
-
box-shadow:
|
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:
|
102
|
+
border-width: 0;
|
107
103
|
}
|
108
104
|
.poll-wrapper {
|
109
|
-
margin: 9px
|
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:
|
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:
|
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:
|
193
|
+
padding: 0;
|
198
194
|
}
|
199
195
|
|
200
196
|
table .table-checkbox label {
|
201
197
|
height: 100%;
|
202
198
|
width: 100%;
|
203
|
-
padding:
|
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:
|
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:
|
276
|
-
-moz-box-shadow:
|
277
|
-
box-shadow:
|
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:
|
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:
|
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:
|
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.
|
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-
|
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
|