2012-08-16 42 views
0

的Rails 3.2.6Rspec的控制器测试失败时零

我已经设置了联系控制器和形式这篇文章的建议: http://matharvard.ca/posts/2011/aug/22/contact-form-in-rails-3/

Message模型类:

class Message 
    include ActiveModel::Validations 
    include ActiveModel::Conversion 
    extend ActiveModel::Naming 

    attr_accessor :name, :email, :message_body 

    # validations are here... 

    def initialize(attributes = {}) 
     attributes.each do |name, value| 
     send("#{name}=", value) 
     end 
    end 

    def persisted? 
     false 
    end 
end 

The Contact Controller

def create 
    @message = Message.new(params[:message]) 

    if @message.valid? 
     ContactMailer.new_message(@message).deliver 
     redirect_to(root_path, :notice => "Message was successfully sent.") 
    else 
     render :new 
    end 
end 

我的RSpec的控制器测试:

describe "POST create" do 
    context "with valid information" do 
     let (:message) { FactoryGirl.build(:message) } 
     it "creates a new message" do 
     post :create, params: message 
     assigns(:message).should eq(message) 
    end 
    end 
end 

我已经通过let验证了message变量不是零,所以FactoryGirl正在做的工作。

测试结果:

ContactController POST create with valid information creates a new message 
Failure/Error: post :create, params: message 
NoMethodError: 
    undefined method `each' for nil:NilClass 
# ./app/models/message.rb:14:in `initialize' 
# ./app/controllers/contact_controller.rb:8:in `new' 
# ./app/controllers/contact_controller.rb:8:in `create' 
# ./spec/controllers/contact_controller_spec.rb:22:in `block (4 levels) in <top (required)>' 

我知道它的失败是因为Messageinitialize方法。但我现在确定它为什么初始化为零。当我在开发环境中测试表单时,它似乎工作正常。谢谢你的帮助。

+0

您似乎错过了'it'块中的'end'。 – Finbarr 2012-08-16 22:45:47

+0

当我将代码粘贴到现在已更正的StackOverflow中时发生错字。 – 2012-08-16 22:57:01

+0

我建议调试params [:message]的值 – Finbarr 2012-08-16 22:59:37

回答

0

我不得不将一个散列作为值传递给post的散列选项。像这样:

message_attributes = FactoryGirl.attributes_for(:message) 
post :create, message: message_attributes 

为了测试两个消息比较,我不得不修改Message模型像这样:

attr_accessor :attributes, :name, :email, :message_body 
def initialize(attributes = {}) 
    if not attributes.nil? 
     attributes.each do |name, value| 
      send("#{name}=", value) 
     end 
    end 
    @attributes = attributes 
end 

所以最后一个经过测试是这样的:

it "creates a new message" do 
    message_attributes = FactoryGirl.attributes_for(:message) 
    post :create, message: message_attributes 
    assigns(:message).attributes.symbolize_keys.should eq(message_attributes) 
end 

我不知道这是否是最佳的解决方案,但它确实创造通过的测试和验证控制器正确地创建消息。

0

我想你应该将消息传递与它明确的名称,就像这样:

it "creates a new message" do 
    post :create, :message => message 
    assigns(:message).should eq(message) 
end 

或短期风格:

post :create, message: message 

,那么你将有PARAMS哈希键'message'

params[:message] = ... # here's your message 

否则你不会有一个关键的params和hense params[:message]返回零

+0

这对我来说不起作用。看到我的答案。 – 2012-08-17 20:07:50

+0

我犯了一个错误,看到更新 – megas 2012-08-18 02:40:44

+0

这仍然不适合我。当我用这种方法尝试时,@message被创建时没有任何属性。 – 2012-08-24 17:14:01