sidekiq 2.4.0 → 2.5.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.
- data/.gitignore +1 -0
- data/Changes.md +11 -0
- data/Gemfile +6 -0
- data/lib/sidekiq.rb +1 -0
- data/lib/sidekiq/api.rb +184 -0
- data/lib/sidekiq/cli.rb +3 -2
- data/lib/sidekiq/client.rb +16 -10
- data/lib/sidekiq/core_ext.rb +30 -0
- data/lib/sidekiq/exception_handler.rb +9 -0
- data/lib/sidekiq/processor.rb +1 -1
- data/lib/sidekiq/testing/inline.rb +3 -3
- data/lib/sidekiq/util.rb +1 -11
- data/lib/sidekiq/version.rb +1 -1
- data/lib/sidekiq/web.rb +49 -1
- data/lib/sidekiq/worker.rb +3 -9
- data/sidekiq.gemspec +3 -0
- data/test/config.yml +1 -1
- data/test/test_api.rb +75 -0
- data/test/test_cli.rb +6 -1
- data/test/test_exception_handler.rb +16 -0
- data/test/test_testing_inline.rb +3 -3
- data/test/test_web.rb +27 -6
- data/web/assets/images/bootstrap/glyphicons-halflings-white.png +0 -0
- data/web/assets/images/bootstrap/glyphicons-halflings.png +0 -0
- data/web/assets/images/logo.png +0 -0
- data/web/assets/images/status-sd8051fd480.png +0 -0
- data/web/assets/images/status/active.png +0 -0
- data/web/assets/images/status/idle.png +0 -0
- data/web/assets/javascripts/application.js +44 -16
- data/web/assets/javascripts/vendor/bootstrap.js +0 -0
- data/web/assets/javascripts/vendor/jquery.js +0 -0
- data/web/assets/javascripts/vendor/jquery.timeago.js +0 -0
- data/web/assets/stylesheets/application.css +1 -3
- data/web/assets/stylesheets/layout.css +0 -0
- data/web/assets/stylesheets/partials/_base.scss +116 -0
- data/web/assets/stylesheets/partials/_colors.scss +46 -0
- data/web/assets/stylesheets/partials/_fonts.scss +3 -0
- data/web/assets/stylesheets/partials/_h5bp.scss +293 -0
- data/web/assets/stylesheets/partials/_layout.scss +197 -0
- data/web/assets/stylesheets/partials/_navbar.scss +100 -0
- data/web/assets/stylesheets/partials/_normalize.scss +504 -0
- data/web/assets/stylesheets/partials/_prettify.css +30 -0
- data/web/assets/stylesheets/partials/_variables.scss +28 -0
- data/web/assets/stylesheets/public.scss +74 -0
- data/web/assets/stylesheets/style.scss +69 -0
- data/web/assets/stylesheets/vendor/bootstrap-responsive.css +0 -0
- data/web/assets/stylesheets/vendor/bootstrap.css +0 -0
- data/web/assets/stylesheets/vendors.scss +2 -0
- data/web/views/_nav.slim +13 -0
- data/web/views/_paging.slim +0 -0
- data/web/views/_status.slim +3 -0
- data/web/views/_summary.slim +19 -8
- data/web/views/_workers.slim +12 -5
- data/web/views/index.slim +8 -9
- data/web/views/layout.slim +27 -19
- data/web/views/poll.slim +1 -0
- data/web/views/queue.slim +22 -9
- data/web/views/queues.slim +7 -7
- data/web/views/retries.slim +13 -10
- data/web/views/retry.slim +5 -5
- data/web/views/scheduled.slim +13 -10
- metadata +71 -2
data/lib/sidekiq/worker.rb
CHANGED
@@ -31,6 +31,7 @@ module Sidekiq
|
|
31
31
|
end
|
32
32
|
|
33
33
|
module ClassMethods
|
34
|
+
|
34
35
|
def perform_async(*args)
|
35
36
|
client_push('class' => self, 'args' => args)
|
36
37
|
end
|
@@ -52,7 +53,7 @@ module Sidekiq
|
|
52
53
|
# :backtrace - whether to save any error backtrace in the retry payload to display in web UI,
|
53
54
|
# can be true, false or an integer number of lines to save, default *false*
|
54
55
|
def sidekiq_options(opts={})
|
55
|
-
self.sidekiq_options_hash = get_sidekiq_options.merge(
|
56
|
+
self.sidekiq_options_hash = get_sidekiq_options.merge((opts || {}).stringify_keys)
|
56
57
|
end
|
57
58
|
|
58
59
|
DEFAULT_OPTIONS = { 'retry' => true, 'queue' => 'default' }
|
@@ -61,15 +62,8 @@ module Sidekiq
|
|
61
62
|
self.sidekiq_options_hash ||= DEFAULT_OPTIONS
|
62
63
|
end
|
63
64
|
|
64
|
-
def stringify_keys(hash) # :nodoc:
|
65
|
-
hash.keys.each do |key|
|
66
|
-
hash[key.to_s] = hash.delete(key)
|
67
|
-
end
|
68
|
-
hash
|
69
|
-
end
|
70
|
-
|
71
65
|
def client_push(item) # :nodoc:
|
72
|
-
Sidekiq::Client.push(stringify_keys
|
66
|
+
Sidekiq::Client.push(item.stringify_keys)
|
73
67
|
end
|
74
68
|
|
75
69
|
end
|
data/sidekiq.gemspec
CHANGED
@@ -19,6 +19,9 @@ Gem::Specification.new do |gem|
|
|
19
19
|
gem.add_dependency 'connection_pool', '~> 0.9.2'
|
20
20
|
gem.add_dependency 'celluloid', '~> 0.12.0'
|
21
21
|
gem.add_dependency 'multi_json', '~> 1'
|
22
|
+
gem.add_dependency 'sprockets-sass'
|
23
|
+
gem.add_dependency 'sass'
|
24
|
+
gem.add_dependency 'compass'
|
22
25
|
gem.add_development_dependency 'minitest', '~> 3'
|
23
26
|
gem.add_development_dependency 'sinatra'
|
24
27
|
gem.add_development_dependency 'slim'
|
data/test/config.yml
CHANGED
data/test/test_api.rb
ADDED
@@ -0,0 +1,75 @@
|
|
1
|
+
require 'helper'
|
2
|
+
|
3
|
+
class TestApi < MiniTest::Unit::TestCase
|
4
|
+
describe 'with an empty database' do
|
5
|
+
before do
|
6
|
+
Sidekiq.redis {|c| c.flushdb }
|
7
|
+
end
|
8
|
+
|
9
|
+
it 'shows queue as empty' do
|
10
|
+
q = Sidekiq::Queue.new
|
11
|
+
assert_equal 0, q.size
|
12
|
+
end
|
13
|
+
|
14
|
+
class ApiWorker
|
15
|
+
include Sidekiq::Worker
|
16
|
+
end
|
17
|
+
|
18
|
+
it 'can enumerate jobs' do
|
19
|
+
q = Sidekiq::Queue.new
|
20
|
+
ApiWorker.perform_async(1, 'mike')
|
21
|
+
assert_equal ['TestApi::ApiWorker'], q.map(&:klass)
|
22
|
+
|
23
|
+
job = q.first
|
24
|
+
assert_equal 24, job.jid.size
|
25
|
+
assert_equal [1, 'mike'], job.args
|
26
|
+
|
27
|
+
q = Sidekiq::Queue.new('other')
|
28
|
+
assert_equal 0, q.size
|
29
|
+
end
|
30
|
+
|
31
|
+
it 'can delete jobs' do
|
32
|
+
q = Sidekiq::Queue.new
|
33
|
+
ApiWorker.perform_async(1, 'mike')
|
34
|
+
assert_equal 1, q.size
|
35
|
+
assert_equal [true], q.map(&:delete)
|
36
|
+
assert_equal 0, q.size
|
37
|
+
end
|
38
|
+
|
39
|
+
it 'shows empty retries' do
|
40
|
+
r = Sidekiq::RetrySet.new
|
41
|
+
assert_equal 0, r.size
|
42
|
+
end
|
43
|
+
|
44
|
+
it 'can enumerate retries' do
|
45
|
+
add_retry
|
46
|
+
|
47
|
+
r = Sidekiq::RetrySet.new
|
48
|
+
assert_equal 1, r.size
|
49
|
+
array = r.to_a
|
50
|
+
assert_equal 1, array.size
|
51
|
+
|
52
|
+
retri = array.first
|
53
|
+
assert_equal 'ApiWorker', retri.klass
|
54
|
+
assert_equal 'default', retri.queue
|
55
|
+
assert_equal 'bob', retri.jid
|
56
|
+
assert_in_delta Time.now.to_f, retri.at.to_f, 0.01
|
57
|
+
end
|
58
|
+
|
59
|
+
it 'can delete retries' do
|
60
|
+
add_retry
|
61
|
+
r = Sidekiq::RetrySet.new
|
62
|
+
assert_equal 1, r.size
|
63
|
+
r.map(&:delete)
|
64
|
+
assert_equal 0, r.size
|
65
|
+
end
|
66
|
+
|
67
|
+
def add_retry
|
68
|
+
at = Time.now.to_f
|
69
|
+
payload = Sidekiq.dump_json('class' => 'ApiWorker', 'args' => [1, 'mike'], 'queue' => 'default', 'jid' => 'bob')
|
70
|
+
Sidekiq.redis do |conn|
|
71
|
+
conn.zadd('retry', at.to_s, payload)
|
72
|
+
end
|
73
|
+
end
|
74
|
+
end
|
75
|
+
end
|
data/test/test_cli.rb
CHANGED
@@ -70,6 +70,11 @@ class TestCli < MiniTest::Unit::TestCase
|
|
70
70
|
assert_equal %w(queue_one queue-two), Sidekiq.options[:queues]
|
71
71
|
end
|
72
72
|
|
73
|
+
it 'handles queues with dots in the name' do
|
74
|
+
@cli.parse(['sidekiq', '-q', 'foo.bar', '-r', './test/fake_env.rb'])
|
75
|
+
assert_equal ['foo.bar'], Sidekiq.options[:queues]
|
76
|
+
end
|
77
|
+
|
73
78
|
it 'sets verbose' do
|
74
79
|
old = Sidekiq.logger.level
|
75
80
|
@cli.parse(['sidekiq', '-v', '-r', './test/fake_env.rb'])
|
@@ -130,7 +135,7 @@ class TestCli < MiniTest::Unit::TestCase
|
|
130
135
|
end
|
131
136
|
|
132
137
|
it 'sets queues' do
|
133
|
-
assert_equal 2, Sidekiq.options[:queues].count { |q| q == '
|
138
|
+
assert_equal 2, Sidekiq.options[:queues].count { |q| q == 'very_often' }
|
134
139
|
assert_equal 1, Sidekiq.options[:queues].count { |q| q == 'seldom' }
|
135
140
|
end
|
136
141
|
end
|
@@ -55,6 +55,22 @@ class TestExceptionHandler < MiniTest::Unit::TestCase
|
|
55
55
|
end
|
56
56
|
end
|
57
57
|
|
58
|
+
describe "with fake Honeybadger" do
|
59
|
+
before do
|
60
|
+
::Honeybadger = MiniTest::Mock.new
|
61
|
+
end
|
62
|
+
|
63
|
+
after do
|
64
|
+
Object.send(:remove_const, "Honeybadger") # HACK should probably inject Honeybadger etc into this class in the future
|
65
|
+
end
|
66
|
+
|
67
|
+
it "notifies Honeybadger" do
|
68
|
+
::Honeybadger.expect(:notify_or_ignore,nil,[TEST_EXCEPTION,:parameters => { :a => 1 }])
|
69
|
+
Component.new.invoke_exception(:a => 1)
|
70
|
+
::Honeybadger.verify
|
71
|
+
end
|
72
|
+
end
|
73
|
+
|
58
74
|
describe "with fake ExceptionNotifier" do
|
59
75
|
before do
|
60
76
|
::ExceptionNotifier = Module.new
|
data/test/test_testing_inline.rb
CHANGED
@@ -46,9 +46,9 @@ class TestInline < MiniTest::Unit::TestCase
|
|
46
46
|
|
47
47
|
after do
|
48
48
|
Sidekiq::Client.singleton_class.class_eval do
|
49
|
-
remove_method :
|
50
|
-
alias_method :
|
51
|
-
remove_method :
|
49
|
+
remove_method :raw_push
|
50
|
+
alias_method :raw_push, :raw_push_old
|
51
|
+
remove_method :raw_push_old
|
52
52
|
end
|
53
53
|
end
|
54
54
|
|
data/test/test_web.rb
CHANGED
@@ -27,14 +27,14 @@ class TestWeb < MiniTest::Unit::TestCase
|
|
27
27
|
it 'can display home' do
|
28
28
|
get '/'
|
29
29
|
assert_equal 200, last_response.status
|
30
|
-
assert_match /
|
30
|
+
assert_match /status-idle/, last_response.body
|
31
31
|
refute_match /default/, last_response.body
|
32
32
|
end
|
33
33
|
|
34
34
|
it 'can display poll' do
|
35
35
|
get '/poll'
|
36
36
|
assert_equal 200, last_response.status
|
37
|
-
assert_match /
|
37
|
+
assert_match /summary/, last_response.body
|
38
38
|
assert_match /workers/, last_response.body
|
39
39
|
refute_match /navbar/, last_response.body
|
40
40
|
end
|
@@ -74,6 +74,24 @@ class TestWeb < MiniTest::Unit::TestCase
|
|
74
74
|
refute conn.smembers('queues').include?('foo')
|
75
75
|
end
|
76
76
|
end
|
77
|
+
|
78
|
+
it 'can delete a job' do
|
79
|
+
Sidekiq.redis do |conn|
|
80
|
+
conn.rpush('queue:foo', "{}")
|
81
|
+
conn.rpush('queue:foo', "{\"foo\":\"bar\"}")
|
82
|
+
conn.rpush('queue:foo', "{\"foo2\":\"bar2\"}")
|
83
|
+
end
|
84
|
+
|
85
|
+
get '/queues/foo'
|
86
|
+
assert_equal 200, last_response.status
|
87
|
+
|
88
|
+
post '/queues/foo/delete', key_val: "{\"foo\":\"bar\"}"
|
89
|
+
assert_equal 302, last_response.status
|
90
|
+
|
91
|
+
Sidekiq.redis do |conn|
|
92
|
+
refute conn.lrange('queue:foo', 0, -1).include?("{\"foo\":\"bar\"}")
|
93
|
+
end
|
94
|
+
end
|
77
95
|
|
78
96
|
it 'can display scheduled' do
|
79
97
|
get '/scheduled'
|
@@ -149,12 +167,15 @@ class TestWeb < MiniTest::Unit::TestCase
|
|
149
167
|
end
|
150
168
|
|
151
169
|
it 'can show user defined tab' do
|
152
|
-
|
170
|
+
begin
|
171
|
+
Sidekiq::Web.tabs['Custom Tab'] = '/custom'
|
153
172
|
|
154
|
-
|
155
|
-
|
173
|
+
get '/'
|
174
|
+
assert_match 'Custom Tab', last_response.body
|
156
175
|
|
157
|
-
|
176
|
+
ensure
|
177
|
+
Sidekiq::Web.tabs.delete 'Custom Tab'
|
178
|
+
end
|
158
179
|
end
|
159
180
|
|
160
181
|
def add_scheduled
|
File without changes
|
File without changes
|
Binary file
|
Binary file
|
Binary file
|
Binary file
|
@@ -25,31 +25,59 @@ $(function() {
|
|
25
25
|
|
26
26
|
$('a[name=poll]').data('polling', false);
|
27
27
|
|
28
|
+
pollStatus = $('.poll-status')
|
29
|
+
|
30
|
+
pollStatusText = pollStatus.find('.text')
|
31
|
+
pollStatusBadge = pollStatus.find('.badge')
|
32
|
+
pollStatusBadge.hide();
|
33
|
+
pollStatusMarkup = pollStatus.html();
|
34
|
+
|
28
35
|
$('a[name=poll]').on('click', function(e) {
|
29
36
|
e.preventDefault();
|
37
|
+
|
30
38
|
var pollLink = $(this);
|
39
|
+
|
31
40
|
if (pollLink.data('polling')) {
|
41
|
+
|
42
|
+
$(this).removeClass('active');
|
43
|
+
|
32
44
|
clearInterval(pollLink.data('interval'));
|
33
|
-
pollLink.text('
|
34
|
-
|
35
|
-
|
36
|
-
|
45
|
+
pollLink.text(pollLink.data('text'));
|
46
|
+
|
47
|
+
pollStatus.html(pollStatusMarkup);
|
48
|
+
pollStatusBadge.hide();
|
49
|
+
|
50
|
+
} else {
|
51
|
+
|
52
|
+
$(this).addClass('active');
|
53
|
+
|
37
54
|
var href = pollLink.attr('href');
|
38
|
-
|
39
|
-
|
40
|
-
var responseHtml = $(data);
|
41
|
-
$('.hero-unit').replaceWith(responseHtml.find('.hero-unit'));
|
42
|
-
$('.workers').replaceWith(responseHtml.find('.workers'));
|
43
|
-
$('time').timeago();
|
44
|
-
});
|
45
|
-
var currentTime = new Date();
|
46
|
-
$('.poll-status').text('Last polled at: ' + currentTime.getHours() + ':' + pad(currentTime.getMinutes()) + ':' + pad(currentTime.getSeconds()));
|
47
|
-
}, 2000));
|
48
|
-
$('.poll-status').text('Starting to poll...');
|
55
|
+
|
56
|
+
pollLink.data('text', pollLink.text());
|
49
57
|
pollLink.text('Stop Polling');
|
58
|
+
pollLink.data('interval', setInterval(function(){
|
59
|
+
livePoll(href);
|
60
|
+
}, 2000));
|
61
|
+
|
62
|
+
pollStatusText.text('Starting to poll...');
|
50
63
|
}
|
64
|
+
|
51
65
|
pollLink.data('polling', !pollLink.data('polling'));
|
52
|
-
|
66
|
+
|
67
|
+
});
|
68
|
+
|
69
|
+
livePoll = function livePoll(href){
|
70
|
+
$.get(href, function(data) {
|
71
|
+
var responseHtml = $(data);
|
72
|
+
$('.summary').replaceWith(responseHtml.find('.summary'));
|
73
|
+
$('.status').html(responseHtml.find('.status').html().toString());
|
74
|
+
$('.workers').replaceWith(responseHtml.find('.workers'));
|
75
|
+
$('time').timeago();
|
76
|
+
});
|
77
|
+
var currentTime = new Date();
|
78
|
+
$('.poll-status .text').text('Last polled: ')
|
79
|
+
$('.poll-status .time').show().text(currentTime.getHours() + ':' + pad(currentTime.getMinutes()) + ':' + pad(currentTime.getSeconds()));
|
80
|
+
}
|
53
81
|
});
|
54
82
|
|
55
83
|
$(function() {
|
File without changes
|
File without changes
|
File without changes
|
File without changes
|
@@ -0,0 +1,116 @@
|
|
1
|
+
/* baseline */
|
2
|
+
@include establish-baseline;
|
3
|
+
|
4
|
+
/* apply a natural box layout model to all elements */
|
5
|
+
* { @include box-sizing(border-box); }
|
6
|
+
|
7
|
+
/* styyyling */
|
8
|
+
body {
|
9
|
+
color: $dark_gray;
|
10
|
+
padding : 0;
|
11
|
+
text-rendering : optimizeLegibility;
|
12
|
+
-webkit-font-smoothing: antialiased;
|
13
|
+
font: {
|
14
|
+
family:"Helvetica Neue", Helvetica, Arial, sans-serif;
|
15
|
+
}
|
16
|
+
|
17
|
+
|
18
|
+
background-color: $background;
|
19
|
+
background-image: url();
|
20
|
+
}
|
21
|
+
|
22
|
+
|
23
|
+
.public{
|
24
|
+
body{
|
25
|
+
font: {
|
26
|
+
family:'Gudea', "Helvetica Neue", Helvetica, Arial, sans-serif;
|
27
|
+
weight: 400;
|
28
|
+
}
|
29
|
+
}
|
30
|
+
}
|
31
|
+
|
32
|
+
|
33
|
+
|
34
|
+
a{
|
35
|
+
color: $link_color;
|
36
|
+
&:active, &:hover, &:visited{
|
37
|
+
color: $link_active_color;
|
38
|
+
}
|
39
|
+
&.btn{
|
40
|
+
color: $dark_gray;
|
41
|
+
}
|
42
|
+
&.btn-primary{
|
43
|
+
color:white;
|
44
|
+
}
|
45
|
+
}
|
46
|
+
|
47
|
+
h1, h2, h3, h4, h5, h6, strong{
|
48
|
+
font-weight: 700;
|
49
|
+
}
|
50
|
+
|
51
|
+
.brand, h5, h4, h3, h2, h1{
|
52
|
+
font-family: Armata;
|
53
|
+
font-weight: 400;
|
54
|
+
}
|
55
|
+
|
56
|
+
.title{
|
57
|
+
color: $highlight_color;
|
58
|
+
}
|
59
|
+
|
60
|
+
.subheader{
|
61
|
+
font-weight:200;
|
62
|
+
font-size: 24px;
|
63
|
+
@media (min-width: $split_width){
|
64
|
+
line-height:1.5em;
|
65
|
+
font-size: 32px;
|
66
|
+
}
|
67
|
+
}
|
68
|
+
.navbar .brand{
|
69
|
+
color: $btn_primary;
|
70
|
+
@include text-shadow(none);
|
71
|
+
@extend .brand;
|
72
|
+
}
|
73
|
+
.title, .admin .brand{
|
74
|
+
@include background($title_primary_gradient);
|
75
|
+
-webkit-background-clip: text;
|
76
|
+
-webkit-text-fill-color: transparent;
|
77
|
+
}
|
78
|
+
|
79
|
+
|
80
|
+
.hero-unit{
|
81
|
+
background:none;
|
82
|
+
@media (max-width: $split_width) {
|
83
|
+
padding: 20px;
|
84
|
+
}
|
85
|
+
}
|
86
|
+
|
87
|
+
.public{
|
88
|
+
.hero-unit{
|
89
|
+
.title{
|
90
|
+
@media (min-width: $split_width){
|
91
|
+
font-size :96px;
|
92
|
+
}
|
93
|
+
}
|
94
|
+
}
|
95
|
+
}
|
96
|
+
|
97
|
+
.page-header{
|
98
|
+
h1, h2, h3, h4, h5, h6{
|
99
|
+
@include text-shadow($highlight);
|
100
|
+
color:$title_color;
|
101
|
+
}
|
102
|
+
}
|
103
|
+
|
104
|
+
#downlad_btn{
|
105
|
+
font-weight:700;
|
106
|
+
}
|
107
|
+
|
108
|
+
|
109
|
+
pre{
|
110
|
+
font-size:11px;
|
111
|
+
}
|
112
|
+
|
113
|
+
code{
|
114
|
+
background:none;
|
115
|
+
border:none;
|
116
|
+
}
|