aboutsummaryrefslogtreecommitdiffstats
path: root/actionpack
diff options
context:
space:
mode:
authorDavid Heinemeier Hansson <david@loudthinking.com>2007-02-19 00:27:19 +0000
committerDavid Heinemeier Hansson <david@loudthinking.com>2007-02-19 00:27:19 +0000
commit7ec0204ecd7557a63267ff64cc412338176a0805 (patch)
tree82ba2ca3e3e1d5367267db828581a096a425eb04 /actionpack
parentaad7fbde68684547959dcccc2102c978d5347a78 (diff)
downloadrails-7ec0204ecd7557a63267ff64cc412338176a0805.tar.gz
rails-7ec0204ecd7557a63267ff64cc412338176a0805.tar.bz2
rails-7ec0204ecd7557a63267ff64cc412338176a0805.zip
Move etagging down to response, so renders with layouts dont screw it up [DHH]
git-svn-id: http://svn-commit.rubyonrails.org/rails/trunk@6165 5ecf4fe2-1ee6-0310-87b1-e25e094e27de
Diffstat (limited to 'actionpack')
-rwxr-xr-xactionpack/lib/action_controller/base.rb17
-rw-r--r--actionpack/lib/action_controller/cgi_process.rb22
-rwxr-xr-xactionpack/lib/action_controller/response.rb53
-rw-r--r--actionpack/test/controller/render_test.rb1
4 files changed, 50 insertions, 43 deletions
diff --git a/actionpack/lib/action_controller/base.rb b/actionpack/lib/action_controller/base.rb
index c012abc448..360e115f16 100755
--- a/actionpack/lib/action_controller/base.rb
+++ b/actionpack/lib/action_controller/base.rb
@@ -7,7 +7,6 @@ require 'action_controller/url_rewriter'
require 'action_controller/status_codes'
require 'drb'
require 'set'
-require 'digest/md5'
module ActionController #:nodoc:
class ActionControllerError < StandardError #:nodoc:
@@ -474,6 +473,9 @@ module ActionController #:nodoc:
send(method, *arguments)
assign_default_content_type_and_charset
+
+ response.request = request
+ response.prepare!
response
ensure
process_cleanup
@@ -876,20 +878,7 @@ module ActionController #:nodoc:
response.body << text
else
response.body = text
-
- if text.is_a?(String)
- if response.headers['Status'][0..2] == '200' && !response.body.empty?
- response.headers['Etag'] = %("#{Digest::MD5.hexdigest(text)}")
-
- if request.headers['HTTP_IF_NONE_MATCH'] == response.headers['Etag']
- response.headers['Status'] = "304 Not Modified"
- response.body = ''
- end
- end
- end
end
-
- response.body
end
def render_javascript(javascript, status = nil, append_response = true) #:nodoc:
diff --git a/actionpack/lib/action_controller/cgi_process.rb b/actionpack/lib/action_controller/cgi_process.rb
index 2af1654eec..aa4cba1066 100644
--- a/actionpack/lib/action_controller/cgi_process.rb
+++ b/actionpack/lib/action_controller/cgi_process.rb
@@ -184,9 +184,6 @@ end_msg
end
def out(output = $stdout)
- convert_content_type!
- set_content_length!
-
output.binmode if output.respond_to?(:binmode)
output.sync = false if output.respond_to?(:sync=)
@@ -209,24 +206,5 @@ end_msg
# lost connection to parent process, ignore output
end
end
-
- private
- def convert_content_type!
- if content_type = @headers.delete("Content-Type")
- @headers["type"] = content_type
- end
- if content_type = @headers.delete("Content-type")
- @headers["type"] = content_type
- end
- if content_type = @headers.delete("content-type")
- @headers["type"] = content_type
- end
- end
-
- # Don't set the Content-Length for block-based bodies as that would mean reading it all into memory. Not nice
- # for, say, a 2GB streaming file.
- def set_content_length!
- @headers["Content-Length"] = @body.size unless @body.respond_to?(:call)
- end
end
end
diff --git a/actionpack/lib/action_controller/response.rb b/actionpack/lib/action_controller/response.rb
index a6438a132b..8c4aec8d61 100755
--- a/actionpack/lib/action_controller/response.rb
+++ b/actionpack/lib/action_controller/response.rb
@@ -1,6 +1,9 @@
+require 'digest/md5'
+
module ActionController
class AbstractResponse #:nodoc:
DEFAULT_HEADERS = { "Cache-Control" => "no-cache" }
+ attr_accessor :request
attr_accessor :body, :headers, :session, :cookies, :assigns, :template, :redirected_to, :redirected_to_method_params, :layout
def initialize
@@ -8,28 +11,64 @@ module ActionController
end
def content_type=(mime_type)
- @headers["Content-Type"] = charset ? "#{mime_type}; charset=#{charset}" : mime_type
+ self.headers["Content-Type"] = charset ? "#{mime_type}; charset=#{charset}" : mime_type
end
def content_type
- content_type = String(@headers["Content-Type"]).split(";")[0]
+ content_type = String(headers["Content-Type"] || headers["type"]).split(";")[0]
content_type.blank? ? nil : content_type
end
def charset=(encoding)
- @headers["Content-Type"] = "#{content_type || "text/html"}; charset=#{encoding}"
+ self.headers["Content-Type"] = "#{content_type || "text/html"}; charset=#{encoding}"
end
def charset
- charset = String(@headers["Content-Type"]).split(";")[1]
+ charset = String(headers["Content-Type"] || headers["type"]).split(";")[1]
charset.blank? ? nil : charset.strip.split("=")[1]
end
def redirect(to_url, permanently = false)
- @headers["Status"] = "302 Found" unless @headers["Status"] == "301 Moved Permanently"
- @headers["Location"] = to_url
+ self.headers["Status"] = "302 Found" unless headers["Status"] == "301 Moved Permanently"
+ self.headers["Location"] = to_url
+
+ self.body = "<html><body>You are being <a href=\"#{to_url}\">redirected</a>.</body></html>"
+ end
- @body = "<html><body>You are being <a href=\"#{to_url}\">redirected</a>.</body></html>"
+ def prepare!
+ handle_conditional_get!
+ convert_content_type!
+ set_content_length!
end
+
+ private
+ def handle_conditional_get!
+ if body.is_a?(String) && headers['Status'][0..2] == '200' && !body.empty?
+ self.headers['Etag'] ||= %("#{Digest::MD5.hexdigest(body)}")
+
+ if request.headers['HTTP_IF_NONE_MATCH'] == headers['Etag']
+ self.headers['Status'] = '304 Not Modified'
+ self.body = ''
+ end
+ end
+ end
+
+ def convert_content_type!
+ if content_type = headers.delete("Content-Type")
+ self.headers["type"] = content_type
+ end
+ if content_type = headers.delete("Content-type")
+ self.headers["type"] = content_type
+ end
+ if content_type = headers.delete("content-type")
+ self.headers["type"] = content_type
+ end
+ end
+
+ # Don't set the Content-Length for block-based bodies as that would mean reading it all into memory. Not nice
+ # for, say, a 2GB streaming file.
+ def set_content_length!
+ self.headers["Content-Length"] = body.size unless body.respond_to?(:call)
+ end
end
end \ No newline at end of file
diff --git a/actionpack/test/controller/render_test.rb b/actionpack/test/controller/render_test.rb
index 21f5fb1f51..495e48a34a 100644
--- a/actionpack/test/controller/render_test.rb
+++ b/actionpack/test/controller/render_test.rb
@@ -337,6 +337,7 @@ class RenderTest < Test::Unit::TestCase
def test_etag_should_govern_renders_with_layouts_too
get :builder_layout_test
+ assert_equal "<wrapper>\n<html>\n <p>Hello </p>\n<p>This is grand!</p>\n</html>\n</wrapper>\n", @response.body
assert_equal etag_for("<wrapper>\n<html>\n <p>Hello </p>\n<p>This is grand!</p>\n</html>\n</wrapper>\n"), @response.headers['Etag']
end