Skip to content

Commit

Permalink
Take variants into account when calculating template digests in Actio…
Browse files Browse the repository at this point in the history
…nView::Digestor
  • Loading branch information
dhh committed Mar 5, 2014
1 parent f441ac7 commit d7cf95c
Show file tree
Hide file tree
Showing 6 changed files with 104 additions and 24 deletions.
4 changes: 2 additions & 2 deletions actionpack/test/controller/caching_test.rb
Expand Up @@ -243,8 +243,8 @@ def test_xml_formatted_fragment_caching
end end


private private
def template_digest(name, format) def template_digest(name, format, variant = nil)
ActionView::Digestor.digest(name, format, @controller.lookup_context) ActionView::Digestor.digest(name: name, format: format, variant: variant, finder: @controller.lookup_context)
end end
end end


Expand Down
8 changes: 8 additions & 0 deletions actionview/CHANGELOG.md
@@ -1,3 +1,11 @@
* Take variants into account when calculating template digests in ActionView::Digestor.

The arguments to ActionView::Digestor#digest are now being passed as a hash
to support variants and allow more flexibility in the future. The support for
regular (required) arguments is deprecated and will be removed in Rails 5.0 or later.

*Piotr Chmolowski, Łukasz Strzałkowski*

* Fixed a problem where the default options for the `button_tag` helper is not * Fixed a problem where the default options for the `button_tag` helper is not
applied correctly. applied correctly.


Expand Down
64 changes: 51 additions & 13 deletions actionview/lib/action_view/digestor.rb
Expand Up @@ -9,23 +9,56 @@ class Digestor
@@digest_monitor = Monitor.new @@digest_monitor = Monitor.new


class << self class << self
def digest(name, format, finder, options = {}) # Supported options:
#
# * <tt>name</tt> - Template name
# * <tt>format</tt> - Template format
# * <tt>variant</tt> - Variant of +format+ (optional)
# * <tt>finder</tt> - An instance of ActionView::LookupContext
# * <tt>dependencies</tt> - An array of dependent views
# * <tt>partial</tt> - Specifies whether the template is a partial
def digest(*args)
options = _setup_options(*args)

name = options[:name]
format = options[:format]
variant = options[:variant]
finder = options[:finder]

details_key = finder.details_key.hash details_key = finder.details_key.hash
dependencies = Array.wrap(options[:dependencies]) dependencies = Array.wrap(options[:dependencies])
cache_key = ([name, details_key, format] + dependencies).join('.') cache_key = ([name, details_key, format, variant].compact + dependencies).join('.')


# this is a correctly done double-checked locking idiom # this is a correctly done double-checked locking idiom
# (ThreadSafe::Cache's lookups have volatile semantics) # (ThreadSafe::Cache's lookups have volatile semantics)
@@cache[cache_key] || @@digest_monitor.synchronize do @@cache[cache_key] || @@digest_monitor.synchronize do
@@cache.fetch(cache_key) do # re-check under lock @@cache.fetch(cache_key) do # re-check under lock
compute_and_store_digest(cache_key, name, format, finder, options) compute_and_store_digest(cache_key, options)
end end
end end
end end


def _setup_options(*args)
unless args.first.is_a?(Hash)
ActiveSupport::Deprecation.warn("Arguments to ActionView::Digestor should be provided as a hash. The support for regular arguments will be removed in Rails 5.0 or later")

{
name: args.first,
format: args.second,
finder: args.third,
}.merge(args.fourth || {})
else
options = args.first
options.assert_valid_keys(:name, :format, :variant, :finder, :dependencies, :partial)

options
end
end

private private
def compute_and_store_digest(cache_key, name, format, finder, options) # called under @@digest_monitor lock
klass = if options[:partial] || name.include?("/_") def compute_and_store_digest(cache_key, options) # called under @@digest_monitor lock
klass = if options[:partial] || options[:name].include?("/_")
# Prevent re-entry or else recursive templates will blow the stack. # Prevent re-entry or else recursive templates will blow the stack.
# There is no need to worry about other threads seeing the +false+ value, # There is no need to worry about other threads seeing the +false+ value,
# as they will then have to wait for this thread to let go of the @@digest_monitor lock. # as they will then have to wait for this thread to let go of the @@digest_monitor lock.
Expand All @@ -35,20 +68,25 @@ def compute_and_store_digest(cache_key, name, format, finder, options) # called
Digestor Digestor
end end


digest = klass.new(name, format, finder, options).digest digest = klass.new(options).digest
# Store the actual digest if config.cache_template_loading is true # Store the actual digest if config.cache_template_loading is true
@@cache[cache_key] = stored_digest = digest if ActionView::Resolver.caching? @@cache[cache_key] = stored_digest = digest if ActionView::Resolver.caching?
digest digest
ensure ensure
# something went wrong or ActionView::Resolver.caching? is false, make sure not to corrupt the @@cache # something went wrong or ActionView::Resolver.caching? is false, make sure not to corrupt the @@cache
@@cache.delete_pair(cache_key, false) if pre_stored && !stored_digest @@cache.delete_pair(cache_key, false) if pre_stored && !stored_digest
end end
end end


attr_reader :name, :format, :finder, :options attr_reader :name, :format, :variant, :finder, :options

def initialize(*args)
@options = self.class._setup_options(*args)


def initialize(name, format, finder, options={}) @name = @options.delete(:name)
@name, @format, @finder, @options = name, format, finder, options @format = @options.delete(:format)
@variant = @options.delete(:variant)
@finder = @options.delete(:finder)
end end


def digest def digest
Expand All @@ -68,7 +106,7 @@ def dependencies


