Rails条件get(开发模式)仍然返回“Not Modified”(`#fresh_when()`)

时间:2011-11-07 17:44:36

标签: ruby-on-rails http caching ruby-on-rails-3.1

为什么以下(在开发模式下)错误地返回“304 not modified” - 在开发模式下运行时,Rails默认情况下不应禁用此类功能吗?

我的控制器看起来像这样:

class WidgetController < ApplicationController

  def show
    @widget = Widget.find(params[:id])
    fresh_when(etag: etag_for(@widget), last_modified: @widget.updated_at)
  end

  private

    def etag_for(*args)
      args.flatten + [current_user, last_deploy]
    end

    def last_deploy
      `git log --pretty=format:%H`.chomp
    end

end

我不清楚为什么在我的Rails应用程序中Development模式,这将返回“304 Not Modified”标题,我认为根据开发模式,这些事情没有启用?

我在本地使用瘦网络服务器,我认为这有点不寻常,否则这是一个没有特殊条件的典型应用程序,或者Rails 3.1.1上运行的案例

2 个答案:

答案 0 :(得分:2)

所以答案比那更容易:

./config/environments/development.rb我添加了一行:

  config.middleware.delete(Rack::ConditionalGet)

我向Rails提交了这个“错误”以获得官方回复,问题可以找到here at their Github issue tracker

答案 1 :(得分:0)

fresh_when不会被环境改变。每the source for fresh_when

def fresh_when(options)
  options.assert_valid_keys(:etag, :last_modified, :public)

  response.etag          = options[:etag]          if options[:etag]
  response.last_modified = options[:last_modified] if options[:last_modified]
  response.cache_control[:public] = true if options[:public]

  head :not_modified if request.fresh?(response)
end

然后是request.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

但是,您可以做的是在开发模式下向etag添加随机数(或时间戳),以强制每个请求都是新鲜的。