rcelery 1.0.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (47) hide show
  1. data/Gemfile +3 -0
  2. data/LICENSE +27 -0
  3. data/Rakefile +34 -0
  4. data/bin/rceleryd +7 -0
  5. data/lib/rcelery/async_result.rb +23 -0
  6. data/lib/rcelery/configuration.rb +25 -0
  7. data/lib/rcelery/daemon.rb +82 -0
  8. data/lib/rcelery/eager_result.rb +10 -0
  9. data/lib/rcelery/events.rb +103 -0
  10. data/lib/rcelery/pool.rb +59 -0
  11. data/lib/rcelery/rails.rb +29 -0
  12. data/lib/rcelery/railtie.rb +11 -0
  13. data/lib/rcelery/task/context.rb +25 -0
  14. data/lib/rcelery/task/runner.rb +73 -0
  15. data/lib/rcelery/task.rb +118 -0
  16. data/lib/rcelery/task_support.rb +56 -0
  17. data/lib/rcelery/version.rb +6 -0
  18. data/lib/rcelery/worker.rb +76 -0
  19. data/lib/rcelery.rb +86 -0
  20. data/rails/init.rb +4 -0
  21. data/rcelery.gemspec +28 -0
  22. data/spec/bin/ci +81 -0
  23. data/spec/integration/Procfile +4 -0
  24. data/spec/integration/bin/celeryd +4 -0
  25. data/spec/integration/bin/rabbitmq-server +4 -0
  26. data/spec/integration/bin/rceleryd +4 -0
  27. data/spec/integration/python_components/celery_client.py +5 -0
  28. data/spec/integration/python_components/celery_deferred_client.py +10 -0
  29. data/spec/integration/python_components/celeryconfig.py +26 -0
  30. data/spec/integration/python_components/requirements.txt +9 -0
  31. data/spec/integration/python_components/tasks.py +13 -0
  32. data/spec/integration/ruby_client_python_worker_spec.rb +17 -0
  33. data/spec/integration/ruby_worker_spec.rb +115 -0
  34. data/spec/integration/spec_helper.rb +43 -0
  35. data/spec/integration/tasks.rb +37 -0
  36. data/spec/spec_helper.rb +34 -0
  37. data/spec/unit/configuration_spec.rb +91 -0
  38. data/spec/unit/daemon_spec.rb +21 -0
  39. data/spec/unit/eager_spec.rb +46 -0
  40. data/spec/unit/events_spec.rb +149 -0
  41. data/spec/unit/pool_spec.rb +124 -0
  42. data/spec/unit/rails_spec.rb +0 -0
  43. data/spec/unit/rcelery_spec.rb +154 -0
  44. data/spec/unit/task_spec.rb +192 -0
  45. data/spec/unit/task_support_spec.rb +94 -0
  46. data/spec/unit/worker_spec.rb +63 -0
  47. metadata +290 -0
