Module: ActionController::Caching::Pages

Defined in:
lib/action_controller/caching/pages.rb

Overview

Page caching is an approach to caching where the entire action output of is stored as a HTML file that the web server can serve without going through Action Pack. This is the fastest way to cache your content as opposed to going dynamically through the process of generating the content. Unfortunately, this incredible speed-up is only available to stateless pages where all visitors are treated the same. Content management systems – including weblogs and wikis – have many pages that are a great fit for this approach, but account-based systems where people log in and manipulate their own data are often less likely candidates.

Specifying which actions to cache is done through the caches_page class method:

class WeblogController < ActionController::Base
  caches_page :show, :new
end

This will generate cache files such as weblog/show/5.html and weblog/new.html, which match the URLs used to trigger the dynamic generation. This is how the web server is able pick up a cache file when it exists and otherwise let the request pass on to Action Pack to generate it.

Expiration of the cache is handled by deleting the cached file, which results in a lazy regeneration approach where the cache is not restored before another hit is made against it. The API for doing so mimics the options from url_for and friends:

class WeblogController < ActionController::Base
  def update
    List.update(params[:list][:id], params[:list])
    expire_page :action => "show", :id => params[:list][:id]
    redirect_to :action => "show", :id => params[:list][:id]
  end
end

Additionally, you can expire caches using Sweepers that act on changes in the model to determine when a cache is supposed to be expired.

Defined Under Namespace

Modules: ClassMethods

Class Method Summary collapse

Instance Method Summary collapse

Class Method Details

.included(base) ⇒ Object

:nodoc:



37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
# File 'lib/action_controller/caching/pages.rb', line 37

def self.included(base) #:nodoc:
  base.extend(ClassMethods)
  base.class_eval do
    @@page_cache_directory = defined?(Rails.public_path) ? Rails.public_path : ""
    ##
    # :singleton-method:
    # The cache directory should be the document root for the web server and is set using <tt>Base.page_cache_directory = "/document/root"</tt>.
    # For Rails, this directory has already been set to Rails.public_path (which is usually set to <tt>RAILS_ROOT + "/public"</tt>). Changing
    # this setting can be useful to avoid naming conflicts with files in <tt>public/</tt>, but doing so will likely require configuring your
    # web server to look in the new location for cached files.
    cattr_accessor :page_cache_directory

    @@page_cache_extension = '.html'
    ##
    # :singleton-method:
    # Most Rails requests do not have an extension, such as <tt>/weblog/new</tt>. In these cases, the page caching mechanism will add one in
    # order to make it easy for the cached files to be picked up properly by the web server. By default, this cache extension is <tt>.html</tt>.
    # If you want something else, like <tt>.php</tt> or <tt>.shtml</tt>, just set Base.page_cache_extension. In cases where a request already has an
    # extension, such as <tt>.xml</tt> or <tt>.rss</tt>, page caching will not add an extension. This allows it to work well with RESTful apps.
    cattr_accessor :page_cache_extension
  end
end

Instance Method Details

#cache_page(content = nil, options = nil) ⇒ Object

Manually cache the content in the key determined by options. If no content is provided, the contents of response.body is used If no options are provided, the requested url is used. Example:

cache_page "I'm the cached content", :controller => "lists", :action => "show"


131
132
133
134
135
136
137
138
139
140
141
142
143
144
# File 'lib/action_controller/caching/pages.rb', line 131

def cache_page(content = nil, options = nil)
  return unless perform_caching && caching_allowed

  path = case options
    when Hash
      url_for(options.merge(:only_path => true, :skip_relative_url_root => true, :format => params[:format]))
    when String
      options
    else
      request.path
  end

  self.class.cache_page(content || response.body, path)
end

#expire_page(options = {}) ⇒ Object

Expires the page that was cached with the options as a key. Example:

expire_page :controller => "lists", :action => "show"


112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
# File 'lib/action_controller/caching/pages.rb', line 112

def expire_page(options = {})
  return unless perform_caching

  if options.is_a?(Hash)
    if options[:action].is_a?(Array)
      options[:action].dup.each do |action|
        self.class.expire_page(url_for(options.merge(:only_path => true, :skip_relative_url_root => true, :action => action)))
      end
    else
      self.class.expire_page(url_for(options.merge(:only_path => true, :skip_relative_url_root => true)))
    end
  else
    self.class.expire_page(options)
  end
end