def nested_dependencies def nested_dependencies
dependencies.collect do |dependency| dependencies.collect do |dependency|
dependencies = PartialDigestor.new(dependency, format, finder).nested_dependencies dependencies = PartialDigestor.new(name: dependency, format: format, finder: finder).nested_dependencies
dependencies.any? ? { dependency => dependencies } : dependency dependencies.any? ? { dependency => dependencies } : dependency
end end
end end
Expand All @@ -88,7 +126,7 @@ def partial?
end end


def template def template
@template ||= finder.find(logical_name, [], partial?, formats: [ format ]) @template ||= finder.find(logical_name, [], partial?, formats: [ format ], variants: [ variant ])
end end


def source def source
Expand All @@ -97,7 +135,7 @@ def source


def dependency_digest def dependency_digest
template_digests = dependencies.collect do |template_name| template_digests = dependencies.collect do |template_name|
Digestor.digest(template_name, format, finder, partial: true) Digestor.digest(name: template_name, format: format, finder: finder, partial: true)
end end


(template_digests + injected_dependencies).join("-") (template_digests + injected_dependencies).join("-")
Expand Down
2 changes: 1 addition & 1 deletion actionview/lib/action_view/helpers/cache_helper.rb
Expand Up @@ -167,7 +167,7 @@ def fragment_name_with_digest(name) #:nodoc:
if @virtual_path if @virtual_path
[ [
*Array(name.is_a?(Hash) ? controller.url_for(name).split("://").last : name), *Array(name.is_a?(Hash) ? controller.url_for(name).split("://").last : name),
Digestor.digest(@virtual_path, formats.last.to_sym, lookup_context, dependencies: view_cache_dependencies) Digestor.digest(name: @virtual_path, format: formats.last.to_sym, variant: request.variant, finder: lookup_context, dependencies: view_cache_dependencies)
] ]
else else
name name
Expand Down
15 changes: 15 additions & 0 deletions actionview/test/fixtures/digestor/messages/new.html+iphone.erb
@@ -0,0 +1,15 @@
<%# Template Dependency: messages/message %>
<%= render "header" %>
<%= render "comments/comments" %>
<%= render "messages/actions/move" %>
<%= render @message.history.events %>
<%# render "something_missing" %>
<%# render "something_missing_1" %>
<%
# Template Dependency: messages/form
%>
35 changes: 27 additions & 8 deletions actionview/test/template/digestor_test.rb
Expand Up @@ -26,7 +26,11 @@ def details_key
end end


def find(logical_name, keys, partial, options) def find(logical_name, keys, partial, options)
FixtureTemplate.new("digestor/#{partial ? logical_name.gsub(%r|/([^/]+)$|, '/_\1') : logical_name}.#{options[:formats].first}.erb") partial_name = partial ? logical_name.gsub(%r|/([^/]+)$|, '/_\1') : logical_name
format = options[:formats].first.to_s
format += "+#{options[:variants].first}" if options[:variants].any?

FixtureTemplate.new("digestor/#{partial_name}.#{format}.erb")
end end
end end


Expand Down Expand Up @@ -194,6 +198,13 @@ def test_old_style_hash_in_render_invocation
end end
end end


def test_variants
assert_digest_difference("messages/new", false, variant: :iphone) do
change_template("messages/new", :iphone)
change_template("messages/_header", :iphone)
end
end

def test_dependencies_via_options_results_in_different_digest def test_dependencies_via_options_results_in_different_digest
digest_plain = digest("comments/_comment") digest_plain = digest("comments/_comment")
digest_fridge = digest("comments/_comment", dependencies: ["fridge"]) digest_fridge = digest("comments/_comment", dependencies: ["fridge"])
Expand Down Expand Up @@ -242,6 +253,11 @@ def test_digest_cache_cleanup_with_recursion_and_template_caching_off
ActionView::Resolver.caching = resolver_before ActionView::Resolver.caching = resolver_before
end end


def test_arguments_deprecation
assert_deprecated(/should be provided as a hash/) { ActionView::Digestor.digest('messages/show', :html, finder) }
assert_deprecated(/should be provided as a hash/) { ActionView::Digestor.new('messages/show', :html, finder) }
end

private private
def assert_logged(message) def assert_logged(message)
old_logger = ActionView::Base.logger old_logger = ActionView::Base.logger
Expand All @@ -258,26 +274,29 @@ def assert_logged(message)
end end
end end


def assert_digest_difference(template_name, persistent = false) def assert_digest_difference(template_name, persistent = false, options = {})
previous_digest = digest(template_name) previous_digest = digest(template_name, options)
ActionView::Digestor.cache.clear unless persistent ActionView::Digestor.cache.clear unless persistent


yield yield


assert previous_digest != digest(template_name), "digest didn't change" assert previous_digest != digest(template_name, options), "digest didn't change"
ActionView::Digestor.cache.clear ActionView::Digestor.cache.clear
end end


def digest(template_name, options={}) def digest(template_name, options = {})
ActionView::Digestor.digest(template_name, :html, finder, options) options = options.dup
ActionView::Digestor.digest({ name: template_name, format: :html, finder: finder }.merge(options))
end end


def finder def finder
@finder ||= FixtureFinder.new @finder ||= FixtureFinder.new
end end


def change_template(template_name) def change_template(template_name, variant = nil)
File.open("digestor/#{template_name}.html.erb", "w") do |f| variant = "+#{variant}" if variant.present?

File.open("digestor/#{template_name}.html#{variant}.erb", "w") do |f|
f.write "\nTHIS WAS CHANGED!" f.write "\nTHIS WAS CHANGED!"
end end
end end
Expand Down

0 comments on commit d7cf95c

Please sign in to comment.