gpt4 book ai didi

ruby-on-rails - RSpec 没有看到对 ActiveRecord 对象的更改,语法为 `change(object, :message)`

转载 作者:行者123 更新时间:2023-12-05 04:09:45 26 4
gpt4 key购买 nike

我有三个模型:

  • 用户
  • 公司
  • promise

CommitmentUsersCompanies 的 HABTM 连接表(即,User 加入了一个 Company,它创建了一个新的 Commitment)。它还具有一些额外的列/属性:

  • admin(用户是否拥有该公司的管理员权限?)
  • confirmed_by_admin(公司管理员是否已确认此用户加入公司的请求?)
  • confirmed_by_member(用户本人是否确认了加入公司的邀请?)

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

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

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

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)>'

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

最佳答案

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

  • .change 的“ block ”形式运行两次(expect block “之前”和“之后”),无论该 block 内部是什么:

    .change{ car​​ol.commitments.first.confirmed? }

  • .change 的“方法”形式为第一个参数运行一次:carol.commitments.first,但为第二个参数运行两次:确认?。但是,问题是规范文件中此时的 carol.commitments.firstcommitments_controller 中实际更新的对象不共享相同的内存空间#update(很可能该对象名为 @commitment)。虽然它们是相同的Commitment记录,但它们是独立的实例,并且另一个的属性值不会在另一个更改时自动更改。

考虑以下演示此“方法”形式工作的场景:

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

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

关于ruby-on-rails - RSpec 没有看到对 ActiveRecord 对象的更改,语法为 `change(object, :message)`,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/45274875/

26 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com