Work around strange cast/type checking issue

For some reason, line 421 (the responds to call check) excluded the stub's call type.
Luckily this line doesn't seem to be necessary anymore.
Removed the unecessary quick check.

The tests from spec/spectator/mocks/double_spec:88-96 were failing when they're the only tests in the file.
The non-matching stub wouldn't raise.
Stepping through, attempting to access the value would segfault.
This is because it accessed a stub with String instead of its real Int32 type.
Removing the aforementioned check fixes this.
This commit is contained in:
Michael Miller 2022-11-27 19:43:03 -07:00
parent 318e4eba89
commit 015d36ea4c
No known key found for this signature in database
GPG key ID: 32B47AE8F388A1FF

View file

@ -398,7 +398,7 @@ module Spectator
{% end %} {% end %}
end end
# Utility macro for casting a stub (and it's return value) to the correct type. # Utility macro for casting a stub (and its return value) to the correct type.
# #
# *stub* is the variable holding the stub. # *stub* is the variable holding the stub.
# *call* is the variable holding the captured method call. # *call* is the variable holding the captured method call.
@ -408,49 +408,37 @@ module Spectator
# - `:raise` - raise a `TypeCastError`. # - `:raise` - raise a `TypeCastError`.
# - `:no_return` - raise as no value should be returned. # - `:no_return` - raise as no value should be returned.
private macro _spectator_cast_stub_value(stub, call, type, fail_cast = :nil) private macro _spectator_cast_stub_value(stub, call, type, fail_cast = :nil)
# Attempt to cast the stub to the method's return type. {% if fail_cast == :no_return %}
# If successful, return the value of the stub. {{stub}}.call({{call}})
# This is a common usage where the return type is simple and matches the stub type exactly. raise TypeCastError.new("#{_spectator_stubbed_name} received message #{ {{call}} } and is attempting to return a value, but it shouldn't have returned (`NoReturn`).")
if %typed = {{stub}}.as?(::Spectator::TypedStub({{type}})) {% else %}
%typed.call({{call}}) # Get the value as-is from the stub.
else # This will be compiled as a union of all known stubbed value types.
# The stub couldn't be easily cast to match the return type. %value = {{stub}}.call({{call}})
# Even though all stubs will have a `#call` method, the compiler doesn't seem to agree. # Attempt to cast the value to the method's return type.
# Assert that it will (this should never fail). # If successful, which it will be in most cases, return it.
raise TypeCastError.new("Stub has no value") unless {{stub}}.responds_to?(:call) # The caller will receive a properly typed value without unions or other side-effects.
%cast = %value.as?({{type}})
{% if fail_cast == :no_return %} if %cast.is_a?({{type}})
{{stub}}.call({{call}}) %cast
raise TypeCastError.new("#{_spectator_stubbed_name} received message #{ {{call}} } and is attempting to return a value, but it shouldn't have returned (`NoReturn`).") else
{% else %} {% if fail_cast == :nil %}
# Get the value as-is from the stub. nil
# This will be compiled as a union of all known stubbed value types. {% elsif fail_cast == :raise %}
%value = {{stub}}.call({{call}}) # The stubbed value was something else entirely and cannot be cast to the return type.
# There's something weird going on (compiler bug?) that sometimes causes this class lookup to fail.
# Attempt to cast the value to the method's return type. %type = begin
# If successful, which it will be in most cases, return it. %value.class.to_s
# The caller will receive a properly typed value without unions or other side-effects. rescue
if %cast = %value.as?({{type}}) "<Unknown>"
%cast end
else raise TypeCastError.new("#{_spectator_stubbed_name} received message #{ {{call}} } and is attempting to return a `#{%type}`, but returned type must be `#{ {{type}} }`.")
{% if fail_cast == :nil %} {% else %}
nil {% raise "fail_cast must be :nil, :raise, or :no_return, but got: #{fail_cast}" %}
{% elsif fail_cast == :raise %} {% end %}
# The stubbed value was something else entirely and cannot be cast to the return type. end
# There's something weird going on (compiler bug?) that sometimes causes this class lookup to fail. {% end %}
%type = begin
%value.class.to_s
rescue
"<Unknown>"
end
raise TypeCastError.new("#{_spectator_stubbed_name} received message #{ {{call}} } and is attempting to return a `#{%type}`, but returned type must be `#{ {{type}} }`.")
{% else %}
{% raise "fail_cast must be :nil, :raise, or :no_return, but got: #{fail_cast}" %}
{% end %}
end
{% end %}
end
end end
end end
end end