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

GH-3344: Treat kotlin.Unit return as null in MMIH #3346

Merged
merged 3 commits into from
Jul 20, 2020

Conversation

artembilan
Copy link
Member

Fixes #3344

When function lambda doesn't return anything (e.g. a void method call is the last one),
Kotlin produces a kotlin.Unit instance as a return value which is not null and produced
as a reply message payload.

  • Fix MessagingMethodInvokerHelper to treat a kotlin.Unit as null for reply
    making Kotlin lambdas working the same way as Java lambdas when we don't return anything
    from from there

Cherry-pick to 5.3.x

Fixes spring-projects#3344

When function lambda doesn't return anything (e.g. a `void` method call is the last one),
Kotlin produces a `kotlin.Unit` instance as a return value which is not null and produced
as a reply message payload.

* Fix `MessagingMethodInvokerHelper` to treat a `kotlin.Unit` as `null` for reply
making Kotlin lambdas working the same way as Java lambdas when we don't return anything
from from there

**Cherry-pick to `5.3.x`**
public Object invoke(ParametersWrapper parameters) {
Message<?> message = parameters.getMessage();
if (this.canProcessMessageList) {
message = new MutableMessage<>(parameters.getMessages(), parameters.getHeaders());
}
try {
return this.invocableHandlerMethod.invoke(message);
Object result = this.invocableHandlerMethod.invoke(message);
if (result != null && result.getClass().getName().equals("kotlin.Unit")) {
Copy link
Contributor

Choose a reason for hiding this comment

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

Can we not capture this condition as a final boolean by checking the return type in a CTOR (to avoid running this code on every non-null return)?

Copy link
Member Author

Choose a reason for hiding this comment

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

Well, we can avoid something indeed, when a return type is void or Unit. However in this case the return type is Object so end-user can return null ()or just don't have return in the function lambda and we don't produce a reply message.
In this case Kotlin for non-return in function lambda produces for us that Unit which is not a null apparently.
So, we produce some weird reply message.
Therefore even if we have some final checks for return type, we still would have some logic to check the returned value to be sure that we don't produce a reply message with Unit payload.

To summarize: when have void return this result is always going to be null, therefore we don't go to check a class name for Unit. If the value not null, it already doesn't matter what void check we have in advance: we still have to check this result for Unit type.

Does it make sense?

Copy link
Contributor

Choose a reason for hiding this comment

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

Could we at least do something like...

if (result != null && this.isKotlinLambda && result.getClass().getName().equals("kotlin.Unit")) {

?

This result.getClass().getName().equals("kotlin.Unit") just seems like a lot of overhead for this one corner case and not good for the 99.9% of calls.

Copy link
Member Author

Choose a reason for hiding this comment

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

Ah! OK. Makes sense.
I'll think what we can do...
Thanks for the pointer!

Verified

This commit was created on GitHub.com and signed with GitHub’s verified signature.
use it in the `MessagingMethodInvokerHelper` instead of
`.getName().equals()`
@artembilan artembilan requested a review from garyrussell July 20, 2020 16:53
@artembilan
Copy link
Member Author

@garyrussell ,

how about this solution?

@garyrussell
Copy link
Contributor

Yes, that's better but you've tagged it @since 5.4 so no more backport?

Verified

This commit was created on GitHub.com and signed with GitHub’s verified signature.
@artembilan
Copy link
Member Author

Fixed here: db6786c.

Thanks for pointing that out!

@garyrussell garyrussell merged commit 6a865b5 into spring-projects:master Jul 20, 2020
garyrussell pushed a commit that referenced this pull request Jul 20, 2020
* GH-3344: Treat kotlin.Unit return as null in MMIH

Fixes #3344

When function lambda doesn't return anything (e.g. a `void` method call is the last one),
Kotlin produces a `kotlin.Unit` instance as a return value which is not null and produced
as a reply message payload.

* Fix `MessagingMethodInvokerHelper` to treat a `kotlin.Unit` as `null` for reply
making Kotlin lambdas working the same way as Java lambdas when we don't return anything
from from there

**Cherry-pick to `5.3.x`**

* * Introduce `ClassUtils.isKotlinUnit(Class)` API;
use it in the `MessagingMethodInvokerHelper` instead of
`.getName().equals()`

* * Fix since on new `isKotlinUnit()` API
@garyrussell
Copy link
Contributor

And cherry-picked.

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
2 participants