require 'sinatra' require 'active_support' require 'active_record' require 'delayed_job' require 'haml' module DelayedJobMonitor class App < Sinatra::Base set :root, File.dirname(__FILE__) set :static, true set :public_folder, File.expand_path('../public', __FILE__) set :views, File.expand_path('../views', __FILE__) set :haml, { :format => :html5 } set :port, 4567 def delayed_job begin if DelayedJobMonitor.connected? DelayedJobMonitor::Job else nil end rescue end end ############################## SINTRA APP ################################ def current_page url_path request.path_info.sub('/','') end def start params[:start].to_i end def per_page 25 end def url_path(*path_parts) [ path_prefix, path_parts ].join("/").squeeze('/') end alias_method :u, :url_path def path_prefix request.env['SCRIPT_NAME'] end def delayed_jobs(type) delayed_job.where(delayed_job_sql(type)) end def delayed_job_sql(type) case type when :enqueued '' when :working 'locked_at is not null' when :failed 'last_error is not null' when :pending 'attempts = 0' end end def partial(template, local_vars = {}) @partial = true haml(template.to_sym, {:layout => false}, local_vars) ensure @partial = false end def poll if @polling text = "Last Updated: #{Time.now.strftime("%H:%M:%S")}" else text = "Live Poll" end "

#{text}

" end def show_for_polling(page) content_type "text/html" @polling = true # show(page.to_sym, false).gsub(/\s{1,}/, ' ') @jobs = delayed_jobs(page.to_sym) haml(page.to_sym, {:layout => false}) end ####################### SINATRA ROUTES/ACTIONS ########################## def tabs [ {:name => 'Overview', :path => '/overview'}, {:name => 'Enqueued', :path => '/enqueued'}, {:name => 'Working', :path => '/working'}, {:name => 'Pending', :path => '/pending'}, {:name => 'Failed', :path => '/failed'} # {:name => 'Servers', :path => '/servers'}, # {:name => 'Workers', :path => '/workers'} ] end get "/?" do redirect u(:overview) end #Static Page Rendering %w(enqueued working pending failed).each do |page| get "/#{page}" do @jobs = delayed_jobs(page.to_sym).order('created_at desc, id desc').offset(start).limit(per_page) @all_jobs = delayed_jobs(page.to_sym) haml page.to_sym end end #Polling Page Rendering %w(overview enqueued working pending failed stats) .each do |page| get "/#{page}.poll" do show_for_polling(page) end get "/#{page}/:id.poll" do show_for_polling(page) end end get '/overview' do if delayed_job haml :overview else @message = "Unable to connected to Delayed::Job database" haml :error end end # get "/servers" do # @servers = delayed_job_monitor::Server.servers # p @servers.class # p @servers.count # p @servers.first # end # get "/workers" do # @workers = delayed_job_monitor::Server.servers.map{|x| x.delayed_job_workers} # end get "/queue/:queue" do @jobs = delayed_job.where(:queue=>params[:queue]).order('created_at desc, id desc').offset(start).limit(per_page) @all_jobs = delayed_job.where(:queue=>params[:queue]).count haml :queue end get "/remove/:id" do delayed_job.find(params[:id]).delete redirect back end get "/requeue/:id" do job = delayed_job.find(params[:id]) job.update_attributes(:run_at =>Time.now,:failed_at=>nil,:locked_at=>nil,:attempts=>0) redirect back end post "/failed/clear" do delayed_job.where("last_error IS NOT NULL").destroy_all redirect back end post "/requeue/all" do delayed_job.where("last_error IS NOT NULL").update_all( :run_at=>Time.now, :failed_at => nil, :attempts=>0, :last_error=>nil, :locked_at=>nil) redirect back end get '/update/:id' do "#{params[:id]}" end post '/update/all' do "#{params}" end post '/kill/:pid' do delayed_job_monitor::Worker.kill_worker(params[:pid]) redirect back end end end