6

这个问题可能看起来像这个问题的重复,接受的答案对我的问题没有帮助。

语境

由于 Rails 5 不再支持在控制器测试中直接操作会话(现在继承自 ActionDispatch::IntegrationTest),因此我将走上模拟和存根的黑暗道路。

我知道这是不好的做法,并且有更好的方法来测试控制器(我确实理解他们转向集成测试),但我不想运行完整的集成测试并在单个测试中调用多个操作只是为了设置一个特定的会话变量。

设想

使用Mocha模拟/存根会话变量实际上非常容易:

ActionDispatch::Request::Session.any_instance.stubs(:[]).with(:some_variable).returns("some value")

问题是,Rails 在会话中存储了很多东西(只需session.inspect在您的一个视图中的任何地方做一个)并且存根该:[]方法显然会阻止访问它们中的任何一个(因此session[:some_other_variable]在测试中将不再起作用)。

问题

:[]有没有办法仅在使用特定参数调用时存根/模拟该方法,并使所有其他调用不被存根?

我本来希望像

ActionDispatch::Request::Session.any_instance.stubs(:[]).with(:some_variable).returns("some value")
ActionDispatch::Request::Session.any_instance.stubs(:[]).with(anything).returns(original_value)

但我找不到完成它的方法。

4

1 回答 1

1

据我所知,这是 mocha 中没有的功能

https://github.com/freerange/mocha/issues/334

我知道这确实存在于rspec-mock

https://github.com/rspec/rspec-mocks/blob/97c972be57f2c060a4a7fb8a3c5700a5ede693f0/spec/rspec/mocks/stub_implementation_spec.rb#L29

但是,您可以使用的一种 hacky 方法是将原始会话存储在一个对象中,然后模拟每当控制器接收到会话时,它会返回另一个模拟对象,在这种情况下,您可以返回一个模拟的 veue,或者委托调用到原来的会话

class MySession
  def initialize(original)
    @original = original
  end
  def [](key)
    if key == :mocked_key
      2
    else
      original[key]
    end
  end
end

let!(original_session) { controller.send(:session) }
let(:my_session) { MySession.new(original_session) }

before do
  controller.stubs(:session) { my_session }
end

猜猜摩卡也允许你做块模拟,所以你不需要类,但你需要original_session调用它

但我没有看到一个干净的方式

于 2021-05-24T21:22:41.670 回答