mirror of
https://github.com/rails/rails.git
synced 2022-11-09 12:12:34 -05:00
Fixes Sam Ruby tests suite.
Signed-off-by: Yehuda Katz <wycats@gmail.com>
This commit is contained in:
parent
1bbb9b2db0
commit
2407479603
4 changed files with 113 additions and 56 deletions
|
@ -2,7 +2,7 @@ module ActionDispatch
|
||||||
class Callbacks
|
class Callbacks
|
||||||
include ActiveSupport::NewCallbacks
|
include ActiveSupport::NewCallbacks
|
||||||
|
|
||||||
define_callbacks :call, :terminator => "result == false", :scope => :kind
|
define_callbacks :call, :terminator => "result == false", :rescuable => true
|
||||||
define_callbacks :prepare, :scope => :name
|
define_callbacks :prepare, :scope => :name
|
||||||
|
|
||||||
# Add a preparation callback. Preparation callbacks are run before every
|
# Add a preparation callback. Preparation callbacks are run before every
|
||||||
|
@ -25,10 +25,6 @@ module ActionDispatch
|
||||||
set_callback(:call, :before, *args, &block)
|
set_callback(:call, :before, *args, &block)
|
||||||
end
|
end
|
||||||
|
|
||||||
def self.around(*args, &block)
|
|
||||||
set_callback(:call, :around, *args, &block)
|
|
||||||
end
|
|
||||||
|
|
||||||
def self.after(*args, &block)
|
def self.after(*args, &block)
|
||||||
set_callback(:call, :after, *args, &block)
|
set_callback(:call, :after, *args, &block)
|
||||||
end
|
end
|
||||||
|
@ -36,7 +32,6 @@ module ActionDispatch
|
||||||
class << self
|
class << self
|
||||||
# DEPRECATED
|
# DEPRECATED
|
||||||
alias_method :before_dispatch, :before
|
alias_method :before_dispatch, :before
|
||||||
alias_method :around_dispatch, :around
|
|
||||||
alias_method :after_dispatch, :after
|
alias_method :after_dispatch, :after
|
||||||
end
|
end
|
||||||
|
|
||||||
|
|
|
@ -631,9 +631,8 @@ class FragmentCachingTest < ActionController::TestCase
|
||||||
buffer = 'generated till now -> '
|
buffer = 'generated till now -> '
|
||||||
@controller.fragment_for(buffer, 'expensive') { fragment_computed = true }
|
@controller.fragment_for(buffer, 'expensive') { fragment_computed = true }
|
||||||
|
|
||||||
assert_equal 2, listener.size
|
assert_equal 1, listener.count { |e| e.name == :fragment_exist? }
|
||||||
assert_equal :fragment_exist?, listener[0].name
|
assert_equal 1, listener.count { |e| e.name == :write_fragment }
|
||||||
assert_equal :write_fragment, listener[1].name
|
|
||||||
|
|
||||||
assert fragment_computed
|
assert fragment_computed
|
||||||
assert_equal 'generated till now -> ', buffer
|
assert_equal 'generated till now -> ', buffer
|
||||||
|
|
|
@ -92,10 +92,10 @@ module ActiveSupport
|
||||||
class Callback
|
class Callback
|
||||||
@@_callback_sequence = 0
|
@@_callback_sequence = 0
|
||||||
|
|
||||||
attr_accessor :name, :filter, :kind, :options, :per_key, :klass
|
attr_accessor :chain, :filter, :kind, :options, :per_key, :klass
|
||||||
|
|
||||||
def initialize(name, filter, kind, options, klass)
|
def initialize(chain, filter, kind, options, klass)
|
||||||
@name, @kind, @klass = name, kind, klass
|
@chain, @kind, @klass = chain, kind, klass
|
||||||
normalize_options!(options)
|
normalize_options!(options)
|
||||||
|
|
||||||
@per_key = options.delete(:per_key)
|
@per_key = options.delete(:per_key)
|
||||||
|
@ -107,9 +107,9 @@ module ActiveSupport
|
||||||
_compile_per_key_options
|
_compile_per_key_options
|
||||||
end
|
end
|
||||||
|
|
||||||
def clone(klass)
|
def clone(chain, klass)
|
||||||
obj = super()
|
obj = super()
|
||||||
obj.name = name
|
obj.chain = chain
|
||||||
obj.klass = klass
|
obj.klass = klass
|
||||||
obj.per_key = @per_key.dup
|
obj.per_key = @per_key.dup
|
||||||
obj.options = @options.dup
|
obj.options = @options.dup
|
||||||
|
@ -117,7 +117,6 @@ module ActiveSupport
|
||||||
obj.per_key[:unless] = @per_key[:unless].dup
|
obj.per_key[:unless] = @per_key[:unless].dup
|
||||||
obj.options[:if] = @options[:if].dup
|
obj.options[:if] = @options[:if].dup
|
||||||
obj.options[:unless] = @options[:unless].dup
|
obj.options[:unless] = @options[:unless].dup
|
||||||
obj.options[:scope] = @options[:scope].dup
|
|
||||||
obj
|
obj
|
||||||
end
|
end
|
||||||
|
|
||||||
|
@ -125,14 +124,15 @@ module ActiveSupport
|
||||||
options[:if] = Array.wrap(options[:if])
|
options[:if] = Array.wrap(options[:if])
|
||||||
options[:unless] = Array.wrap(options[:unless])
|
options[:unless] = Array.wrap(options[:unless])
|
||||||
|
|
||||||
options[:scope] ||= [:kind]
|
|
||||||
options[:scope] = Array.wrap(options[:scope])
|
|
||||||
|
|
||||||
options[:per_key] ||= {}
|
options[:per_key] ||= {}
|
||||||
options[:per_key][:if] = Array.wrap(options[:per_key][:if])
|
options[:per_key][:if] = Array.wrap(options[:per_key][:if])
|
||||||
options[:per_key][:unless] = Array.wrap(options[:per_key][:unless])
|
options[:per_key][:unless] = Array.wrap(options[:per_key][:unless])
|
||||||
end
|
end
|
||||||
|
|
||||||
|
def name
|
||||||
|
chain.name
|
||||||
|
end
|
||||||
|
|
||||||
def next_id
|
def next_id
|
||||||
@@_callback_sequence += 1
|
@@_callback_sequence += 1
|
||||||
end
|
end
|
||||||
|
@ -168,15 +168,12 @@ module ActiveSupport
|
||||||
|
|
||||||
# This will supply contents for before and around filters, and no
|
# This will supply contents for before and around filters, and no
|
||||||
# contents for after filters (for the forward pass).
|
# contents for after filters (for the forward pass).
|
||||||
def start(key = nil, options = {})
|
def start(key=nil, object=nil)
|
||||||
object, terminator = (options || {}).values_at(:object, :terminator)
|
|
||||||
return if key && !object.send("_one_time_conditions_valid_#{@callback_id}?")
|
return if key && !object.send("_one_time_conditions_valid_#{@callback_id}?")
|
||||||
|
|
||||||
terminator ||= false
|
|
||||||
|
|
||||||
# options[0] is the compiled form of supplied conditions
|
# options[0] is the compiled form of supplied conditions
|
||||||
# options[1] is the "end" for the conditional
|
# options[1] is the "end" for the conditional
|
||||||
|
#
|
||||||
if @kind == :before || @kind == :around
|
if @kind == :before || @kind == :around
|
||||||
if @kind == :before
|
if @kind == :before
|
||||||
# if condition # before_save :filter_name, :if => :condition
|
# if condition # before_save :filter_name, :if => :condition
|
||||||
|
@ -185,7 +182,7 @@ module ActiveSupport
|
||||||
filter = <<-RUBY_EVAL
|
filter = <<-RUBY_EVAL
|
||||||
unless halted
|
unless halted
|
||||||
result = #{@filter}
|
result = #{@filter}
|
||||||
halted = (#{terminator})
|
halted = (#{chain.config[:terminator]})
|
||||||
end
|
end
|
||||||
RUBY_EVAL
|
RUBY_EVAL
|
||||||
|
|
||||||
|
@ -226,8 +223,7 @@ module ActiveSupport
|
||||||
|
|
||||||
# This will supply contents for around and after filters, but not
|
# This will supply contents for around and after filters, but not
|
||||||
# before filters (for the backward pass).
|
# before filters (for the backward pass).
|
||||||
def end(key = nil, options = {})
|
def end(key=nil, object=nil)
|
||||||
object = (options || {})[:object]
|
|
||||||
return if key && !object.send("_one_time_conditions_valid_#{@callback_id}?")
|
return if key && !object.send("_one_time_conditions_valid_#{@callback_id}?")
|
||||||
|
|
||||||
if @kind == :around || @kind == :after
|
if @kind == :around || @kind == :after
|
||||||
|
@ -302,7 +298,8 @@ module ActiveSupport
|
||||||
@klass.send(:define_method, "#{method_name}_object") { filter }
|
@klass.send(:define_method, "#{method_name}_object") { filter }
|
||||||
|
|
||||||
_normalize_legacy_filter(kind, filter)
|
_normalize_legacy_filter(kind, filter)
|
||||||
method_to_call = @options[:scope].map{ |s| s.is_a?(Symbol) ? send(s) : s }.join("_")
|
scopes = Array.wrap(chain.config[:scope])
|
||||||
|
method_to_call = scopes.map{ |s| s.is_a?(Symbol) ? send(s) : s }.join("_")
|
||||||
|
|
||||||
@klass.class_eval <<-RUBY_EVAL, __FILE__, __LINE__ + 1
|
@klass.class_eval <<-RUBY_EVAL, __FILE__, __LINE__ + 1
|
||||||
def #{method_name}(&blk)
|
def #{method_name}(&blk)
|
||||||
|
@ -331,37 +328,52 @@ module ActiveSupport
|
||||||
|
|
||||||
# An Array with a compile method
|
# An Array with a compile method
|
||||||
class CallbackChain < Array
|
class CallbackChain < Array
|
||||||
attr_reader :symbol, :config
|
attr_reader :name, :config
|
||||||
|
|
||||||
def initialize(symbol, config)
|
def initialize(name, config)
|
||||||
@symbol = symbol
|
@name = name
|
||||||
@config = config
|
@config = {
|
||||||
|
:terminator => "false",
|
||||||
|
:rescuable => false,
|
||||||
|
:scope => [ :kind ]
|
||||||
|
}.merge(config)
|
||||||
end
|
end
|
||||||
|
|
||||||
def compile(key=nil, options={})
|
def compile(key=nil, object=nil)
|
||||||
options = config.merge(options)
|
|
||||||
|
|
||||||
method = []
|
method = []
|
||||||
method << "value = nil"
|
method << "value = nil"
|
||||||
method << "halted = false"
|
method << "halted = false"
|
||||||
|
|
||||||
each do |callback|
|
each do |callback|
|
||||||
method << callback.start(key, options)
|
method << callback.start(key, object)
|
||||||
|
end
|
||||||
|
|
||||||
|
if config[:rescuable]
|
||||||
|
method << "rescued_error = nil"
|
||||||
|
method << "begin"
|
||||||
end
|
end
|
||||||
|
|
||||||
method << "value = yield if block_given? && !halted"
|
method << "value = yield if block_given? && !halted"
|
||||||
|
|
||||||
reverse_each do |callback|
|
if config[:rescuable]
|
||||||
method << callback.end(key, options)
|
method << "rescue Exception => e"
|
||||||
|
method << "rescued_error = e"
|
||||||
|
method << "end"
|
||||||
end
|
end
|
||||||
|
|
||||||
|
reverse_each do |callback|
|
||||||
|
method << callback.end(key, object)
|
||||||
|
end
|
||||||
|
|
||||||
|
method << "raise rescued_error if rescued_error" if config[:rescuable]
|
||||||
method << "halted ? false : (block_given? ? value : true)"
|
method << "halted ? false : (block_given? ? value : true)"
|
||||||
method.compact.join("\n")
|
method.compact.join("\n")
|
||||||
end
|
end
|
||||||
|
|
||||||
def clone(klass)
|
def clone(klass)
|
||||||
chain = CallbackChain.new(@symbol, @config.dup)
|
chain = CallbackChain.new(@name, @config.dup)
|
||||||
chain.push(*map {|c| c.clone(klass)})
|
callbacks = map { |c| c.clone(chain, klass) }
|
||||||
|
chain.push(*callbacks)
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
|
@ -407,32 +419,35 @@ module ActiveSupport
|
||||||
# key. It creates a new callback method for the key, calculating
|
# key. It creates a new callback method for the key, calculating
|
||||||
# which callbacks can be omitted because of per_key conditions.
|
# which callbacks can be omitted because of per_key conditions.
|
||||||
#
|
#
|
||||||
def __create_keyed_callback(name, kind, obj, &blk) #:nodoc:
|
def __create_keyed_callback(name, kind, object, &blk) #:nodoc:
|
||||||
@_keyed_callbacks ||= {}
|
@_keyed_callbacks ||= {}
|
||||||
@_keyed_callbacks[name] ||= begin
|
@_keyed_callbacks[name] ||= begin
|
||||||
str = send("_#{kind}_callbacks").compile(name, :object => obj)
|
str = send("_#{kind}_callbacks").compile(name, object)
|
||||||
class_eval "def #{name}() #{str} end", __FILE__, __LINE__
|
class_eval "def #{name}() #{str} end", __FILE__, __LINE__
|
||||||
true
|
true
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
def __update_callbacks(name, filters = CallbackChain.new(name, {}), block = nil)
|
# This is used internally to append, prepend and skip callbacks to the
|
||||||
|
# CallbackChain.
|
||||||
|
#
|
||||||
|
def __update_callbacks(name, filters = [], block = nil) #:nodoc:
|
||||||
type = [:before, :after, :around].include?(filters.first) ? filters.shift : :before
|
type = [:before, :after, :around].include?(filters.first) ? filters.shift : :before
|
||||||
options = filters.last.is_a?(Hash) ? filters.pop : {}
|
options = filters.last.is_a?(Hash) ? filters.pop : {}
|
||||||
filters.unshift(block) if block
|
filters.unshift(block) if block
|
||||||
|
|
||||||
callbacks = send("_#{name}_callbacks")
|
chain = send("_#{name}_callbacks")
|
||||||
yield callbacks, type, filters, options if block_given?
|
yield chain, type, filters, options if block_given?
|
||||||
|
|
||||||
__define_runner(name)
|
__define_runner(name)
|
||||||
end
|
end
|
||||||
|
|
||||||
# Define callbacks.
|
# Set callbacks for a previously defined callback.
|
||||||
#
|
#
|
||||||
# Syntax:
|
# Syntax:
|
||||||
# set_callback :save, :before, :before_meth
|
# set_callback :save, :before, :before_meth
|
||||||
# set_callback :save, :after, :after_meth, :if => :condition
|
# set_callback :save, :after, :after_meth, :if => :condition
|
||||||
# set_callback :save, :around {|r| stuff; yield; stuff }
|
# set_callback :save, :around, lambda { |r| stuff; yield; stuff }
|
||||||
#
|
#
|
||||||
# It also updates the _run_<name>_callbacks method, which is the public
|
# It also updates the _run_<name>_callbacks method, which is the public
|
||||||
# API to run the callbacks. Use skip_callback to skip any defined one.
|
# API to run the callbacks. Use skip_callback to skip any defined one.
|
||||||
|
@ -448,43 +463,91 @@ module ActiveSupport
|
||||||
# Per-Key conditions are evaluated only once per use of a given key.
|
# Per-Key conditions are evaluated only once per use of a given key.
|
||||||
# In the case of the above example, you would do:
|
# In the case of the above example, you would do:
|
||||||
#
|
#
|
||||||
# run_dispatch_callbacks(action_name) { ... dispatch stuff ... }
|
# _run_dispatch_callbacks(action_name) { ... dispatch stuff ... }
|
||||||
#
|
#
|
||||||
# In that case, each action_name would get its own compiled callback
|
# In that case, each action_name would get its own compiled callback
|
||||||
# method that took into consideration the per_key conditions. This
|
# method that took into consideration the per_key conditions. This
|
||||||
# is a speed improvement for ActionPack.
|
# is a speed improvement for ActionPack.
|
||||||
#
|
#
|
||||||
def set_callback(name, *filters, &block)
|
def set_callback(name, *filters, &block)
|
||||||
__update_callbacks(name, filters, block) do |callbacks, type, filters, options|
|
__update_callbacks(name, filters, block) do |chain, type, filters, options|
|
||||||
filters.map! do |filter|
|
filters.map! do |filter|
|
||||||
callbacks.delete_if {|c| c.matches?(type, filter) }
|
chain.delete_if {|c| c.matches?(type, filter) }
|
||||||
Callback.new(name, filter, type, options.merge(callbacks.config), self)
|
Callback.new(chain, filter, type, options.dup, self)
|
||||||
end
|
end
|
||||||
|
|
||||||
options[:prepend] ? callbacks.unshift(*filters) : callbacks.push(*filters)
|
options[:prepend] ? chain.unshift(*filters) : chain.push(*filters)
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
|
# Skip a previously defined callback for a given type.
|
||||||
|
#
|
||||||
def skip_callback(name, *filters, &block)
|
def skip_callback(name, *filters, &block)
|
||||||
__update_callbacks(name, filters, block) do |callbacks, type, filters, options|
|
__update_callbacks(name, filters, block) do |chain, type, filters, options|
|
||||||
|
chain = send("_#{name}_callbacks=", chain.clone(self))
|
||||||
|
|
||||||
filters.each do |filter|
|
filters.each do |filter|
|
||||||
callbacks = send("_#{name}_callbacks=", callbacks.clone(self))
|
filter = chain.find {|c| c.matches?(type, filter) }
|
||||||
filter = callbacks.find {|c| c.matches?(type, filter) }
|
|
||||||
|
|
||||||
if filter && options.any?
|
if filter && options.any?
|
||||||
filter.recompile!(options, options[:per_key] || {})
|
filter.recompile!(options, options[:per_key] || {})
|
||||||
else
|
else
|
||||||
callbacks.delete(filter)
|
chain.delete(filter)
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
|
# Reset callbacks for a given type.
|
||||||
|
#
|
||||||
def reset_callbacks(symbol)
|
def reset_callbacks(symbol)
|
||||||
send("_#{symbol}_callbacks").clear
|
send("_#{symbol}_callbacks").clear
|
||||||
__define_runner(symbol)
|
__define_runner(symbol)
|
||||||
end
|
end
|
||||||
|
|
||||||
|
# Define callbacks types.
|
||||||
|
#
|
||||||
|
# ==== Example
|
||||||
|
#
|
||||||
|
# define_callbacks :validate
|
||||||
|
#
|
||||||
|
# ==== Options
|
||||||
|
#
|
||||||
|
# * <tt>:terminator</tt> - Indicates when a before filter is considered
|
||||||
|
# to be halted.
|
||||||
|
#
|
||||||
|
# define_callbacks :validate, :terminator => "result == false"
|
||||||
|
#
|
||||||
|
# In the example above, if any before validate callbacks returns false,
|
||||||
|
# other callbacks are not executed. Defaults to "false".
|
||||||
|
#
|
||||||
|
# * <tt>:rescuable</tt> - By default, after filters are not executed if
|
||||||
|
# the given block or an before_filter raises an error. Supply :rescuable => true
|
||||||
|
# to change this behavior.
|
||||||
|
#
|
||||||
|
# * <tt>:scope</tt> - Show which methods should be executed when a class
|
||||||
|
# is giben as callback:
|
||||||
|
#
|
||||||
|
# define_callbacks :filters, :scope => [ :kind ]
|
||||||
|
#
|
||||||
|
# When a class is given:
|
||||||
|
#
|
||||||
|
# before_filter MyFilter
|
||||||
|
#
|
||||||
|
# It will call the type of the filter in the given class, which in this
|
||||||
|
# case, is "before".
|
||||||
|
#
|
||||||
|
# If, for instance, you supply the given scope:
|
||||||
|
#
|
||||||
|
# define_callbacks :validate, :scope => [ :kind, :name ]
|
||||||
|
#
|
||||||
|
# It will call "#{kind}_#{name}" in the given class. So "before_validate"
|
||||||
|
# will be called in the class below:
|
||||||
|
#
|
||||||
|
# before_validate MyValidation
|
||||||
|
#
|
||||||
|
# Defaults to :kind.
|
||||||
|
#
|
||||||
def define_callbacks(*symbols)
|
def define_callbacks(*symbols)
|
||||||
config = symbols.last.is_a?(Hash) ? symbols.pop : {}
|
config = symbols.last.is_a?(Hash) ? symbols.pop : {}
|
||||||
symbols.each do |symbol|
|
symbols.each do |symbol|
|
||||||
|
|
Loading…
Reference in a new issue