12

利用 ActionController 的新respond_with方法...它如何确定当操作(保存)成功时要渲染什么,什么时候不成功?

我问是因为我试图让一个脚手架生成的规范(包括在下面)通过,只要我能理解它。该应用程序运行良好,但奇怪的是,当验证失败时,它似乎正在呈现/carriers(至少浏览器的 URL 是这样说的)。然而,规范正在期待"new"(我也是,就此而言),而是正在接收<"">. 如果我更改规范以期望""它仍然失败。

当它呈现/carriers该页面时,会在验证失败的字段旁边显示 error_messages,正如人们所期望的那样。

任何熟悉的人都可以respond_with看到这里发生了什么吗?

#carrier.rb
  validates :name, :presence => true 

#carriers_controller.rb
class CarriersController < ApplicationController
  respond_to :html, :json

...

  def new
    respond_with(@carrier = Carrier.new)
  end

  def create
     @carrier = Carrier.new(params[:carrier])
     flash[:success] = 'Carrier was successfully created.' if @carrier.save
     respond_with(@carrier) 
  end

规范失败:

#carriers_controller_spec.rb
require 'spec_helper'

describe CarriersController do

  def mock_carrier(stubs={})
    (@mock_carrier ||= mock_model(Carrier).as_null_object).tap do |carrier|
      carrier.stub(stubs) unless stubs.empty?
    end
  end


  describe "POST create" do
    describe "with invalid params" do
      it "re-renders the 'new' template" do
        Carrier.stub(:new) { mock_carrier(:save => false) }
        post :create, :carrier => {}
        response.should render_template("new")
      end
    end
  end
end

出现此错误:

  1) CarriersController POST create with invalid params re-renders the 'new' template
     Failure/Error: response.should render_template("new")
     expecting <"new"> but rendering with <"">.
     Expected block to return true value.
     # (eval):2:in `assert_block'
     # ./spec/controllers/carriers_controller_spec.rb:81:in `block (4 levels) in <top (required)>'
4

1 回答 1

23

tl:博士

向模拟添加错误哈希:

Carrier.stub(:new) { mock_carrier(:save => false, 
                       :errors => { :anything => "any value (even nil)" })}

这将触发respond_with.

这里发生了什么

在后面加上这个post :create

response.code.should == "200"

它失败了expected: "200", got: "302"。所以它在不应该的时候重定向而不是渲染新模板。它要去哪里?给它一条我们知道会失败的路径:

response.should redirect_to("/")

现在它失败了Expected response to be a redirect to <http://test.host/> but was a redirect to <http://test.host/carriers/1001>

规范应该通过渲染模板来传递,这是模拟 Carrier 对象返回 falsenew后的正常事件过程。save而是respond_with最终重定向到show_carrier_path. 这是完全错误的。但为什么?

在对源代码进行一些挖掘之后,控制器似乎试图呈现“运营商/创建”。没有这样的模板,因此引发了异常。救援块确定请求是 POST 并且错误哈希中没有任何内容,控制器会在此基础上重定向到默认资源,即 mock Carrier

这令人费解,因为控制器不应假定存在有效的模型实例。这create毕竟是一个。在这一点上,我只能推测测试环境以某种方式走捷径。

所以解决方法是提供一个虚假的错误哈希。通常情况下,失败后会有一些东西在哈希中save,所以这有点道理。

于 2010-12-15T06:22:27.870 回答