Ruby on rails RSpec/Capybara未在测试中提交给DB

Ruby on rails RSpec/Capybara未在测试中提交给DB,ruby-on-rails,rspec,capybara,Ruby On Rails,Rspec,Capybara,自从将我们的测试套件从Minitest迁移到RSpec后,我非常绝望。到目前为止,所有控制器和模型测试都运行良好,但由于尝试移植(以前通过/工作)如下特性测试,我遇到了麻烦 feature 'Create place' do scenario 'create valid place as user' do login_as_user visit '/places/new' fill_in_valid_place_information click_button

自从将我们的测试套件从Minitest迁移到RSpec后,我非常绝望。到目前为止,所有控制器和模型测试都运行良好,但由于尝试移植(以前通过/工作)如下特性测试,我遇到了麻烦

feature 'Create place' do
  scenario 'create valid place as user' do
    login_as_user
    visit '/places/new'
    fill_in_valid_place_information
    click_button('Create Place')
    visit '/places'
    expect(page).to have_content('Any place', count: 1)
  end

  ...

  def fill_in_valid_place_information
      fill_in('place_name', with: 'Any place')
      fill_in('place_street', with: 'Magdalenenstr.')
      fill_in('place_house_number', with: '19')
      fill_in('place_postal_code', with: '10963')
      fill_in('place_city', with: 'Berlin')
      fill_in('place_email', with: 'schnipp@schnapp.com')
      fill_in('place_homepage', with: 'http://schnapp.com')
      fill_in('place_phone', with: '03081763253')
  end
end
不幸的是,这不会导致DB提交,从而导致测试失败。如果我将撬入测试并手动创建请求的位置,则不会失败。我尝试了不同的方法来触发按钮,但到目前为止没有任何效果

这就是我的
rails\u helper.rb
的样子:

ENV['RAILS_ENV'] ||= 'test'
require File.expand_path('../../config/environment', __FILE__)
abort("The Rails environment is running in production mode!") if Rails.env.production?
require 'spec_helper'
require 'rspec/rails'
require 'capybara/rspec'
require 'capybara/rails'
require 'capybara/poltergeist'
require 'pry'

def validate_captcha
  fill_in 'captcha', with: SimpleCaptcha::SimpleCaptchaData.first.value
end

def login_as_user
  user = create :user, email: 'user@example.com'
  visit 'login/'
  fill_in 'sessions_email', with: 'user@example.com'
  fill_in 'sessions_password', with: 'secret'
  click_on 'Login'
end

Dir[Rails.root.join('spec/support/**/*.rb')].each { |f| require f }

ActiveRecord::Migration.maintain_test_schema!

Capybara.register_driver :poltergeist do |app|
    Capybara::Poltergeist::Driver.new(app, phantomjs_options: ['--ignore-ssl-errors=true'])
end

Capybara.javascript_driver = :poltergeist

RSpec.configure do |config|
  config.use_transactional_fixtures = false

  config.before(:suite) do
    DatabaseCleaner.clean
  end

  config.before(:each) do
    DatabaseCleaner.strategy = :transaction
  end

  config.before(:each, no_transaction: true) do
    DatabaseCleaner.strategy = :truncation
  end

  config.before(:each, js: true) do
    DatabaseCleaner.strategy = :truncation
  end

  config.before(:each) do
    DatabaseCleaner.start
  end

  config.after(:each) do
    DatabaseCleaner.clean
  end

  config.include Capybara::DSL
  config.include Rails.application.routes.url_helpers
  config.fixture_path = "#{::Rails.root}/spec/fixtures"
  config.infer_spec_type_from_file_location!
  config.filter_rails_from_backtrace!
end
有人知道可能的原因吗?创业板版本:

  • 水豚2.12.0
  • rspec 3.5.0
  • 轨道4.2.7.1
最好的,谢谢, 安迪

---更新

我添加了
fill\u in\u valid\u place\u信息
method

这就是在启用或不启用Capybara JS驱动程序的情况下测试失败的原因(在本测试中,这并不重要,因为该功能不使用任何JS)。不幸的是,它没有给出任何真正的提示来使用

1) Create place create valid place as user
     Failure/Error: expect(page).to have_content('Any place', count: 1)

       expected to find text "Any place" 1 time but found 0 times in "KIEZ KARTE Find places Here comes a list of all POIs currently available in our database. If you are looking for a specific location please enter parts of its descriptive features into the 'Search' field. Search: Name Postal code Categories No data available in table"

       Timeout reached while running a *waiting* Capybara finder...perhaps you wanted to return immediately? Use a non-waiting Capybara finder. More info: http://blog.codeship.com/faster-rails-tests?utm_source=gem_exception
---更新2


我发现了与卡皮亚拉无关的问题。实际上,我忘了为我们正在调用的API传输存根响应。谢谢大家参与我的斗争

您的测试中有许多潜在的问题可能会导致您看到的问题,如果您包含测试产生的实际错误消息,将来将更容易缩小范围

  • 您的场景/功能未标记:js元数据,无法使用Capybara驱动程序激活。您可能在某个地方指定了
    Capybara.default\u驱动程序,但如果是这样,那么您的DatabaseCleaner配置是错误的

  • 使用中推荐的DatabaseCleaner配置。如果您指定了#1中提到的
    Capybara.default_driver
    ,并且使用了:js/:driver元数据使用模式,则驱动程序名称检测将起作用。此外,后/后差异对于减少测试片状非常重要

  • 您的
    login\u as\u user
    方法需要在返回之前验证登录是否已完成。这是因为
    单击“登录”
    可以异步触发并在实际登录之前返回。这会导致您在中止登录后立即调用
    visit
    ,阻止发送会话cookie,并在您希望用户登录时以未登录用户结束。要解决这个问题,你需要像

    def login_as_user
      ...
      click_on 'Login'
      expect(page).to have_text('You are now logged in!') #whatever message is shown on successful login, or use have_css with some element on the page that only exists when a user is logged in (user menu, etc)
    end
    
    同样的问题也存在于
    点击按钮(“创建地点”)
    访问“/places”
    之间,访问可以有效地取消按钮点击的效果


  • 你能用有效的信息方法发布填写信息吗?嘿,谢谢。我更新了答案,包括方法和失败消息(这没有多大帮助)嘿,托马斯,谢谢你的全面回答。1.测试的功能不包含任何JS。无论如何,即使使用:js标记,它也会失败。2.我做到了,什么也没有改变。3.登录已经完成,我确认了。即使没有登录,软件的逻辑也允许提交,因此这不是一个问题。另外:测试在Minitest下“概念上”运行--@A.Neumann您的代码是否依赖于提交后的
    回调?确实如此,我在创建后有两个
    callbacks@A.Neumann另外,您可以在测试日志中显示您的创建操作和相关输出。注意-如果使用JS驱动程序,您在
    单击按钮('Create Place')
    访问'/places'
    与登录时遇到相同的问题,那么访问可以有效地取消创建。如果在单击按钮(“创建位置”)后输出page.html,它是否显示任何错误?@A.Neumann
    after\u Create
    不同于
    after\u commit
    -after\u Create应该可以,但在事务模式下不会调用
    after\u commit