resque-better_unique 1.0.0
Sign up to get free protection for your applications and to get access to all the features.
- checksums.yaml +7 -0
- data/.gitignore +12 -0
- data/.rspec +2 -0
- data/.travis.yml +5 -0
- data/Gemfile +4 -0
- data/README.md +50 -0
- data/Rakefile +6 -0
- data/bin/console +14 -0
- data/bin/setup +8 -0
- data/lib/resque/plugins/better_unique.rb +98 -0
- data/lib/resque/plugins/better_unique/version.rb +7 -0
- data/resque-better_unique.gemspec +29 -0
- metadata +140 -0
checksums.yaml
ADDED
@@ -0,0 +1,7 @@
|
|
1
|
+
---
|
2
|
+
SHA1:
|
3
|
+
metadata.gz: 49c3786567e5fac7afaf1952494e57dab802498c
|
4
|
+
data.tar.gz: c2c493ae52134c85d258f263f8da2a64e35ab843
|
5
|
+
SHA512:
|
6
|
+
metadata.gz: 6718dbab47f5b8f8cd6806981370aba54b36c53369c6126968139f9ea14a1d805e83330cdd646970df77ff096149500728efbf6f902dec2ab6cea16977dd198c
|
7
|
+
data.tar.gz: b433207f6254cb640c3ed511743fddc4b5b2a07786cb73e2ffb2f068160c00a6d711c4db4f0e034ac04551a63278e31d8143e3ea5d71268b0239d061f2674204
|
data/.gitignore
ADDED
data/.rspec
ADDED
data/.travis.yml
ADDED
data/Gemfile
ADDED
data/README.md
ADDED
@@ -0,0 +1,50 @@
|
|
1
|
+
# Resque::BetterUnique
|
2
|
+
|
3
|
+
There are currently a number of resque plugins that provide this functionality in some form or another, but one thing they all lack is the ability to control how the unique constraint is defined. Sometimes, a job should only be unique until a worker begins processing it, in other cases you will want the job to remain unique until the job completes, or maybe even long after the job has completed. This allows you to do all of the above and more with a single gem.
|
4
|
+
|
5
|
+
The functionality of this gem is based on the sidekiq equivalent [sidekiq-unique-jobs](https://github.com/mhenrixon/sidekiq-unique-jobs).
|
6
|
+
|
7
|
+
## Installation
|
8
|
+
|
9
|
+
Add this line to your application's Gemfile:
|
10
|
+
|
11
|
+
```ruby
|
12
|
+
gem 'resque-better_unique'
|
13
|
+
```
|
14
|
+
|
15
|
+
And then execute:
|
16
|
+
|
17
|
+
$ bundle
|
18
|
+
|
19
|
+
Or install it yourself as:
|
20
|
+
|
21
|
+
$ gem install resque-better_unique
|
22
|
+
|
23
|
+
## Usage
|
24
|
+
Include this plugin into your job class and call the `unique` method
|
25
|
+
```ruby
|
26
|
+
class MyWorker
|
27
|
+
include Resque::Plugins::BetterUnique
|
28
|
+
unique :while_executing, timeout: 5.minutes
|
29
|
+
end
|
30
|
+
```
|
31
|
+
|
32
|
+
The unique method takes up to two arguments:
|
33
|
+
- mode: (default=:until_executed)
|
34
|
+
* while_executing: only one distinct job can be processed at a time
|
35
|
+
* until_executing: only one job can be queued at a time
|
36
|
+
* until_executed: only one job can be queued or processed at a time
|
37
|
+
* until_timeout: only one job can be queued or processed in a given time period
|
38
|
+
- options: Hash of options
|
39
|
+
* timeout - integer or object that responds to to_i - How long should a lock live
|
40
|
+
* unique_args - a proc or a symbol which takes the arguments of perform and returns the arguments that should be used to determine uniqueness
|
41
|
+
|
42
|
+
## Development
|
43
|
+
|
44
|
+
After checking out the repo, run `bin/setup` to install dependencies. Then, run `rake spec` to run the tests. You can also run `bin/console` for an interactive prompt that will allow you to experiment.
|
45
|
+
|
46
|
+
To install this gem onto your local machine, run `bundle exec rake install`. To release a new version, update the version number in `version.rb`, and then run `bundle exec rake release`, which will create a git tag for the version, push git commits and tags, and push the `.gem` file to [rubygems.org](https://rubygems.org).
|
47
|
+
|
48
|
+
## Contributing
|
49
|
+
|
50
|
+
Bug reports and pull requests are welcome on GitHub at https://github.com/[USERNAME]/resque-better_unique.
|
data/Rakefile
ADDED
data/bin/console
ADDED
@@ -0,0 +1,14 @@
|
|
1
|
+
#!/usr/bin/env ruby
|
2
|
+
|
3
|
+
require "bundler/setup"
|
4
|
+
require "resque/better_unique"
|
5
|
+
|
6
|
+
# You can add fixtures and/or initialization code here to make experimenting
|
7
|
+
# with your gem easier. You can also use a different console, if you like.
|
8
|
+
|
9
|
+
# (If you use this, don't forget to add pry to your Gemfile!)
|
10
|
+
# require "pry"
|
11
|
+
# Pry.start
|
12
|
+
|
13
|
+
require "irb"
|
14
|
+
IRB.start(__FILE__)
|
data/bin/setup
ADDED
@@ -0,0 +1,98 @@
|
|
1
|
+
module Resque
|
2
|
+
module Plugins
|
3
|
+
module BetterUnique
|
4
|
+
|
5
|
+
|
6
|
+
def self.included(base_klass)
|
7
|
+
base_klass.extend(ClassMethods)
|
8
|
+
end
|
9
|
+
|
10
|
+
module ClassMethods
|
11
|
+
# Override in your job to control the lock key. It is
|
12
|
+
# passed the same arguments as `perform`, that is, your job's
|
13
|
+
# payload.
|
14
|
+
def lock_key(*args)
|
15
|
+
unique_args = unique_options[:unique_args]
|
16
|
+
lock_args = case unique_args
|
17
|
+
when Proc
|
18
|
+
unique_args.call(*args)
|
19
|
+
when Symbol
|
20
|
+
self.send(unique_args, *args)
|
21
|
+
else
|
22
|
+
args
|
23
|
+
end
|
24
|
+
"#{lock_key_base}-#{lock_args.to_s}"
|
25
|
+
end
|
26
|
+
|
27
|
+
def lock_key_base
|
28
|
+
"lock:#{name}"
|
29
|
+
end
|
30
|
+
|
31
|
+
def locked?(*args)
|
32
|
+
Resque.redis.exists(lock_key(*args))
|
33
|
+
end
|
34
|
+
|
35
|
+
def unique_options
|
36
|
+
@unique_options || {}
|
37
|
+
end
|
38
|
+
|
39
|
+
def unique_options=(options)
|
40
|
+
@unique_options = options
|
41
|
+
end
|
42
|
+
|
43
|
+
def unique_mode
|
44
|
+
unique_options[:mode].to_sym || :none
|
45
|
+
end
|
46
|
+
|
47
|
+
def unique(mode=:until_executed, options={})
|
48
|
+
self.unique_options = {mode: mode}.merge(options)
|
49
|
+
end
|
50
|
+
|
51
|
+
def before_enqueue_unique_lock(*args)
|
52
|
+
if [:until_executing, :until_executed, :until_timeout].include?(unique_mode)
|
53
|
+
return false if locked?(*args)
|
54
|
+
set_lock(*args)
|
55
|
+
end
|
56
|
+
true
|
57
|
+
end
|
58
|
+
|
59
|
+
def around_perform_unique_lock(*args)
|
60
|
+
case unique_mode
|
61
|
+
when :until_executing
|
62
|
+
release_lock(*args)
|
63
|
+
when :while_executing
|
64
|
+
return if locked?(*args) || !set_lock(*args)
|
65
|
+
end
|
66
|
+
yield
|
67
|
+
ensure
|
68
|
+
if [:until_executed, :while_executing].include?(unique_mode)
|
69
|
+
release_lock(*args)
|
70
|
+
end
|
71
|
+
end
|
72
|
+
|
73
|
+
def release_lock(*args)
|
74
|
+
Resque.redis.del(lock_key(*args))
|
75
|
+
end
|
76
|
+
|
77
|
+
def set_lock(*args)
|
78
|
+
is_now_locked = Resque.redis.setnx(lock_key(*args), true)
|
79
|
+
if is_now_locked && unique_options[:timeout]
|
80
|
+
Resque.redis.expire(lock_key(*args), unique_options[:timeout].to_i)
|
81
|
+
end
|
82
|
+
is_now_locked
|
83
|
+
end
|
84
|
+
|
85
|
+
def release_all_locks(offset=nil)
|
86
|
+
return if offset == '0'
|
87
|
+
new_offset, keys = Resque.redis.scan(offset || 0)
|
88
|
+
keys.each do |key|
|
89
|
+
Resque.redis.del(key) if key.start_with?(lock_key_base)
|
90
|
+
end
|
91
|
+
release_all_locks(new_offset)
|
92
|
+
rescue Redis::CommandError
|
93
|
+
Resque.logger.error 'This command only works with versions of redis-server over 2.8'
|
94
|
+
end
|
95
|
+
end
|
96
|
+
end
|
97
|
+
end
|
98
|
+
end
|
@@ -0,0 +1,29 @@
|
|
1
|
+
# coding: utf-8
|
2
|
+
lib = File.expand_path('../lib', __FILE__)
|
3
|
+
$LOAD_PATH.unshift(lib) unless $LOAD_PATH.include?(lib)
|
4
|
+
require 'resque/plugins/better_unique/version'
|
5
|
+
|
6
|
+
Gem::Specification.new do |spec|
|
7
|
+
spec.name = 'resque-better_unique'
|
8
|
+
spec.version = Resque::Plugins::BetterUnique::VERSION
|
9
|
+
spec.authors = ['Will Bryant']
|
10
|
+
spec.email = ['william@tout.com']
|
11
|
+
|
12
|
+
spec.summary = %q{A resque plugin for better control over unique jobs}
|
13
|
+
spec.description = %q{There are a number of plugins which allow you define unique jobs, but each only handle on use-case. This allows you have full control over how uniqueness is defined}
|
14
|
+
spec.homepage = 'http://github.com/will3216/resque-better_unique'
|
15
|
+
|
16
|
+
spec.files = `git ls-files -z`.split("\x0").reject do |f|
|
17
|
+
f.match(%r{^(test|spec|features)/})
|
18
|
+
end
|
19
|
+
spec.bindir = 'exe'
|
20
|
+
spec.executables = spec.files.grep(%r{^exe/}) { |f| File.basename(f) }
|
21
|
+
spec.require_paths = ['lib']
|
22
|
+
|
23
|
+
spec.add_development_dependency 'bundler', '~> 1.14'
|
24
|
+
spec.add_development_dependency 'rake', '~> 10.0'
|
25
|
+
spec.add_development_dependency 'rspec', '~> 3.0'
|
26
|
+
spec.add_development_dependency 'resque'
|
27
|
+
spec.add_development_dependency 'redis'
|
28
|
+
spec.add_development_dependency 'simplecov'
|
29
|
+
end
|
metadata
ADDED
@@ -0,0 +1,140 @@
|
|
1
|
+
--- !ruby/object:Gem::Specification
|
2
|
+
name: resque-better_unique
|
3
|
+
version: !ruby/object:Gem::Version
|
4
|
+
version: 1.0.0
|
5
|
+
platform: ruby
|
6
|
+
authors:
|
7
|
+
- Will Bryant
|
8
|
+
autorequire:
|
9
|
+
bindir: exe
|
10
|
+
cert_chain: []
|
11
|
+
date: 2017-07-18 00:00:00.000000000 Z
|
12
|
+
dependencies:
|
13
|
+
- !ruby/object:Gem::Dependency
|
14
|
+
name: bundler
|
15
|
+
requirement: !ruby/object:Gem::Requirement
|
16
|
+
requirements:
|
17
|
+
- - "~>"
|
18
|
+
- !ruby/object:Gem::Version
|
19
|
+
version: '1.14'
|
20
|
+
type: :development
|
21
|
+
prerelease: false
|
22
|
+
version_requirements: !ruby/object:Gem::Requirement
|
23
|
+
requirements:
|
24
|
+
- - "~>"
|
25
|
+
- !ruby/object:Gem::Version
|
26
|
+
version: '1.14'
|
27
|
+
- !ruby/object:Gem::Dependency
|
28
|
+
name: rake
|
29
|
+
requirement: !ruby/object:Gem::Requirement
|
30
|
+
requirements:
|
31
|
+
- - "~>"
|
32
|
+
- !ruby/object:Gem::Version
|
33
|
+
version: '10.0'
|
34
|
+
type: :development
|
35
|
+
prerelease: false
|
36
|
+
version_requirements: !ruby/object:Gem::Requirement
|
37
|
+
requirements:
|
38
|
+
- - "~>"
|
39
|
+
- !ruby/object:Gem::Version
|
40
|
+
version: '10.0'
|
41
|
+
- !ruby/object:Gem::Dependency
|
42
|
+
name: rspec
|
43
|
+
requirement: !ruby/object:Gem::Requirement
|
44
|
+
requirements:
|
45
|
+
- - "~>"
|
46
|
+
- !ruby/object:Gem::Version
|
47
|
+
version: '3.0'
|
48
|
+
type: :development
|
49
|
+
prerelease: false
|
50
|
+
version_requirements: !ruby/object:Gem::Requirement
|
51
|
+
requirements:
|
52
|
+
- - "~>"
|
53
|
+
- !ruby/object:Gem::Version
|
54
|
+
version: '3.0'
|
55
|
+
- !ruby/object:Gem::Dependency
|
56
|
+
name: resque
|
57
|
+
requirement: !ruby/object:Gem::Requirement
|
58
|
+
requirements:
|
59
|
+
- - ">="
|
60
|
+
- !ruby/object:Gem::Version
|
61
|
+
version: '0'
|
62
|
+
type: :development
|
63
|
+
prerelease: false
|
64
|
+
version_requirements: !ruby/object:Gem::Requirement
|
65
|
+
requirements:
|
66
|
+
- - ">="
|
67
|
+
- !ruby/object:Gem::Version
|
68
|
+
version: '0'
|
69
|
+
- !ruby/object:Gem::Dependency
|
70
|
+
name: redis
|
71
|
+
requirement: !ruby/object:Gem::Requirement
|
72
|
+
requirements:
|
73
|
+
- - ">="
|
74
|
+
- !ruby/object:Gem::Version
|
75
|
+
version: '0'
|
76
|
+
type: :development
|
77
|
+
prerelease: false
|
78
|
+
version_requirements: !ruby/object:Gem::Requirement
|
79
|
+
requirements:
|
80
|
+
- - ">="
|
81
|
+
- !ruby/object:Gem::Version
|
82
|
+
version: '0'
|
83
|
+
- !ruby/object:Gem::Dependency
|
84
|
+
name: simplecov
|
85
|
+
requirement: !ruby/object:Gem::Requirement
|
86
|
+
requirements:
|
87
|
+
- - ">="
|
88
|
+
- !ruby/object:Gem::Version
|
89
|
+
version: '0'
|
90
|
+
type: :development
|
91
|
+
prerelease: false
|
92
|
+
version_requirements: !ruby/object:Gem::Requirement
|
93
|
+
requirements:
|
94
|
+
- - ">="
|
95
|
+
- !ruby/object:Gem::Version
|
96
|
+
version: '0'
|
97
|
+
description: There are a number of plugins which allow you define unique jobs, but
|
98
|
+
each only handle on use-case. This allows you have full control over how uniqueness
|
99
|
+
is defined
|
100
|
+
email:
|
101
|
+
- william@tout.com
|
102
|
+
executables: []
|
103
|
+
extensions: []
|
104
|
+
extra_rdoc_files: []
|
105
|
+
files:
|
106
|
+
- ".gitignore"
|
107
|
+
- ".rspec"
|
108
|
+
- ".travis.yml"
|
109
|
+
- Gemfile
|
110
|
+
- README.md
|
111
|
+
- Rakefile
|
112
|
+
- bin/console
|
113
|
+
- bin/setup
|
114
|
+
- lib/resque/plugins/better_unique.rb
|
115
|
+
- lib/resque/plugins/better_unique/version.rb
|
116
|
+
- resque-better_unique.gemspec
|
117
|
+
homepage: http://github.com/will3216/resque-better_unique
|
118
|
+
licenses: []
|
119
|
+
metadata: {}
|
120
|
+
post_install_message:
|
121
|
+
rdoc_options: []
|
122
|
+
require_paths:
|
123
|
+
- lib
|
124
|
+
required_ruby_version: !ruby/object:Gem::Requirement
|
125
|
+
requirements:
|
126
|
+
- - ">="
|
127
|
+
- !ruby/object:Gem::Version
|
128
|
+
version: '0'
|
129
|
+
required_rubygems_version: !ruby/object:Gem::Requirement
|
130
|
+
requirements:
|
131
|
+
- - ">="
|
132
|
+
- !ruby/object:Gem::Version
|
133
|
+
version: '0'
|
134
|
+
requirements: []
|
135
|
+
rubyforge_project:
|
136
|
+
rubygems_version: 2.6.8
|
137
|
+
signing_key:
|
138
|
+
specification_version: 4
|
139
|
+
summary: A resque plugin for better control over unique jobs
|
140
|
+
test_files: []
|