我如何模拟或覆盖Kernel.system?

oba*_*iro 1 ruby mocking testunit

如何以正确的方式模拟或覆盖Kernel.system方法,以便在调用时:

system("some command")
Run Code Online (Sandbox Code Playgroud)

而不是执行命令,它执行一些预定义的代码?

我尝试将以下内容添加到我的Test类中:

module Kernel
    def system
        puts "SYSTEM CALL!!"
    end
end
Run Code Online (Sandbox Code Playgroud)

但它没有按预期工作,而是在执行测试时运行系统调用.

Art*_*cki 7

在某些情况下,做法expect(Kernel).to receive(:system)还不够.

考虑这个例子:

foo_component.rb

class FooComponent
  def run
    system('....')
  end
end
Run Code Online (Sandbox Code Playgroud)

foo_component_spec.rb

require 'spec_helper'

describe FooComponent do
  let(:foo_component) { described_class.new }

  describe '#run' do
    it 'does some awesome things' do
      expect(Kernel).to receive(:system).with('....')
      foo_component.run
    end
  end
end
Run Code Online (Sandbox Code Playgroud)

不起作用.这是因为Kernel模块和Object(父类)混合在Kernel模块中,使得所有Kernel方法都可以在"全局"范围内使用.

这就是为什么正确的测试应该是这样的:

require 'spec_helper'

describe FooComponent do
  let(:foo_component) { described_class.new }

  describe '#run' do
    it 'does some awesome things' do
      expect(foo_component).to receive(:system).with('....')
      foo_component.run
    end
  end
end
Run Code Online (Sandbox Code Playgroud)


Spa*_*jus 5

如果您正在谈论单元测试并使用Rspec,那么您应该可以这样做:

Kernel.should_receive(:system)
Run Code Online (Sandbox Code Playgroud)

或更宽松:

Kernel.stub(:system)
Run Code Online (Sandbox Code Playgroud)

更多信息:https : //www.relishapp.com/rspec/rspec-mocks/v/2-13/docs/message-expectations/expect-a-message


Pet*_*ete 5

由于提出了这个问题,RSpec 3 提出了一种新的语法,您可以在其中编写:

expect(Kernel).to receive(:system)
Run Code Online (Sandbox Code Playgroud)

如果您的代码检查系统调用是否成功,您可以像这样指定结果:

expect(Kernel).to receive(:system).and_return(true)
Run Code Online (Sandbox Code Playgroud)

宽松版:

allow(Kernel).to receive(:system).and_return(true)
Run Code Online (Sandbox Code Playgroud)

  • 仅供参考 使用 ruby​​ 2.3.0 和 rspec-core 3.4.2,我不得不使用 `Kernel.system(...)` 而不是 `system(...)` 来模拟通过。 (2认同)

Joh*_* C. 5

如果它在一个类中,内核就会混合在一起。所以你只需模拟它,就好像它是对象的一部分一样。

例如

expect(subject).to receive(:system).and_return(foo)
Run Code Online (Sandbox Code Playgroud)