RSpec 没有看到使用 `change(object, :message)` 语法对 ActiveRecord 对象的更改
RSpec not seeing changes to ActiveRecord object with `change(object, :message)` syntax
我有三个模型:
User
Company
Commitment
Commitment
是 Users
和 Companies
的 HABTM 连接 table ( 即 当 User
加入一个 Company
,它创建一个新的 Commitment
)。它还有一些额外的 columns/attributes:
admin
(用户是否拥有该公司的管理员权限?)
confirmed_by_admin
(公司管理员是否已确认该用户加入公司的请求?)
confirmed_by_member
(用户本人是否确认受邀加入公司?)
我还定义了一个方便的方法来快速确定承诺是否已完全确认:
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.first
与 commitments_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
免责声明:这是未经验证的,但因为它对我来说太复杂了作为评论(包含所有示例测试代码),我把它写在这里作为答案。如果有人知道更好,请告诉我。
我有三个模型:
User
Company
Commitment
Commitment
是 Users
和 Companies
的 HABTM 连接 table ( 即 当 User
加入一个 Company
,它创建一个新的 Commitment
)。它还有一些额外的 columns/attributes:
admin
(用户是否拥有该公司的管理员权限?)confirmed_by_admin
(公司管理员是否已确认该用户加入公司的请求?)confirmed_by_member
(用户本人是否确认受邀加入公司?)
我还定义了一个方便的方法来快速确定承诺是否已完全确认:
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.first
与commitments_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
免责声明:这是未经验证的,但因为它对我来说太复杂了作为评论(包含所有示例测试代码),我把它写在这里作为答案。如果有人知道更好,请告诉我。