2010-03-28 08:15:02 -04:00
|
|
|
require 'active_support/core_ext/object/blank'
|
|
|
|
|
2010-01-16 07:17:03 -05:00
|
|
|
module ActionDispatch
|
|
|
|
module Http
|
|
|
|
module Cache
|
|
|
|
module Request
|
|
|
|
def if_modified_since
|
|
|
|
if since = env['HTTP_IF_MODIFIED_SINCE']
|
|
|
|
Time.rfc2822(since) rescue nil
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def if_none_match
|
|
|
|
env['HTTP_IF_NONE_MATCH']
|
|
|
|
end
|
|
|
|
|
|
|
|
def not_modified?(modified_at)
|
|
|
|
if_modified_since && modified_at && if_modified_since >= modified_at
|
|
|
|
end
|
|
|
|
|
|
|
|
def etag_matches?(etag)
|
|
|
|
if_none_match && if_none_match == etag
|
|
|
|
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
|
2010-10-17 17:08:41 -04:00
|
|
|
attr_reader :cache_control, :etag
|
|
|
|
alias :etag? :etag
|
2010-02-19 22:19:20 -05:00
|
|
|
|
|
|
|
def initialize(*)
|
|
|
|
status, header, body = super
|
|
|
|
|
|
|
|
@cache_control = {}
|
|
|
|
@etag = self["ETag"]
|
|
|
|
|
|
|
|
if cache_control = self["Cache-Control"]
|
|
|
|
cache_control.split(/,\s*/).each do |segment|
|
|
|
|
first, last = segment.split("=")
|
2010-09-22 15:40:14 -04:00
|
|
|
@cache_control[first.to_sym] = last || true
|
2010-02-19 22:19:20 -05:00
|
|
|
end
|
|
|
|
end
|
2010-01-16 07:17:03 -05:00
|
|
|
end
|
|
|
|
|
|
|
|
def last_modified
|
|
|
|
if last = headers['Last-Modified']
|
|
|
|
Time.httpdate(last)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def last_modified?
|
|
|
|
headers.include?('Last-Modified')
|
|
|
|
end
|
|
|
|
|
|
|
|
def last_modified=(utc_time)
|
|
|
|
headers['Last-Modified'] = utc_time.httpdate
|
|
|
|
end
|
|
|
|
|
|
|
|
def etag=(etag)
|
|
|
|
key = ActiveSupport::Cache.expand_cache_key(etag)
|
2010-02-19 22:19:20 -05:00
|
|
|
@etag = self["ETag"] = %("#{Digest::MD5.hexdigest(key)}")
|
2010-01-16 07:17:03 -05:00
|
|
|
end
|
|
|
|
|
|
|
|
private
|
|
|
|
|
|
|
|
def handle_conditional_get!
|
|
|
|
if etag? || last_modified? || !@cache_control.empty?
|
|
|
|
set_conditional_cache_control!
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
DEFAULT_CACHE_CONTROL = "max-age=0, private, must-revalidate"
|
|
|
|
|
|
|
|
def set_conditional_cache_control!
|
2010-02-19 22:19:20 -05:00
|
|
|
return if self["Cache-Control"].present?
|
|
|
|
|
2010-08-11 17:54:15 -04:00
|
|
|
control = @cache_control
|
|
|
|
|
2010-01-16 07:17:03 -05:00
|
|
|
if control.empty?
|
|
|
|
headers["Cache-Control"] = DEFAULT_CACHE_CONTROL
|
2010-10-17 17:08:41 -04:00
|
|
|
elsif control[:no_cache]
|
2010-01-16 07:17:03 -05:00
|
|
|
headers["Cache-Control"] = "no-cache"
|
|
|
|
else
|
|
|
|
extras = control[:extras]
|
|
|
|
max_age = control[:max_age]
|
|
|
|
|
|
|
|
options = []
|
|
|
|
options << "max-age=#{max_age.to_i}" if max_age
|
|
|
|
options << (control[:public] ? "public" : "private")
|
|
|
|
options << "must-revalidate" if control[:must_revalidate]
|
|
|
|
options.concat(extras) if extras
|
|
|
|
|
|
|
|
headers["Cache-Control"] = options.join(", ")
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
2010-06-28 02:42:13 -04:00
|
|
|
end
|