Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

'Returns' regression with null function callback #602

Merged
merged 3 commits into from
Mar 7, 2018

Conversation

Caraul
Copy link
Contributor

@Caraul Caraul commented Mar 6, 2018

Restore 4.7 behavior, add tests and update CHANGELOG.md

Closes #600.

Copy link
Contributor

@stakx stakx left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This looks good to me, good work! 👍 If you don't mind, just two minor things that I'd like you to take a look at.

CHANGELOG.md Outdated
@@ -20,6 +20,7 @@ The format is loosely based on [Keep a Changelog](http://keepachangelog.com/en/1

* Usage of `ReturnsExtensions.ThrowsAsync()` can cause `UnobservedTaskException` (@snrnats, #595)
* `ReturnsAsync` and `ThrowsAsync` with delay parameter starts timer at setup (@snrnats, #595)
* `Returns` regression with null function callback (@Caraul, #600)
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Please replace the issue number (#600) with the PR number (#602) to keep the changelog formatting consistent.

if (value == null)
{
// The following may seem overly cautious, but we don't throw an `ArgumentNullException`
// here because Moq has been very forgiving with incorrect `Returns` in the past.
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Let's rewrite these two comment lines so it becomes clear that we're not just being "overly cautious", but that this is in fact a required piece of code. For example:

// A `null` reference (instead of a valid delegate) is interpreted as the actual return value.
// This is necessary because the compiler might have picked the wrong overload for calls like
// `Returns(null)`, and instead of in `Returns(TResult)`, we ended up in `Returns(Delegate)` or
// `Returns(Func<TResult>)`, which likely isn't what the user intended. So here we do what
// we would've done in `Returns(TResult)`:

(We're essentially dealing with the same problem—C# picking the "wrong" overload—as already mentioned here:)

https://github.com/moq/moq4/blob/f86759225c46b072c4a721b200b31992d4c54b8b/Source/MethodCallReturn.cs#L92-L101

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Well, the comment is subject to discuss :)

I've already mentioned "Returns(null) doesn't compile due to ambiguous invocation" so there is no wrong pick - anyway the user should clearly choose one of Returns overloads with typecasting or "closing" generics etc. Rock on which we split is Returns<TResult>(null) which seems like Returns((TResult)null) but in fact it's IReturns.Returns<T>((Func<T, TResult>)null) - surprisingly, isn't it? So I'd propose sightly modified version:

// A `null` reference (instead of a valid delegate) is interpreted as the actual return value.
// This is necessary because the user might have picked an ambiguous overload like `Returns<TResult>(null)`,
// but instead of in `Returns(TResult)`, we ended up in `Returns<T>(Func<T, TResult>)`,
// which likely isn't what the user intended. So here we do what we would've done in `Returns(TResult)`

What do you think?

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

"Returns(null) doesn't compile due to ambiguous invocation" so there is no wrong pick

Returns(null) compiles just fine in some quick scenarios that I've tried, so I can't really confirm this.

anyway the user should clearly choose one of Returns overloads with typecasting or "closing" generics etc.

That may be so, but we have no control over whether they think so, too, nor whether users will actually do so. If everyone explicitly chose the right overload, then passing null instead of a delegate should trigger an ArgumentNullException. But let's stay on the safe side and assume that some people let the compiler pick the right overload. And because that can go wrong, we need to "correct" those situations.

Regarding your new comment, that's principally also fine with me. But let's be clear that to the compiler, the overload isn't ambiguous at all. It's simply that users might expect a different overload to be called than what the compiler will choose.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Returns(null) compiles just fine in some quick scenarios that I've tried, so I can't really confirm this.

The truth appears to be in between - it's ambiguous for references but for value types compiler chooses Returns(Func<TResult>).

users might expect a different overload

My point exactly. The only note that reasons are "compiler chooses unexpected overload" and "user chooses an overload that behaves unexpectedly". I'd also try to avoid the word "wrong" as it may be incorrectly interpreted as "bug in compiler". So what about version:

// A `null` reference (instead of a valid delegate) is interpreted as the actual return value.
// This is necessary because the compiler might have picked the unexpected overload for calls like
// `Returns(null)`, or the user might have picked an overload like `Returns<TResult>(null)`,
// and instead of in `Returns(TResult)`, we ended up in `Returns(Delegate)` or
// `Returns(Func<TResult>)`, which likely isn't what the user intended. So here we do what
// we would've done in `Returns(TResult)`:

Copy link
Contributor

@stakx stakx Mar 7, 2018

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

That's fine with me, too, except I think that when the user picks a specific overload e.g. by explicitly specifying the generic type argument, then AFAIK we cannot end up in any other overloads. So I'd remove the part that says or the user might .... (How can the chosen overload be unexpected if the choice was explicitly made by the user?)

I agree that avoiding the word wrong is a good idea.

I really don't mean to be too fussy about the comment, main thing is that we're more specific than my original overly cautious comment, which clearly no longer applies.

Copy link
Contributor

@stakx stakx left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Thanks for the changes!

@stakx stakx merged commit d4aea9c into devlooped:master Mar 7, 2018
@stakx
Copy link
Contributor

stakx commented Mar 7, 2018

... and merged! Thank you for fixing this regression!

@stakx
Copy link
Contributor

stakx commented Jun 9, 2018

@Caraul - I just pushed Moq 4.8.3 to NuGet, which includes your changes. Sorry it's taken so long.

@Caraul
Copy link
Contributor Author

Caraul commented Jun 14, 2018 via email

@devlooped devlooped locked and limited conversation to collaborators Sep 7, 2024
Sign up for free to subscribe to this conversation on GitHub. Already have an account? Sign in.
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

2 participants