aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--actionpack/lib/action_controller/caching.rb97
-rw-r--r--actionpack/lib/action_controller/test_process.rb1
-rw-r--r--actionpack/lib/action_view/helpers/text_helper.rb5
-rw-r--r--actionpack/test/controller/caching_filestore.rb62
4 files changed, 94 insertions, 71 deletions
diff --git a/actionpack/lib/action_controller/caching.rb b/actionpack/lib/action_controller/caching.rb
index 88529d8008..d31343dd71 100644
--- a/actionpack/lib/action_controller/caching.rb
+++ b/actionpack/lib/action_controller/caching.rb
@@ -57,8 +57,7 @@ module ActionController #:nodoc:
# By default, the cache extension is .html, which makes it easy for the cached files to be picked up by the web server. If you want
# something else, like .php or .shtml, just set Base.page_cache_extension.
module Pages
- def self.append_features(base) #:nodoc:
- super
+ def self.included(base) #:nodoc:
base.extend(ClassMethods)
base.class_eval do
@@page_cache_directory = defined?(RAILS_ROOT) ? "#{RAILS_ROOT}/public" : ""
@@ -270,7 +269,7 @@ module ActionController #:nodoc:
end
# Called by CacheHelper#cache
- def cache_erb_fragment(block, name = {}, options = {})
+ def cache_erb_fragment(block, name = {}, options = nil)
unless perform_caching then block.call; return end
buffer = eval("_erbout", block.binding)
@@ -284,33 +283,30 @@ module ActionController #:nodoc:
end
end
- def write_fragment(name, content, options = {})
+ def write_fragment(name, content, options = nil)
return unless perform_caching
key = fragment_cache_key(name)
self.class.benchmark "Cached fragment: #{key}" do
fragment_cache_store.write(key, content, options)
end
-
content
end
- def read_fragment(name, options = {})
+ def read_fragment(name, options = nil)
return unless perform_caching
- key, cache = fragment_cache_key(name), nil
+ key = fragment_cache_key(name)
self.class.benchmark "Fragment read: #{key}" do
- cache = fragment_cache_store.read(key, options)
+ fragment_cache_store.read(key, options)
end
-
- cache || false
end
# Name can take one of three forms:
# * String: This would normally take the form of a path like "pages/45/notes"
# * Hash: Is treated as an implicit call to url_for, like { :controller => "pages", :action => "notes", :id => 45 }
# * Regexp: Will destroy all the matched fragments, example: %r{pages/\d*/notes}
- def expire_fragment(name, options = {})
+ def expire_fragment(name, options = nil)
return unless perform_caching
key = fragment_cache_key(name)
@@ -327,29 +323,58 @@ module ActionController #:nodoc:
end
# Deprecated -- just call expire_fragment with a regular expression
- def expire_matched_fragments(matcher = /.*/, options = {}) #:nodoc:
+ def expire_matched_fragments(matcher = /.*/, options = nil) #:nodoc:
expire_fragment(matcher, options)
end
- class MemoryStore #:nodoc:
- def initialize
- @data, @mutex = { }, Mutex.new
+
+ class UnthreadedMemoryStore #:nodoc:
+ def initialize #:nodoc:
+ @data = {}
end
- def read(name, options = {}) #:nodoc:
- @mutex.synchronize { @data[name] } rescue nil
+ def read(name, options=nil) #:nodoc:
+ @data[name]
end
- def write(name, value, options = {}) #:nodoc:
- @mutex.synchronize { @data[name] = value }
+ def write(name, value, options=nil) #:nodoc:
+ @data[name] = value
end
- def delete(name, options = {}) #:nodoc:
- @mutex.synchronize { @data.delete(name) }
+ def delete(name, options=nil) #:nodoc:
+ @data.delete(name)
end
- def delete_matched(matcher, options) #:nodoc:
- @mutex.synchronize { @data.delete_if { |k,v| k =~ matcher } }
+ def delete_matched(matcher, options=nil) #:nodoc:
+ @data.delete_if { |k,v| k =~ matcher }
+ end
+ end
+
+ module ThreadSafety
+ def read(name, options=nil) #:nodoc:
+ @mutex.synchronize { super }
+ end
+
+ def write(name, value, options=nil) #:nodoc:
+ @mutex.synchronize { super }
+ end
+
+ def delete(name, options=nil) #:nodoc:
+ @mutex.synchronize { super }
+ end
+
+ def delete_matched(matcher, options=nil) #:nodoc:
+ @mutex.synchronize { super }
+ end
+ end
+
+ class MemoryStore < UnthreadedMemoryStore #:nodoc:
+ def initialize #:nodoc:
+ super
+ if ActionController::Base.allow_concurrency
+ @mutex = Mutex.new
+ MemoryStore.send(:include, ThreadSafety)
+ end
end
end
@@ -357,8 +382,9 @@ module ActionController #:nodoc:
attr_reader :address
def initialize(address = 'druby://localhost:9192')
+ super()
@address = address
- @data, @mutex = DRbObject.new(nil, address), Mutex.new
+ @data = DRbObject.new(nil, address)
end
end
@@ -366,26 +392,27 @@ module ActionController #:nodoc:
attr_reader :address
def initialize(address = 'localhost')
+ super()
@address = address
- @data, @mutex = MemCache.new(address), Mutex.new
+ @data = MemCache.new(address)
end
end
- class FileStore #:nodoc:
+ class UnthreadedFileStore #:nodoc:
attr_reader :cache_path
def initialize(cache_path)
@cache_path = cache_path
end
- def write(name, value, options = {}) #:nodoc:
+ def write(name, value, options = nil) #:nodoc:
ensure_cache_path(File.dirname(real_file_path(name)))
File.open(real_file_path(name), "w+") { |f| f.write(value) }
rescue => e
- Base.logger.error "Couldn't create cache directory: #{name} (#{e.message})" unless Base.logger.nil?
+ Base.logger.error "Couldn't create cache directory: #{name} (#{e.message})" if Base.logger
end
- def read(name, options = {}) #:nodoc:
+ def read(name, options = nil) #:nodoc:
IO.read(real_file_path(name)) rescue nil
end
@@ -427,7 +454,17 @@ module ActionController #:nodoc:
end
end
end
- end
+ end
+
+ class FileStore < UnthreadedFileStore #:nodoc:
+ def initialize(cache_path)
+ super(cache_path)
+ if ActionController::Base.allow_concurrency
+ @mutex = Mutex.new
+ FileStore.send(:include, ThreadSafety)
+ end
+ end
+ end
end
# Sweepers are the terminators of the caching world and responsible for expiring caches when model objects change.
diff --git a/actionpack/lib/action_controller/test_process.rb b/actionpack/lib/action_controller/test_process.rb
index 59604e7c34..783d21ff53 100644
--- a/actionpack/lib/action_controller/test_process.rb
+++ b/actionpack/lib/action_controller/test_process.rb
@@ -116,6 +116,7 @@ module ActionController #:nodoc:
@request_uri = "/"
self.remote_addr = "0.0.0.0"
@env["SERVER_PORT"] = 80
+ @env['REQUEST_METHOD'] = "GET"
end
end
diff --git a/actionpack/lib/action_view/helpers/text_helper.rb b/actionpack/lib/action_view/helpers/text_helper.rb
index 732e87078f..16f9450df1 100644
--- a/actionpack/lib/action_view/helpers/text_helper.rb
+++ b/actionpack/lib/action_view/helpers/text_helper.rb
@@ -8,7 +8,7 @@ module ActionView
# <% for post in @posts %>
# Title: <%= truncate(post.title, 20) %>
# <% end %>
- module TextHelper
+ module TextHelper
# The regular puts and print are outlawed in eRuby. It's recommended to use the <%= "hello" %> form instead of print "hello".
# If you absolutely must use a method-based output, you can use concat. It's use like this <% concat "hello", binding %>. Notice that
# it doesn't have an equal sign in front. Using <%= concat "hello" %> would result in a double hello.
@@ -265,6 +265,9 @@ module ActionView
end
end
+ class << self
+ include ActionView::Helpers::TextHelper
+ end
private
# The cycle helpers need to store the cycles in a place that is
diff --git a/actionpack/test/controller/caching_filestore.rb b/actionpack/test/controller/caching_filestore.rb
index e2478e9546..389ebe02fa 100644
--- a/actionpack/test/controller/caching_filestore.rb
+++ b/actionpack/test/controller/caching_filestore.rb
@@ -1,16 +1,15 @@
require 'fileutils'
require File.dirname(__FILE__) + '/../abstract_unit'
-#generate the greatest logging class that ever lived
class TestLogDevice < Logger::LogDevice
attr :last_message, true
def initialize
- @last_message=String.new
+ @last_message=String.new
end
def write(message)
- @last_message << message
+ @last_message << message
end
def clear
@@ -23,14 +22,13 @@ TestLog = TestLogDevice.new
RAILS_DEFAULT_LOGGER = Logger.new(TestLog)
ActionController::Base.logger = RAILS_DEFAULT_LOGGER
-#generate a random key to ensure the cache is always in a different location
-RANDOM_KEY = rand(99999999).to_s
-FILE_STORE_PATH = File.dirname(__FILE__) + '/../temp/' + RANDOM_KEY
-ActionController::Base.perform_caching = true
-ActionController::Base.fragment_cache_store = ActionController::Caching::Fragments::FileStore.new(FILE_STORE_PATH)
-
-#setup the routing information...not sure if this does anything
-ActionController::Routing::Routes.connect "test", :controller => 'test', :action => 'render_to_cache'
+def use_store
+ #generate a random key to ensure the cache is always in a different location
+ RANDOM_KEY = rand(99999999).to_s
+ FILE_STORE_PATH = File.dirname(__FILE__) + '/../temp/' + RANDOM_KEY
+ ActionController::Base.perform_caching = true
+ ActionController::Base.fragment_cache_store = :file_store, FILE_STORE_PATH
+end
class TestController < ActionController::Base
caches_action :render_to_cache, :index
@@ -45,48 +43,32 @@ class FileStoreTest < Test::Unit::TestCase
def setup
@request = ActionController::TestRequest.new
@response = ActionController::TestResponse.new
+ @controller = TestController.new
@request.host = "hostname.com"
end
-
- #To prime the cache with hostname.com/test
- def test_render_to_cache_prime_a
- @request.path_parameters = {:controller => "test"}
- assert_fragment_cached do process_request end
- end
-
- #To prime the cache with hostname.com/test/render_to_cache
- def test_render_to_cache_prime_b
- @request.path_parameters = {:action => "render_to_cache", :controller => "test"}
- assert_fragment_cached do process_request end
+
+ def teardown
+ FileUtils.rm_rf(FILE_STORE_PATH)
end
- #To hit the cache with hostname.com/test
- def test_render_to_cache_zhit_a
- @request.path_parameters = {:controller => "test"}
- assert_fragment_hit do process_request end
+ def test_render_cached
+ assert_fragment_cached { get :render_to_cache }
+ assert_fragment_hit { get :render_to_cache }
end
- #To hit the cache with hostname.com/test/render_to_cache
- def test_render_to_cache_zhit_b
- @request.path_parameters = {:action => "render_to_cache", :controller => "test"}
- assert_fragment_hit do process_request end
- end
private
- def process_request
- TestController.process(@request, @response)
- end
-
- def assert_fragment_cached(&proc)
- proc.call
+ def assert_fragment_cached
+ yield
assert(TestLog.last_message.include?("Cached fragment:"), "--ERROR-- FileStore write failed ----")
assert(!TestLog.last_message.include?("Couldn't create cache directory:"), "--ERROR-- FileStore create directory failed ----")
TestLog.clear
end
- def assert_fragment_hit(&proc)
- proc.call
- assert(TestLog.last_message.include?( "Fragment hit:"), "--ERROR-- Fragment not found in FileStore ----")
+ def assert_fragment_hit
+ yield
+ assert(TestLog.last_message.include?("Fragment read:"), "--ERROR-- Fragment not found in FileStore ----")
+ assert(!TestLog.last_message.include?("Cached fragment:"), "--ERROR-- Did cache ----")
TestLog.clear
end
end \ No newline at end of file