data/Gemfile ADDED
@@ -0,0 +1,3 @@
1
+ source 'http://rubygems.org/'
2
+
3
+ gemspec
data/LICENSE ADDED
@@ -0,0 +1,27 @@
1
+ Copyright (c) 2011, Leapfrog Direct Response, LLC
2
+ All rights reserved.
3
+
4
+ Redistribution and use in source and binary forms, with or without
5
+ modification, are permitted provided that the following conditions are met:
6
+ * Redistributions of source code must retain the above copyright
7
+ notice, this list of conditions and the following disclaimer.
8
+ * Redistributions in binary form must reproduce the above copyright
9
+ notice, this list of conditions and the following disclaimer in the
10
+ documentation and/or other materials provided with the distribution.
11
+ * Neither the name of the Leapfrog Direct Response, LLC, including
12
+ its subsidiaries and affiliates nor the names of its
13
+ contributors, may be used to endorse or promote products derived
14
+ from this software without specific prior written permission.
15
+
16
+ THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
17
+ "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
18
+ LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
19
+ A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL LEAPFROG DIRECT
20
+ RESPONSE, LLC, INCLUDING ITS SUBSIDIARIES AND AFFILIATES, BE LIABLE
21
+ FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
22
+ CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
23
+ SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR
24
+ BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY,
25
+ WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE
26
+ OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN
27
+ IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
data/Rakefile ADDED
@@ -0,0 +1,34 @@
1
+ require 'rubygems'
2
+ require 'bundler'
3
+ Bundler.setup
4
+
5
+ require 'rake'
6
+ require 'rspec/core/rake_task'
7
+ require 'rake/gempackagetask'
8
+
9
+ def gemspec
10
+ eval(File.new('rcelery.gemspec').read)
11
+ end
12
+ Rake::GemPackageTask.new(gemspec).define
13
+
14
+ desc 'Run ruby worker integration specs'
15
+ RSpec::Core::RakeTask.new('spec:integration:ruby_worker') do |t|
16
+ t.pattern = 'spec/integration/ruby_worker_spec.rb'
17
+ t.rspec_opts = ["--color"]
18
+ end
19
+
20
+ desc 'Run python worker integration specs'
21
+ RSpec::Core::RakeTask.new('spec:integration:python_worker') do |t|
22
+ t.pattern = 'spec/integration/ruby_client_python_worker_spec.rb'
23
+ t.rspec_opts = ["--color"]
24
+ end
25
+
26
+ desc 'Run unit specs'
27
+ RSpec::Core::RakeTask.new('spec:unit') do |t|
28
+ t.pattern = 'spec/unit/*_spec.rb'
29
+ t.rspec_opts = ["--color"]
30
+ end
31
+
32
+ desc 'Run all specs'
33
+ task :spec => ['spec:unit',]
34
+
data/bin/rceleryd ADDED
@@ -0,0 +1,7 @@
1
+ #!/usr/bin/env ruby
2
+ require 'rubygems'
3
+ require 'rcelery'
4
+
5
+ daemon = RCelery::Daemon.new(ARGV)
6
+ daemon.run
7
+
@@ -0,0 +1,23 @@
1
+ module RCelery
2
+ class AsyncResult
3
+ def initialize(task_id)
4
+ @task_id = task_id
5
+ @queue = Task.result_queue(task_id)
6
+ end
7
+
8
+ def wait
9
+ result_value = :no_result
10
+ @queue.subscribe do |payload|
11
+ result_value = JSON.parse(payload)['result']
12
+ end
13
+
14
+ while(result_value == :no_result)
15
+ sleep(0.05)
16
+ end
17
+
18
+ @queue.unsubscribe
19
+ result_value
20
+ end
21
+ end
22
+ end
23
+
@@ -0,0 +1,25 @@
1
+ require 'configtoolkit'
2
+ require 'configtoolkit/hashreader'
3
+ require 'configtoolkit/hashwriter'
4
+
5
+ module RCelery
6
+ class Configuration < ConfigToolkit::BaseConfig
7
+ add_optional_param(:host, String, 'localhost')
8
+ add_optional_param(:port, Integer, 5672)
9
+ add_optional_param(:vhost, String, '/')
10
+ add_optional_param(:username, String, 'guest')
11
+ add_optional_param(:password, String, 'guest')
12
+ add_optional_param(:application, String, 'application')
13
+ add_optional_param(:worker_count, Integer, 1)
14
+ add_optional_param(:eager_mode, ConfigToolkit::Boolean, false)
15
+
16
+ def initialize(options = {})
17
+ load(ConfigToolkit::HashReader.new(options))
18
+ end
19
+
20
+ def to_hash
21
+ dump(ConfigToolkit::HashWriter.new)
22
+ end
23
+ end
24
+ end
25
+
@@ -0,0 +1,82 @@
1
+ require 'optparse'
2
+ require 'rcelery/pool'
3
+
4
+ module RCelery
5
+ class Daemon
6
+ def initialize(args)
7
+ @config = RCelery::Configuration.new
8
+ opts = OptionParser.new do |opt|
9
+ opt.on('-n', '--hostname HOSTNAME', 'Hostname of the AMQP broker') do |host|
10
+ @config.host = host
11
+ end
12
+
13
+ opt.on('-p', '--port PORT', 'Port of the AMQP broker') do |port|
14
+ @config.port = port
15
+ end
16
+
17
+ opt.on('-v', '--vhost VHOST', 'Vhost of the AMQP broker') do |vhost|
18
+ @config.vhost = vhost
19
+ end
20
+
21
+ opt.on('-u', '--username USERNAME', 'Username to use during authentication with the AMQP broker') do |username|
22
+ @config.username = username
23
+ end
24
+
25
+ opt.on('-w', '--password PASSWORD', 'Password to use during authentication with the AMQP broker') do |password|
26
+ @config.password = password
27
+ end
28
+
29
+ opt.on('-a', '--application APPLICATION', 'Name of the application') do |application|
30
+ @config.application = application
31
+ end
32
+
33
+ opt.on('-t', '--tasks lib1,lib2,...', Array, 'List of libraries to require that contain task definitions') do |requires|
34
+ requires.each do |lib|
35
+ require lib
36
+ end
37
+ end
38
+
39
+ opt.on('-r', '--rails', 'Require \'config/environment\' to provide the Rails environment') do
40
+ require 'config/environment'
41
+ require 'rcelery/rails'
42
+ RCelery::Rails.initialize
43
+ ::Rails.logger.auto_flushing = true
44
+ end
45
+
46
+ opt.on('-W', '--workers NUMBER', 'The number of workers to launch (default 1)') do |num|
47
+ @config.worker_count = num
48
+ end
49
+
50
+ opt.on_tail('-h', '--help', 'Show this message') do
51
+ puts opts
52
+ exit
53
+ end
54
+ end
55
+ opts.parse!(args)
56
+ end
57
+
58
+ def run
59
+ pool = RCelery::Pool.new(@config)
60
+ @config.worker_count.times do
61
+ Thread.new do
62
+ @worker = RCelery::Worker.new
63
+ @worker.start pool
64
+ end
65
+ end
66
+
67
+ pool.start
68
+ trap_signals
69
+ RCelery.thread.join
70
+ end
71
+
72
+ def trap_signals
73
+ block = proc do
74
+ @worker.stop
75
+ RCelery.stop
76
+ exit
77
+ end
78
+ Signal.trap('INT', &block)
79
+ Signal.trap('TERM', &block)
80
+ end
81
+ end
82
+ end
@@ -0,0 +1,10 @@
1
+ module RCelery
2
+ class EagerResult
3
+ attr_accessor :wait
4
+
5
+ def initialize(value)
6
+ @wait = value
7
+ end
8
+
9
+ end
10
+ end
@@ -0,0 +1,103 @@
1
+ require 'socket'
2
+
3
+ module RCelery
4
+ class Events
5
+ def self.hostname
6
+ Socket.gethostname
7
+ end
8
+
9
+ def self.timestamp
10
+ Time.now.to_f
11
+ end
12
+
13
+ def self.task_received(uuid, name, args, kwargs, retries = 0, eta = nil)
14
+ RCelery.publish(:event, {
15
+ :type => 'task-received',
16
+ :uuid => uuid,
17
+ :name => name,
18
+ :args => args,
19
+ :kwargs => kwargs,
20
+ :retries => retries,
21
+ :eta => eta,
22
+ :hostname => hostname,
23
+ :timestamp => timestamp
24
+ }, :routing_key => 'task.received')
25
+ end
26
+
27
+ def self.task_started(uuid, pid)
28
+ RCelery.publish(:event, {
29
+ :type => 'task-started',
30
+ :uuid => uuid,
31
+ :hostname => hostname,
32
+ :timestamp => timestamp,
33
+ :pid => pid
34
+ }, :routing_key => 'task.started')
35
+ end
36
+
37
+ def self.task_succeeded(uuid, result, runtime)
38
+ RCelery.publish(:event, {
39
+ :type => 'task-succeeded',
40
+ :uuid => uuid,
41
+ :result => result,
42
+ :hostname => hostname,
43
+ :timestamp => timestamp,
44
+ :runtime => runtime,
45
+ }, :routing_key => 'task.succeeded')
46
+ end
47
+
48
+ def self.task_failed(uuid, exception, traceback)
49
+ RCelery.publish(:event, {
50
+ :type => 'task-failed',
51
+ :uuid => uuid,
52
+ :exception => exception,
53
+ :traceback => traceback,
54
+ :hostname => hostname,
55
+ :timestamp => timestamp
56
+ }, :routing_key => 'task.failed')
57
+ end
58
+
59
+ def self.task_retried(uuid, exception, traceback)
60
+ RCelery.publish(:event, {
61
+ :type => 'task-retried',
62
+ :uuid => uuid,
63
+ :exception => exception,
64
+ :traceback => traceback,
65
+ :hostname => hostname,
66
+ :timestamp => timestamp
67
+ }, :routing_key => 'task.retried')
68
+ end
69
+
70
+ def self.worker_online(sw_ident, sw_ver, sw_sys)
71
+ RCelery.publish(:event, {
72
+ :type => 'worker-online',
73
+ :sw_ident => sw_ident,
74
+ :sw_ver => sw_ver,
75
+ :sw_sys => sw_sys,
76
+ :hostname => hostname,
77
+ :timestamp => timestamp
78
+ }, :routing_key => 'worker.online')
79
+ end
80
+
81
+ def self.worker_heartbeat(sw_ident, sw_ver, sw_sys)
82
+ RCelery.publish(:event, {
83
+ :type => 'worker-heartbeat',
84
+ :sw_ident => sw_ident,
85
+ :sw_ver => sw_ver,
86
+ :sw_sys => sw_sys,
87
+ :hostname => hostname,
88
+ :timestamp => timestamp
89
+ }, :routing_key => 'worker.heartbeat')
90
+ end
91
+
92
+ def self.worker_offline(sw_ident, sw_ver, sw_sys)
93
+ RCelery.publish(:event, {
94
+ :type => 'worker-offline',
95
+ :sw_ident => sw_ident,
96
+ :sw_ver => sw_ver,
97
+ :sw_sys => sw_sys,
98
+ :hostname => hostname,
99
+ :timestamp => timestamp
100
+ }, :routing_key => 'worker.offline')
101
+ end
102
+ end
103
+ end
@@ -0,0 +1,59 @@
1
+ require 'thread'
2
+ require 'time'
3
+
4
+ module RCelery
5
+ class Pool
6
+
7
+ def initialize(options={})
8
+ @task_queue = Queue.new
9
+ RCelery.start(options) unless RCelery.running?
10
+ end
11
+
12
+ def start
13
+ subscribe
14
+ end
15
+
16
+ def subscribe
17
+ # amqp-client has a nice fat TODO in the delivery handler to
18
+ # ack if necessary; we'll just manually do it, however, the
19
+ # call to subscribe still needs :ack => true so the server
20
+ # expects our ack
21
+ RCelery.queue.subscribe(:ack => true) do |header, payload|
22
+ begin
23
+ message = JSON.parse(payload)
24
+ RCelery::Events.task_received(message['id'], message['task'], message['args'], message['kwargs'], nil, message['eta'])
25
+
26
+ if message['eta'] && Time.parse(message['eta']) > Time.now
27
+ defer({:message => message, :header => header})
28
+ else
29
+ @task_queue.push({:message => message, :header => header})
30
+ end
31
+ rescue JSON::ParserError
32
+ # not a message we care about
33
+ header.ack
34
+ end
35
+ end
36
+ end
37
+
38
+ def defer(task)
39
+ time_difference = (Time.parse(task[:message]['eta']) - Time.now).to_i
40
+ EM.add_timer(time_difference) do
41
+ @task_queue.push(task)
42
+ end
43
+ end
44
+
45
+ def poll
46
+ @task_queue.pop
47
+ end
48
+
49
+ def unsubscribe
50
+ RCelery.queue.unsubscribe
51
+ end
52
+
53
+ def stop
54
+ unsubscribe
55
+ RCelery.stop
56
+ end
57
+
58
+ end
59
+ end
@@ -0,0 +1,29 @@
1
+ require 'qusion'
2
+
3
+ module RCelery
4
+ def self.thread
5
+ @thread ||= Qusion.thread
6
+ end
7
+
8
+ module Rails
9
+ def self.initialize
10
+ config_file = File.join(::Rails.root, 'config', 'rcelery.yml')
11
+ raw_config = nil
12
+
13
+ if File.exists?(config_file)
14
+ raw_config = YAML.load_file(config_file)[::Rails.env]
15
+ end
16
+
17
+ unless raw_config.nil?
18
+ config = RCelery::Configuration.new(raw_config)
19
+ if config.eager_mode
20
+ RCelery.start(config)
21
+ else
22
+ Qusion.start(config.to_hash) do
23
+ RCelery.start(config)
24
+ end
25
+ end
26
+ end
27
+ end
28
+ end
29
+ end
@@ -0,0 +1,11 @@
1
+ require 'rcelery/rails'
2
+
3
+ module RCelery
4
+ class Railtie < ::Rails::Railtie
5
+ railtie_name :rcelery
6
+
7
+ initializer "rcelery.rails" do |app|
8
+ RCelery::Rails.initialize
9
+ end
10
+ end
11
+ end
@@ -0,0 +1,25 @@
1
+ module RCelery
2
+ class Task
3
+ class Context
4
+ def initialize(name)
5
+ @key = "#{name}.request"
6
+ end
7
+
8
+ def update(options = {})
9
+ Thread.current[@key] = options
10
+ end
11
+
12
+ def clear
13
+ update
14
+ end
15
+
16
+ def method_missing(method, *args)
17
+ if args.length.zero?
18
+ Thread.current[@key][method]
19
+ else
20
+ super(method, *args)
21
+ end
22
+ end
23
+ end
24
+ end
25
+ end
@@ -0,0 +1,73 @@
1
+ module RCelery
2
+ class Task
3
+ module States
4
+ SUCCESS = 'SUCCESS'.freeze
5
+ RETRY = 'RETRY'.freeze
6
+ FAILURE = 'FAILURE'.freeze
7
+ end
8
+
9
+ class Runner
10
+ include States
11
+
12
+ attr_reader :task, :result, :status
13
+
14
+ def initialize(message)
15
+ @task = Task.all_tasks[message['task']]
16
+ @task_id = message['id']
17
+ @eager = message['eager'].nil? ? false : message['eager']
18
+
19
+ @args = [message['args'], message['kwargs']].flatten.compact
20
+ @args.pop if @args.last.is_a?(Hash) && @args.last.empty?
21
+
22
+ @queue = Task.result_queue(@task_id) unless eager_mode?
23
+ @task.request.update(
24
+ :task_id => @task_id,
25
+ :retries => message['retries'] || 0,
26
+ :args => message['args'],
27
+ :kwargs => message['kwargs']
28
+ )
29
+ end
30
+
31
+ def execute
32
+ result = @task.method.call(*@args)
33
+ @status = SUCCESS
34
+ @result = result
35
+ publish_result if publish_result?
36
+ rescue RetryError => raised
37
+ @result = raised
38
+ @status = RETRY
39
+ rescue Exception => raised
40
+ @result = raised
41
+ @status = FAILURE
42
+ publish_result if publish_result?
43
+ ensure
44
+ @task.request.clear
45
+ end
46
+
47
+ private
48
+ def publish_result
49
+ traceback = []
50
+
51
+ if @status == FAILURE
52
+ traceback = result.backtrace
53
+ end
54
+
55
+ RCelery.publish(:result, {
56
+ :result => @result,
57
+ :status => @status,
58
+ :task_id => @task_id,
59
+ :traceback => traceback },
60
+ :routing_key => @task_id.gsub('-', ''),
61
+ :persistent => true)
62
+ end
63
+
64
+ def eager_mode?
65
+ @eager == true
66
+ end
67
+
68
+ def publish_result?
69
+ @task.ignore_result? == false && eager_mode? == false
70
+ end
71
+ end
72
+ end
73
+ end
@@ -0,0 +1,118 @@
1
+ require 'uuid'
2
+ require 'json'
3
+ require 'rcelery/task/runner'
4
+ require 'rcelery/task/context'
5
+
6
+ module RCelery
7
+ class Task
8
+ class RetryError < StandardError; end
9
+ class MaxRetriesExceededError < StandardError; end
10
+
11
+ class << self
12
+ attr_accessor :result_queue_expires, :max_retries
13
+ end
14
+
15
+ self.max_retries = 3
16
+ self.result_queue_expires = 3600000
17
+
18
+ attr_reader :name, :method, :request
19
+
20
+ def self.all_tasks
21
+ @all_tasks ||= {}
22
+ end
23
+
24
+ def self.execute(message)
25
+ runner = Runner.new(message)
26
+ runner.execute
27
+ runner
28
+ end
29
+
30
+ def self.result_queue(task_id)
31
+ queue_name = task_id.gsub('-', '')
32
+ RCelery.channel.queue(
33
+ queue_name,
34
+ :durable => true,
35
+ :auto_delete => true,
36
+ :arguments => {'x-expires' => result_queue_expires}
37
+ ).bind(
38
+ RCelery.exchanges[:result],
39
+ :routing_key => queue_name
40
+ )
41
+ end
42
+
43
+ def initialize(options = {})
44
+ @name = options[:name]
45
+ @method = options[:method]
46
+ @routing_key = options[:routing_key]
47
+ @ignore_result = options[:ignore_result].nil? ?
48
+ true : options[:ignore_result]
49
+ @request = Context.new(@name)
50
+ end
51
+
52
+ def delay(*args)
53
+ kwargs = args.pop if args.last.is_a?(Hash)
54
+ apply_async(:args => args, :kwargs => kwargs)
55
+ end
56
+
57
+ def retry(options = {})
58
+ args = options[:args] || request.args
59
+ kwargs = options[:kwargs] || request.kwargs
60
+ max_retries = options[:max_retries] || self.class.max_retries
61
+
62
+ if (request.retries + 1) > max_retries
63
+ if options[:exc]
64
+ raise options[:exc]
65
+ else
66
+ raise MaxRetriesExceededError
67
+ end
68
+ end
69
+
70
+ apply_async(
71
+ :args => args,
72
+ :kwargs => kwargs,
73
+ :task_id => request.task_id,
74
+ :retries => request.retries + 1,
75
+ :eta => options[:eta] || default_eta
76
+ )
77
+
78
+ raise RetryError
79
+ end
80
+
81
+ def apply_async(options = {})
82
+ task_id = options[:task_id] || UUID.generate
83
+ task = {
84
+ :id => task_id,
85
+ :task => @name,
86
+ :args => options[:args],
87
+ :kwargs => options[:kwargs] || {}
88
+ }
89
+ task[:eta] = options[:eta].strftime("%Y-%m-%dT%H:%M:%S") if options[:eta]
90
+ task[:retries] = options[:retries] if options[:retries]
91
+
92
+ if RCelery.eager_mode?
93
+ task[:eager] = true
94
+ runner = Task.execute(JSON.parse(task.to_json))
95
+ return (EagerResult.new(runner.result) unless ignore_result?)
96
+ end
97
+
98
+ pub_opts = {
99
+ :persistent => true,
100
+ :routing_key => options[:routing_key] || @routing_key
101
+ }
102
+
103
+ # initialize result queue first to avoid races
104
+ res = ignore_result? ? nil: AsyncResult.new(task_id)
105
+ RCelery.publish(:request, task, pub_opts)
106
+ res
107
+ end
108
+
109
+ def ignore_result?
110
+ @ignore_result
111
+ end
112
+
113
+ private
114
+ def default_eta
115
+ Time.at(Time.now + (60 * 3))
116
+ end
117
+ end
118
+ end
@@ -0,0 +1,56 @@
1
+ module RCelery
2
+ module TaskSupport
3
+ def self.included(base)
4
+ base.extend(ClassMethods)
5
+ end
6
+
7
+ def self.task_name(mod, method)
8
+ parts = mod.split('::') << method.to_s
9
+ klass_name = parts.map do |part|
10
+ part.gsub(/([^\a])([A-Z])/) { "#{$1}_#{$2}" }.downcase
11
+ end.join('.')
12
+ end
13
+
14
+ module ClassMethods
15
+ attr_accessor :current_options
16
+
17
+ def method_added(method)
18
+ return if @current_options.nil?
19
+
20
+ mod = self
21
+ klass = Class.new { include mod }
22
+ bound_method = klass.instance_method(method).bind(klass.new)
23
+
24
+ task_name = @current_options[:name] ||
25
+ TaskSupport.task_name(mod.name, method)
26
+
27
+ task = Task.new(@current_options.merge(
28
+ :name => task_name,
29
+ :method => bound_method
30
+ ))
31
+
32
+ # current_options must be nil'ed before we redefine
33
+ # the method as doing so would trigger this method
34
+ # again and cause an infinite loop
35
+ @current_options = nil
36
+ mod.module_eval do
37
+ alias_method :"_#{method}", method
38
+
39
+ define_method(method) do |*args|
40
+ if args.length.zero?
41
+ task
42
+ else
43
+ send(:"_#{method}", *args)
44
+ end
45
+ end
46
+ end
47
+
48
+ RCelery::Task.all_tasks[task_name] = task
49
+ end
50
+
51
+ def task(options = {})
52
+ @current_options = options
53
+ end
54
+ end
55
+ end
56
+ end