0

我的测试如下所示:

def setup
    @period_registration= FactoryGirl.create(:period_registration)
  end


 test "should post save_period" do
    sign_in(FactoryGirl.create(:user))
     assert_difference('PeriodRegistration.count') do
      post :save_period, period_registration: FactoryGirl.attributes_for(:period_registration)
    end
    assert_not_nil assigns(:period_registration)

  end

但是当我运行它时,我得到了这个错误:

 1) Error:
test_should_post_save_period(PeriodRegistrationsControllerTest):
NoMethodError: undefined method `event' for nil:NilClass

这是我的控制器:

  def save_period
    @period_registration = PeriodRegistration.new(params[:registration])
    @period_registration.save
    flash[:success] = "Successfully Registered for Session."
    redirect_to event_url(@period_registration.period.event)
  end

我的工厂是这样的:

factory :event do
    name 'First Event'
    street '123 street'
    city 'Chicago'
    state 'Iowa'
    date Date.today
  end


  factory :period do
    name 'First Period'
    description 'This is a description'
    start_time Time.now + 10.days
    end_time Time.now + 10.days + 2.hours
    event
    product
  end

factory :period_registration do
    user
    period
  end

我需要创建一个周期对象和一个事件对象吗?如果有怎么办?我认为这不是问题,因为我相信在各个工厂中通过“周期”、“产品”和“事件”自动创建这些。

关于从这里看哪里的任何想法?

4

1 回答 1

1

简短的回答 - 是的,您确实创建了对象。

长答案:

  1. 在控制器中:

    @period_registration.period.event
    

    这行代码违反了得墨忒耳法则。这不是好的设计。这行代码应如下所示:

    @period_registration.event
    

    但是您必须在 PeriodRegistration 模型中创建新方法。最简单的方法变体可以是:

    def event
      period.event
    end
    
  2. 在控制器中:您不检查 PeriodRegistration 模型是否已保存。

  3. 据我了解,PeriodRegistration 模型有 2 个关联,当您使用 FactoryGirl.attributes_for 时,工厂不会创建关联对象,它只是为您提供 PeriodRegistration 的一组属性。要通过此测试,您应该在调用控制器之前创建这两个对象。此外,最佳实践是 - 测试应该只有一个断言。例如:

    def setup
      @user = FactoryGirl.create(:user)
      @period = FactoryGirl.create(:period)
    end
    
    test "should post save_period" do
      sign_in(@user)
      assert_difference('PeriodRegistration.count') do
        post :save_period, period_registration: FactoryGirl.attributes_for(:period_registration, user: @user, period: @period)
      end
    end
    
    test "should assings @period_registration" do
      sign_in(@user)
      post :save_period, period_registration: FactoryGirl.attributes_for(:period_registration, user: @user, period: @period)
      assert_not_nil assigns(:period_registration)
    end
    
  4. 测试控制器时,您可以使用模拟对象而不是真实模型。

于 2012-09-01T01:28:24.287 回答