2015-12-31 18:33:35 -05:00
|
|
|
# frozen_string_literal: true
|
2013-08-22 03:32:42 -04:00
|
|
|
require 'erb'
|
2013-04-29 17:56:05 -04:00
|
|
|
require 'yaml'
|
2012-03-04 15:58:16 -05:00
|
|
|
require 'sinatra/base'
|
2013-05-16 11:27:10 -04:00
|
|
|
|
2013-04-29 17:56:05 -04:00
|
|
|
require 'sidekiq'
|
|
|
|
require 'sidekiq/api'
|
2012-07-18 01:14:15 -04:00
|
|
|
require 'sidekiq/paginator'
|
2013-10-13 18:56:17 -04:00
|
|
|
require 'sidekiq/web_helpers'
|
2012-07-18 01:14:15 -04:00
|
|
|
|
2012-03-04 15:58:16 -05:00
|
|
|
module Sidekiq
|
|
|
|
class Web < Sinatra::Base
|
2012-07-18 01:14:15 -04:00
|
|
|
include Sidekiq::Paginator
|
|
|
|
|
2015-07-28 17:58:40 -04:00
|
|
|
enable :sessions
|
2015-12-21 05:24:00 -05:00
|
|
|
use ::Rack::Protection, :use => :authenticity_token unless ENV['RACK_ENV'] == 'test'
|
2015-07-06 15:51:15 -04:00
|
|
|
|
2013-07-28 18:02:43 -04:00
|
|
|
set :root, File.expand_path(File.dirname(__FILE__) + "/../../web")
|
2015-06-10 16:09:16 -04:00
|
|
|
set :public_folder, proc { "#{root}/assets" }
|
|
|
|
set :views, proc { "#{root}/views" }
|
2013-10-21 15:07:18 -04:00
|
|
|
set :locales, ["#{root}/locales"]
|
2012-09-30 08:57:51 -04:00
|
|
|
|
2013-10-13 18:56:17 -04:00
|
|
|
helpers WebHelpers
|
2012-03-06 23:17:42 -05:00
|
|
|
|
2013-10-13 18:56:17 -04:00
|
|
|
DEFAULT_TABS = {
|
|
|
|
"Dashboard" => '',
|
2014-03-07 00:30:11 -05:00
|
|
|
"Busy" => 'busy',
|
2013-10-13 18:56:17 -04:00
|
|
|
"Queues" => 'queues',
|
|
|
|
"Retries" => 'retries',
|
|
|
|
"Scheduled" => 'scheduled',
|
2014-02-09 17:56:01 -05:00
|
|
|
"Dead" => 'morgue',
|
2013-10-13 18:56:17 -04:00
|
|
|
}
|
2013-05-30 00:07:49 -04:00
|
|
|
|
2013-10-13 18:56:17 -04:00
|
|
|
class << self
|
|
|
|
def default_tabs
|
|
|
|
DEFAULT_TABS
|
2013-04-15 13:43:31 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
def custom_tabs
|
2013-10-13 18:56:17 -04:00
|
|
|
@custom_tabs ||= {}
|
2013-08-22 03:32:42 -04:00
|
|
|
end
|
2013-10-13 18:56:17 -04:00
|
|
|
alias_method :tabs, :custom_tabs
|
2014-08-01 19:40:42 -04:00
|
|
|
|
|
|
|
attr_accessor :app_url
|
2012-03-05 23:53:14 -05:00
|
|
|
end
|
|
|
|
|
2014-03-07 00:30:11 -05:00
|
|
|
get "/busy" do
|
|
|
|
erb :busy
|
2012-03-04 15:58:16 -05:00
|
|
|
end
|
2012-03-05 23:53:14 -05:00
|
|
|
|
2014-05-16 00:12:44 -04:00
|
|
|
post "/busy" do
|
2014-12-30 20:25:55 -05:00
|
|
|
if params['identity']
|
|
|
|
p = Sidekiq::Process.new('identity' => params['identity'])
|
2014-06-25 19:59:18 -04:00
|
|
|
p.quiet! if params[:quiet]
|
|
|
|
p.stop! if params[:stop]
|
2014-05-16 00:12:44 -04:00
|
|
|
else
|
2015-01-16 13:59:41 -05:00
|
|
|
processes.each do |pro|
|
2014-05-16 00:12:44 -04:00
|
|
|
pro.quiet! if params[:quiet]
|
|
|
|
pro.stop! if params[:stop]
|
|
|
|
end
|
|
|
|
end
|
|
|
|
redirect "#{root_path}busy"
|
|
|
|
end
|
|
|
|
|
2012-06-08 20:20:11 -04:00
|
|
|
get "/queues" do
|
2013-06-01 17:54:29 -04:00
|
|
|
@queues = Sidekiq::Queue.all
|
2013-08-21 18:38:52 -04:00
|
|
|
erb :queues
|
2012-06-08 20:20:11 -04:00
|
|
|
end
|
|
|
|
|
2012-03-05 23:53:14 -05:00
|
|
|
get "/queues/:name" do
|
2012-03-17 12:41:24 -04:00
|
|
|
halt 404 unless params[:name]
|
2012-07-18 01:14:15 -04:00
|
|
|
@count = (params[:count] || 25).to_i
|
2012-03-05 23:53:14 -05:00
|
|
|
@name = params[:name]
|
2014-04-25 01:16:34 -04:00
|
|
|
@queue = Sidekiq::Queue.new(@name)
|
2012-07-18 01:14:15 -04:00
|
|
|
(@current_page, @total_size, @messages) = page("queue:#{@name}", params[:page], @count)
|
2014-12-17 15:09:20 -05:00
|
|
|
@messages = @messages.map { |msg| Sidekiq::Job.new(msg, @name) }
|
2013-08-21 18:38:52 -04:00
|
|
|
erb :queue
|
2012-03-05 23:53:14 -05:00
|
|
|
end
|
2012-03-29 13:48:06 -04:00
|
|
|
|
2012-05-02 23:19:59 -04:00
|
|
|
post "/queues/:name" do
|
2012-11-25 21:43:48 -05:00
|
|
|
Sidekiq::Queue.new(params[:name]).clear
|
2012-06-08 20:20:11 -04:00
|
|
|
redirect "#{root_path}queues"
|
2012-05-02 23:19:59 -04:00
|
|
|
end
|
|
|
|
|
2012-10-23 19:11:14 -04:00
|
|
|
post "/queues/:name/delete" do
|
2012-12-02 16:23:05 -05:00
|
|
|
Sidekiq::Job.new(params[:key_val], params[:name]).delete
|
2013-10-13 18:56:17 -04:00
|
|
|
redirect_with_query("#{root_path}queues/#{params[:name]}")
|
2012-10-23 19:11:14 -04:00
|
|
|
end
|
|
|
|
|
2014-02-09 17:56:01 -05:00
|
|
|
get '/morgue' do
|
|
|
|
@count = (params[:count] || 25).to_i
|
2014-12-17 15:09:20 -05:00
|
|
|
(@current_page, @total_size, @dead) = page("dead", params[:page], @count, reverse: true)
|
|
|
|
@dead = @dead.map { |msg, score| Sidekiq::SortedEntry.new(nil, score, msg) }
|
2014-02-09 17:56:01 -05:00
|
|
|
erb :morgue
|
|
|
|
end
|
|
|
|
|
|
|
|
get "/morgue/:key" do
|
|
|
|
halt 404 unless params['key']
|
|
|
|
@dead = Sidekiq::DeadSet.new.fetch(*parse_params(params['key'])).first
|
|
|
|
redirect "#{root_path}morgue" if @dead.nil?
|
|
|
|
erb :dead
|
|
|
|
end
|
|
|
|
|
|
|
|
post '/morgue' do
|
2015-07-10 12:29:31 -04:00
|
|
|
redirect request.path unless params['key']
|
2014-02-09 17:56:01 -05:00
|
|
|
|
|
|
|
params['key'].each do |key|
|
|
|
|
job = Sidekiq::DeadSet.new.fetch(*parse_params(key)).first
|
2014-07-28 16:42:06 -04:00
|
|
|
retry_or_delete_or_kill job, params if job
|
2014-02-09 17:56:01 -05:00
|
|
|
end
|
|
|
|
redirect_with_query("#{root_path}morgue")
|
|
|
|
end
|
|
|
|
|
|
|
|
post "/morgue/all/delete" do
|
|
|
|
Sidekiq::DeadSet.new.clear
|
|
|
|
redirect "#{root_path}morgue"
|
|
|
|
end
|
|
|
|
|
|
|
|
post "/morgue/all/retry" do
|
|
|
|
Sidekiq::DeadSet.new.retry_all
|
|
|
|
redirect "#{root_path}morgue"
|
|
|
|
end
|
|
|
|
|
|
|
|
post "/morgue/:key" do
|
|
|
|
halt 404 unless params['key']
|
|
|
|
job = Sidekiq::DeadSet.new.fetch(*parse_params(params['key'])).first
|
2014-07-28 16:42:06 -04:00
|
|
|
retry_or_delete_or_kill job, params if job
|
2014-02-09 17:56:01 -05:00
|
|
|
redirect_with_query("#{root_path}morgue")
|
|
|
|
end
|
|
|
|
|
|
|
|
|
2012-05-11 19:48:03 -04:00
|
|
|
get '/retries' do
|
2012-07-18 01:14:15 -04:00
|
|
|
@count = (params[:count] || 25).to_i
|
|
|
|
(@current_page, @total_size, @retries) = page("retry", params[:page], @count)
|
2014-12-17 15:09:20 -05:00
|
|
|
@retries = @retries.map { |msg, score| Sidekiq::SortedEntry.new(nil, score, msg) }
|
2013-08-21 18:38:52 -04:00
|
|
|
erb :retries
|
2012-05-11 19:48:03 -04:00
|
|
|
end
|
|
|
|
|
2012-11-26 14:53:22 -05:00
|
|
|
get "/retries/:key" do
|
|
|
|
@retry = Sidekiq::RetrySet.new.fetch(*parse_params(params['key'])).first
|
2012-11-26 11:22:48 -05:00
|
|
|
redirect "#{root_path}retries" if @retry.nil?
|
2013-08-21 18:38:52 -04:00
|
|
|
erb :retry
|
2012-05-25 23:21:42 -04:00
|
|
|
end
|
|
|
|
|
2012-05-11 19:48:03 -04:00
|
|
|
post '/retries' do
|
2014-09-16 13:55:50 -04:00
|
|
|
redirect request.path unless params['key']
|
2012-11-26 14:53:22 -05:00
|
|
|
|
|
|
|
params['key'].each do |key|
|
|
|
|
job = Sidekiq::RetrySet.new.fetch(*parse_params(key)).first
|
2014-07-28 16:42:06 -04:00
|
|
|
retry_or_delete_or_kill job, params if job
|
2012-05-11 19:48:03 -04:00
|
|
|
end
|
2013-10-13 18:56:17 -04:00
|
|
|
redirect_with_query("#{root_path}retries")
|
2012-05-11 19:48:03 -04:00
|
|
|
end
|
|
|
|
|
2012-11-25 21:43:48 -05:00
|
|
|
post "/retries/all/delete" do
|
|
|
|
Sidekiq::RetrySet.new.clear
|
|
|
|
redirect "#{root_path}retries"
|
|
|
|
end
|
|
|
|
|
|
|
|
post "/retries/all/retry" do
|
2013-02-19 23:36:59 -05:00
|
|
|
Sidekiq::RetrySet.new.retry_all
|
2012-11-25 21:43:48 -05:00
|
|
|
redirect "#{root_path}retries"
|
2012-05-11 19:48:03 -04:00
|
|
|
end
|
|
|
|
|
2012-11-26 14:53:22 -05:00
|
|
|
post "/retries/:key" do
|
|
|
|
job = Sidekiq::RetrySet.new.fetch(*parse_params(params['key'])).first
|
2014-07-28 16:42:06 -04:00
|
|
|
retry_or_delete_or_kill job, params if job
|
2013-10-13 18:56:17 -04:00
|
|
|
redirect_with_query("#{root_path}retries")
|
2012-05-11 19:48:03 -04:00
|
|
|
end
|
|
|
|
|
2012-11-26 11:22:48 -05:00
|
|
|
get '/scheduled' do
|
|
|
|
@count = (params[:count] || 25).to_i
|
|
|
|
(@current_page, @total_size, @scheduled) = page("schedule", params[:page], @count)
|
2014-12-17 15:09:20 -05:00
|
|
|
@scheduled = @scheduled.map { |msg, score| Sidekiq::SortedEntry.new(nil, score, msg) }
|
2013-08-21 18:38:52 -04:00
|
|
|
erb :scheduled
|
2012-11-26 11:22:48 -05:00
|
|
|
end
|
|
|
|
|
2013-07-27 12:49:21 -04:00
|
|
|
get "/scheduled/:key" do
|
|
|
|
@job = Sidekiq::ScheduledSet.new.fetch(*parse_params(params['key'])).first
|
|
|
|
redirect "#{root_path}scheduled" if @job.nil?
|
2013-08-21 18:38:52 -04:00
|
|
|
erb :scheduled_job_info
|
2013-07-27 12:49:21 -04:00
|
|
|
end
|
|
|
|
|
2012-11-26 11:22:48 -05:00
|
|
|
post '/scheduled' do
|
2014-09-16 13:55:50 -04:00
|
|
|
redirect request.path unless params['key']
|
2013-06-21 22:43:06 -04:00
|
|
|
|
2012-11-26 14:53:22 -05:00
|
|
|
params['key'].each do |key|
|
2013-06-21 22:43:06 -04:00
|
|
|
job = Sidekiq::ScheduledSet.new.fetch(*parse_params(key)).first
|
2014-05-13 13:35:37 -04:00
|
|
|
delete_or_add_queue job, params if job
|
2012-11-26 14:53:22 -05:00
|
|
|
end
|
2013-10-13 18:56:17 -04:00
|
|
|
redirect_with_query("#{root_path}scheduled")
|
2012-03-29 23:55:16 -04:00
|
|
|
end
|
2012-05-11 19:48:03 -04:00
|
|
|
|
2013-07-27 12:49:21 -04:00
|
|
|
post "/scheduled/:key" do
|
|
|
|
halt 404 unless params['key']
|
|
|
|
job = Sidekiq::ScheduledSet.new.fetch(*parse_params(params['key'])).first
|
2014-05-13 13:35:37 -04:00
|
|
|
delete_or_add_queue job, params if job
|
2013-10-13 18:56:17 -04:00
|
|
|
redirect_with_query("#{root_path}scheduled")
|
2013-07-27 12:49:21 -04:00
|
|
|
end
|
|
|
|
|
2013-02-20 11:56:53 -05:00
|
|
|
get '/' do
|
2014-07-27 20:36:12 -04:00
|
|
|
@redis_info = redis_info.select{ |k, v| REDIS_KEYS.include? k }
|
2012-12-05 11:28:55 -05:00
|
|
|
stats_history = Sidekiq::Stats::History.new((params[:days] || 30).to_i)
|
|
|
|
@processed_history = stats_history.processed
|
|
|
|
@failed_history = stats_history.failed
|
2013-08-21 18:38:52 -04:00
|
|
|
erb :dashboard
|
2012-12-05 11:28:55 -05:00
|
|
|
end
|
|
|
|
|
2014-05-09 19:03:50 -04:00
|
|
|
REDIS_KEYS = %w(redis_version uptime_in_days connected_clients used_memory_human used_memory_peak_human)
|
2013-10-13 18:56:17 -04:00
|
|
|
|
2012-12-12 11:49:20 -05:00
|
|
|
get '/dashboard/stats' do
|
2014-12-26 12:46:35 -05:00
|
|
|
redirect "#{root_path}stats"
|
|
|
|
end
|
|
|
|
|
|
|
|
get '/stats' do
|
2013-01-26 14:34:46 -05:00
|
|
|
sidekiq_stats = Sidekiq::Stats.new
|
2014-12-26 12:46:35 -05:00
|
|
|
redis_stats = redis_info.select { |k, v| REDIS_KEYS.include? k }
|
2013-01-26 14:34:46 -05:00
|
|
|
|
2012-12-05 11:28:55 -05:00
|
|
|
content_type :json
|
2014-12-26 12:46:35 -05:00
|
|
|
Sidekiq.dump_json(
|
2013-01-26 14:34:46 -05:00
|
|
|
sidekiq: {
|
2015-01-16 20:54:34 -05:00
|
|
|
processed: sidekiq_stats.processed,
|
|
|
|
failed: sidekiq_stats.failed,
|
|
|
|
busy: sidekiq_stats.workers_size,
|
|
|
|
processes: sidekiq_stats.processes_size,
|
|
|
|
enqueued: sidekiq_stats.enqueued,
|
|
|
|
scheduled: sidekiq_stats.scheduled_size,
|
|
|
|
retries: sidekiq_stats.retry_size,
|
|
|
|
dead: sidekiq_stats.dead_size,
|
|
|
|
default_latency: sidekiq_stats.default_queue_latency
|
2013-01-26 14:34:46 -05:00
|
|
|
},
|
|
|
|
redis: redis_stats
|
2014-12-24 14:02:47 -05:00
|
|
|
)
|
|
|
|
end
|
|
|
|
|
|
|
|
get '/stats/queues' do
|
2015-01-16 20:54:34 -05:00
|
|
|
queue_stats = Sidekiq::Stats::Queues.new
|
2014-12-24 14:02:47 -05:00
|
|
|
|
|
|
|
content_type :json
|
|
|
|
Sidekiq.dump_json(
|
2015-01-16 20:54:34 -05:00
|
|
|
queue_stats.lengths
|
2014-12-24 14:02:47 -05:00
|
|
|
)
|
|
|
|
end
|
|
|
|
|
2014-05-13 13:35:37 -04:00
|
|
|
private
|
|
|
|
|
2014-07-28 16:42:06 -04:00
|
|
|
def retry_or_delete_or_kill job, params
|
2014-05-13 13:35:37 -04:00
|
|
|
if params['retry']
|
|
|
|
job.retry
|
|
|
|
elsif params['delete']
|
|
|
|
job.delete
|
2014-07-28 16:42:06 -04:00
|
|
|
elsif params['kill']
|
|
|
|
job.kill
|
2014-05-13 13:35:37 -04:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def delete_or_add_queue job, params
|
|
|
|
if params['delete']
|
|
|
|
job.delete
|
|
|
|
elsif params['add_to_queue']
|
|
|
|
job.add_to_queue
|
|
|
|
end
|
|
|
|
end
|
2012-03-04 15:58:16 -05:00
|
|
|
end
|
|
|
|
end
|
2015-08-31 13:57:03 -04:00
|
|
|
|
|
|
|
if defined?(::ActionDispatch::Request::Session) &&
|
|
|
|
!::ActionDispatch::Request::Session.respond_to?(:each)
|
|
|
|
# mperham/sidekiq#2460
|
|
|
|
# Rack apps can't reuse the Rails session store without
|
|
|
|
# this monkeypatch
|
|
|
|
class ActionDispatch::Request::Session
|
|
|
|
def each(&block)
|
|
|
|
hash = self.to_hash
|
|
|
|
hash.each(&block)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|