为什么这个请求间歇性地失败,或者我该如何调试呢?

时间:2013-02-17 23:55:01

标签: ruby-on-rails capybara factory-bot rspec-rails

这是间歇性失败的测试:

  context "as a regular_user" do
    before :each do
      @user = FactoryGirl.create(:user, :role => 'regular_user')
      visit new_user_session_path unless current_path == new_user_session_path
      fill_in "Email", :with => @user.email
      fill_in "Password", :with => @user.password
      click_button "Sign In"
    end

    it "removes thing from favorites while on thing index page" do
      @things = FactoryGirl.create_list(:thing, 5)
      @user.set_mark :favorite, @things.first
      @user.reload.favorite_things
      visit things_path unless current_path == things_path
      expect{
        first(:link, "Remove from favorites").click # Sometimes this fails
        @user.reload.favorite_things
      }.to change { @user.favorite_things.count }.by(-1)
      page.should have_content("Sort by")
    end
  end

我正在使用gem markable来提供收藏夹功能。

当我使用rspec运行规范时,似乎没有任何成功/失败的模式(它不像失败后总是失败,反之亦然)。当规范通过guardspork也正在使用)时,可能会出现一种模式,如果我启动guard并且测试通过,那么它将始终通过guard正在运行... Sames用于失败 - 如果我启动guard并且失败,那么guard实例中的所有后续运行也将失败。

刚才我跑rspec spec/requests/users_spec.rb:148来运行那个规范,但失败了:

1) Users as a regular_user removes thing from favorites while on thing index page
     Failure/Error: click_link "Remove from favorites" # Sometimes this fails
     Capybara::ElementNotFound:
       Unable to find link "Remove from favorites"
     # ./spec/requests/users_spec.rb:155:in `block (4 levels) in <top (required)>'
     # ./spec/requests/users_spec.rb:153:in `block (3 levels) in <top (required)>'

之后我又跑了rspec spec/requests/users_spec.rb:148,并且成功了:

Finished in 0.83754 seconds
1 example, 0 failures

我尝试在<% @things.each do |thing| %>视图文件的index.html.erb部分中添加“asdfasdf”,并使用page.should have_content("asdfasdf")检查该部分,但有时会失败。我尝试在page.should have_selector('.favoritesblock', visible: true)块之前添加expectfavoritesblock存在于视图中的同一@things块中),这有时会失败。但是,它永远不会在<% @things.each do |thing| %>循环之外找到文本。

我在save_and_open_page行之前添加了first(:link, "Remove from favorites").click # Sometimes this fails,并且能够生成规范失败的实例。该页面只包含两件事(我的FactoryGirl调用应该创建5)并且他们都没有“从收藏夹中删除”链接(取决于该内容是否被收藏,它显示“添加到收藏夹”或“从收藏中删除”)。

基于此,我尝试稍微修改规范的前两行,但它没有帮助(有时仍然失败):

5.times { FactoryGirl.create(:thing) }
@user.set_mark :favorite, Thing.first

我注意到的是,当它成功时,它会显示“从收藏夹中移除”链接,但它并不总是显示所有5件事。

所以,我需要知道为什么这个规范会间歇性地失败,或者我还能做些什么来找出造成问题的原因。有什么想法吗?

1 个答案:

答案 0 :(得分:1)

我设置工厂的方式是间歇性的rspec失败的原因。

该项目有一个布尔字段active,它可以显示网站上的内容(如果它是true),或者不显示网站上的内容(如果它是false )。

我对我的工厂非常聪明,并将该字段设置为f.active { [true, false].sample }。因此,有时工厂会创建活动的东西,有时它会创建不活动的东西。我有目的地创建我的工厂以这种方式行事,因为我希望默认工厂能够尽可能多地创建可能的现实世界。我只需要记住,某些方面有时需要手动设置 - 就像在这种情况下,我需要只创建活动的东西。

修复方法是将我的规范更改为仅创建活动内容:

@things = FactoryGirl.create_list(:thing, 5, :active => true)