-
-
Notifications
You must be signed in to change notification settings - Fork 2.6k
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
Remove VarargMatcher
#2835
Remove VarargMatcher
#2835
Conversation
fixes: mockito#1593 Remove broken `VarargMatcher` internal interface. BREAKING CHANGE: This changes the default behaviour of the `any()` matcher when passed to a varargs parameter. Previously, the `any()` matcher would match each element in the varargs parameter, matching any number of elements 0...n. From Mockito v5 onwards `any()`, when passed to a varargs parameter, will match invocations where a single value is passed to the varargs parameter. To match any number of values passed to the varargs parameter, pass the type of the varargs parameter to `any()`. For example, given a `String...` varargs parameter, use `any(String[].class)`.
Codecov ReportAll modified and coverable lines are covered by tests ✅
Additional details and impacted files@@ Coverage Diff @@
## main #2835 +/- ##
============================================
- Coverage 84.90% 84.88% -0.03%
+ Complexity 2877 2875 -2
============================================
Files 327 327
Lines 8773 8761 -12
Branches 1063 1060 -3
============================================
- Hits 7449 7437 -12
Misses 1047 1047
Partials 277 277 ☔ View full report in Codecov by Sentry. |
@TimvdLippe I'm working through current implementations of FYI: I can't get a green build locally on main branch. Seeing loads of errors about like: java.lang.IllegalStateException: Could not initialize plugin: interface org.mockito.plugins.MockMaker (alternate: null)
...
Could not initialize inline Byte Buddy mock maker.
It appears as if your JDK does not supply a working agent attachment mechanism.
Java : 15
JVM vendor name : AdoptOpenJDK
JVM vendor version : 15.0.2+7
JVM name : OpenJDK 64-Bit Server VM
JVM version : 15.0.2+7
JVM info : mixed mode, sharing
OS name : Mac OS X
OS version : 10.16
... What JDK do you recommend using with Mockito? |
...tests duplicate others in `VarargsTest`
BREAKING CHANGE: The behaviour of argument captors when passed to a varargs parameter has changed. Prior to Mockito v5, passing an argument captor to a varargs parameter would capture any number of values passed to the varargs parameter. From Mockito v5, the same call will match only a single value passed to the varargs parameter. To capture any number of values, change to the type of the argument captor to match the array type of the varargs parameter. For example, given a `String...` varargs parameter, change the argument captor from type `String` to type `String[]`.
src/test/java/org/mockitousage/bugs/varargs/VarargsNotPlayingWithAnyTest.java
Show resolved
Hide resolved
... fix tests!
... fix tests!
BREAKING CHANGE: The behaviour of all methods in `MockitoHamcrest` when passed to a varargs parameter has changed. Prior to Mockito v5, these matchers would match each element in the varargs parameter, matching any number of elements 0...n. From Mockito v5 onwards these matchers, when passed to a varargs parameter, will match invocations where a single value is passed to the varargs parameter. To match any number of values passed to the varargs parameter, pass use suitable Matcher that matches the vararg's array type. For example, given a `String...` varargs parameter, use a Matcher that handles `String[].class`.
... required to support matchers that match a vararg parameter, rather than values in a varargs parameter.
All done. Over to you @TimvdLippe :) |
It's only a breaking change for `any()` where:
* it’s used for a varargs parameter, and
* no additional marchers occur after this matcher, and
* it’s used to match either zero or more than two values being passed to
the vararg parameter.
My previous PR added support for `any(String[].class)`, and added a `isA(String[].class)` as well. This is the new preferred way of matching a varargs parameter of any length.
However, without this PR it still won’t be possible to use `any()` to
match exactly one value passed to the varargs parameter. Even using something like `any(String.class)` would match zero or more values. Fixing that has to be a breaking change.
That said, you can not use `isA(String.class)` to match exactly one value.
We could release 5,0 workout this change, but that means waiting until 6.0
before fixing this issue. Which is… a year away?
IMHO, there is no point in holding off unless users know they need to migrate to the new solution. We can't mark `any()` as deprecated, as it's not. But we could add a log message when `any()` is being used in a deprecated way. Is this something Mockito has done before:? The log message could also include details of how to use `isA` for matching exactly one value.
Up to you.
…On Fri, 23 Dec 2022 at 18:22, Tim van der Lippe ***@***.***> wrote:
***@***.**** commented on this pull request.
------------------------------
In src/test/java/org/mockitousage/matchers/VarargsTest.java
<#2835 (comment)>:
> @@ -363,20 +363,6 @@ public void shouldMockVarargsInvocation_multiple_vararg_matcher() {
assertThat(mock.methodWithVarargAndNonVarargVariants("a", "b", "c")).isNull();
}
- @test
- public void shouldMockVarargsInvocationUsingCasts() {
Hm, breaking any() sounds a bit scary. Is it possible to write code that
works both before and after? E.g. is it possible to use
any(String[].class) before this change as well? Ideally these are not
mutually exclusive, so our users can prepare for the breaking change up
front.
—
Reply to this email directly, view it on GitHub
<#2835 (comment)>, or
unsubscribe
<https://github.com/notifications/unsubscribe-auth/AB5EE3UZZG5MA3WD2VKEVYLWOWDQTANCNFSM6AAAAAATHNWRR4>
.
You are receiving this because you authored the thread.Message ID:
***@***.***>
|
Thoughts @TimvdLippe ? |
@big-andy-coates with regards to the breakage: it seems like users have alternatives that will work on both 4.10.0 and 5.0.0. This is essentially my only requirement: it should be possible to write that will work before and after this change. That way, users can first update their usage on 4.10.0 to get it to work, then upgrade to 5.0.0 and then potentially update their code again if they so desire. As I understand your comment, this would be possible with With regards to your JDK issue: you are using JDK 15. Can you try out either 11 or 17? That should hopefully work. If that doesn't work, maybe @reta can share their local setup? They have been contributing lately as well, so I am assuming their local setup works. |
@TimvdLippe #2807 is not released yet, so there is no way users can express ‘any number
of varargs’ in 4.10.0, except the broken `any()` method. Using
`isA(String[].class)` would not work without #2807.
I’m guessing it’s not (easily) possible to release #2807 as 4.11.0…?
…On Sun, 25 Dec 2022 at 20:13, Tim van der Lippe ***@***.***> wrote:
@big-andy-coates <https://github.com/big-andy-coates> with regards to the
breakage: it seems like users have alternatives that will work on both
4.10.0 and 5.0.0. This is essentially my only requirement: it should be
possible to write that will work before and after this change. That way,
users can first update their usage on 4.10.0 to get it to work, then
upgrade to 5.0.0 and then potentially update their code again if they so
desire. As I understand your comment, this would be possible with isA? In
any case, if it wasn't possible to write these kind of matchers before and
this PR enables more use cases, then that's fine by me as well.
With regards to your JDK issue: you are using JDK 15. Can you try out
either 11 or 17? That should hopefully work. If that doesn't work, maybe
@reta <https://github.com/reta> can share their local setup? They have
been contributing lately as well, so I am assuming their local setup works.
—
Reply to this email directly, view it on GitHub
<#2835 (comment)>,
or unsubscribe
<https://github.com/notifications/unsubscribe-auth/AB5EE3V55VZEJ47DUHHJOPDWPBB6FANCNFSM6AAAAAATHNWRR4>
.
You are receiving this because you were mentioned.Message ID:
***@***.***>
|
@big-andy-coates the error message could be a bit misleading (at least so far the issues I've run into have not been related to JDK), happy to help here. |
Hi @reta, Yes, please. Do you need more info? It was all running fine until recently when work started on v5... |
Hi @big-andy-coates , could you please share the command line you use to run the tests and which module or test(s) fail with |
Ah, I think I know what's going in. There's an error in the build script. PR inbound... See #2839 |
@TimvdLippe, incase this was lost in the chatter about build failures... |
@big-andy-coates If I understand correctly, #2807 is not a breaking change, is it? If so, I can publish it as part of 4.11.0. That will require some tinkering on my end, but I think that is better for our users. Then we have a feasible alternative to the breaking change here, so that they can always have code running on both 4.11.0 and 5.0.0, without needing a big bang approach. |
That is correct @TimvdLippe |
I created https://github.com/mockito/mockito/tree/release/4.11.0 and cherry-picked #2807 onto there. Then I pushed tag v4.11.0 which is being published as we speak: https://github.com/mockito/mockito/actions/runs/3794091624 |
https://repo1.maven.org/maven2/org/mockito/mockito-core/4.11.0/ has been published successfully. I will now do a full review of this PR so that we can get it in 😄 |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This all looks great to me! I only have one question with regards to a test you changed. I would like to know if that is a required regression or if we can keep the pre-existing behavior.
...demonstrating how exactly two, or any number, of varargs values can be matched with argument captors.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This all LGTM. Unfortunately I didn't realize this would clash with the other PR I just merged, so we have to fix the merge conflicts 😢
I wasn't expecting the other PR to be merged until after 5.0.0 was released, but ... no worries. |
fixes: #2836
fixes: #1593
fixes: #585
Remove broken
VarargMatcher
internal interface.any()
behaviour changeBREAKING CHANGE: The behaviour of
any()
when passed to a varargs parameter has changed. Prior to Mockito v5, theany()
matcher would match each element in the varargs parameter, matching any number of elements 0...n. From Mockito v5 onwardsany()
, when passed to a varargs parameter, will match invocations where a single value is passed to the varargs parameter. To match any number of values passed to the varargs parameter, pass the type of the varargs parameter toany()
. For example, given aString...
varargs parameter, useany(String[].class)
.Example:
ArgumentCaptor.forClass()
behaviour changeBREAKING CHANGE: The behaviour of argument captors when passed to a varargs parameter has changed. Prior to Mockito v5, passing an argument captor to a varargs parameter would capture any number of values passed to the varargs parameter. From Mockito v5 onwards, the same call will match only a single value passed to the varargs parameter. To capture any number of values, change to the type of the argument captor to match the array type of the varargs parameter. For example, given a
String...
varargs parameter, change the argument captor from typeString
to typeString[]
.Example:
argThat()
behaviour changeBREAKING CHANGE: The behaviour of all methods in
MockitoHamcrest
, such asargThat()
, when passed to a varargs parameter has changed. Prior to Mockito v5, these matchers would match each element in the varargs parameter, matching any number of elements 0...n. From Mockito v5 onwards these matchers, when passed to a varargs parameter, will match invocations where a single value is passed to the varargs parameter. To match any number of values passed to the varargs parameter, pass use suitable Hamcrest Matcher and the varargs array type toargThat
. For example, given aString...
varargs parameter, use:argThat(isA(String[].class), String[].class)
.Checklist
including project members to get a better picture of the change
commit is meaningful and help the people that will explore a change in 2 years
Fixes #<issue number>
in the description if relevantFixes #<issue number>
if relevant