RSpec 没有看到使用 `change(object, :message)` 语法对 ActiveRecord 对象的更改

RSpec not seeing changes to ActiveRecord object with `change(object, :message)` syntax

我有三个模型:

CommitmentUsersCompanies 的 HABTM 连接 table (User 加入一个 Company,它创建一个新的 Commitment)。它还有一些额外的 columns/attributes:

我还定义了一个方便的方法来快速确定承诺是否已完全确认:

class Commitment < ApplicationRecord
  def confirmed?
    confirmed_by_admin? && confirmed_by_member?
  end
end

现在我正在编写请求规范,但由于某种原因,change 匹配器只能使用其两种语法之一:

let :carol   { FactoryGirl.create(:user) }
let :company { FactoryGirl.create(:company) }

it 'confirms invitation to join company' do
  # Initialize unconfirmed commitment
  FactoryGirl.create(:commitment, user: carol,
                                  company: company,
                                  confirmed_by_admin: true)

  expect do
    patch commitment_path(carol.commitments.first),
          params: { commitment: { confirmed_by_member: true } }

  # for the following syntaxes, ------------------------------------------------
  # this works:
  end.to change { carol.commitments.first.confirmed?) }.from(false).to(true)
  # and this fails:
  end.to change(carol.commitments.first, :confirmed?).from(false).to(true)
  # ----------------------------------------------------------------------------

end

似乎 carol.commitments.first 在 RSpec 测试更改时没有重新加载 — 我得到以下测试输出:

Failure/Error:
  expect do
    patch commitment_path(carol.commitments.first),
          params: { commitment: { confirmed_by_member: true } }
  end.to change(Commitment.find_by(user: carol, company: company), :confirmed?).from(false).to(true)

  expected #confirmed? to have changed from false to true, but did not change
# ./spec/requests/commitments_spec.rb:69:in `block (3 levels) in <top (required)>'

什么给了?很明显,我可以坚持使用花括号/块语法,但我想了解为什么其中一种有效,而另一种无效。

检查 docs 并自己尝试一个新的 rails 项目来复制您的场景,但也失败了,我 相信 失败是因为

  • .change的"block"形式是运行两次("before"和"after"expect块) ,该块内的任何内容:

    .change{ carol.commitments.first.confirmed? }

  • .change 的 "method" 形式是 运行 一次用于第一个参数:carol.commitments.first,但 运行 两次用于第二个参数 :confirmed?。然而,这个问题是规范文件中的 carol.commitments.firstcommitments_controller#update 中实际更新的对象不共享相同的内存 space(大多数可能该对象被命名为 @commitment)。尽管它们是相同的 Commitment 记录,但它们是独立的实例,并且当另一个更改时,另一个的属性值不会自动更改。

考虑以下演示此 "method" 表单工作的场景:

it 'sometest' do
  commitment = FactoryGirl.create(:commitment, user: carol,
                                  company: company,
                                  confirmed_by_admin: true)
  expect do
    # this commitment object is exactly the same object passed in the `change` below
    commitment.confirmed_by_member = true
  end.to change(commitment, :confirmed?).from(false).to(true)
end

免责声明:这是未经验证的,但因为它对我来说太复杂了作为评论(包含所有示例测试代码),我把它写在这里作为答案。如果有人知道更好,请告诉我。