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.

Files changed (62) hide show
  1. data/.gitignore +1 -0
  2. data/Changes.md +11 -0
  3. data/Gemfile +6 -0
  4. data/lib/sidekiq.rb +1 -0
  5. data/lib/sidekiq/api.rb +184 -0
  6. data/lib/sidekiq/cli.rb +3 -2
  7. data/lib/sidekiq/client.rb +16 -10
  8. data/lib/sidekiq/core_ext.rb +30 -0
  9. data/lib/sidekiq/exception_handler.rb +9 -0
  10. data/lib/sidekiq/processor.rb +1 -1
  11. data/lib/sidekiq/testing/inline.rb +3 -3
  12. data/lib/sidekiq/util.rb +1 -11
  13. data/lib/sidekiq/version.rb +1 -1
  14. data/lib/sidekiq/web.rb +49 -1
  15. data/lib/sidekiq/worker.rb +3 -9
  16. data/sidekiq.gemspec +3 -0
  17. data/test/config.yml +1 -1
  18. data/test/test_api.rb +75 -0
  19. data/test/test_cli.rb +6 -1
  20. data/test/test_exception_handler.rb +16 -0
  21. data/test/test_testing_inline.rb +3 -3
  22. data/test/test_web.rb +27 -6
  23. data/web/assets/images/bootstrap/glyphicons-halflings-white.png +0 -0
  24. data/web/assets/images/bootstrap/glyphicons-halflings.png +0 -0
  25. data/web/assets/images/logo.png +0 -0
  26. data/web/assets/images/status-sd8051fd480.png +0 -0
  27. data/web/assets/images/status/active.png +0 -0
  28. data/web/assets/images/status/idle.png +0 -0
  29. data/web/assets/javascripts/application.js +44 -16
  30. data/web/assets/javascripts/vendor/bootstrap.js +0 -0
  31. data/web/assets/javascripts/vendor/jquery.js +0 -0
  32. data/web/assets/javascripts/vendor/jquery.timeago.js +0 -0
  33. data/web/assets/stylesheets/application.css +1 -3
  34. data/web/assets/stylesheets/layout.css +0 -0
  35. data/web/assets/stylesheets/partials/_base.scss +116 -0
  36. data/web/assets/stylesheets/partials/_colors.scss +46 -0
  37. data/web/assets/stylesheets/partials/_fonts.scss +3 -0
  38. data/web/assets/stylesheets/partials/_h5bp.scss +293 -0
  39. data/web/assets/stylesheets/partials/_layout.scss +197 -0
  40. data/web/assets/stylesheets/partials/_navbar.scss +100 -0
  41. data/web/assets/stylesheets/partials/_normalize.scss +504 -0
  42. data/web/assets/stylesheets/partials/_prettify.css +30 -0
  43. data/web/assets/stylesheets/partials/_variables.scss +28 -0
  44. data/web/assets/stylesheets/public.scss +74 -0
  45. data/web/assets/stylesheets/style.scss +69 -0
  46. data/web/assets/stylesheets/vendor/bootstrap-responsive.css +0 -0
  47. data/web/assets/stylesheets/vendor/bootstrap.css +0 -0
  48. data/web/assets/stylesheets/vendors.scss +2 -0
  49. data/web/views/_nav.slim +13 -0
  50. data/web/views/_paging.slim +0 -0
  51. data/web/views/_status.slim +3 -0
  52. data/web/views/_summary.slim +19 -8
  53. data/web/views/_workers.slim +12 -5
  54. data/web/views/index.slim +8 -9
  55. data/web/views/layout.slim +27 -19
  56. data/web/views/poll.slim +1 -0
  57. data/web/views/queue.slim +22 -9
  58. data/web/views/queues.slim +7 -7
  59. data/web/views/retries.slim +13 -10
  60. data/web/views/retry.slim +5 -5
  61. data/web/views/scheduled.slim +13 -10
  62. metadata +71 -2
@@ -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(stringify_keys(opts || {}))
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(item))
66
+ Sidekiq::Client.push(item.stringify_keys)
73
67
  end
74
68
 
75
69
  end
@@ -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'
@@ -5,5 +5,5 @@
5
5
  :pidfile: /tmp/sidekiq-config-test.pid
6
6
  :concurrency: 50
7
7
  :queues:
8
- - [often, 2]
8
+ - [<%="very_"%>often, 2]
9
9
  - [seldom, 1]
@@ -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
@@ -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 == 'often' }
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
@@ -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 :push
50
- alias_method :push, :push_old
51
- remove_method :push_old
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
 
@@ -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 /Sidekiq is idle/, last_response.body
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 /hero-unit/, last_response.body
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
- Sidekiq::Web.tabs << 'Custom Tab'
170
+ begin
171
+ Sidekiq::Web.tabs['Custom Tab'] = '/custom'
153
172
 
154
- get '/'
155
- assert_match 'Custom Tab', last_response.body
173
+ get '/'
174
+ assert_match 'Custom Tab', last_response.body
156
175
 
157
- Sidekiq::Web.tabs.delete 'Custom Tab'
176
+ ensure
177
+ Sidekiq::Web.tabs.delete 'Custom Tab'
178
+ end
158
179
  end
159
180
 
160
181
  def add_scheduled
@@ -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('Live Poll');
34
- $('.poll-status').text('');
35
- }
36
- else {
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
- pollLink.data('interval', setInterval(function() {
39
- $.get(href, function(data) {
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
@@ -1,6 +1,4 @@
1
1
  /*
2
- *= require vendor/bootstrap
3
- *= require layout
4
- *= require vendor/bootstrap-responsive
2
+ *= require style
5
3
  *= require_self
6
4
  */
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
+ }