1
0
Fork 0
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:
José Valim 2009-09-21 21:47:21 -03:00 committed by Yehuda Katz
parent 1bbb9b2db0
commit 2407479603
4 changed files with 113 additions and 56 deletions

View file

@ -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

View file

@ -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

View file

@ -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|

View file

@ -112,4 +112,4 @@ class InheritedCallbacksTest2 < Test::Unit::TestCase
def test_crazy_mix_off def test_crazy_mix_off
assert_equal %w(before1 before2 update after2 after1), @update2.log assert_equal %w(before1 before2 update after2 after1), @update2.log
end end
end end