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

Best-effort error reporting for interactions on final methods #2040

Merged
merged 15 commits into from
Dec 28, 2024

Conversation

AndreasTu
Copy link
Member

@AndreasTu AndreasTu commented Nov 7, 2024

Add error reporting code to the byte-buddy mock maker to report interaction on final methods, which can't be handled.

The IMockInteractionValidator interface allows IMockMakers to implement different kinds of validations on mock interactions.

Copy link

codecov bot commented Nov 7, 2024

Codecov Report

Attention: Patch coverage is 88.18898% with 15 lines in your changes missing coverage. Please review.

Project coverage is 81.84%. Comparing base (2c7db77) to head (c148647).
Report is 180 commits behind head on master.

Files with missing lines Patch % Lines
...untime/extension/builtin/FailsWithInterceptor.java 12.50% 4 Missing and 3 partials ⚠️
...va/org/spockframework/mock/runtime/MockObject.java 71.42% 1 Missing and 3 partials ⚠️
...ock/runtime/ByteBuddyMockInteractionValidator.java 95.74% 1 Missing and 1 partial ⚠️
...ckframework/spring/mock/DelegatingInterceptor.java 33.33% 2 Missing ⚠️
Additional details and impacted files

Impacted file tree graph

@@             Coverage Diff              @@
##             master    #2040      +/-   ##
============================================
+ Coverage     80.44%   81.84%   +1.39%     
- Complexity     4337     4686     +349     
============================================
  Files           441      452      +11     
  Lines         13534    14659    +1125     
  Branches       1707     1845     +138     
============================================
+ Hits          10888    11998    +1110     
+ Misses         2008     1979      -29     
- Partials        638      682      +44     
Files with missing lines Coverage Δ
...main/java/org/spockframework/compiler/AstUtil.java 80.13% <ø> (+0.26%) ⬆️
...java/org/spockframework/compiler/SpecRewriter.java 93.91% <100.00%> (+0.21%) ⬆️
...n/java/org/spockframework/compiler/SpockNames.java 0.00% <ø> (ø)
...in/java/org/spockframework/lang/SpecInternals.java 86.86% <100.00%> (+5.85%) ⬆️
...java/org/spockframework/mock/ISpockMockObject.java 100.00% <100.00%> (ø)
.../main/java/org/spockframework/mock/MockNature.java 100.00% <100.00%> (ø)
...rc/main/java/org/spockframework/mock/MockUtil.java 100.00% <100.00%> (ø)
...ork/mock/constraint/EqualMethodNameConstraint.java 100.00% <100.00%> (ø)
...k/mock/constraint/EqualPropertyNameConstraint.java 100.00% <100.00%> (ø)
...ockframework/mock/constraint/TargetConstraint.java 100.00% <100.00%> (+6.66%) ⬆️
... and 12 more

... and 20 files with indirect coverage changes

@AndreasTu AndreasTu force-pushed the fix_2039 branch 2 times, most recently from 2b54442 to a50eb7f Compare November 7, 2024 13:04
}

@Issue("https://github.com/spockframework/spock/issues/2039")
def "cannot stub final methods with byteBuddy without error message when one overload is non final"() {
Copy link
Member

Choose a reason for hiding this comment

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

The method name suggests that we should get an error here, but the test does not reflect that.

Copy link
Member Author

Choose a reason for hiding this comment

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

It fails to mock the method,. because one of the overloads is final, so there is not error message, because the parameters could have matched both the final or the non-final method, so we can't detect that right now => No error message.

This is the best-effort part of this PR.

Copy link
Member

Choose a reason for hiding this comment

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

Yes, but this doesn't become obvious. The confusing part is what without error message refers to. It can be read in two ways: one, that you can't do this without getting an error message, or it is a scenario where you don't expect to get an error message.

Copy link
Member

Choose a reason for hiding this comment

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

It might help to use block labels in the test.

* @author Andreas Turban
*/
@ThreadSafe
final class ByteBuddyMockInteractionValidation implements IMockInteractionValidation {
Copy link
Member

Choose a reason for hiding this comment

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

Is this really specific to ByteBuddy, or does it apply to CgLib as well?

Copy link
Member Author

@AndreasTu AndreasTu Nov 9, 2024

Choose a reason for hiding this comment

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

It would also work for the cglib mock maker, but I did not want to implement the static field cache in cglib, because it is already deprecated.

If you want that supported, let me know, then I will have a look.

@Vampire
Copy link
Member

Vampire commented Nov 8, 2024

I didn't have time for a proper review, but is the validation done at compile-time or runtime?
If it was done at runtime, how much is the performance impact?

@AndreasTu
Copy link
Member Author

@Vampire It is done at runtime.
During compile-time, it would not be possible to know the used mock maker, therefore it would lead to false positives, if the user would use e.g. mockito or an external mock maker.

During runtime, the expensive operation of calculating the final methods is done, once per class and only if an interaction is specified. Note: Currently I am only checking for the public finals, => class.getMethods() not for anything else, as I said best-effort. But this could be extended in the future.

The final methods Set cache is attached to the generated mock class as static field, so the cache is GCed without any weakrefs etc., when the mock class is GCed.

For each interaction specification the cost is to loop over the contained IInvocationConstraints in the IMockInteraction and make a Set lookup for the method name, and two Set lookups for a property name, if the byteBuddy mock maker is used.

For mockito and java-proxy there is not overhead, except of the getter of the spock mock obj and a null check.

Add error reporting code to the byte-buddy` mock maker
to report interaction on final methods, which can't be handled.

The IMockInteractionValidator interface allows IMockMakers
to implement different kinds of validations on mock interactions.

Fixes spockframework#2039
}

@Issue("https://github.com/spockframework/spock/issues/2039")
def "cannot stub final methods with byteBuddy without error message when one overload is non final"() {
Copy link
Member

Choose a reason for hiding this comment

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

Yes, but this doesn't become obvious. The confusing part is what without error message refers to. It can be read in two ways: one, that you can't do this without getting an error message, or it is a scenario where you don't expect to get an error message.

@AndreasTu AndreasTu requested a review from leonard84 December 19, 2024 17:44
@AndreasTu AndreasTu merged commit e081141 into spockframework:master Dec 28, 2024
29 checks passed
@AndreasTu AndreasTu deleted the fix_2039 branch December 28, 2024 14:21
@AndreasTu
Copy link
Member Author

@leonard84 Thanks for the review and the changes!

@AndreasTu AndreasTu added this to the 2.4-M5 milestone Dec 28, 2024
@leonard84
Copy link
Member

You were too fast, I wanted to split out the @FailsWith into its own PR 😅 .

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

Successfully merging this pull request may close these issues.

3 participants