mirror of
https://gitea.invidious.io/iv-org/shard-spectator.git
synced 2024-08-15 00:53:35 +00:00
Support creating instances of mocked modules via class
This is a bit of a hack. The `.new` method is added to the module, which creates an instance that includes the mocked module. No changes to the def_mock and new_mock methods are necessary. For some reason, infinite recursion occurs when calling `.new` on the class. To get around the issue for now, the internal method of allocation is used. That is, allocate + initialize.
This commit is contained in:
parent
d378583054
commit
fa99987780
4 changed files with 388 additions and 0 deletions
|
@ -1027,4 +1027,262 @@ Spectator.describe "Mock DSL", :smoke do
|
|||
expect(fake.reference).to eq("reference")
|
||||
end
|
||||
end
|
||||
|
||||
describe "mock module" do
|
||||
module Dummy
|
||||
# `extend self` cannot be used.
|
||||
# The Crystal compiler doesn't report the methods as class methods when doing so.
|
||||
|
||||
def self.abstract_method
|
||||
:not_really_abstract
|
||||
end
|
||||
|
||||
def self.default_method
|
||||
:original
|
||||
end
|
||||
|
||||
def self.args(arg)
|
||||
arg
|
||||
end
|
||||
|
||||
def self.method1
|
||||
:original
|
||||
end
|
||||
|
||||
def self.reference
|
||||
method1.to_s
|
||||
end
|
||||
end
|
||||
|
||||
mock(Dummy) do
|
||||
abstract_stub def self.abstract_method
|
||||
:abstract
|
||||
end
|
||||
|
||||
stub def self.default_method
|
||||
:default
|
||||
end
|
||||
end
|
||||
|
||||
let(fake) { class_mock(Dummy) }
|
||||
|
||||
it "raises on abstract stubs" do
|
||||
expect { fake.abstract_method }.to raise_error(Spectator::UnexpectedMessage, /abstract_method/)
|
||||
end
|
||||
|
||||
it "can define default stubs" do
|
||||
expect(fake.default_method).to eq(:default)
|
||||
end
|
||||
|
||||
it "can define new stubs" do
|
||||
expect { allow(fake).to receive(:args).and_return(42) }.to change { fake.args(5) }.from(5).to(42)
|
||||
end
|
||||
|
||||
it "can override class method stubs" do
|
||||
allow(fake).to receive(:method1).and_return(:override)
|
||||
expect(fake.method1).to eq(:override)
|
||||
end
|
||||
|
||||
xit "can reference stubs", pending: "Default stub of module class methods always refer to original" do
|
||||
allow(fake).to receive(:method1).and_return(:reference)
|
||||
expect(fake.reference).to eq("reference")
|
||||
end
|
||||
end
|
||||
|
||||
context "with a class including a mocked module" do
|
||||
module Dummy
|
||||
getter _spectator_invocations = [] of Symbol
|
||||
|
||||
def method1
|
||||
@_spectator_invocations << :method1
|
||||
"original"
|
||||
end
|
||||
|
||||
def method2 : Symbol
|
||||
@_spectator_invocations << :method2
|
||||
:original
|
||||
end
|
||||
|
||||
def method3(arg)
|
||||
@_spectator_invocations << :method3
|
||||
arg
|
||||
end
|
||||
|
||||
def method4 : Symbol
|
||||
@_spectator_invocations << :method4
|
||||
yield
|
||||
end
|
||||
|
||||
def method5
|
||||
@_spectator_invocations << :method5
|
||||
yield.to_i
|
||||
end
|
||||
|
||||
def method6
|
||||
@_spectator_invocations << :method6
|
||||
yield
|
||||
end
|
||||
|
||||
def method7(arg, *args, kwarg, **kwargs)
|
||||
@_spectator_invocations << :method7
|
||||
{arg, args, kwarg, kwargs}
|
||||
end
|
||||
|
||||
def method8(arg, *args, kwarg, **kwargs)
|
||||
@_spectator_invocations << :method8
|
||||
yield
|
||||
{arg, args, kwarg, kwargs}
|
||||
end
|
||||
end
|
||||
|
||||
# method1 stubbed via mock block
|
||||
# method2 stubbed via keyword args
|
||||
# method3 not stubbed (calls original)
|
||||
# method4 stubbed via mock block (yields)
|
||||
# method5 stubbed via keyword args (yields)
|
||||
# method6 not stubbed (calls original and yields)
|
||||
# method7 not stubbed (calls original) testing args
|
||||
# method8 not stubbed (calls original and yields) testing args
|
||||
mock(Dummy, method2: :stubbed, method5: 42) do
|
||||
stub def method1
|
||||
"stubbed"
|
||||
end
|
||||
|
||||
stub def method4 : Symbol
|
||||
yield
|
||||
:block
|
||||
end
|
||||
end
|
||||
|
||||
subject(fake) { mock(Dummy) }
|
||||
|
||||
it "defines a subclass" do
|
||||
expect(fake).to be_a(Dummy)
|
||||
end
|
||||
|
||||
it "defines stubs in the block" do
|
||||
expect(fake.method1).to eq("stubbed")
|
||||
end
|
||||
|
||||
it "can stub methods defined in the block" do
|
||||
stub = Spectator::ValueStub.new(:method1, "override")
|
||||
expect { fake._spectator_define_stub(stub) }.to change { fake.method1 }.from("stubbed").to("override")
|
||||
end
|
||||
|
||||
it "defines stubs from keyword arguments" do
|
||||
expect(fake.method2).to eq(:stubbed)
|
||||
end
|
||||
|
||||
it "can stub methods from keyword arguments" do
|
||||
stub = Spectator::ValueStub.new(:method2, :override)
|
||||
expect { fake._spectator_define_stub(stub) }.to change { fake.method2 }.from(:stubbed).to(:override)
|
||||
end
|
||||
|
||||
it "calls the original implementation for methods not provided a stub" do
|
||||
expect(fake.method3(:xyz)).to eq(:xyz)
|
||||
end
|
||||
|
||||
it "can stub methods after declaration" do
|
||||
stub = Spectator::ValueStub.new(:method3, :abc)
|
||||
expect { fake._spectator_define_stub(stub) }.to change { fake.method3(:xyz) }.from(:xyz).to(:abc)
|
||||
end
|
||||
|
||||
it "defines stubs with yield in the block" do
|
||||
expect(fake.method4 { :wrong }).to eq(:block)
|
||||
end
|
||||
|
||||
it "can stub methods with yield in the block" do
|
||||
stub = Spectator::ValueStub.new(:method4, :override)
|
||||
expect { fake._spectator_define_stub(stub) }.to change { fake.method4 { :wrong } }.from(:block).to(:override)
|
||||
end
|
||||
|
||||
it "defines stubs with yield from keyword arguments" do
|
||||
expect(fake.method5 { :wrong }).to eq(42)
|
||||
end
|
||||
|
||||
it "can stub methods with yield from keyword arguments" do
|
||||
stub = Spectator::ValueStub.new(:method5, 123)
|
||||
expect { fake._spectator_define_stub(stub) }.to change { fake.method5 { "0" } }.from(42).to(123)
|
||||
end
|
||||
|
||||
it "can stub yielding methods after declaration" do
|
||||
stub = Spectator::ValueStub.new(:method6, :abc)
|
||||
expect { fake._spectator_define_stub(stub) }.to change { fake.method6 { :xyz } }.from(:xyz).to(:abc)
|
||||
end
|
||||
|
||||
it "handles arguments correctly" do
|
||||
args1 = fake.method7(1, 2, 3, kwarg: 4, x: 5, y: 6, z: 7)
|
||||
args2 = fake.method8(1, 2, 3, kwarg: 4, x: 5, y: 6, z: 7) { :block }
|
||||
aggregate_failures do
|
||||
expect(args1).to eq({1, {2, 3}, 4, {x: 5, y: 6, z: 7}})
|
||||
expect(args2).to eq({1, {2, 3}, 4, {x: 5, y: 6, z: 7}})
|
||||
end
|
||||
end
|
||||
|
||||
it "handles arguments correctly with stubs" do
|
||||
stub1 = Spectator::ProcStub.new(:method7, args_proc)
|
||||
stub2 = Spectator::ProcStub.new(:method8, args_proc)
|
||||
fake._spectator_define_stub(stub1)
|
||||
fake._spectator_define_stub(stub2)
|
||||
args1 = fake.method7(1, 2, 3, kwarg: 4, x: 5, y: 6, z: 7)
|
||||
args2 = fake.method8(1, 2, 3, kwarg: 4, x: 5, y: 6, z: 7) { :block }
|
||||
aggregate_failures do
|
||||
expect(args1).to eq({1, {2, 3}, 4, {x: 5, y: 6, z: 7}})
|
||||
expect(args2).to eq({1, {2, 3}, 4, {x: 5, y: 6, z: 7}})
|
||||
end
|
||||
end
|
||||
|
||||
it "compiles types without unions" do
|
||||
aggregate_failures do
|
||||
expect(fake.method1).to compile_as(String)
|
||||
expect(fake.method2).to compile_as(Symbol)
|
||||
expect(fake.method3(42)).to compile_as(Int32)
|
||||
expect(fake.method4 { :foo }).to compile_as(Symbol)
|
||||
expect(fake.method5 { "123" }).to compile_as(Int32)
|
||||
expect(fake.method6 { "123" }).to compile_as(String)
|
||||
end
|
||||
end
|
||||
|
||||
def restricted(thing : Dummy)
|
||||
thing.method1
|
||||
end
|
||||
|
||||
it "can be used in type restricted methods" do
|
||||
expect(restricted(fake)).to eq("stubbed")
|
||||
end
|
||||
|
||||
it "does not call the original method when stubbed" do
|
||||
fake.method1
|
||||
fake.method2
|
||||
fake.method3("foo")
|
||||
fake.method4 { :foo }
|
||||
fake.method5 { "42" }
|
||||
fake.method6 { 42 }
|
||||
fake.method7(1, 2, 3, kwarg: 4, x: 5, y: 6, z: 7)
|
||||
fake.method8(1, 2, 3, kwarg: 4, x: 5, y: 6, z: 7) { :block }
|
||||
expect(fake._spectator_invocations).to contain_exactly(:method3, :method6, :method7, :method8)
|
||||
end
|
||||
|
||||
# Cannot test unexpected messages - will not compile due to missing methods.
|
||||
|
||||
describe "deferred default stubs" do
|
||||
mock(Dummy)
|
||||
|
||||
let(fake2) do
|
||||
mock(Dummy,
|
||||
method1: "stubbed",
|
||||
method3: 123,
|
||||
method4: :xyz)
|
||||
end
|
||||
|
||||
it "uses the keyword arguments as stubs" do
|
||||
aggregate_failures do
|
||||
expect(fake2.method1).to eq("stubbed")
|
||||
expect(fake2.method2).to eq(:original)
|
||||
expect(fake2.method3(42)).to eq(123)
|
||||
expect(fake2.method4 { :foo }).to eq(:xyz)
|
||||
end
|
||||
end
|
||||
end
|
||||
end
|
||||
end
|
||||
|
|
Loading…
Add table
Add a link
Reference in a new issue