2012-03-04 12:58:16 -08:00
|
|
|
require 'sinatra/base'
|
|
|
|
require 'slim'
|
2012-07-17 22:14:15 -07:00
|
|
|
require 'sidekiq/paginator'
|
|
|
|
|
2012-03-04 12:58:16 -08:00
|
|
|
module Sidekiq
|
|
|
|
class Web < Sinatra::Base
|
2012-07-17 22:14:15 -07:00
|
|
|
include Sidekiq::Paginator
|
|
|
|
|
2012-03-05 19:27:27 -08:00
|
|
|
dir = File.expand_path(File.dirname(__FILE__) + "/../../web")
|
2012-11-02 11:11:34 -07:00
|
|
|
set :public_folder, "#{dir}/assets"
|
2012-03-04 12:58:16 -08:00
|
|
|
set :views, "#{dir}/views"
|
|
|
|
set :root, "#{dir}/public"
|
2012-03-05 20:53:14 -08:00
|
|
|
set :slim, :pretty => true
|
2012-09-30 14:57:51 +02:00
|
|
|
|
2012-03-05 20:53:14 -08:00
|
|
|
helpers do
|
2012-03-28 21:36:16 +02:00
|
|
|
|
2012-06-05 07:51:18 -07:00
|
|
|
def reset_worker_list
|
|
|
|
Sidekiq.redis do |conn|
|
|
|
|
workers = conn.smembers('workers')
|
|
|
|
workers.each do |name|
|
|
|
|
conn.srem('workers', name)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2012-03-05 20:53:14 -08:00
|
|
|
def workers
|
2012-03-06 20:17:42 -08:00
|
|
|
@workers ||= begin
|
2012-03-17 09:41:24 -07:00
|
|
|
Sidekiq.redis do |conn|
|
2012-03-07 20:22:21 -08:00
|
|
|
conn.smembers('workers').map do |w|
|
|
|
|
msg = conn.get("worker:#{w}")
|
2012-04-24 07:15:29 -07:00
|
|
|
msg ? [w, Sidekiq.load_json(msg)] : nil
|
|
|
|
end.compact.sort { |x| x[1] ? -1 : 1 }
|
2012-03-07 20:22:21 -08:00
|
|
|
end
|
2012-03-06 20:17:42 -08:00
|
|
|
end
|
2012-03-05 20:53:14 -08:00
|
|
|
end
|
2012-03-06 20:17:42 -08:00
|
|
|
|
2012-09-04 09:45:35 -07:00
|
|
|
def info
|
|
|
|
@info ||= Sidekiq.info
|
|
|
|
end
|
|
|
|
|
2012-03-07 20:22:21 -08:00
|
|
|
def processed
|
2012-09-04 09:45:35 -07:00
|
|
|
info[:processed]
|
2012-03-07 20:22:21 -08:00
|
|
|
end
|
|
|
|
|
|
|
|
def failed
|
2012-09-04 09:45:35 -07:00
|
|
|
info[:failed]
|
2012-03-07 20:22:21 -08:00
|
|
|
end
|
|
|
|
|
2012-05-25 20:21:42 -07:00
|
|
|
def zcard(name)
|
|
|
|
Sidekiq.redis { |conn| conn.zcard(name) }
|
2012-03-18 12:29:09 -07:00
|
|
|
end
|
|
|
|
|
2012-03-05 20:53:14 -08:00
|
|
|
def queues
|
2012-09-03 22:41:32 -07:00
|
|
|
@queues ||= Sidekiq.info[:queues_with_sizes]
|
2012-03-05 20:53:14 -08:00
|
|
|
end
|
2012-03-06 20:17:42 -08:00
|
|
|
|
2012-05-22 07:16:08 -07:00
|
|
|
def backlog
|
2012-09-04 09:45:35 -07:00
|
|
|
info[:backlog]
|
2012-05-22 07:16:08 -07:00
|
|
|
end
|
|
|
|
|
2012-03-29 13:48:06 -04:00
|
|
|
def retries_with_score(score)
|
|
|
|
Sidekiq.redis do |conn|
|
2012-03-29 20:55:16 -07:00
|
|
|
results = conn.zrangebyscore('retry', score, score)
|
2012-04-22 14:02:35 -07:00
|
|
|
results.map { |msg| Sidekiq.load_json(msg) }
|
2012-03-29 13:48:06 -04:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2012-03-05 20:53:14 -08:00
|
|
|
def location
|
2012-03-17 09:41:24 -07:00
|
|
|
Sidekiq.redis { |conn| conn.client.location }
|
2012-03-05 20:53:14 -08:00
|
|
|
end
|
2012-03-06 20:17:42 -08:00
|
|
|
|
2012-03-05 20:53:14 -08:00
|
|
|
def root_path
|
|
|
|
"#{env['SCRIPT_NAME']}/"
|
|
|
|
end
|
2012-03-06 20:17:42 -08:00
|
|
|
|
2012-09-30 14:57:51 +02:00
|
|
|
def current_path
|
|
|
|
@current_path ||= request.path_info.gsub(/^\//,'')
|
|
|
|
end
|
|
|
|
|
2012-03-17 09:41:24 -07:00
|
|
|
def current_status
|
2012-04-24 07:15:29 -07:00
|
|
|
return 'idle' if workers.size == 0
|
2012-03-06 20:17:42 -08:00
|
|
|
return 'active'
|
|
|
|
end
|
2012-03-29 16:57:32 -04:00
|
|
|
|
|
|
|
def relative_time(time)
|
2012-03-29 22:24:22 -07:00
|
|
|
%{<time datetime="#{time.getutc.iso8601}">#{time}</time>}
|
2012-03-29 16:57:32 -04:00
|
|
|
end
|
2012-05-11 16:48:03 -07:00
|
|
|
|
|
|
|
def display_args(args, count=100)
|
|
|
|
args.map { |arg| a = arg.inspect; a.size > count ? "#{a[0..count]}..." : a }.join(", ")
|
|
|
|
end
|
2012-08-20 19:28:14 -03:00
|
|
|
|
|
|
|
def tabs
|
2012-08-21 12:06:20 -03:00
|
|
|
self.class.tabs
|
2012-08-20 19:28:14 -03:00
|
|
|
end
|
2012-08-31 10:53:08 -04:00
|
|
|
|
2012-08-31 21:50:19 -07:00
|
|
|
def number_with_delimiter(number)
|
2012-08-31 10:53:08 -04:00
|
|
|
begin
|
|
|
|
Float(number)
|
|
|
|
rescue ArgumentError, TypeError
|
2012-08-31 21:50:19 -07:00
|
|
|
return number
|
2012-08-31 10:53:08 -04:00
|
|
|
end
|
|
|
|
|
2012-08-31 21:50:19 -07:00
|
|
|
options = {:delimiter => ',', :separator => '.'}
|
2012-08-31 10:53:08 -04:00
|
|
|
parts = number.to_s.to_str.split('.')
|
|
|
|
parts[0].gsub!(/(\d)(?=(\d\d\d)+(?!\d))/, "\\1#{options[:delimiter]}")
|
2012-08-31 13:14:00 -04:00
|
|
|
parts.join(options[:separator])
|
2012-08-31 10:53:08 -04:00
|
|
|
end
|
2012-03-05 20:53:14 -08:00
|
|
|
end
|
|
|
|
|
2012-03-04 12:58:16 -08:00
|
|
|
get "/" do
|
|
|
|
slim :index
|
|
|
|
end
|
2012-03-05 20:53:14 -08:00
|
|
|
|
2012-07-26 16:08:15 -07:00
|
|
|
get "/poll" do
|
|
|
|
slim :poll, layout: false
|
|
|
|
end
|
|
|
|
|
2012-06-08 17:20:11 -07:00
|
|
|
get "/queues" do
|
|
|
|
@queues = queues
|
|
|
|
slim :queues
|
|
|
|
end
|
|
|
|
|
2012-03-05 20:53:14 -08:00
|
|
|
get "/queues/:name" do
|
2012-03-17 09:41:24 -07:00
|
|
|
halt 404 unless params[:name]
|
2012-07-17 22:14:15 -07:00
|
|
|
@count = (params[:count] || 25).to_i
|
2012-03-05 20:53:14 -08:00
|
|
|
@name = params[:name]
|
2012-07-17 22:14:15 -07:00
|
|
|
(@current_page, @total_size, @messages) = page("queue:#{@name}", params[:page], @count)
|
|
|
|
@messages = @messages.map {|msg| Sidekiq.load_json(msg) }
|
2012-03-05 20:53:14 -08:00
|
|
|
slim :queue
|
|
|
|
end
|
2012-03-29 13:48:06 -04:00
|
|
|
|
2012-06-05 07:51:18 -07:00
|
|
|
post "/reset" do
|
|
|
|
reset_worker_list
|
|
|
|
redirect root_path
|
|
|
|
end
|
|
|
|
|
2012-05-02 20:19:59 -07:00
|
|
|
post "/queues/:name" do
|
|
|
|
Sidekiq.redis do |conn|
|
|
|
|
conn.del("queue:#{params[:name]}")
|
|
|
|
conn.srem("queues", params[:name])
|
|
|
|
end
|
2012-06-08 17:20:11 -07:00
|
|
|
redirect "#{root_path}queues"
|
2012-05-02 20:19:59 -07:00
|
|
|
end
|
|
|
|
|
2012-10-23 16:11:14 -07:00
|
|
|
post "/queues/:name/delete" do
|
|
|
|
Sidekiq.redis do |conn|
|
|
|
|
conn.lrem("queue:#{params[:name]}", 0, params[:key_val])
|
|
|
|
end
|
|
|
|
redirect "#{root_path}queues/#{params[:name]}"
|
|
|
|
end
|
|
|
|
|
2012-03-29 13:48:06 -04:00
|
|
|
get "/retries/:score" do
|
|
|
|
halt 404 unless params[:score]
|
2012-03-29 20:55:16 -07:00
|
|
|
@score = params[:score].to_f
|
2012-05-04 21:00:48 -07:00
|
|
|
@retries = retries_with_score(@score)
|
2012-05-11 16:48:03 -07:00
|
|
|
redirect "#{root_path}retries" if @retries.empty?
|
2012-03-29 13:48:06 -04:00
|
|
|
slim :retry
|
|
|
|
end
|
2012-03-29 20:55:16 -07:00
|
|
|
|
2012-05-11 16:48:03 -07:00
|
|
|
get '/retries' do
|
2012-07-17 22:14:15 -07:00
|
|
|
@count = (params[:count] || 25).to_i
|
|
|
|
(@current_page, @total_size, @retries) = page("retry", params[:page], @count)
|
|
|
|
@retries = @retries.map {|msg, score| [Sidekiq.load_json(msg), score] }
|
2012-05-11 16:48:03 -07:00
|
|
|
slim :retries
|
|
|
|
end
|
|
|
|
|
2012-05-25 20:21:42 -07:00
|
|
|
get '/scheduled' do
|
2012-07-17 22:14:15 -07:00
|
|
|
@count = (params[:count] || 25).to_i
|
|
|
|
(@current_page, @total_size, @scheduled) = page("schedule", params[:page], @count)
|
|
|
|
@scheduled = @scheduled.map {|msg, score| [Sidekiq.load_json(msg), score] }
|
2012-05-25 20:21:42 -07:00
|
|
|
slim :scheduled
|
|
|
|
end
|
|
|
|
|
|
|
|
post '/scheduled' do
|
|
|
|
halt 404 unless params[:score]
|
|
|
|
halt 404 unless params['delete']
|
|
|
|
params[:score].each do |score|
|
|
|
|
s = score.to_f
|
|
|
|
process_score('schedule', s, :delete)
|
|
|
|
end
|
2012-06-18 15:21:51 +03:00
|
|
|
redirect "#{root_path}scheduled"
|
2012-05-25 20:21:42 -07:00
|
|
|
end
|
|
|
|
|
2012-05-11 16:48:03 -07:00
|
|
|
post '/retries' do
|
|
|
|
halt 404 unless params[:score]
|
|
|
|
params[:score].each do |score|
|
|
|
|
s = score.to_f
|
|
|
|
if params['retry']
|
2012-05-25 20:21:42 -07:00
|
|
|
process_score('retry', s, :retry)
|
2012-05-11 16:48:03 -07:00
|
|
|
elsif params['delete']
|
2012-05-25 20:21:42 -07:00
|
|
|
process_score('retry', s, :delete)
|
2012-05-11 16:48:03 -07:00
|
|
|
end
|
|
|
|
end
|
2012-06-18 14:59:25 +03:00
|
|
|
redirect "#{root_path}retries"
|
2012-05-11 16:48:03 -07:00
|
|
|
end
|
|
|
|
|
2012-03-29 20:55:16 -07:00
|
|
|
post "/retries/:score" do
|
|
|
|
halt 404 unless params[:score]
|
|
|
|
score = params[:score].to_f
|
|
|
|
if params['retry']
|
2012-06-12 21:59:43 -04:00
|
|
|
process_score('retry', score, :retry)
|
2012-05-11 16:48:03 -07:00
|
|
|
elsif params['delete']
|
2012-06-12 21:59:43 -04:00
|
|
|
process_score('retry', score, :delete)
|
2012-05-11 16:48:03 -07:00
|
|
|
end
|
2012-06-18 14:59:25 +03:00
|
|
|
redirect "#{root_path}retries"
|
2012-05-11 16:48:03 -07:00
|
|
|
end
|
|
|
|
|
2012-05-25 20:21:42 -07:00
|
|
|
def process_score(set, score, operation)
|
2012-05-11 16:48:03 -07:00
|
|
|
case operation
|
|
|
|
when :retry
|
2012-03-29 20:55:16 -07:00
|
|
|
Sidekiq.redis do |conn|
|
2012-05-25 20:21:42 -07:00
|
|
|
results = conn.zrangebyscore(set, score, score)
|
|
|
|
conn.zremrangebyscore(set, score, score)
|
2012-03-29 20:55:16 -07:00
|
|
|
results.map do |message|
|
2012-04-22 14:02:35 -07:00
|
|
|
msg = Sidekiq.load_json(message)
|
2012-05-25 15:30:15 -07:00
|
|
|
msg['retry_count'] = msg['retry_count'] - 1
|
|
|
|
conn.rpush("queue:#{msg['queue']}", Sidekiq.dump_json(msg))
|
2012-03-29 20:55:16 -07:00
|
|
|
end
|
|
|
|
end
|
2012-05-11 16:48:03 -07:00
|
|
|
when :delete
|
2012-03-29 20:55:16 -07:00
|
|
|
Sidekiq.redis do |conn|
|
2012-05-25 20:21:42 -07:00
|
|
|
conn.zremrangebyscore(set, score, score)
|
2012-03-29 20:55:16 -07:00
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
2012-05-11 16:48:03 -07:00
|
|
|
|
2012-08-20 19:28:14 -03:00
|
|
|
def self.tabs
|
2012-09-30 14:57:51 +02:00
|
|
|
@tabs ||= {
|
|
|
|
"Workers" =>'',
|
|
|
|
"Queues" =>'queues',
|
|
|
|
"Retries" =>'retries',
|
|
|
|
"Scheduled" =>'scheduled'
|
|
|
|
}
|
2012-08-20 19:28:14 -03:00
|
|
|
end
|
|
|
|
|
2012-03-04 12:58:16 -08:00
|
|
|
end
|
|
|
|
|
|
|
|
end
|
2012-11-02 11:11:34 -07:00
|
|
|
|