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 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
@@ -0,0 +1,12 @@
1
+ /.bundle/
2
+ /.yardoc
3
+ /Gemfile.lock
4
+ /_yardoc/
5
+ /coverage/
6
+ /doc/
7
+ /pkg/
8
+ /spec/reports/
9
+ /tmp/
10
+
11
+ # rspec failure tracking
12
+ .rspec_status
data/.rspec ADDED
@@ -0,0 +1,2 @@
1
+ --format documentation
2
+ --color
data/.travis.yml ADDED
@@ -0,0 +1,5 @@
1
+ sudo: false
2
+ language: ruby
3
+ rvm:
4
+ - 2.3.3
5
+ before_install: gem install bundler -v 1.14.5
data/Gemfile ADDED
@@ -0,0 +1,4 @@
1
+ source 'https://rubygems.org'
2
+
3
+ # Specify your gem's dependencies in resque-better_unique.gemspec
4
+ gemspec
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
@@ -0,0 +1,6 @@
1
+ require "bundler/gem_tasks"
2
+ require "rspec/core/rake_task"
3
+
4
+ RSpec::Core::RakeTask.new(:spec)
5
+
6
+ task :default => :spec
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,8 @@
1
+ #!/usr/bin/env bash
2
+ set -euo pipefail
3
+ IFS=$'\n\t'
4
+ set -vx
5
+
6
+ bundle install
7
+
8
+ # Do any other automated setup that you need to do here
@@ -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,7 @@
1
+ module Resque
2
+ module Plugins
3
+ module BetterUnique
4
+ VERSION = '1.0.0'
5
+ end
6
+ end
7
+ 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: []