actions.rb 7.8 KB
Newer Older
1 2 3 4 5 6 7 8 9 10 11
require 'set'

module ActionController #:nodoc:
  module Caching
    # Action caching is similar to page caching by the fact that the entire output of the response is cached, but unlike page caching,
    # every request still goes through the Action Pack. The key benefit of this is that filters are run before the cache is served, which
    # allows for authentication and other restrictions on whether someone is allowed to see the cache. Example:
    #
    #   class ListsController < ApplicationController
    #     before_filter :authenticate, :except => :public
    #     caches_page   :public
12
    #     caches_action :index, :show, :feed
13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29
    #   end
    #
    # In this example, the public action doesn't require authentication, so it's possible to use the faster page caching method. But both the
    # show and feed action are to be shielded behind the authenticate filter, so we need to implement those as action caches.
    #
    # Action caching internally uses the fragment caching and an around filter to do the job. The fragment cache is named according to both
    # the current host and the path. So a page that is accessed at http://david.somewhere.com/lists/show/1 will result in a fragment named
    # "david.somewhere.com/lists/show/1". This allows the cacher to differentiate between "david.somewhere.com/lists/" and
    # "jamis.somewhere.com/lists/" -- which is a helpful way of assisting the subdomain-as-account-key pattern.
    #
    # Different representations of the same resource, e.g. <tt>http://david.somewhere.com/lists</tt> and <tt>http://david.somewhere.com/lists.xml</tt>
    # are treated like separate requests and so are cached separately. Keep in mind when expiring an action cache that <tt>:action => 'lists'</tt> is not the same
    # as <tt>:action => 'list', :format => :xml</tt>.
    #
    # You can set modify the default action cache path by passing a :cache_path option.  This will be passed directly to ActionCachePath.path_for.  This is handy
    # for actions with multiple possible routes that should be cached differently.  If a block is given, it is called with the current controller instance.
    #
30 31 32
    # And you can also use :if (or :unless) to pass a Proc that specifies when the action should be cached.
    #
    # Finally, if you are using memcached, you can also pass :expires_in.
33
    #
34 35 36
    #   class ListsController < ApplicationController
    #     before_filter :authenticate, :except => :public
    #     caches_page   :public
37
    #     caches_action :index, :if => Proc.new { |c| !c.request.format.json? } # cache if is not a JSON request
38
    #     caches_action :show, :cache_path => { :project => 1 }, :expires_in => 1.hour
39 40
    #     caches_action :feed, :cache_path => Proc.new { |controller|
    #       controller.params[:user_id] ?
P
Pratik Naik 已提交
41 42
    #         controller.send(:user_list_url, controller.params[:user_id], controller.params[:id]) :
    #         controller.send(:list_url, controller.params[:id]) }
43
    #   end
44
    #
45 46
    # If you pass :layout => false, it will only cache your action content. It is useful when your layout has dynamic information.
    #
47 48 49 50 51 52 53 54 55 56 57 58 59
    module Actions
      def self.included(base) #:nodoc:
        base.extend(ClassMethods)
          base.class_eval do
            attr_accessor :rendered_action_cache, :action_cache_path
          end
      end

      module ClassMethods
        # Declares that +actions+ should be cached.
        # See ActionController::Caching::Actions for details.
        def caches_action(*actions)
          return unless cache_configured?
60
          options = actions.extract_options!
61 62 63
          filter_options = { :only => actions, :if => options.delete(:if), :unless => options.delete(:unless) }

          cache_filter = ActionCacheFilter.new(:layout => options.delete(:layout), :cache_path => options.delete(:cache_path), :store_options => options)
64 65 66
          around_filter(filter_options) do |controller, action|
            cache_filter.filter(controller, action)
          end
67 68 69 70 71 72 73 74 75
        end
      end

      protected
        def expire_action(options = {})
          return unless cache_configured?

          if options[:action].is_a?(Array)
            options[:action].dup.each do |action|
76
              expire_fragment(ActionCachePath.path_for(self, options.merge({ :action => action }), false))
77 78
            end
          else
79
            expire_fragment(ActionCachePath.path_for(self, options, false))
80 81 82 83
          end
        end

      class ActionCacheFilter #:nodoc:
84 85
        def initialize(options, &block)
          @options = options
86 87
        end

88 89 90 91 92 93
        def filter(controller, action)
          should_continue = before(controller)
          action.call if should_continue
          after(controller)
        end

94
        def before(controller)
95 96
          cache_path = ActionCachePath.new(controller, path_options_for(controller, @options.slice(:cache_path)))
          if cache = controller.read_fragment(cache_path.path, @options[:store_options])
97 98
            controller.rendered_action_cache = true
            set_content_type!(controller, cache_path.extension)
99 100
            options = { :text => cache }
            options.merge!(:layout => true) if cache_layout?
101
            controller.__send__(:render, options)
102 103 104 105 106 107 108
            false
          else
            controller.action_cache_path = cache_path
          end
        end

        def after(controller)
109
          return if controller.rendered_action_cache || !caching_allowed(controller)
110
          action_content = cache_layout? ? content_for_layout(controller) : controller.response.body
111
          controller.write_fragment(controller.action_cache_path.path, action_content, @options[:store_options])
112 113 114 115 116 117 118 119 120 121 122 123
        end

        private
          def set_content_type!(controller, extension)
            controller.response.content_type = Mime::Type.lookup_by_extension(extension).to_s if extension
          end

          def path_options_for(controller, options)
            ((path_options = options[:cache_path]).respond_to?(:call) ? path_options.call(controller) : path_options) || {}
          end

          def caching_allowed(controller)
124
            controller.request.get? && controller.response.status.to_i == 200
125
          end
126 127 128 129 130 131

          def cache_layout?
            @options[:layout] == false
          end

          def content_for_layout(controller)
132
            controller.template.layout && controller.template.instance_variable_get('@cached_content_for_layout')
133
          end
134
      end
135

136 137
      class ActionCachePath
        attr_reader :path, :extension
138

139
        class << self
140
          def path_for(controller, options, infer_extension = true)
141
            new(controller, options, infer_extension).path
142 143
          end
        end
144
        
L
lifo 已提交
145
        # If +infer_extension+ is true, the cache path extension is looked up from the request's path & format.
146 147 148 149 150 151
        # This is desirable when reading and writing the cache, but not when expiring the cache -
        # expire_action should expire the same files regardless of the request format.
        def initialize(controller, options = {}, infer_extension = true)
          if infer_extension
            extract_extension(controller.request)
            options = options.reverse_merge(:format => @extension) if options.is_a?(Hash)
152
          end
153

154 155
          path = controller.url_for(options).split('://').last
          normalize!(path)
156
          add_extension!(path, @extension)
157 158
          @path = URI.unescape(path)
        end
159

160 161 162 163
        private
          def normalize!(path)
            path << 'index' if path[-1] == ?/
          end
164

165
          def add_extension!(path, extension)
166
            path << ".#{extension}" if extension and !path.ends_with?(extension)
167
          end
168 169
          
          def extract_extension(request)
170 171
            # Don't want just what comes after the last '.' to accommodate multi part extensions
            # such as tar.gz.
172
            @extension = request.path[/^[^.]+\.(.+)$/, 1] || request.cache_format
173 174 175 176
          end
      end
    end
  end
177
end