ruby-on-rails - Ruby on Rails 测试失败或成功取决于测试套件的调用方式

标签 ruby-on-rails unit-testing testing railstutorial.org

我正在学习 Hartl Ruby on Rails 教程(第 4 版),但在我尝试运行的一些测试中遇到了问题 - 特别是第 11.2.3 节中的练习。到目前为止,我所有的测试都是绿色的。

我测试新的代码:

vagrant@ubuntu-18:/vagrant/sample_app/test$ rails test:mailers
Started with run options --seed 59483

  1/1: [======================================================================] 100% Time: 00:00:00, Time: 00:00:00

Finished in 0.33304s
1 tests, 9 assertions, 0 failures, 0 errors, 0 skips

一切都很正常。

我尝试运行完整的测试套件:

vagrant@ubuntu-18:/vagrant/sample_app/test$ rails test
Running via Spring preloader in process 11914
Started with run options --seed 56533

ERROR["test_account_activation", UserMailerTest, 1.2505091589991935]
 test_account_activation#UserMailerTest (1.25s)
ActionView::Template::Error:         ActionView::Template::Error: undefined method `merge' for nil:NilClass
            app/views/user_mailer/account_activation.html.erb:6:in `_app_views_user_mailer_account_activation_html_erb__2910351700178259135_47130470135880'
            app/mailers/user_mailer.rb:10:in `account_activation'
            test/mailers/user_mailer_test.rb:9:in `block in <class:UserMailerTest>'

  48/48: [========================================================================] 100% Time: 00:00:01, Time: 00:00:01

Finished in 1.71036s
48 tests, 206 assertions, 0 failures, 1 errors, 0 skips

SCSS 。我不会在这里复制的测试确认它绝对是同一段代码,在一个地方失败而在另一个地方工作。

但是,如果我调用 rails test:run 而不是调用 rails test,则会发生这种情况:

vagrant@ubuntu-18:/vagrant/sample_app/test$ rails test:run
Started with run options --seed 50303

  48/48: [========================================================================] 100% Time: 00:00:01, Time: 00:00:01

Finished in 1.72194s
48 tests, 215 assertions, 0 failures, 0 errors, 0 skips

同样,如果我使用 rake testrake test:run,那么一切都很好。

vagrant@ubuntu-18:/vagrant/sample_app/test$ rake test
(in /vagrant/sample_app)
Started with run options --seed 9429

  48/48: [========================================================================] 100% Time: 00:00:01, Time: 00:00:01

Finished in 1.70475s
48 tests, 215 assertions, 0 failures, 0 errors, 0 skips
vagrant@ubuntu-18:/vagrant/sample_app/test$ rake test:run
(in /vagrant/sample_app)
Started with run options --seed 50280

  48/48: [========================================================================] 100% Time: 00:00:01, Time: 00:00:01

Finished in 1.73529s
48 tests, 215 assertions, 0 failures, 0 errors, 0 skips

谁能解释一下这是怎么回事?理想情况下如何让 rails test 调用再次工作?

编辑

使用 rails test:run 使用的相同种子启动 rails test 以表明它与测试顺序无关。

vagrant@ubuntu-18:/vagrant/sample_app/test$ rails test:run 
Started with run options --seed 326

  48/48: [=======================================================] 100% Time: 00:00:01, Time: 00:00:01

Finished in 1.66690s
48 tests, 215 assertions, 0 failures, 0 errors, 0 skips
vagrant@ubuntu-18:/vagrant/sample_app/test$ rails test --seed 326
Running via Spring preloader in process 28117
Started with run options --seed 326

ERROR["test_account_activation", UserMailerTest, 1.605607868055813]
 test_account_activation#UserMailerTest (1.61s)
ActionView::Template::Error:         ActionView::Template::Error: undefined method `merge' for nil:NilClass
            app/views/user_mailer/account_activation.html.erb:6:in `_app_views_user_mailer_account_activation_html_erb__2910351700178259135_47130458820540'
            app/mailers/user_mailer.rb:10:in `account_activation'
            test/mailers/user_mailer_test.rb:9:in `block in <class:UserMailerTest>'

  48/48: [=======================================================] 100% Time: 00:00:01, Time: 00:00:01

Finished in 1.64637s
48 tests, 206 assertions, 0 failures, 1 errors, 0 skips

rake TESTOPTS="--seed=326" 也成功了。

编辑 2

测试:

require 'test_helper'

class UserMailerTest < ActionMailer::TestCase

  test "account_activation" do
    user = users(:michael)
    user.activation_token = User.new_token
    mail = UserMailer.account_activation(user)
    assert_equal "Account Activation", mail.subject
    assert_equal [user.email], mail.to
    assert_equal ['noreply@example.com'], mail.from
    assert_match user.name, mail.body.encoded
    assert_match user.activation_token, mail.body.encoded
    assert_match CGI.escape(user.email), mail.body.encoded
  end


end

测试方法: (其中以 mail to: 开头的行是 user_mailer.rb:10)

class UserMailer < ApplicationMailer

  def account_activation(user)
    @user = user
    mail to: @user.email, subject: "Account Activation"
  end

end

夹具:

michael:
    name: Michael Example
    email: michael@example.com
    password_digest: <%= User.digest('password') %>
    admin: true
    activated: true
    activated_at: <%= Time.zone.now %>

模板:

<h1>Sample App</h1>

<p>Hi <%= @user.name %>, </p>
<p>Welcome to the Sample App! Please click on the link belowe to activate your account.</p>

<%= link_to "Activate", edit_account_activation_url(@user.activation_token, email: @user.email) %>

最佳答案

感谢@AJFaraday 在评论中指出 Spring 预加载器正在运行。

我调用了 spring stop,问题自行解决了。

Spring 似乎又重新启动了,但是 rails test 现在给出了与其他三种方法相同的测试结果。

关于ruby-on-rails - Ruby on Rails 测试失败或成功取决于测试套件的调用方式,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/58454713/

相关文章:

javascript - Jest 仅显示一个测试套件,即使测试文件中存在对 describe() 的多个顶级调用也是如此

ruby - 无法测试我的模型测试

ruby-on-rails - Rails 委托(delegate)更新调用

ruby-on-rails - 加速 ruby​​ on rails 网站

ruby-on-rails - 设计 : translation missing: fr. devise.failure.user.not_found_in_database

java - 测试 Spring Boot 存储库时不会引发预期的异常

testing - Flutter:测试 FutureBuilder

javascript - 在javascript中访问rails路由

node.js - 使用 promise 解析/拒绝对 Mongoose 验证进行单元测试

c++ - 如何使用私有(private)构造函数 C++ 测试私有(private)成员