FEATURE: add a simple queue Scheduler::Defer.later {}

For quick jobs that do not need to be sent to sidekiq,
runs inline in a single thread but does not block
This commit is contained in:
Sam 2014-03-17 11:59:34 +11:00
parent fe63db7953
commit 2c8ae22b87
4 changed files with 95 additions and 3 deletions

View File

@ -33,9 +33,12 @@ class Auth::DefaultCurrentUserProvider
if current_user if current_user
Jobs.enqueue(:update_user_info, u = current_user
user_id: current_user.id, Scheduler::Defer.later do
ip: request.ip) u.update_last_seen!
u.update_ip_address!(request.ip)
end
end end
# possible we have an api call, impersonate # possible we have an api call, impersonate

48
lib/scheduler/defer.rb Normal file
View File

@ -0,0 +1,48 @@
module Scheduler
module Deferrable
def initialize
@async = Rails.env != "test"
@queue = Queue.new
@thread = Thread.new {
while true
do_work
end
}
end
# for test
def async=(val)
@async = val
end
def later(&blk)
if @async
@queue << [RailsMultisite::ConnectionManagement.current_db, blk]
else
blk.call
end
end
def stop!
@thread.kill
end
private
def do_work
db, job = @queue.deq
RailsMultisite::ConnectionManagement.establish_connection(db: db)
job.call
rescue => ex
Discourse.handle_exception(ex)
ensure
ActiveRecord::Base.connection_handler.clear_active_connections!
end
end
class Defer
extend Deferrable
initialize
end
end

View File

@ -4,3 +4,4 @@ end
require_dependency 'scheduler/schedule' require_dependency 'scheduler/schedule'
require_dependency 'scheduler/schedule_info' require_dependency 'scheduler/schedule_info'
require_dependency 'scheduler/manager' require_dependency 'scheduler/manager'
require_dependency 'scheduler/defer'

View File

@ -0,0 +1,40 @@
# encoding: utf-8
require 'spec_helper'
require 'scheduler/scheduler'
describe Scheduler::Defer do
class DeferInstance
include Scheduler::Deferrable
end
def wait_for(timeout, &blk)
till = Time.now + (timeout.to_f / 1000)
while Time.now < till && !blk.call
sleep 0.001
end
end
before do
@defer = DeferInstance.new
@defer.async = true
end
after do
@defer.stop!
end
it "can queue jobs properly" do
s = nil
@defer.later do
s = "good"
end
wait_for(10) do
s == "good"
end
s.should == "good"
end
end