2012-03-10 16:07:19 -05:00
|
|
|
require 'helper'
|
|
|
|
require 'sidekiq'
|
|
|
|
require 'sidekiq/web'
|
|
|
|
require 'rack/test'
|
|
|
|
|
2013-09-22 17:38:33 -04:00
|
|
|
class TestWeb < Sidekiq::Test
|
2012-03-10 16:07:19 -05:00
|
|
|
describe 'sidekiq web' do
|
|
|
|
include Rack::Test::Methods
|
|
|
|
|
|
|
|
def app
|
|
|
|
Sidekiq::Web
|
|
|
|
end
|
|
|
|
|
2012-12-02 23:18:31 -05:00
|
|
|
def job_params(job, score)
|
|
|
|
"#{score}-#{job['jid']}"
|
|
|
|
end
|
|
|
|
|
2012-03-10 16:07:19 -05:00
|
|
|
before do
|
2012-03-14 00:19:46 -04:00
|
|
|
Sidekiq.redis = REDIS
|
2012-03-28 22:16:54 -04:00
|
|
|
Sidekiq.redis {|c| c.flushdb }
|
2012-03-10 16:07:19 -05:00
|
|
|
end
|
|
|
|
|
|
|
|
class WebWorker
|
|
|
|
include Sidekiq::Worker
|
|
|
|
|
|
|
|
def perform(a, b)
|
|
|
|
a + b
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2013-02-20 15:04:19 -05:00
|
|
|
it 'can display workers' do
|
2013-04-22 17:14:19 -04:00
|
|
|
Sidekiq.redis do |conn|
|
|
|
|
identity = 'foo:1234-123abc:default'
|
|
|
|
conn.sadd('workers', identity)
|
|
|
|
conn.setex("worker:#{identity}:started", 10, Time.now.to_s)
|
|
|
|
hash = {:queue => 'critical', :payload => { 'class' => WebWorker.name, 'args' => [1,'abc'] }, :run_at => Time.now.to_i }
|
|
|
|
conn.setex("worker:#{identity}", 10, Sidekiq.dump_json(hash))
|
|
|
|
end
|
|
|
|
|
2013-02-20 15:04:19 -05:00
|
|
|
get '/workers'
|
2012-03-10 16:07:19 -05:00
|
|
|
assert_equal 200, last_response.status
|
2013-04-22 17:14:19 -04:00
|
|
|
assert_match /status-active/, last_response.body
|
|
|
|
assert_match /critical/, last_response.body
|
|
|
|
assert_match /WebWorker/, last_response.body
|
2012-06-08 20:20:11 -04:00
|
|
|
end
|
2012-03-10 16:07:19 -05:00
|
|
|
|
2012-06-08 20:20:11 -04:00
|
|
|
it 'can display queues' do
|
2012-04-01 22:53:45 -04:00
|
|
|
assert Sidekiq::Client.push('queue' => :foo, 'class' => WebWorker, 'args' => [1, 3])
|
2012-03-10 16:07:19 -05:00
|
|
|
|
2012-06-08 20:20:11 -04:00
|
|
|
get '/queues'
|
2012-03-10 16:07:19 -05:00
|
|
|
assert_equal 200, last_response.status
|
2012-05-12 02:28:30 -04:00
|
|
|
assert_match /foo/, last_response.body
|
2012-06-08 20:20:11 -04:00
|
|
|
refute_match /HardWorker/, last_response.body
|
2012-03-17 12:41:24 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
it 'handles queue view' do
|
|
|
|
get '/queues/default'
|
|
|
|
assert_equal 200, last_response.status
|
|
|
|
end
|
|
|
|
|
2012-05-05 01:23:43 -04:00
|
|
|
it 'can delete a queue' do
|
|
|
|
Sidekiq.redis do |conn|
|
|
|
|
conn.rpush('queue:foo', '{}')
|
|
|
|
conn.sadd('queues', 'foo')
|
|
|
|
end
|
|
|
|
|
|
|
|
get '/queues/foo'
|
|
|
|
assert_equal 200, last_response.status
|
|
|
|
|
|
|
|
post '/queues/foo'
|
|
|
|
assert_equal 302, last_response.status
|
|
|
|
|
|
|
|
Sidekiq.redis do |conn|
|
|
|
|
refute conn.smembers('queues').include?('foo')
|
2012-11-25 21:43:48 -05:00
|
|
|
refute conn.exists('queues:foo')
|
2012-05-05 01:23:43 -04:00
|
|
|
end
|
|
|
|
end
|
2012-11-25 21:43:48 -05:00
|
|
|
|
2013-06-24 14:28:13 -04:00
|
|
|
it 'can clear an empty worker list' do
|
2013-06-24 11:14:06 -04:00
|
|
|
post '/reset'
|
|
|
|
assert_equal 302, last_response.status
|
|
|
|
end
|
|
|
|
|
2013-06-24 14:28:13 -04:00
|
|
|
it 'can clear a non-empty worker list' do
|
|
|
|
Sidekiq.redis do |conn|
|
|
|
|
identity = 'foo'
|
|
|
|
conn.sadd('workers', identity)
|
|
|
|
end
|
|
|
|
|
|
|
|
post '/reset'
|
|
|
|
|
|
|
|
assert_equal 302, last_response.status
|
|
|
|
|
|
|
|
Sidekiq.redis do |conn|
|
|
|
|
refute conn.smembers('workers').any?
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2012-10-23 19:11:14 -04:00
|
|
|
it 'can delete a job' do
|
|
|
|
Sidekiq.redis do |conn|
|
|
|
|
conn.rpush('queue:foo', "{}")
|
|
|
|
conn.rpush('queue:foo', "{\"foo\":\"bar\"}")
|
|
|
|
conn.rpush('queue:foo', "{\"foo2\":\"bar2\"}")
|
|
|
|
end
|
|
|
|
|
|
|
|
get '/queues/foo'
|
|
|
|
assert_equal 200, last_response.status
|
|
|
|
|
|
|
|
post '/queues/foo/delete', key_val: "{\"foo\":\"bar\"}"
|
|
|
|
assert_equal 302, last_response.status
|
|
|
|
|
|
|
|
Sidekiq.redis do |conn|
|
|
|
|
refute conn.lrange('queue:foo', 0, -1).include?("{\"foo\":\"bar\"}")
|
|
|
|
end
|
|
|
|
end
|
2012-05-11 19:48:03 -04:00
|
|
|
|
|
|
|
it 'can display retries' do
|
|
|
|
get '/retries'
|
|
|
|
assert_equal 200, last_response.status
|
2012-05-12 02:28:30 -04:00
|
|
|
assert_match /found/, last_response.body
|
|
|
|
refute_match /HardWorker/, last_response.body
|
2012-05-11 19:48:03 -04:00
|
|
|
|
|
|
|
add_retry
|
|
|
|
|
|
|
|
get '/retries'
|
|
|
|
assert_equal 200, last_response.status
|
2012-05-12 02:28:30 -04:00
|
|
|
refute_match /found/, last_response.body
|
|
|
|
assert_match /HardWorker/, last_response.body
|
2012-05-11 19:48:03 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
it 'can display a single retry' do
|
2012-11-26 14:53:22 -05:00
|
|
|
params = add_retry
|
2013-07-27 19:04:06 -04:00
|
|
|
get '/retries/0-shouldntexist'
|
2012-05-11 19:48:03 -04:00
|
|
|
assert_equal 302, last_response.status
|
2012-11-26 14:53:22 -05:00
|
|
|
get "/retries/#{job_params(*params)}"
|
2012-05-11 19:48:03 -04:00
|
|
|
assert_equal 200, last_response.status
|
2012-05-12 02:28:30 -04:00
|
|
|
assert_match /HardWorker/, last_response.body
|
2012-05-11 19:48:03 -04:00
|
|
|
end
|
2012-07-26 19:08:15 -04:00
|
|
|
|
2012-11-26 14:53:22 -05:00
|
|
|
it 'handles missing retry' do
|
2013-07-27 19:04:06 -04:00
|
|
|
get "/retries/0-shouldntexist"
|
2012-11-26 14:53:22 -05:00
|
|
|
assert_equal 302, last_response.status
|
|
|
|
end
|
|
|
|
|
2012-06-12 21:59:43 -04:00
|
|
|
it 'can delete a single retry' do
|
2012-11-26 14:53:22 -05:00
|
|
|
params = add_retry
|
|
|
|
post "/retries/#{job_params(*params)}", 'delete' => 'Delete'
|
2012-06-12 21:59:43 -04:00
|
|
|
assert_equal 302, last_response.status
|
2012-06-18 07:59:25 -04:00
|
|
|
assert_equal 'http://example.org/retries', last_response.header['Location']
|
2012-07-26 19:08:15 -04:00
|
|
|
|
2012-06-12 21:59:43 -04:00
|
|
|
get "/retries"
|
|
|
|
assert_equal 200, last_response.status
|
2012-11-26 14:53:22 -05:00
|
|
|
refute_match /#{params.first['args'][2]}/, last_response.body
|
2012-06-12 21:59:43 -04:00
|
|
|
end
|
2012-07-26 19:08:15 -04:00
|
|
|
|
2012-11-25 21:43:48 -05:00
|
|
|
it 'can delete all retries' do
|
|
|
|
3.times { add_retry }
|
|
|
|
|
|
|
|
post "/retries/all/delete", 'delete' => 'Delete'
|
|
|
|
assert_equal 0, Sidekiq::RetrySet.new.size
|
|
|
|
assert_equal 302, last_response.status
|
|
|
|
assert_equal 'http://example.org/retries', last_response.header['Location']
|
|
|
|
end
|
|
|
|
|
2012-06-12 23:32:13 -04:00
|
|
|
it 'can retry a single retry now' do
|
2012-11-26 14:53:22 -05:00
|
|
|
params = add_retry
|
|
|
|
post "/retries/#{job_params(*params)}", 'retry' => 'Retry'
|
2012-06-12 23:32:13 -04:00
|
|
|
assert_equal 302, last_response.status
|
2012-06-18 07:59:25 -04:00
|
|
|
assert_equal 'http://example.org/retries', last_response.header['Location']
|
2012-07-26 19:08:15 -04:00
|
|
|
|
2012-06-12 23:32:13 -04:00
|
|
|
get '/queues/default'
|
|
|
|
assert_equal 200, last_response.status
|
2012-11-26 14:53:22 -05:00
|
|
|
assert_match /#{params.first['args'][2]}/, last_response.body
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'can display scheduled' do
|
|
|
|
get '/scheduled'
|
|
|
|
assert_equal 200, last_response.status
|
|
|
|
assert_match /found/, last_response.body
|
|
|
|
refute_match /HardWorker/, last_response.body
|
|
|
|
|
|
|
|
add_scheduled
|
|
|
|
|
|
|
|
get '/scheduled'
|
|
|
|
assert_equal 200, last_response.status
|
|
|
|
refute_match /found/, last_response.body
|
|
|
|
assert_match /HardWorker/, last_response.body
|
2012-06-12 23:32:13 -04:00
|
|
|
end
|
2012-05-11 19:48:03 -04:00
|
|
|
|
2013-07-27 18:49:35 -04:00
|
|
|
it 'can display a single scheduled job' do
|
|
|
|
params = add_scheduled
|
2013-07-27 19:04:06 -04:00
|
|
|
get '/scheduled/0-shouldntexist'
|
2013-07-27 18:49:35 -04:00
|
|
|
assert_equal 302, last_response.status
|
|
|
|
get "/scheduled/#{job_params(*params)}"
|
|
|
|
assert_equal 200, last_response.status
|
|
|
|
assert_match /HardWorker/, last_response.body
|
|
|
|
end
|
|
|
|
|
2013-07-27 19:04:06 -04:00
|
|
|
it 'handles missing scheduled job' do
|
|
|
|
get "/scheduled/0-shouldntexist"
|
|
|
|
assert_equal 302, last_response.status
|
|
|
|
end
|
|
|
|
|
2013-07-27 18:49:35 -04:00
|
|
|
it 'can add to queue a single scheduled job' do
|
|
|
|
params = add_scheduled
|
|
|
|
post "/scheduled/#{job_params(*params)}", 'add_to_queue' => true
|
|
|
|
assert_equal 302, last_response.status
|
|
|
|
assert_equal 'http://example.org/scheduled', last_response.header['Location']
|
|
|
|
|
|
|
|
get '/queues/default'
|
|
|
|
assert_equal 200, last_response.status
|
|
|
|
assert_match /#{params.first['args'][2]}/, last_response.body
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'can delete a single scheduled job' do
|
|
|
|
params = add_scheduled
|
|
|
|
post "/scheduled/#{job_params(*params)}", 'delete' => 'Delete'
|
|
|
|
assert_equal 302, last_response.status
|
|
|
|
assert_equal 'http://example.org/scheduled', last_response.header['Location']
|
|
|
|
|
|
|
|
get "/scheduled"
|
|
|
|
assert_equal 200, last_response.status
|
|
|
|
refute_match /#{params.first['args'][2]}/, last_response.body
|
|
|
|
end
|
|
|
|
|
2012-11-26 11:22:48 -05:00
|
|
|
it 'can delete scheduled' do
|
2012-11-26 14:53:22 -05:00
|
|
|
params = add_scheduled
|
2012-11-26 11:22:48 -05:00
|
|
|
Sidekiq.redis do |conn|
|
|
|
|
assert_equal 1, conn.zcard('schedule')
|
2012-11-26 14:53:22 -05:00
|
|
|
post '/scheduled', 'key' => [job_params(*params)], 'delete' => 'Delete'
|
2012-11-26 11:22:48 -05:00
|
|
|
assert_equal 302, last_response.status
|
|
|
|
assert_equal 'http://example.org/scheduled', last_response.header['Location']
|
|
|
|
assert_equal 0, conn.zcard('schedule')
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2013-06-23 19:22:54 -04:00
|
|
|
it "can move scheduled to default queue" do
|
2013-07-17 00:20:42 -04:00
|
|
|
q = Sidekiq::Queue.new
|
2013-06-23 19:22:54 -04:00
|
|
|
params = add_scheduled
|
|
|
|
Sidekiq.redis do |conn|
|
|
|
|
assert_equal 1, conn.zcard('schedule')
|
2013-07-17 00:20:42 -04:00
|
|
|
assert_equal 0, q.size
|
2013-06-23 19:22:54 -04:00
|
|
|
post '/scheduled', 'key' => [job_params(*params)], 'add_to_queue' => 'AddToQueue'
|
|
|
|
assert_equal 302, last_response.status
|
|
|
|
assert_equal 'http://example.org/scheduled', last_response.header['Location']
|
|
|
|
assert_equal 0, conn.zcard('schedule')
|
2013-07-17 00:20:42 -04:00
|
|
|
assert_equal 1, q.size
|
2013-06-23 19:22:54 -04:00
|
|
|
get '/queues/default'
|
|
|
|
assert_equal 200, last_response.status
|
|
|
|
assert_match /#{params[0]['args'][2]}/, last_response.body
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2012-11-25 21:43:48 -05:00
|
|
|
it 'can retry all retries' do
|
|
|
|
msg, score = add_retry
|
|
|
|
add_retry
|
|
|
|
|
|
|
|
post "/retries/all/retry", 'retry' => 'Retry'
|
|
|
|
assert_equal 302, last_response.status
|
|
|
|
assert_equal 'http://example.org/retries', last_response.header['Location']
|
|
|
|
assert_equal 2, Sidekiq::Queue.new("default").size
|
|
|
|
|
|
|
|
get '/queues/default'
|
|
|
|
assert_equal 200, last_response.status
|
|
|
|
assert_match /#{msg['args'][2]}/, last_response.body
|
|
|
|
end
|
|
|
|
|
2012-08-20 18:28:14 -04:00
|
|
|
it 'can show user defined tab' do
|
2012-10-28 22:49:43 -04:00
|
|
|
begin
|
|
|
|
Sidekiq::Web.tabs['Custom Tab'] = '/custom'
|
2012-08-20 18:28:14 -04:00
|
|
|
|
2012-10-28 22:49:43 -04:00
|
|
|
get '/'
|
|
|
|
assert_match 'Custom Tab', last_response.body
|
2012-08-20 18:28:14 -04:00
|
|
|
|
2012-10-28 22:49:43 -04:00
|
|
|
ensure
|
|
|
|
Sidekiq::Web.tabs.delete 'Custom Tab'
|
|
|
|
end
|
2012-08-20 18:28:14 -04:00
|
|
|
end
|
|
|
|
|
2013-02-20 15:04:19 -05:00
|
|
|
it 'can display home' do
|
|
|
|
get '/'
|
2012-12-11 09:49:16 -05:00
|
|
|
assert_equal 200, last_response.status
|
|
|
|
end
|
|
|
|
|
2013-10-21 15:07:18 -04:00
|
|
|
Sidekiq::Web.settings.locales << File.join(File.dirname(__FILE__), "fixtures")
|
|
|
|
it 'can show user defined tab with custom locales' do
|
|
|
|
begin
|
|
|
|
Sidekiq::Web.tabs['Custom Tab'] = '/custom'
|
|
|
|
Sidekiq::Web.get('/custom') do
|
|
|
|
t('translated_text')
|
|
|
|
end
|
|
|
|
|
|
|
|
get '/custom'
|
|
|
|
assert_match /Changed text/, last_response.body
|
|
|
|
|
|
|
|
ensure
|
|
|
|
Sidekiq::Web.tabs.delete 'Custom Tab'
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2013-01-26 14:34:46 -05:00
|
|
|
describe 'stats' do
|
|
|
|
before do
|
|
|
|
Sidekiq.redis do |conn|
|
|
|
|
conn.set("stat:processed", 5)
|
|
|
|
conn.set("stat:failed", 2)
|
|
|
|
end
|
|
|
|
2.times { add_retry }
|
|
|
|
3.times { add_scheduled }
|
2013-02-18 10:15:51 -05:00
|
|
|
4.times { add_worker }
|
|
|
|
|
2013-01-26 14:34:46 -05:00
|
|
|
get '/dashboard/stats'
|
|
|
|
@response = Sidekiq.load_json(last_response.body)
|
2012-12-13 06:53:54 -05:00
|
|
|
end
|
2012-12-28 20:55:00 -05:00
|
|
|
|
2013-01-26 14:34:46 -05:00
|
|
|
it 'can refresh dashboard stats' do
|
|
|
|
assert_equal 200, last_response.status
|
|
|
|
end
|
|
|
|
|
|
|
|
describe "for sidekiq" do
|
|
|
|
it 'are namespaced' do
|
|
|
|
assert_includes @response.keys, "sidekiq"
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'reports processed' do
|
2013-02-18 10:15:51 -05:00
|
|
|
assert_equal 5, @response["sidekiq"]["processed"]
|
2013-01-26 14:34:46 -05:00
|
|
|
end
|
|
|
|
|
|
|
|
it 'reports failed' do
|
2013-02-18 10:15:51 -05:00
|
|
|
assert_equal 2, @response["sidekiq"]["failed"]
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'reports busy' do
|
|
|
|
assert_equal 4, @response["sidekiq"]["busy"]
|
2013-01-26 14:34:46 -05:00
|
|
|
end
|
|
|
|
|
|
|
|
it 'reports retries' do
|
2013-02-18 10:15:51 -05:00
|
|
|
assert_equal 2, @response["sidekiq"]["retries"]
|
2013-01-26 14:34:46 -05:00
|
|
|
end
|
|
|
|
|
|
|
|
it 'reports scheduled' do
|
2013-02-18 10:15:51 -05:00
|
|
|
assert_equal 3, @response["sidekiq"]["scheduled"]
|
2013-01-26 14:34:46 -05:00
|
|
|
end
|
2013-10-04 03:18:04 -04:00
|
|
|
|
|
|
|
it 'reports latency' do
|
2013-10-04 12:44:45 -04:00
|
|
|
assert_equal 0, @response["sidekiq"]["default_latency"]
|
2013-10-04 03:18:04 -04:00
|
|
|
end
|
2013-01-26 14:34:46 -05:00
|
|
|
end
|
|
|
|
|
|
|
|
describe "for redis" do
|
|
|
|
it 'are namespaced' do
|
|
|
|
assert_includes @response.keys, "redis"
|
|
|
|
end
|
|
|
|
end
|
2012-12-13 06:53:54 -05:00
|
|
|
end
|
|
|
|
|
2012-05-26 21:38:55 -04:00
|
|
|
def add_scheduled
|
2012-11-26 11:22:48 -05:00
|
|
|
score = Time.now.to_f
|
2012-05-26 21:38:55 -04:00
|
|
|
msg = { 'class' => 'HardWorker',
|
|
|
|
'args' => ['bob', 1, Time.now.to_f],
|
2012-11-26 14:53:22 -05:00
|
|
|
'jid' => 'f39af2a05e8f4b24dbc0f1e4' }
|
2012-05-26 21:38:55 -04:00
|
|
|
Sidekiq.redis do |conn|
|
|
|
|
conn.zadd('schedule', score, Sidekiq.dump_json(msg))
|
|
|
|
end
|
2012-11-26 14:53:22 -05:00
|
|
|
[msg, score]
|
2012-05-26 21:38:55 -04:00
|
|
|
end
|
|
|
|
|
2012-05-11 19:48:03 -04:00
|
|
|
def add_retry
|
|
|
|
msg = { 'class' => 'HardWorker',
|
|
|
|
'args' => ['bob', 1, Time.now.to_f],
|
|
|
|
'queue' => 'default',
|
|
|
|
'error_message' => 'Some fake message',
|
|
|
|
'error_class' => 'RuntimeError',
|
|
|
|
'retry_count' => 0,
|
2012-11-26 11:22:48 -05:00
|
|
|
'failed_at' => Time.now.utc,
|
2012-11-26 14:53:22 -05:00
|
|
|
'jid' => 'f39af2a05e8f4b24dbc0f1e4'}
|
2012-05-11 19:48:03 -04:00
|
|
|
score = Time.now.to_f
|
|
|
|
Sidekiq.redis do |conn|
|
|
|
|
conn.zadd('retry', score, Sidekiq.dump_json(msg))
|
|
|
|
end
|
2012-11-26 14:53:22 -05:00
|
|
|
[msg, score]
|
2012-05-11 19:48:03 -04:00
|
|
|
end
|
2013-02-18 10:15:51 -05:00
|
|
|
|
|
|
|
def add_worker
|
|
|
|
process_id = rand(1000)
|
|
|
|
msg = "{\"queue\":\"default\",\"payload\":{\"retry\":true,\"queue\":\"default\",\"timeout\":20,\"backtrace\":5,\"class\":\"HardWorker\",\"args\":[\"bob\",10,5],\"jid\":\"2b5ad2b016f5e063a1c62872\"},\"run_at\":1361208995}"
|
|
|
|
Sidekiq.redis do |conn|
|
|
|
|
conn.sadd("workers", "mercury.home:#{process_id}-70215157189060:started")
|
|
|
|
conn.set("worker:mercury.home:#{process_id}-70215157189060:started", msg)
|
|
|
|
end
|
|
|
|
end
|
2012-03-10 16:07:19 -05:00
|
|
|
end
|
|
|
|
end
|