mirror of
https://github.com/rails/rails.git
synced 2022-11-09 12:12:34 -05:00
758ad13c58
In applications which use :all helpers (the default), most controllers won't be making modifications to their _helpers module. In CRuby this created many ICLASS objects which could cause a large increase in memory usage in applications with many controllers and helpers. To avoid creating unnecessary modules this PR builds modules only when a modification is being made: ethier by calling `helper`, `helper_method`, or through having a default helper (one matching the controller's name) included onto it.
292 lines
8.2 KiB
Ruby
292 lines
8.2 KiB
Ruby
# frozen_string_literal: true
|
|
|
|
require "active_support/core_ext/module/redefine_method"
|
|
require "action_controller"
|
|
require "action_controller/test_case"
|
|
require "action_view"
|
|
|
|
require "rails-dom-testing"
|
|
|
|
module ActionView
|
|
# = Action View Test Case
|
|
class TestCase < ActiveSupport::TestCase
|
|
class TestController < ActionController::Base
|
|
include ActionDispatch::TestProcess
|
|
|
|
attr_accessor :request, :response, :params
|
|
|
|
class << self
|
|
# Overrides AbstractController::Base#controller_path
|
|
attr_accessor :controller_path
|
|
end
|
|
|
|
def controller_path=(path)
|
|
self.class.controller_path = path
|
|
end
|
|
|
|
def initialize
|
|
super
|
|
self.class.controller_path = ""
|
|
@request = ActionController::TestRequest.create(self.class)
|
|
@response = ActionDispatch::TestResponse.new
|
|
|
|
@request.env.delete("PATH_INFO")
|
|
@params = ActionController::Parameters.new
|
|
end
|
|
end
|
|
|
|
module Behavior
|
|
extend ActiveSupport::Concern
|
|
|
|
include ActionDispatch::Assertions, ActionDispatch::TestProcess
|
|
include Rails::Dom::Testing::Assertions
|
|
include ActionController::TemplateAssertions
|
|
include ActionView::Context
|
|
|
|
include ActionDispatch::Routing::PolymorphicRoutes
|
|
|
|
include AbstractController::Helpers
|
|
include ActionView::Helpers
|
|
include ActionView::RecordIdentifier
|
|
include ActionView::RoutingUrlFor
|
|
|
|
include ActiveSupport::Testing::ConstantLookup
|
|
|
|
delegate :lookup_context, to: :controller
|
|
attr_accessor :controller, :output_buffer, :rendered
|
|
|
|
module ClassMethods
|
|
def tests(helper_class)
|
|
case helper_class
|
|
when String, Symbol
|
|
self.helper_class = "#{helper_class.to_s.underscore}_helper".camelize.safe_constantize
|
|
when Module
|
|
self.helper_class = helper_class
|
|
end
|
|
end
|
|
|
|
def determine_default_helper_class(name)
|
|
determine_constant_from_test_name(name) do |constant|
|
|
Module === constant && !(Class === constant)
|
|
end
|
|
end
|
|
|
|
def helper_method(*methods)
|
|
# Almost a duplicate from ActionController::Helpers
|
|
methods.flatten.each do |method|
|
|
_helpers_for_modification.module_eval <<-end_eval, __FILE__, __LINE__ + 1
|
|
def #{method}(*args, &block) # def current_user(*args, &block)
|
|
_test_case.send(:'#{method}', *args, &block) # _test_case.send(:'current_user', *args, &block)
|
|
end # end
|
|
ruby2_keywords(:'#{method}') if respond_to?(:ruby2_keywords, true)
|
|
end_eval
|
|
end
|
|
end
|
|
|
|
attr_writer :helper_class
|
|
|
|
def helper_class
|
|
@helper_class ||= determine_default_helper_class(name)
|
|
end
|
|
|
|
def new(*)
|
|
include_helper_modules!
|
|
super
|
|
end
|
|
|
|
private
|
|
def include_helper_modules!
|
|
helper(helper_class) if helper_class
|
|
include _helpers
|
|
end
|
|
end
|
|
|
|
def setup_with_controller
|
|
controller_class = Class.new(ActionView::TestCase::TestController)
|
|
@controller = controller_class.new
|
|
@request = @controller.request
|
|
@view_flow = ActionView::OutputFlow.new
|
|
# empty string ensures buffer has UTF-8 encoding as
|
|
# new without arguments returns ASCII-8BIT encoded buffer like String#new
|
|
@output_buffer = ActiveSupport::SafeBuffer.new ""
|
|
@rendered = +""
|
|
|
|
test_case_instance = self
|
|
controller_class.define_method(:_test_case) { test_case_instance }
|
|
end
|
|
|
|
def config
|
|
@controller.config if @controller.respond_to?(:config)
|
|
end
|
|
|
|
def render(options = {}, local_assigns = {}, &block)
|
|
view.assign(view_assigns)
|
|
@rendered << output = view.render(options, local_assigns, &block)
|
|
output
|
|
end
|
|
|
|
def rendered_views
|
|
@_rendered_views ||= RenderedViewsCollection.new
|
|
end
|
|
|
|
def _routes
|
|
@controller._routes if @controller.respond_to?(:_routes)
|
|
end
|
|
|
|
# Need to experiment if this priority is the best one: rendered => output_buffer
|
|
class RenderedViewsCollection
|
|
def initialize
|
|
@rendered_views ||= Hash.new { |hash, key| hash[key] = [] }
|
|
end
|
|
|
|
def add(view, locals)
|
|
@rendered_views[view] ||= []
|
|
@rendered_views[view] << locals
|
|
end
|
|
|
|
def locals_for(view)
|
|
@rendered_views[view]
|
|
end
|
|
|
|
def rendered_views
|
|
@rendered_views.keys
|
|
end
|
|
|
|
def view_rendered?(view, expected_locals)
|
|
locals_for(view).any? do |actual_locals|
|
|
expected_locals.all? { |key, value| value == actual_locals[key] }
|
|
end
|
|
end
|
|
end
|
|
|
|
included do
|
|
setup :setup_with_controller
|
|
ActiveSupport.run_load_hooks(:action_view_test_case, self)
|
|
|
|
helper do
|
|
def protect_against_forgery?
|
|
false
|
|
end
|
|
|
|
def _test_case
|
|
controller._test_case
|
|
end
|
|
end
|
|
end
|
|
|
|
private
|
|
# Need to experiment if this priority is the best one: rendered => output_buffer
|
|
def document_root_element
|
|
Nokogiri::HTML::Document.parse(@rendered.blank? ? @output_buffer : @rendered).root
|
|
end
|
|
|
|
module Locals
|
|
attr_accessor :rendered_views
|
|
|
|
def render(options = {}, local_assigns = {})
|
|
case options
|
|
when Hash
|
|
if block_given?
|
|
rendered_views.add options[:layout], options[:locals]
|
|
elsif options.key?(:partial)
|
|
rendered_views.add options[:partial], options[:locals]
|
|
end
|
|
else
|
|
rendered_views.add options, local_assigns
|
|
end
|
|
|
|
super
|
|
end
|
|
end
|
|
|
|
# The instance of ActionView::Base that is used by +render+.
|
|
def view
|
|
@view ||= begin
|
|
view = @controller.view_context
|
|
view.singleton_class.include(_helpers)
|
|
view.extend(Locals)
|
|
view.rendered_views = rendered_views
|
|
view.output_buffer = output_buffer
|
|
view
|
|
end
|
|
end
|
|
|
|
alias_method :_view, :view
|
|
|
|
INTERNAL_IVARS = [
|
|
:@NAME,
|
|
:@failures,
|
|
:@assertions,
|
|
:@__io__,
|
|
:@_assertion_wrapped,
|
|
:@_assertions,
|
|
:@_result,
|
|
:@_routes,
|
|
:@controller,
|
|
:@_layouts,
|
|
:@_files,
|
|
:@_rendered_views,
|
|
:@method_name,
|
|
:@output_buffer,
|
|
:@_partials,
|
|
:@passed,
|
|
:@rendered,
|
|
:@request,
|
|
:@routes,
|
|
:@tagged_logger,
|
|
:@_templates,
|
|
:@options,
|
|
:@test_passed,
|
|
:@view,
|
|
:@view_context_class,
|
|
:@view_flow,
|
|
:@_subscribers,
|
|
:@html_document
|
|
]
|
|
|
|
def _user_defined_ivars
|
|
instance_variables - INTERNAL_IVARS
|
|
end
|
|
|
|
# Returns a Hash of instance variables and their values, as defined by
|
|
# the user in the test case, which are then assigned to the view being
|
|
# rendered. This is generally intended for internal use and extension
|
|
# frameworks.
|
|
def view_assigns
|
|
Hash[_user_defined_ivars.map do |ivar|
|
|
[ivar[1..-1].to_sym, instance_variable_get(ivar)]
|
|
end]
|
|
end
|
|
|
|
def method_missing(selector, *args)
|
|
begin
|
|
routes = @controller.respond_to?(:_routes) && @controller._routes
|
|
rescue
|
|
# Don't call routes, if there is an error on _routes call
|
|
end
|
|
|
|
if routes &&
|
|
(routes.named_routes.route_defined?(selector) ||
|
|
routes.mounted_helpers.method_defined?(selector))
|
|
@controller.__send__(selector, *args)
|
|
else
|
|
super
|
|
end
|
|
end
|
|
|
|
def respond_to_missing?(name, include_private = false)
|
|
begin
|
|
routes = defined?(@controller) && @controller.respond_to?(:_routes) && @controller._routes
|
|
rescue
|
|
# Don't call routes, if there is an error on _routes call
|
|
end
|
|
|
|
routes &&
|
|
(routes.named_routes.route_defined?(name) ||
|
|
routes.mounted_helpers.method_defined?(name))
|
|
end
|
|
end
|
|
|
|
include Behavior
|
|
end
|
|
end
|