aboutsummaryrefslogtreecommitdiffstats
path: root/activesupport/lib
diff options
context:
space:
mode:
authorJosé Valim <jose.valim@gmail.com>2011-12-25 07:55:24 -0800
committerJosé Valim <jose.valim@gmail.com>2011-12-25 07:55:24 -0800
commitd04428df4904822962c587525843ee0a11ec2cd5 (patch)
treed18e4b31ca5dba8ba5d54c040f57c9360cbc4dc5 /activesupport/lib
parent1acdecf6234b6a8d6dc01d1ada1b8e153c7c60fb (diff)
parent66a587cf5590d37483d8aba64da5022df08ecf07 (diff)
downloadrails-d04428df4904822962c587525843ee0a11ec2cd5.tar.gz
rails-d04428df4904822962c587525843ee0a11ec2cd5.tar.bz2
rails-d04428df4904822962c587525843ee0a11ec2cd5.zip
Merge pull request #4182 from bogdan/non_keyed_callbacks
AS::Callbacks: improve __define_runner perfomance for non-keyed callbacks.
Diffstat (limited to 'activesupport/lib')
-rw-r--r--activesupport/lib/active_support/callbacks.rb21
1 files changed, 11 insertions, 10 deletions
diff --git a/activesupport/lib/active_support/callbacks.rb b/activesupport/lib/active_support/callbacks.rb
index 11069301f1..ba9fb4ecce 100644
--- a/activesupport/lib/active_support/callbacks.rb
+++ b/activesupport/lib/active_support/callbacks.rb
@@ -378,18 +378,15 @@ module ActiveSupport
module ClassMethods
# Generate the internal runner method called by +run_callbacks+.
def __define_runner(symbol) #:nodoc:
- body = send("_#{symbol}_callbacks").compile
- runner_method = "_run_#{symbol}_callbacks"
+ name = __callback_runner_name(nil, symbol)
+ undef_method(name) if method_defined?(name)
silence_warnings do
+ runner_method = "_run_#{symbol}_callbacks"
undef_method runner_method if method_defined?(runner_method)
class_eval <<-RUBY_EVAL, __FILE__, __LINE__ + 1
def #{runner_method}(key = nil, &blk)
- if key
- self.class.__run_keyed_callback(key, :#{symbol}, self, &blk)
- else
- #{body}
- end
+ self.class.__run_callback(key, :#{symbol}, self, &blk)
end
private :#{runner_method}
RUBY_EVAL
@@ -400,10 +397,10 @@ module ActiveSupport
# If this called first time it creates a new callback method for the key,
# calculating which callbacks can be omitted because of per_key conditions.
#
- def __run_keyed_callback(key, kind, object, &blk) #:nodoc:
- name = "_run__#{self.name.hash.abs}__#{kind}__#{key.hash.abs}__callbacks"
+ def __run_callback(key, kind, object, &blk) #:nodoc:
+ name = __callback_runner_name(key, kind)
unless object.respond_to?(name)
- str = send("_#{kind}_callbacks").compile(name, object)
+ str = send("_#{kind}_callbacks").compile(key, object)
class_eval <<-RUBY_EVAL, __FILE__, __LINE__ + 1
def #{name}() #{str} end
protected :#{name}
@@ -412,6 +409,10 @@ module ActiveSupport
object.send(name, &blk)
end
+ def __callback_runner_name(key, kind)
+ "_run__#{self.name.hash.abs}__#{kind}__#{key.hash.abs}__callbacks"
+ end
+
# This is used internally to append, prepend and skip callbacks to the
# CallbackChain.
#