qu-delayed 0.1.4
Sign up to get free protection for your applications and to get access to all the features.
- data/.document +5 -0
- data/.rspec +1 -0
- data/Gemfile +21 -0
- data/Gemfile.lock +64 -0
- data/Guardfile +16 -0
- data/LICENSE.txt +20 -0
- data/README.rdoc +40 -0
- data/Rakefile +49 -0
- data/VERSION +1 -0
- data/lib/qu-delayed-immediate.rb +2 -0
- data/lib/qu-delayed-mongo.rb +4 -0
- data/lib/qu-delayed.rb +1 -0
- data/lib/qu/delayed.rb +20 -0
- data/lib/qu/delayed/backend/immediate.rb +15 -0
- data/lib/qu/delayed/backend/mongo.rb +44 -0
- data/lib/qu/delayed/payload.rb +27 -0
- data/lib/qu/delayed/worker.rb +64 -0
- data/qu-delayed.gemspec +99 -0
- data/spec/qu/delayed/backend/immediate_spec.rb +23 -0
- data/spec/qu/delayed/backend/mongo_spec.rb +63 -0
- data/spec/qu/delayed/payload_spec.rb +29 -0
- data/spec/qu/delayed/worker_spec.rb +5 -0
- data/spec/qu/delayed_spec.rb +44 -0
- data/spec/spec_helper.rb +31 -0
- metadata +217 -0
data/.document
ADDED
data/.rspec
ADDED
@@ -0,0 +1 @@
|
|
1
|
+
--color
|
data/Gemfile
ADDED
@@ -0,0 +1,21 @@
|
|
1
|
+
source "http://rubygems.org"
|
2
|
+
|
3
|
+
gem 'qu'
|
4
|
+
|
5
|
+
group :development do
|
6
|
+
gem "rspec", "~> 2.8.0"
|
7
|
+
gem "rdoc", "~> 3.12"
|
8
|
+
gem "bundler", "~> 1.0.0"
|
9
|
+
gem "jeweler", "~> 1.8.3"
|
10
|
+
gem "rcov", ">= 0"
|
11
|
+
|
12
|
+
gem 'timecop', :require => 'timecop'
|
13
|
+
|
14
|
+
gem 'spork'
|
15
|
+
gem 'guard'
|
16
|
+
gem 'guard-rspec'
|
17
|
+
gem 'guard-spork'
|
18
|
+
|
19
|
+
gem 'qu-mongo'
|
20
|
+
gem 'bson_ext', :platforms => :ruby
|
21
|
+
end
|
data/Gemfile.lock
ADDED
@@ -0,0 +1,64 @@
|
|
1
|
+
GEM
|
2
|
+
remote: http://rubygems.org/
|
3
|
+
specs:
|
4
|
+
bson (1.5.2)
|
5
|
+
bson_ext (1.5.2)
|
6
|
+
bson (= 1.5.2)
|
7
|
+
diff-lcs (1.1.3)
|
8
|
+
ffi (1.0.11)
|
9
|
+
git (1.2.5)
|
10
|
+
guard (1.0.0)
|
11
|
+
ffi (>= 0.5.0)
|
12
|
+
thor (~> 0.14.6)
|
13
|
+
guard-rspec (0.6.0)
|
14
|
+
guard (>= 0.10.0)
|
15
|
+
guard-spork (0.5.2)
|
16
|
+
guard (>= 0.10.0)
|
17
|
+
spork (>= 0.8.4)
|
18
|
+
jeweler (1.8.3)
|
19
|
+
bundler (~> 1.0)
|
20
|
+
git (>= 1.2.5)
|
21
|
+
rake
|
22
|
+
rdoc
|
23
|
+
json (1.6.5)
|
24
|
+
mongo (1.5.2)
|
25
|
+
bson (= 1.5.2)
|
26
|
+
multi_json (1.0.4)
|
27
|
+
qu (0.1.4)
|
28
|
+
multi_json
|
29
|
+
qu-mongo (0.1.4)
|
30
|
+
mongo
|
31
|
+
qu (= 0.1.4)
|
32
|
+
rake (0.9.2.2)
|
33
|
+
rcov (0.9.11)
|
34
|
+
rdoc (3.12)
|
35
|
+
json (~> 1.4)
|
36
|
+
rspec (2.8.0)
|
37
|
+
rspec-core (~> 2.8.0)
|
38
|
+
rspec-expectations (~> 2.8.0)
|
39
|
+
rspec-mocks (~> 2.8.0)
|
40
|
+
rspec-core (2.8.0)
|
41
|
+
rspec-expectations (2.8.0)
|
42
|
+
diff-lcs (~> 1.1.2)
|
43
|
+
rspec-mocks (2.8.0)
|
44
|
+
spork (0.9.0)
|
45
|
+
thor (0.14.6)
|
46
|
+
timecop (0.3.5)
|
47
|
+
|
48
|
+
PLATFORMS
|
49
|
+
ruby
|
50
|
+
|
51
|
+
DEPENDENCIES
|
52
|
+
bson_ext
|
53
|
+
bundler (~> 1.0.0)
|
54
|
+
guard
|
55
|
+
guard-rspec
|
56
|
+
guard-spork
|
57
|
+
jeweler (~> 1.8.3)
|
58
|
+
qu
|
59
|
+
qu-mongo
|
60
|
+
rcov
|
61
|
+
rdoc (~> 3.12)
|
62
|
+
rspec (~> 2.8.0)
|
63
|
+
spork
|
64
|
+
timecop
|
data/Guardfile
ADDED
@@ -0,0 +1,16 @@
|
|
1
|
+
# A sample Guardfile
|
2
|
+
# More info at https://github.com/guard/guard#readme
|
3
|
+
|
4
|
+
guard 'spork' do
|
5
|
+
watch('Gemfile')
|
6
|
+
watch('Gemfile.lock')
|
7
|
+
watch('spec/spec_helper.rb') { :rspec }
|
8
|
+
end
|
9
|
+
|
10
|
+
guard 'rspec', :version => 2, :cli => '--drb' do
|
11
|
+
watch(%r{^spec/.+_spec\.rb$})
|
12
|
+
watch(%r{^lib/(.+)\.rb$}) { |m| "spec/#{m[1]}_spec.rb" }
|
13
|
+
watch('spec/spec_helper.rb') { "spec" }
|
14
|
+
watch(%r{^spec/support/(.+)\.rb$}) { "spec" }
|
15
|
+
end
|
16
|
+
|
data/LICENSE.txt
ADDED
@@ -0,0 +1,20 @@
|
|
1
|
+
Copyright (c) 2012 Viacheslav Molokov
|
2
|
+
|
3
|
+
Permission is hereby granted, free of charge, to any person obtaining
|
4
|
+
a copy of this software and associated documentation files (the
|
5
|
+
"Software"), to deal in the Software without restriction, including
|
6
|
+
without limitation the rights to use, copy, modify, merge, publish,
|
7
|
+
distribute, sublicense, and/or sell copies of the Software, and to
|
8
|
+
permit persons to whom the Software is furnished to do so, subject to
|
9
|
+
the following conditions:
|
10
|
+
|
11
|
+
The above copyright notice and this permission notice shall be
|
12
|
+
included in all copies or substantial portions of the Software.
|
13
|
+
|
14
|
+
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
|
15
|
+
EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
16
|
+
MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
|
17
|
+
NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
|
18
|
+
LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
|
19
|
+
OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
|
20
|
+
WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
data/README.rdoc
ADDED
@@ -0,0 +1,40 @@
|
|
1
|
+
= qu-delayed
|
2
|
+
|
3
|
+
Qu::Delayed is basic sheduler for Qu queue system.
|
4
|
+
|
5
|
+
For now it supports only mongodb backend. Contribution for redis are welcome!
|
6
|
+
|
7
|
+
== Install
|
8
|
+
|
9
|
+
To install just add into your `Gemfile`
|
10
|
+
|
11
|
+
gem 'qu-delayed'
|
12
|
+
|
13
|
+
== Usage
|
14
|
+
|
15
|
+
With Qu::Delayed you can:
|
16
|
+
|
17
|
+
* enqueue job to run _in_ some time (ex: enqueue rebuild job in 5 minutes)
|
18
|
+
* enqueue job to run _at_ some time (ex: enqueue refresh stats job at 01:00AM)
|
19
|
+
|
20
|
+
=== Enqueue job in some time
|
21
|
+
|
22
|
+
It's as simple as:
|
23
|
+
|
24
|
+
Qu.enqueue_in(5.minutes, Rebuild, 'qu-delayed')
|
25
|
+
|
26
|
+
== Contributing to qu-delayed
|
27
|
+
|
28
|
+
* Check out the latest master to make sure the feature hasn't been implemented or the bug hasn't been fixed yet.
|
29
|
+
* Check out the issue tracker to make sure someone already hasn't requested it and/or contributed it.
|
30
|
+
* Fork the project.
|
31
|
+
* Start a feature/bugfix branch.
|
32
|
+
* Commit and push until you are happy with your contribution.
|
33
|
+
* Make sure to add tests for it. This is important so I don't break it in a future version unintentionally.
|
34
|
+
* Please try not to mess with the Rakefile, version, or history. If you want to have your own version, or is otherwise necessary, that is fine, but please isolate to its own commit so I can cherry-pick around it.
|
35
|
+
|
36
|
+
== Copyright
|
37
|
+
|
38
|
+
Copyright (c) 2012 Viacheslav Molokov. See LICENSE.txt for
|
39
|
+
further details.
|
40
|
+
|
data/Rakefile
ADDED
@@ -0,0 +1,49 @@
|
|
1
|
+
# encoding: utf-8
|
2
|
+
|
3
|
+
require 'rubygems'
|
4
|
+
require 'bundler'
|
5
|
+
begin
|
6
|
+
Bundler.setup(:default, :development)
|
7
|
+
rescue Bundler::BundlerError => e
|
8
|
+
$stderr.puts e.message
|
9
|
+
$stderr.puts "Run `bundle install` to install missing gems"
|
10
|
+
exit e.status_code
|
11
|
+
end
|
12
|
+
require 'rake'
|
13
|
+
|
14
|
+
require 'jeweler'
|
15
|
+
Jeweler::Tasks.new do |gem|
|
16
|
+
# gem is a Gem::Specification... see http://docs.rubygems.org/read/chapter/20 for more options
|
17
|
+
gem.name = "qu-delayed"
|
18
|
+
gem.homepage = "http://github.com/imomoisoft/qu-delayed"
|
19
|
+
gem.license = "MIT"
|
20
|
+
gem.summary = %Q{Qu::Delayed is basic sheduler for Qu queue system.}
|
21
|
+
gem.description = %Q{Qu::Delayed is basic sheduler for Qu queue system.}
|
22
|
+
gem.email = "viacheslav.molokov@gmail.com"
|
23
|
+
gem.authors = ["Viacheslav Molokov"]
|
24
|
+
# dependencies defined in Gemfile
|
25
|
+
end
|
26
|
+
Jeweler::RubygemsDotOrgTasks.new
|
27
|
+
|
28
|
+
require 'rspec/core'
|
29
|
+
require 'rspec/core/rake_task'
|
30
|
+
RSpec::Core::RakeTask.new(:spec) do |spec|
|
31
|
+
spec.pattern = FileList['spec/**/*_spec.rb']
|
32
|
+
end
|
33
|
+
|
34
|
+
RSpec::Core::RakeTask.new(:rcov) do |spec|
|
35
|
+
spec.pattern = 'spec/**/*_spec.rb'
|
36
|
+
spec.rcov = true
|
37
|
+
end
|
38
|
+
|
39
|
+
task :default => :spec
|
40
|
+
|
41
|
+
require 'rdoc/task'
|
42
|
+
Rake::RDocTask.new do |rdoc|
|
43
|
+
version = File.exist?('VERSION') ? File.read('VERSION') : ""
|
44
|
+
|
45
|
+
rdoc.rdoc_dir = 'rdoc'
|
46
|
+
rdoc.title = "qu-delayed #{version}"
|
47
|
+
rdoc.rdoc_files.include('README*')
|
48
|
+
rdoc.rdoc_files.include('lib/**/*.rb')
|
49
|
+
end
|
data/VERSION
ADDED
@@ -0,0 +1 @@
|
|
1
|
+
0.1.4
|
data/lib/qu-delayed.rb
ADDED
@@ -0,0 +1 @@
|
|
1
|
+
require 'qu/delayed'
|
data/lib/qu/delayed.rb
ADDED
@@ -0,0 +1,20 @@
|
|
1
|
+
require 'qu'
|
2
|
+
|
3
|
+
require 'qu/delayed/payload'
|
4
|
+
|
5
|
+
module Qu
|
6
|
+
module Delayed
|
7
|
+
autoload :Worker, 'qu/delayed/worker'
|
8
|
+
# Enqueues job to run at given time.
|
9
|
+
def enqueue_at(run_at, klass, *args)
|
10
|
+
backend.enqueue_at Qu::Delayed::Payload.new(:klass => klass, :run_at => run_at, :args => args)
|
11
|
+
end
|
12
|
+
|
13
|
+
# Enqueues job to run in +run_in+ seconds from now.
|
14
|
+
def enqueue_in(run_in, klass, *args)
|
15
|
+
backend.enqueue_at Qu::Delayed::Payload.new(:klass => klass, :run_at => Time.now + run_in, :args => args)
|
16
|
+
end
|
17
|
+
end
|
18
|
+
|
19
|
+
extend Delayed
|
20
|
+
end
|
@@ -0,0 +1,15 @@
|
|
1
|
+
require 'qu/backend/immediate'
|
2
|
+
|
3
|
+
module Qu
|
4
|
+
module Delayed
|
5
|
+
module Backend
|
6
|
+
module Immediate
|
7
|
+
def enqueue_at(payload)
|
8
|
+
payload.perform
|
9
|
+
end
|
10
|
+
end
|
11
|
+
end
|
12
|
+
end
|
13
|
+
end
|
14
|
+
|
15
|
+
Qu::Backend::Immediate.send :include, Qu::Delayed::Backend::Immediate
|
@@ -0,0 +1,44 @@
|
|
1
|
+
require 'qu/backend/mongo'
|
2
|
+
|
3
|
+
module Qu
|
4
|
+
module Delayed
|
5
|
+
module Backend
|
6
|
+
module Mongo
|
7
|
+
def enqueue_at(payload)
|
8
|
+
payload_id = BSON::ObjectId.from_time(payload.run_at, :unique => true)
|
9
|
+
delayed_jobs.insert({
|
10
|
+
:_id => payload_id,
|
11
|
+
:klass => payload.klass.to_s,
|
12
|
+
:queue => payload.queue,
|
13
|
+
:args => payload.args
|
14
|
+
})
|
15
|
+
logger.debug { "Enqueued delayed job #{payload}" }
|
16
|
+
payload
|
17
|
+
end
|
18
|
+
|
19
|
+
# Retrieves next delayed job.
|
20
|
+
#
|
21
|
+
# If there is no job to enqueue returns +nil+.
|
22
|
+
def next_delayed_job
|
23
|
+
doc = delayed_jobs.find_and_modify(:query => {:_id => {'$lte' => BSON::ObjectId.from_time(Time.now) }}, :remove => true, :safe => true)
|
24
|
+
return nil if doc.nil?
|
25
|
+
|
26
|
+
Qu::Delayed::Payload.new(doc).undelay
|
27
|
+
rescue ::Mongo::OperationFailure
|
28
|
+
nil
|
29
|
+
end
|
30
|
+
|
31
|
+
def delayed_jobs
|
32
|
+
self[:delayed_jobs]
|
33
|
+
end
|
34
|
+
|
35
|
+
def clear_delayed
|
36
|
+
logger.info { "Clearing delayed jobs queue" }
|
37
|
+
delayed_jobs.drop
|
38
|
+
end
|
39
|
+
end
|
40
|
+
end
|
41
|
+
end
|
42
|
+
end
|
43
|
+
|
44
|
+
Qu::Backend::Mongo.send :include, Qu::Delayed::Backend::Mongo
|
@@ -0,0 +1,27 @@
|
|
1
|
+
require 'ostruct'
|
2
|
+
|
3
|
+
module Qu
|
4
|
+
module Delayed
|
5
|
+
class Payload < Qu::Payload
|
6
|
+
|
7
|
+
def initialize(options = {})
|
8
|
+
super
|
9
|
+
unless _id.nil?
|
10
|
+
self.run_at ||= _id.generation_time
|
11
|
+
end
|
12
|
+
end
|
13
|
+
|
14
|
+
def perform
|
15
|
+
Qu.enqueue(klass, args)
|
16
|
+
end
|
17
|
+
|
18
|
+
def to_s
|
19
|
+
"#{run_at.to_i}:#{super}"
|
20
|
+
end
|
21
|
+
|
22
|
+
def undelay
|
23
|
+
Qu::Payload.new(:klass => klass, :args => args)
|
24
|
+
end
|
25
|
+
end
|
26
|
+
end
|
27
|
+
end
|
@@ -0,0 +1,64 @@
|
|
1
|
+
require 'qu-delayed'
|
2
|
+
|
3
|
+
module Qu
|
4
|
+
module Delayed
|
5
|
+
class Worker
|
6
|
+
include Qu::Logger
|
7
|
+
|
8
|
+
class Abort < Exception
|
9
|
+
end
|
10
|
+
|
11
|
+
def poll_frequency
|
12
|
+
Qu.backend.poll_frequency
|
13
|
+
end
|
14
|
+
|
15
|
+
def handle_signals
|
16
|
+
logger.debug "Sheduler #{id} registering traps for INT and TERM signals"
|
17
|
+
%W(INT TERM).each do |sig|
|
18
|
+
trap(sig) do
|
19
|
+
logger.info "Sheduler #{id} received #{sig}, shutting down"
|
20
|
+
raise Abort
|
21
|
+
end
|
22
|
+
end
|
23
|
+
end
|
24
|
+
|
25
|
+
def work(options = {:block => true})
|
26
|
+
loop do
|
27
|
+
logger.debug { "Getting next delayed job" }
|
28
|
+
|
29
|
+
if payload = Qu.backend.next_delayed_job
|
30
|
+
return Qu.backend.enqueue(payload)
|
31
|
+
end
|
32
|
+
|
33
|
+
if options[:block]
|
34
|
+
sleep poll_frequency
|
35
|
+
else
|
36
|
+
break
|
37
|
+
end
|
38
|
+
end
|
39
|
+
end
|
40
|
+
|
41
|
+
def start
|
42
|
+
logger.warn "Sheduler #{id} starting"
|
43
|
+
handle_signals
|
44
|
+
loop { work }
|
45
|
+
rescue Abort => e
|
46
|
+
# Ok, we'll shut down, but give us a sec
|
47
|
+
ensure
|
48
|
+
logger.debug "Sheduler #{id} done"
|
49
|
+
end
|
50
|
+
|
51
|
+
def id
|
52
|
+
@id ||= "#{hostname}:#{pid}"
|
53
|
+
end
|
54
|
+
|
55
|
+
def pid
|
56
|
+
@pid ||= Process.pid
|
57
|
+
end
|
58
|
+
|
59
|
+
def hostname
|
60
|
+
@hostname ||= `hostname`.strip
|
61
|
+
end
|
62
|
+
end
|
63
|
+
end
|
64
|
+
end
|
data/qu-delayed.gemspec
ADDED
@@ -0,0 +1,99 @@
|
|
1
|
+
# Generated by jeweler
|
2
|
+
# DO NOT EDIT THIS FILE DIRECTLY
|
3
|
+
# Instead, edit Jeweler::Tasks in Rakefile, and run 'rake gemspec'
|
4
|
+
# -*- encoding: utf-8 -*-
|
5
|
+
|
6
|
+
Gem::Specification.new do |s|
|
7
|
+
s.name = "qu-delayed"
|
8
|
+
s.version = "0.1.4"
|
9
|
+
|
10
|
+
s.required_rubygems_version = Gem::Requirement.new(">= 0") if s.respond_to? :required_rubygems_version=
|
11
|
+
s.authors = ["Viacheslav Molokov"]
|
12
|
+
s.date = "2012-02-12"
|
13
|
+
s.description = "Qu::Delayed is basic sheduler for Qu queue system."
|
14
|
+
s.email = "viacheslav.molokov@gmail.com"
|
15
|
+
s.extra_rdoc_files = [
|
16
|
+
"LICENSE.txt",
|
17
|
+
"README.rdoc"
|
18
|
+
]
|
19
|
+
s.files = [
|
20
|
+
".document",
|
21
|
+
".rspec",
|
22
|
+
"Gemfile",
|
23
|
+
"Gemfile.lock",
|
24
|
+
"Guardfile",
|
25
|
+
"LICENSE.txt",
|
26
|
+
"README.rdoc",
|
27
|
+
"Rakefile",
|
28
|
+
"VERSION",
|
29
|
+
"lib/qu-delayed-immediate.rb",
|
30
|
+
"lib/qu-delayed-mongo.rb",
|
31
|
+
"lib/qu-delayed.rb",
|
32
|
+
"lib/qu/delayed.rb",
|
33
|
+
"lib/qu/delayed/backend/immediate.rb",
|
34
|
+
"lib/qu/delayed/backend/mongo.rb",
|
35
|
+
"lib/qu/delayed/payload.rb",
|
36
|
+
"lib/qu/delayed/worker.rb",
|
37
|
+
"qu-delayed.gemspec",
|
38
|
+
"spec/qu/delayed/backend/immediate_spec.rb",
|
39
|
+
"spec/qu/delayed/backend/mongo_spec.rb",
|
40
|
+
"spec/qu/delayed/payload_spec.rb",
|
41
|
+
"spec/qu/delayed/worker_spec.rb",
|
42
|
+
"spec/qu/delayed_spec.rb",
|
43
|
+
"spec/spec_helper.rb"
|
44
|
+
]
|
45
|
+
s.homepage = "http://github.com/imomoisoft/qu-delayed"
|
46
|
+
s.licenses = ["MIT"]
|
47
|
+
s.require_paths = ["lib"]
|
48
|
+
s.rubygems_version = "1.8.10"
|
49
|
+
s.summary = "Qu::Delayed is basic sheduler for Qu queue system."
|
50
|
+
|
51
|
+
if s.respond_to? :specification_version then
|
52
|
+
s.specification_version = 3
|
53
|
+
|
54
|
+
if Gem::Version.new(Gem::VERSION) >= Gem::Version.new('1.2.0') then
|
55
|
+
s.add_runtime_dependency(%q<qu>, [">= 0"])
|
56
|
+
s.add_development_dependency(%q<rspec>, ["~> 2.8.0"])
|
57
|
+
s.add_development_dependency(%q<rdoc>, ["~> 3.12"])
|
58
|
+
s.add_development_dependency(%q<bundler>, ["~> 1.0.0"])
|
59
|
+
s.add_development_dependency(%q<jeweler>, ["~> 1.8.3"])
|
60
|
+
s.add_development_dependency(%q<rcov>, [">= 0"])
|
61
|
+
s.add_development_dependency(%q<timecop>, [">= 0"])
|
62
|
+
s.add_development_dependency(%q<spork>, [">= 0"])
|
63
|
+
s.add_development_dependency(%q<guard>, [">= 0"])
|
64
|
+
s.add_development_dependency(%q<guard-rspec>, [">= 0"])
|
65
|
+
s.add_development_dependency(%q<guard-spork>, [">= 0"])
|
66
|
+
s.add_development_dependency(%q<qu-mongo>, [">= 0"])
|
67
|
+
s.add_development_dependency(%q<bson_ext>, [">= 0"])
|
68
|
+
else
|
69
|
+
s.add_dependency(%q<qu>, [">= 0"])
|
70
|
+
s.add_dependency(%q<rspec>, ["~> 2.8.0"])
|
71
|
+
s.add_dependency(%q<rdoc>, ["~> 3.12"])
|
72
|
+
s.add_dependency(%q<bundler>, ["~> 1.0.0"])
|
73
|
+
s.add_dependency(%q<jeweler>, ["~> 1.8.3"])
|
74
|
+
s.add_dependency(%q<rcov>, [">= 0"])
|
75
|
+
s.add_dependency(%q<timecop>, [">= 0"])
|
76
|
+
s.add_dependency(%q<spork>, [">= 0"])
|
77
|
+
s.add_dependency(%q<guard>, [">= 0"])
|
78
|
+
s.add_dependency(%q<guard-rspec>, [">= 0"])
|
79
|
+
s.add_dependency(%q<guard-spork>, [">= 0"])
|
80
|
+
s.add_dependency(%q<qu-mongo>, [">= 0"])
|
81
|
+
s.add_dependency(%q<bson_ext>, [">= 0"])
|
82
|
+
end
|
83
|
+
else
|
84
|
+
s.add_dependency(%q<qu>, [">= 0"])
|
85
|
+
s.add_dependency(%q<rspec>, ["~> 2.8.0"])
|
86
|
+
s.add_dependency(%q<rdoc>, ["~> 3.12"])
|
87
|
+
s.add_dependency(%q<bundler>, ["~> 1.0.0"])
|
88
|
+
s.add_dependency(%q<jeweler>, ["~> 1.8.3"])
|
89
|
+
s.add_dependency(%q<rcov>, [">= 0"])
|
90
|
+
s.add_dependency(%q<timecop>, [">= 0"])
|
91
|
+
s.add_dependency(%q<spork>, [">= 0"])
|
92
|
+
s.add_dependency(%q<guard>, [">= 0"])
|
93
|
+
s.add_dependency(%q<guard-rspec>, [">= 0"])
|
94
|
+
s.add_dependency(%q<guard-spork>, [">= 0"])
|
95
|
+
s.add_dependency(%q<qu-mongo>, [">= 0"])
|
96
|
+
s.add_dependency(%q<bson_ext>, [">= 0"])
|
97
|
+
end
|
98
|
+
end
|
99
|
+
|
@@ -0,0 +1,23 @@
|
|
1
|
+
require 'spec_helper'
|
2
|
+
|
3
|
+
require 'qu-delayed-immediate'
|
4
|
+
|
5
|
+
describe Qu::Backend::Immediate do
|
6
|
+
let(:payload) { Qu::Delayed::Payload.new(:klass => SimpleJob) }
|
7
|
+
|
8
|
+
before(:all) do
|
9
|
+
Qu.backend = described_class.new
|
10
|
+
end
|
11
|
+
|
12
|
+
context 'should be extended with Qu::Backend::Delayed::Immediate' do
|
13
|
+
it '#enqueue_at' do
|
14
|
+
subject.should respond_to(:enqueue_at)
|
15
|
+
end
|
16
|
+
end
|
17
|
+
|
18
|
+
it 'performs immediately' do
|
19
|
+
payload.should_receive(:perform)
|
20
|
+
subject.enqueue_at(payload)
|
21
|
+
end
|
22
|
+
|
23
|
+
end
|
@@ -0,0 +1,63 @@
|
|
1
|
+
require 'spec_helper'
|
2
|
+
|
3
|
+
require 'qu-delayed-mongo'
|
4
|
+
|
5
|
+
describe Qu::Backend::Mongo do
|
6
|
+
before do
|
7
|
+
ENV.delete('MONGOHQ_URL')
|
8
|
+
subject.logger.level = 3
|
9
|
+
subject.clear
|
10
|
+
subject.clear_delayed
|
11
|
+
end
|
12
|
+
|
13
|
+
let(:payload) { Qu::Delayed::Payload.new(:run_at => Time.now + 5 * 60, :klass => SimpleJob, :args => [8,9]) }
|
14
|
+
|
15
|
+
before(:all) do
|
16
|
+
Qu.backend = described_class.new
|
17
|
+
end
|
18
|
+
|
19
|
+
context 'should be extended with Qu::Backend::Delayed::Mongo' do
|
20
|
+
it '#enqueue_at' do
|
21
|
+
subject.should respond_to(:enqueue_at)
|
22
|
+
end
|
23
|
+
end
|
24
|
+
|
25
|
+
it 'should be able to enqueue any amount of tasks to same time' do
|
26
|
+
subject.enqueue_at(payload)
|
27
|
+
subject.enqueue_at(payload)
|
28
|
+
subject.delayed_jobs.count.should == 2
|
29
|
+
end
|
30
|
+
|
31
|
+
describe 'enqueued job' do
|
32
|
+
before { subject.enqueue_at(payload) }
|
33
|
+
let(:delayed_payload) { Qu::Delayed::Payload.new(subject.delayed_jobs.find_one) }
|
34
|
+
|
35
|
+
it 'should have run_at field deserialized from _id' do
|
36
|
+
delayed_payload.run_at.to_i.should == payload.run_at.to_i
|
37
|
+
end
|
38
|
+
|
39
|
+
it 'should have klass value deserialized' do
|
40
|
+
delayed_payload.klass.should == payload.klass
|
41
|
+
end
|
42
|
+
|
43
|
+
it 'should have args value deserialized' do
|
44
|
+
delayed_payload.args.should == payload.args
|
45
|
+
end
|
46
|
+
end
|
47
|
+
|
48
|
+
describe 'sheduler' do
|
49
|
+
it 'should enqueue job when time comes' do
|
50
|
+
payload.run_at = Time.now - 5*60
|
51
|
+
subject.enqueue_at(payload)
|
52
|
+
subject.next_delayed_job.should_not be_nil
|
53
|
+
subject.delayed_jobs.count.should == 0
|
54
|
+
end
|
55
|
+
|
56
|
+
it 'should enqueue job when time in the future' do
|
57
|
+
payload.run_at = Time.now + 5*60
|
58
|
+
subject.enqueue_at(payload)
|
59
|
+
subject.next_delayed_job.should be_nil
|
60
|
+
subject.delayed_jobs.count.should == 1
|
61
|
+
end
|
62
|
+
end
|
63
|
+
end
|
@@ -0,0 +1,29 @@
|
|
1
|
+
require 'spec_helper'
|
2
|
+
require 'timecop'
|
3
|
+
require 'mongo'
|
4
|
+
|
5
|
+
describe Qu::Delayed::Payload do
|
6
|
+
before do
|
7
|
+
Timecop.freeze(Time.now)
|
8
|
+
end
|
9
|
+
|
10
|
+
after do
|
11
|
+
Timecop.return
|
12
|
+
end
|
13
|
+
|
14
|
+
|
15
|
+
|
16
|
+
subject { Qu::Delayed::Payload.new(:klass => SimpleJob, :run_at => Time.now, :args => [8,9]) }
|
17
|
+
|
18
|
+
it 'should enqueue it to normal queue' do
|
19
|
+
Qu.should_receive(:enqueue).with(SimpleJob, [8,9])
|
20
|
+
subject.perform
|
21
|
+
end
|
22
|
+
|
23
|
+
it 'should overload to_s' do
|
24
|
+
subject.id = BSON::ObjectId.new
|
25
|
+
subject.to_s.should == "#{Time.now.to_i}:#{subject.id}:#{subject.klass}:#{subject.args.inspect}"
|
26
|
+
end
|
27
|
+
|
28
|
+
|
29
|
+
end
|
@@ -0,0 +1,44 @@
|
|
1
|
+
require 'spec_helper'
|
2
|
+
|
3
|
+
describe Qu::Delayed do
|
4
|
+
context 'should extend Qu with' do
|
5
|
+
it 'enqueue_at' do
|
6
|
+
Qu.should respond_to(:enqueue_at)
|
7
|
+
end
|
8
|
+
|
9
|
+
it 'enqueue_in' do
|
10
|
+
Qu.should respond_to(:enqueue_in)
|
11
|
+
end
|
12
|
+
end
|
13
|
+
|
14
|
+
describe '#enqueue_at' do
|
15
|
+
it 'should send correct message to backend' do
|
16
|
+
run_at = Time.now
|
17
|
+
Qu.backend.should_receive(:enqueue_at) do |payload|
|
18
|
+
payload.should be_instance_of(Qu::Delayed::Payload)
|
19
|
+
payload.klass.should == SimpleJob
|
20
|
+
payload.args.should == [9,8]
|
21
|
+
payload.run_at = run_at
|
22
|
+
end
|
23
|
+
|
24
|
+
Qu.enqueue_at run_at, SimpleJob, 9, 8
|
25
|
+
end
|
26
|
+
end
|
27
|
+
|
28
|
+
describe '#enqueue_in' do
|
29
|
+
before {Timecop.freeze(Time.now)}
|
30
|
+
after {Timecop.return}
|
31
|
+
|
32
|
+
it 'should send correct message to backend' do
|
33
|
+
Qu.backend.should_receive(:enqueue_at) do |payload|
|
34
|
+
payload.should be_instance_of(Qu::Delayed::Payload)
|
35
|
+
payload.klass.should == SimpleJob
|
36
|
+
payload.args.should == [9,8]
|
37
|
+
payload.run_at = Time.now + 5*60
|
38
|
+
end
|
39
|
+
|
40
|
+
Qu.enqueue_in 5*60, SimpleJob, 9, 8
|
41
|
+
end
|
42
|
+
end
|
43
|
+
|
44
|
+
end
|
data/spec/spec_helper.rb
ADDED
@@ -0,0 +1,31 @@
|
|
1
|
+
require 'rubygems'
|
2
|
+
require 'spork'
|
3
|
+
#uncomment the following line to use spork with the debugger
|
4
|
+
#require 'spork/ext/ruby-debug'
|
5
|
+
|
6
|
+
Spork.prefork do
|
7
|
+
$LOAD_PATH.unshift(File.join(File.dirname(__FILE__), '..', 'lib'))
|
8
|
+
$LOAD_PATH.unshift(File.dirname(__FILE__))
|
9
|
+
require 'rspec'
|
10
|
+
require 'qu/backend/spec'
|
11
|
+
require 'timecop'
|
12
|
+
|
13
|
+
# Requires supporting files with custom matchers and macros, etc,
|
14
|
+
# in ./support/ and its subdirectories.
|
15
|
+
Dir["#{File.dirname(__FILE__)}/support/**/*.rb"].each {|f| require f}
|
16
|
+
|
17
|
+
RSpec.configure do |config|
|
18
|
+
config.mock_with :rspec
|
19
|
+
|
20
|
+
config.before do
|
21
|
+
Qu.backend = mock('a backend', :reserve => nil, :failed => nil, :completed => nil,
|
22
|
+
:register_worker => nil, :unregister_worker => nil)
|
23
|
+
Qu.failure = nil
|
24
|
+
end
|
25
|
+
end
|
26
|
+
end
|
27
|
+
|
28
|
+
Spork.each_run do
|
29
|
+
require 'qu-delayed'
|
30
|
+
end
|
31
|
+
|
metadata
ADDED
@@ -0,0 +1,217 @@
|
|
1
|
+
--- !ruby/object:Gem::Specification
|
2
|
+
name: qu-delayed
|
3
|
+
version: !ruby/object:Gem::Version
|
4
|
+
version: 0.1.4
|
5
|
+
prerelease:
|
6
|
+
platform: ruby
|
7
|
+
authors:
|
8
|
+
- Viacheslav Molokov
|
9
|
+
autorequire:
|
10
|
+
bindir: bin
|
11
|
+
cert_chain: []
|
12
|
+
date: 2012-02-12 00:00:00.000000000Z
|
13
|
+
dependencies:
|
14
|
+
- !ruby/object:Gem::Dependency
|
15
|
+
name: qu
|
16
|
+
requirement: &13420180 !ruby/object:Gem::Requirement
|
17
|
+
none: false
|
18
|
+
requirements:
|
19
|
+
- - ! '>='
|
20
|
+
- !ruby/object:Gem::Version
|
21
|
+
version: '0'
|
22
|
+
type: :runtime
|
23
|
+
prerelease: false
|
24
|
+
version_requirements: *13420180
|
25
|
+
- !ruby/object:Gem::Dependency
|
26
|
+
name: rspec
|
27
|
+
requirement: &13419060 !ruby/object:Gem::Requirement
|
28
|
+
none: false
|
29
|
+
requirements:
|
30
|
+
- - ~>
|
31
|
+
- !ruby/object:Gem::Version
|
32
|
+
version: 2.8.0
|
33
|
+
type: :development
|
34
|
+
prerelease: false
|
35
|
+
version_requirements: *13419060
|
36
|
+
- !ruby/object:Gem::Dependency
|
37
|
+
name: rdoc
|
38
|
+
requirement: &13418240 !ruby/object:Gem::Requirement
|
39
|
+
none: false
|
40
|
+
requirements:
|
41
|
+
- - ~>
|
42
|
+
- !ruby/object:Gem::Version
|
43
|
+
version: '3.12'
|
44
|
+
type: :development
|
45
|
+
prerelease: false
|
46
|
+
version_requirements: *13418240
|
47
|
+
- !ruby/object:Gem::Dependency
|
48
|
+
name: bundler
|
49
|
+
requirement: &13417280 !ruby/object:Gem::Requirement
|
50
|
+
none: false
|
51
|
+
requirements:
|
52
|
+
- - ~>
|
53
|
+
- !ruby/object:Gem::Version
|
54
|
+
version: 1.0.0
|
55
|
+
type: :development
|
56
|
+
prerelease: false
|
57
|
+
version_requirements: *13417280
|
58
|
+
- !ruby/object:Gem::Dependency
|
59
|
+
name: jeweler
|
60
|
+
requirement: &13416400 !ruby/object:Gem::Requirement
|
61
|
+
none: false
|
62
|
+
requirements:
|
63
|
+
- - ~>
|
64
|
+
- !ruby/object:Gem::Version
|
65
|
+
version: 1.8.3
|
66
|
+
type: :development
|
67
|
+
prerelease: false
|
68
|
+
version_requirements: *13416400
|
69
|
+
- !ruby/object:Gem::Dependency
|
70
|
+
name: rcov
|
71
|
+
requirement: &13415240 !ruby/object:Gem::Requirement
|
72
|
+
none: false
|
73
|
+
requirements:
|
74
|
+
- - ! '>='
|
75
|
+
- !ruby/object:Gem::Version
|
76
|
+
version: '0'
|
77
|
+
type: :development
|
78
|
+
prerelease: false
|
79
|
+
version_requirements: *13415240
|
80
|
+
- !ruby/object:Gem::Dependency
|
81
|
+
name: timecop
|
82
|
+
requirement: &13405180 !ruby/object:Gem::Requirement
|
83
|
+
none: false
|
84
|
+
requirements:
|
85
|
+
- - ! '>='
|
86
|
+
- !ruby/object:Gem::Version
|
87
|
+
version: '0'
|
88
|
+
type: :development
|
89
|
+
prerelease: false
|
90
|
+
version_requirements: *13405180
|
91
|
+
- !ruby/object:Gem::Dependency
|
92
|
+
name: spork
|
93
|
+
requirement: &13403760 !ruby/object:Gem::Requirement
|
94
|
+
none: false
|
95
|
+
requirements:
|
96
|
+
- - ! '>='
|
97
|
+
- !ruby/object:Gem::Version
|
98
|
+
version: '0'
|
99
|
+
type: :development
|
100
|
+
prerelease: false
|
101
|
+
version_requirements: *13403760
|
102
|
+
- !ruby/object:Gem::Dependency
|
103
|
+
name: guard
|
104
|
+
requirement: &13402900 !ruby/object:Gem::Requirement
|
105
|
+
none: false
|
106
|
+
requirements:
|
107
|
+
- - ! '>='
|
108
|
+
- !ruby/object:Gem::Version
|
109
|
+
version: '0'
|
110
|
+
type: :development
|
111
|
+
prerelease: false
|
112
|
+
version_requirements: *13402900
|
113
|
+
- !ruby/object:Gem::Dependency
|
114
|
+
name: guard-rspec
|
115
|
+
requirement: &13401960 !ruby/object:Gem::Requirement
|
116
|
+
none: false
|
117
|
+
requirements:
|
118
|
+
- - ! '>='
|
119
|
+
- !ruby/object:Gem::Version
|
120
|
+
version: '0'
|
121
|
+
type: :development
|
122
|
+
prerelease: false
|
123
|
+
version_requirements: *13401960
|
124
|
+
- !ruby/object:Gem::Dependency
|
125
|
+
name: guard-spork
|
126
|
+
requirement: &13401120 !ruby/object:Gem::Requirement
|
127
|
+
none: false
|
128
|
+
requirements:
|
129
|
+
- - ! '>='
|
130
|
+
- !ruby/object:Gem::Version
|
131
|
+
version: '0'
|
132
|
+
type: :development
|
133
|
+
prerelease: false
|
134
|
+
version_requirements: *13401120
|
135
|
+
- !ruby/object:Gem::Dependency
|
136
|
+
name: qu-mongo
|
137
|
+
requirement: &13399900 !ruby/object:Gem::Requirement
|
138
|
+
none: false
|
139
|
+
requirements:
|
140
|
+
- - ! '>='
|
141
|
+
- !ruby/object:Gem::Version
|
142
|
+
version: '0'
|
143
|
+
type: :development
|
144
|
+
prerelease: false
|
145
|
+
version_requirements: *13399900
|
146
|
+
- !ruby/object:Gem::Dependency
|
147
|
+
name: bson_ext
|
148
|
+
requirement: &13398400 !ruby/object:Gem::Requirement
|
149
|
+
none: false
|
150
|
+
requirements:
|
151
|
+
- - ! '>='
|
152
|
+
- !ruby/object:Gem::Version
|
153
|
+
version: '0'
|
154
|
+
type: :development
|
155
|
+
prerelease: false
|
156
|
+
version_requirements: *13398400
|
157
|
+
description: Qu::Delayed is basic sheduler for Qu queue system.
|
158
|
+
email: viacheslav.molokov@gmail.com
|
159
|
+
executables: []
|
160
|
+
extensions: []
|
161
|
+
extra_rdoc_files:
|
162
|
+
- LICENSE.txt
|
163
|
+
- README.rdoc
|
164
|
+
files:
|
165
|
+
- .document
|
166
|
+
- .rspec
|
167
|
+
- Gemfile
|
168
|
+
- Gemfile.lock
|
169
|
+
- Guardfile
|
170
|
+
- LICENSE.txt
|
171
|
+
- README.rdoc
|
172
|
+
- Rakefile
|
173
|
+
- VERSION
|
174
|
+
- lib/qu-delayed-immediate.rb
|
175
|
+
- lib/qu-delayed-mongo.rb
|
176
|
+
- lib/qu-delayed.rb
|
177
|
+
- lib/qu/delayed.rb
|
178
|
+
- lib/qu/delayed/backend/immediate.rb
|
179
|
+
- lib/qu/delayed/backend/mongo.rb
|
180
|
+
- lib/qu/delayed/payload.rb
|
181
|
+
- lib/qu/delayed/worker.rb
|
182
|
+
- qu-delayed.gemspec
|
183
|
+
- spec/qu/delayed/backend/immediate_spec.rb
|
184
|
+
- spec/qu/delayed/backend/mongo_spec.rb
|
185
|
+
- spec/qu/delayed/payload_spec.rb
|
186
|
+
- spec/qu/delayed/worker_spec.rb
|
187
|
+
- spec/qu/delayed_spec.rb
|
188
|
+
- spec/spec_helper.rb
|
189
|
+
homepage: http://github.com/imomoisoft/qu-delayed
|
190
|
+
licenses:
|
191
|
+
- MIT
|
192
|
+
post_install_message:
|
193
|
+
rdoc_options: []
|
194
|
+
require_paths:
|
195
|
+
- lib
|
196
|
+
required_ruby_version: !ruby/object:Gem::Requirement
|
197
|
+
none: false
|
198
|
+
requirements:
|
199
|
+
- - ! '>='
|
200
|
+
- !ruby/object:Gem::Version
|
201
|
+
version: '0'
|
202
|
+
segments:
|
203
|
+
- 0
|
204
|
+
hash: -2659276202398035988
|
205
|
+
required_rubygems_version: !ruby/object:Gem::Requirement
|
206
|
+
none: false
|
207
|
+
requirements:
|
208
|
+
- - ! '>='
|
209
|
+
- !ruby/object:Gem::Version
|
210
|
+
version: '0'
|
211
|
+
requirements: []
|
212
|
+
rubyforge_project:
|
213
|
+
rubygems_version: 1.8.10
|
214
|
+
signing_key:
|
215
|
+
specification_version: 3
|
216
|
+
summary: Qu::Delayed is basic sheduler for Qu queue system.
|
217
|
+
test_files: []
|