mirror of
https://github.com/rails/rails.git
synced 2022-11-09 12:12:34 -05:00
7c988f8030
Previously, when content_for is flushed, the content was replaced directly by a new value in ActionView::OutputFlow#set. The problem is this new value passed to the method may not be an instance of ActiveSupport::SafeBuffer. This change forces the value to be set to a new instance of ActiveSupport::SafeBuffer.
76 lines
1.7 KiB
Ruby
76 lines
1.7 KiB
Ruby
require 'active_support/core_ext/string/output_safety'
|
|
|
|
module ActionView
|
|
class OutputFlow #:nodoc:
|
|
attr_reader :content
|
|
|
|
def initialize
|
|
@content = Hash.new { |h,k| h[k] = ActiveSupport::SafeBuffer.new }
|
|
end
|
|
|
|
# Called by _layout_for to read stored values.
|
|
def get(key)
|
|
@content[key]
|
|
end
|
|
|
|
# Called by each renderer object to set the layout contents.
|
|
def set(key, value)
|
|
@content[key] = ActiveSupport::SafeBuffer.new(value)
|
|
end
|
|
|
|
# Called by content_for
|
|
def append(key, value)
|
|
@content[key] << value
|
|
end
|
|
alias_method :append!, :append
|
|
|
|
end
|
|
|
|
class StreamingFlow < OutputFlow #:nodoc:
|
|
def initialize(view, fiber)
|
|
@view = view
|
|
@parent = nil
|
|
@child = view.output_buffer
|
|
@content = view.view_flow.content
|
|
@fiber = fiber
|
|
@root = Fiber.current.object_id
|
|
end
|
|
|
|
# Try to get stored content. If the content
|
|
# is not available and we are inside the layout
|
|
# fiber, we set that we are waiting for the given
|
|
# key and yield.
|
|
def get(key)
|
|
return super if @content.key?(key)
|
|
|
|
if inside_fiber?
|
|
view = @view
|
|
|
|
begin
|
|
@waiting_for = key
|
|
view.output_buffer, @parent = @child, view.output_buffer
|
|
Fiber.yield
|
|
ensure
|
|
@waiting_for = nil
|
|
view.output_buffer, @child = @parent, view.output_buffer
|
|
end
|
|
end
|
|
|
|
super
|
|
end
|
|
|
|
# Appends the contents for the given key. This is called
|
|
# by provides and resumes back to the fiber if it is
|
|
# the key it is waiting for.
|
|
def append!(key, value)
|
|
super
|
|
@fiber.resume if @waiting_for == key
|
|
end
|
|
|
|
private
|
|
|
|
def inside_fiber?
|
|
Fiber.current.object_id != @root
|
|
end
|
|
end
|
|
end
|