ETag Support from Edge-Rails
Added ETag support from http://dev.rubyonrails.org/changeset/6158
This commit is contained in:
parent
e4e26400ef
commit
457ec8627c
|
@ -7,6 +7,7 @@ require 'action_controller/url_rewriter'
|
||||||
require 'action_controller/status_codes'
|
require 'action_controller/status_codes'
|
||||||
require 'drb'
|
require 'drb'
|
||||||
require 'set'
|
require 'set'
|
||||||
|
require 'md5'
|
||||||
|
|
||||||
module ActionController #:nodoc:
|
module ActionController #:nodoc:
|
||||||
class ActionControllerError < StandardError #:nodoc:
|
class ActionControllerError < StandardError #:nodoc:
|
||||||
|
@ -599,6 +600,12 @@ module ActionController #:nodoc:
|
||||||
# _Deprecation_ _notice_: This used to have the signatures
|
# _Deprecation_ _notice_: This used to have the signatures
|
||||||
# <tt>render_partial(partial_path = default_template_name, object = nil, local_assigns = {})</tt> and
|
# <tt>render_partial(partial_path = default_template_name, object = nil, local_assigns = {})</tt> and
|
||||||
# <tt>render_partial_collection(partial_name, collection, partial_spacer_template = nil, local_assigns = {})</tt>.
|
# <tt>render_partial_collection(partial_name, collection, partial_spacer_template = nil, local_assigns = {})</tt>.
|
||||||
|
# == Automatic etagging
|
||||||
|
#
|
||||||
|
# Rendering will automatically insert the etag header on 200 OK responses. The etag is calculated using MD5 of the
|
||||||
|
# response body. If a request comes in that has a matching etag, the response will be changed to a 304 Not Modified
|
||||||
|
# and the response body will be set to an empty string.
|
||||||
|
#
|
||||||
#
|
#
|
||||||
# === Rendering a template
|
# === Rendering a template
|
||||||
#
|
#
|
||||||
|
@ -822,6 +829,16 @@ module ActionController #:nodoc:
|
||||||
else
|
else
|
||||||
response.body = text
|
response.body = text
|
||||||
end
|
end
|
||||||
|
if response.headers['Status'] == "200 OK" && response.body.size > 0
|
||||||
|
response.headers['Etag'] = "\"#{MD5.new(text).to_s}\""
|
||||||
|
|
||||||
|
if request.headers['HTTP_IF_NONE_MATCH'] == response.headers['Etag']
|
||||||
|
response.headers['Status'] = "304 Not Modified"
|
||||||
|
response.body = ''
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
response.body
|
||||||
end
|
end
|
||||||
|
|
||||||
def render_javascript(javascript, status = nil, append_response = true) #:nodoc:
|
def render_javascript(javascript, status = nil, append_response = true) #:nodoc:
|
||||||
|
|
|
@ -48,6 +48,10 @@ module ActionController
|
||||||
# REQUEST_METHOD header directly. Thus, for head, both get? and head? will return true.
|
# REQUEST_METHOD header directly. Thus, for head, both get? and head? will return true.
|
||||||
def head?
|
def head?
|
||||||
@env['REQUEST_METHOD'].downcase.to_sym == :head
|
@env['REQUEST_METHOD'].downcase.to_sym == :head
|
||||||
|
end
|
||||||
|
|
||||||
|
def headers
|
||||||
|
@env
|
||||||
end
|
end
|
||||||
|
|
||||||
# Determine whether the body of a HTTP call is URL-encoded (default)
|
# Determine whether the body of a HTTP call is URL-encoded (default)
|
||||||
|
|
|
@ -69,6 +69,10 @@ class TestController < ActionController::Base
|
||||||
render "test/hello"
|
render "test/hello"
|
||||||
end
|
end
|
||||||
|
|
||||||
|
def heading
|
||||||
|
head :ok
|
||||||
|
end
|
||||||
|
|
||||||
def greeting
|
def greeting
|
||||||
# let's just rely on the template
|
# let's just rely on the template
|
||||||
end
|
end
|
||||||
|
@ -286,8 +290,50 @@ class RenderTest < Test::Unit::TestCase
|
||||||
assert_equal "Goodbye, Local David", @response.body
|
assert_equal "Goodbye, Local David", @response.body
|
||||||
end
|
end
|
||||||
|
|
||||||
|
def test_render_200_should_set_etag
|
||||||
|
get :render_hello_world_from_variable
|
||||||
|
assert_equal etag_for("hello david"), @response.headers['Etag']
|
||||||
|
end
|
||||||
|
|
||||||
|
def test_render_against_etag_request_should_304_when_match
|
||||||
|
@request.headers["HTTP_IF_NONE_MATCH"] = etag_for("hello david")
|
||||||
|
get :render_hello_world_from_variable
|
||||||
|
assert_equal "304 Not Modified", @response.headers['Status']
|
||||||
|
assert @response.body.empty?
|
||||||
|
end
|
||||||
|
|
||||||
|
def test_render_against_etag_request_should_200_when_no_match
|
||||||
|
@request.headers["HTTP_IF_NONE_MATCH"] = etag_for("hello somewhere else")
|
||||||
|
get :render_hello_world_from_variable
|
||||||
|
assert_equal "200 OK", @response.headers['Status']
|
||||||
|
assert !@response.body.empty?
|
||||||
|
end
|
||||||
|
|
||||||
|
def test_render_with_etag
|
||||||
|
get :render_hello_world_from_variable
|
||||||
|
expected_etag = "\"#{MD5.new("hello david").to_s}\""
|
||||||
|
assert_equal expected_etag, @response.headers['Etag']
|
||||||
|
|
||||||
|
@request.headers["HTTP_IF_NONE_MATCH"] = expected_etag
|
||||||
|
get :render_hello_world_from_variable
|
||||||
|
assert_equal "304 Not Modified", @response.headers['Status']
|
||||||
|
|
||||||
|
@request.headers["HTTP_IF_NONE_MATCH"] = "\"diftag\""
|
||||||
|
get :render_hello_world_from_variable
|
||||||
|
assert_equal "200 OK", @response.headers['Status']
|
||||||
|
end
|
||||||
|
|
||||||
|
def render_with_404_shouldnt_have_etag
|
||||||
|
get :render_custom_code
|
||||||
|
assert_nil @response.headers['Etag']
|
||||||
|
end
|
||||||
|
|
||||||
protected
|
protected
|
||||||
def assert_deprecated_render(&block)
|
def assert_deprecated_render(&block)
|
||||||
assert_deprecated(/render/, &block)
|
assert_deprecated(/render/, &block)
|
||||||
end
|
end
|
||||||
|
|
||||||
|
def etag_for(text)
|
||||||
|
"\"#{MD5.new(text).to_s}\""
|
||||||
|
end
|
||||||
end
|
end
|
||||||
|
|
Loading…
Reference in a new issue