cache.rb 4.9 KB
Newer Older
1

2 3 4 5
module ActionDispatch
  module Http
    module Cache
      module Request
6 7 8 9

        HTTP_IF_MODIFIED_SINCE = 'HTTP_IF_MODIFIED_SINCE'.freeze
        HTTP_IF_NONE_MATCH     = 'HTTP_IF_NONE_MATCH'.freeze

10
        def if_modified_since
11
          if since = env[HTTP_IF_MODIFIED_SINCE]
12 13 14 15 16
            Time.rfc2822(since) rescue nil
          end
        end

        def if_none_match
17
          env[HTTP_IF_NONE_MATCH]
18 19
        end

20 21 22 23 24 25
        def if_none_match_etags
          (if_none_match ? if_none_match.split(/\s*,\s*/) : []).collect do |etag|
            etag.gsub(/^\"|\"$/, "")
          end
        end

26 27 28 29 30
        def not_modified?(modified_at)
          if_modified_since && modified_at && if_modified_since >= modified_at
        end

        def etag_matches?(etag)
31 32 33 34
          if etag
            etag = etag.gsub(/^\"|\"$/, "")
            if_none_match_etags.include?(etag)
          end
35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53
        end

        # Check response freshness (Last-Modified and ETag) against request
        # If-Modified-Since and If-None-Match conditions. If both headers are
        # supplied, both must match, or the request is not considered fresh.
        def fresh?(response)
          last_modified = if_modified_since
          etag          = if_none_match

          return false unless last_modified || etag

          success = true
          success &&= not_modified?(response.last_modified) if last_modified
          success &&= etag_matches?(response.etag) if etag
          success
        end
      end

      module Response
54 55
        attr_reader :cache_control, :etag
        alias :etag? :etag
56

57
        def last_modified
58
          if last = headers[LAST_MODIFIED]
59 60 61 62 63
            Time.httpdate(last)
          end
        end

        def last_modified?
64
          headers.include?(LAST_MODIFIED)
65 66 67
        end

        def last_modified=(utc_time)
68
          headers[LAST_MODIFIED] = utc_time.httpdate
69 70
        end

A
arvida 已提交
71 72 73 74 75 76 77 78 79 80 81 82 83 84
        def date
          if date_header = headers['Date']
            Time.httpdate(date_header)
          end
        end

        def date?
          headers.include?('Date')
        end

        def date=(utc_time)
          headers['Date'] = utc_time.httpdate
        end

85 86
        def etag=(etag)
          key = ActiveSupport::Cache.expand_cache_key(etag)
87
          @etag = self[ETAG] = %("#{Digest::MD5.hexdigest(key)}")
88 89 90 91
        end

      private

92 93 94
        LAST_MODIFIED = "Last-Modified".freeze
        ETAG          = "ETag".freeze
        CACHE_CONTROL = "Cache-Control".freeze
95 96
        SPESHUL_KEYS  = %w[extras no-cache max-age public must-revalidate]

97 98 99 100 101 102 103 104
        def cache_control_segments
          if cache_control = self[CACHE_CONTROL]
            cache_control.delete(' ').split(',')
          else
            []
          end
        end

105 106
        def cache_control_headers
          cache_control = {}
107 108 109 110 111 112 113 114 115 116

          cache_control_segments.each do |segment|
            directive, argument = segment.split('=', 2)

            if SPESHUL_KEYS.include? directive
              key = directive.tr('-', '_')
              cache_control[key.to_sym] = argument || true
            else
              cache_control[:extras] ||= []
              cache_control[:extras] << segment
117 118
            end
          end
119

120 121
          cache_control
        end
122

123
        def prepare_cache_control!
124
          @cache_control = cache_control_headers
125
          @etag = self[ETAG]
126 127
        end

128 129 130 131 132 133
        def handle_conditional_get!
          if etag? || last_modified? || !@cache_control.empty?
            set_conditional_cache_control!
          end
        end

134 135 136 137 138
        DEFAULT_CACHE_CONTROL = "max-age=0, private, must-revalidate".freeze
        NO_CACHE              = "no-cache".freeze
        PUBLIC                = "public".freeze
        PRIVATE               = "private".freeze
        MUST_REVALIDATE       = "must-revalidate".freeze
139 140

        def set_conditional_cache_control!
141 142 143 144 145 146 147
          control = {}
          cc_headers = cache_control_headers
          if extras = cc_headers.delete(:extras)
            @cache_control[:extras] ||= []
            @cache_control[:extras] += extras
            @cache_control[:extras].uniq!
          end
148

149 150
          control.merge! cc_headers
          control.merge! @cache_control
151

152
          if control.empty?
153
            headers[CACHE_CONTROL] = DEFAULT_CACHE_CONTROL
154
          elsif control[:no_cache]
155
            headers[CACHE_CONTROL] = NO_CACHE
156 157 158
            if control[:extras]
              headers[CACHE_CONTROL] += ", #{control[:extras].join(', ')}"
            end
159 160 161 162 163 164
          else
            extras  = control[:extras]
            max_age = control[:max_age]

            options = []
            options << "max-age=#{max_age.to_i}" if max_age
165 166
            options << (control[:public] ? PUBLIC : PRIVATE)
            options << MUST_REVALIDATE if control[:must_revalidate]
167 168
            options.concat(extras) if extras

169
            headers[CACHE_CONTROL] = options.join(", ")
170 171 172 173 174
          end
        end
      end
    end
  end
175
end