Initial support for and_yield

This commit is contained in:
Michael Miller 2019-11-17 13:12:39 -07:00
parent b896a7f1d5
commit e49bd0d57a
6 changed files with 48 additions and 4 deletions

View file

@ -12,7 +12,11 @@ module Spectator::Mocks
call = ::Spectator::Mocks::GenericMethodCall.new({{call.name.symbolize}}, args) call = ::Spectator::Mocks::GenericMethodCall.new({{call.name.symbolize}}, args)
::Spectator::Harness.current.mocks.record_call(self, call) ::Spectator::Harness.current.mocks.record_call(self, call)
if (stub = ::Spectator::Harness.current.mocks.find_stub(self, call)) if (stub = ::Spectator::Harness.current.mocks.find_stub(self, call))
{% if call.block.is_a?(Nop) %}
stub.call!(args, typeof(@values.fetch({{call.name.symbolize}}) { raise })) stub.call!(args, typeof(@values.fetch({{call.name.symbolize}}) { raise }))
{% else %}
stub.call!(args, typeof(@values.fetch({{call.name.symbolize}}) { raise })) { |*ya| yield *ya }
{% end %}
else else
@values.fetch({{call.name.symbolize}}) do @values.fetch({{call.name.symbolize}}) do
return nil if ::Spectator::Harness.current.mocks.expected?(self, {{call.name.symbolize}}) return nil if ::Spectator::Harness.current.mocks.expected?(self, {{call.name.symbolize}})

View file

@ -8,7 +8,11 @@ module Spectator::Mocks
call = ::Spectator::Mocks::GenericMethodCall.new({{call.name.symbolize}}, args) call = ::Spectator::Mocks::GenericMethodCall.new({{call.name.symbolize}}, args)
::Spectator::Harness.current.mocks.record_call(self, call) ::Spectator::Harness.current.mocks.record_call(self, call)
if (stub = ::Spectator::Harness.current.mocks.find_stub(self, call)) if (stub = ::Spectator::Harness.current.mocks.find_stub(self, call))
{% if call.block.is_a?(Nop) %}
stub.call!(args, typeof(@values.fetch({{call.name.symbolize}}) { self })) stub.call!(args, typeof(@values.fetch({{call.name.symbolize}}) { self }))
{% else %}
stub.call!(args, typeof(@values.fetch({{call.name.symbolize}}) { self })) { |*ya| yield *ya }
{% end %}
else else
@values.fetch({{call.name.symbolize}}) { self } @values.fetch({{call.name.symbolize}}) { self }
end end

View file

@ -45,12 +45,12 @@ module Spectator::Mocks
end end
end end
def {{name}}({{params.splat}}){% if definition.is_a?(TypeDeclaration) %} : {{definition.type}}{% end %} def {{name}}({{params.splat}} &block){% if definition.is_a?(TypeDeclaration) %} : {{definition.type}}{% end %}
%args = ::Spectator::Mocks::GenericArguments.create({{args.splat}}) %args = ::Spectator::Mocks::GenericArguments.create({{args.splat}})
%call = ::Spectator::Mocks::GenericMethodCall.new({{name.symbolize}}, %args) %call = ::Spectator::Mocks::GenericMethodCall.new({{name.symbolize}}, %args)
::Spectator::Harness.current.mocks.record_call(self, %call) ::Spectator::Harness.current.mocks.record_call(self, %call)
if (%stub = ::Spectator::Harness.current.mocks.find_stub(self, %call)) if (%stub = ::Spectator::Harness.current.mocks.find_stub(self, %call))
%stub.call!(%args, typeof(%method({{args.splat}}) { |*%ya| yield *%ya })) %stub.call!(%args, typeof(%method({{args.splat}}) { |*%ya| yield *%ya })) { |*%ya| yield *%ya }
else else
%method({{args.splat}}) do |*%yield_args| %method({{args.splat}}) do |*%yield_args|
yield *%yield_args yield *%yield_args

View file

@ -16,6 +16,10 @@ module Spectator::Mocks
abstract def call(args : GenericArguments(T, NT), rt : RT.class) forall T, NT, RT abstract def call(args : GenericArguments(T, NT), rt : RT.class) forall T, NT, RT
def call(args : GenericArguments(T, NT), rt : RT.class, &) forall T, NT, RT
call(args, rt)
end
def call!(args : GenericArguments(T, NT), rt : RT.class) : RT forall T, NT, RT def call!(args : GenericArguments(T, NT), rt : RT.class) : RT forall T, NT, RT
value = call(args, rt) value = call(args, rt)
if value.is_a?(RT) if value.is_a?(RT)
@ -25,6 +29,15 @@ module Spectator::Mocks
end end
end end
def call!(args : GenericArguments(T, NT), rt : RT.class) : RT forall T, NT, RT
value = call(args, rt) { |*ya| yield *ya }
if value.is_a?(RT)
value.as(RT)
else
raise TypeCastError.new("The return type of stub #{self} doesn't match the expected type #{RT}")
end
end
def to_s(io) def to_s(io)
io << '#' io << '#'
io << @name io << @name

View file

@ -36,6 +36,10 @@ module Spectator::Mocks
ExceptionMethodStub.new(@name, @source, exception_type.new(*args), @args) ExceptionMethodStub.new(@name, @source, exception_type.new(*args), @args)
end end
def and_yield(*yield_args)
YieldMethodStub.new(@name, @source, yield_args, @args)
end
def with(*args : *T, **opts : **NT) forall T, NT def with(*args : *T, **opts : **NT) forall T, NT
args = GenericArguments.new(args, opts) args = GenericArguments.new(args, opts)
NilMethodStub.new(@name, @source, args) NilMethodStub.new(@name, @source, args)

View file

@ -0,0 +1,19 @@
require "./generic_arguments"
require "./generic_method_stub"
module Spectator::Mocks
class YieldMethodStub(YieldArgs) < GenericMethodStub(Nil)
def initialize(name, source, @yield_args : YieldArgs, args = nil)
super(name, source, args)
end
def call(_args : GenericArguments(T2, NT2), rt : RT.class) forall T2, NT2, RT
raise "Asked to yield |#{@yield_args}| but no block was passed"
end
def call(_args : GenericArguments(T2, NT2), rt : RT.class) forall T2, NT2, RT
yield *@yield_args
nil
end
end
end