gpt4 book ai didi

ruby - 我如何测试我的 Ruby 代码在捕获符号时做了什么? (R规范)

转载 作者:太空宇宙 更新时间:2023-11-03 16:17:56 26 4
gpt4 key购买 nike

我正在处理涉及几个步骤的销售流程 - 折扣、验证、付款、交付等。如果在此过程中出现问题,我希望能够中止操作,但仍会退回销售反对来电者。

代码工作得很好,使用 throw/catch(但如果有帮助,我很乐意使用 raise/rescue。)问题是测试它——理想情况下,我能够编写一个测试说“如果出现问题,销售将停止处理”。相反,似乎我能做的最好的事情就是说“如果出现问题,它会抛出:完成”或“如果第 2 步出现问题,它不会完成第 3 步和第 4 步”。有没有其他方法可以处理这个测试,实际上可以测试更一般的情况?

这是我正在使用的代码的一个略有改动的版本。带有刘海的方法是可能调用 fail!succeed!

的方法
# sale.rb
class Sale
attr_accessor :status, :amount, :discount_code

def process!
catch :done do
validate_params!
amount = apply_code!(discount_code)
verify_delivery!
charge_money!
deliver_goods!
end

self
end

def fail!(error_message: nil)
Rails.logger.error(error_message)
status = :failure
throw :done
end

def succeed!
status = :success
throw :done
end
end

还有一个我想避免的 RSpec 测试的例子:

# sale_spec.rb
describe 'Sale' do
describe '#process!' do
context 'when verification fails' do
before do
allow(sale).to receive(:verify_delivery) { sale.fail! }
end

it "doesn't charge money" do
expect(sale).to_not receive(:charge_money!)
sale.process!
end

it "doesn't deliver the goods" do
expect(sale).to_not receive(:deliver_goods!)
sale.process!
end
end
end
end

有什么方法可以更普遍地测试它吗?

最佳答案

我不会为您的特定代码提供解决方案,但让我们从总体上考虑这个问题。您有一组共同负责销售流程的操作。解决方案是让每个 Action 独立,并让它们进入流程。如果出现错误,每个 Action 都可以恢复。在这种情况下,整个过程将停止在错误指令上,并通过调用它们的还原方法逐个操作后退。所有 Action 都应该共享一些上下文,您可以在其中存储对象(并将它们传递给下一个交互器)。这样您就可以将流程标记为失败并在以后访问其上下文。

有一个很棒的库您应该检查 - https://github.com/collectiveidea/interactor .基本上你可以将交互器定义为 block ,然后从这些类中形成一个组织者。它允许您使用 FooOrganizer.call(some_context) ,它将返回一个上下文对象。此上下文对象响应 success?failure? 方法(您可以在任何交互器中调用 context.fail! 以停止继续,标记进程失败并执行回滚)。您还可以从外部访问上下文中的对象(因此您可以稍后访问销售对象)。

在这种方法中,您可以只测试组织者是否成功(或不成功)和/或检查其上下文中的已处理对象。

关于ruby - 我如何测试我的 Ruby 代码在捕获符号时做了什么? (R规范),我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/39646909/

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