I'm trying to use PowerMockito to create a spy of a final
class but I keep getting the following error, even though I am using PowerMockito's spy()
method in place of Mockito's:
java.lang.IllegalArgumentException: Cannot subclass final class class com.whoever.WidgetUploadClient
My test case looks something like this:
...
import static org.powermock.api.mockito.PowerMockito.mock;
import static org.powermock.api.mockito.PowerMockito.spy;
@RunWith(PowerMockRunner.class)
@PowerMockRunnerDelegate(RobolectricTestRunner.class)
@PowerMockIgnore({"org.mockito.*", "org.robolectric.*", "android.*"})
@PrepareForTest(WidgetUploadClient.class)
@Config(manifest=Config.NONE, sdk = 23)
public class WidgetUploadClientTest {
@Test
public void testUploadWidget() {
WidgetMarshaller mockMarshaller = mock(WidgetMarshaller.class);
WidgetUploadClient client = spy(new WidgetUploadClient(mockMarshaller)); // Exception thrown by spy()
...
}
}
Shouldn't @PrepareForTest(WidgetUploadClient.class)
and using PowerMockito's spy()
method account for WidgetUploadClient
being final?
I have also tried the alternative approach found in Robolectric's PowerMock guide: using RobolectricTestRunner
or RobolectricGradleTestRunner
as the test runner (@RunWith
) with @Rule public PowerMockRule rule = new PowerMockRule()
. When I do that, the test fails to run entirely and a different exception is thrown.
I am using PowerMock/PowerMockito 1.6.5, Robolectric 3.1 and Java 1.8.0_91-b14.
To get this working you have to understand what annotation @PrepareForTest does and make a little changes on your code:
The annotation in used to understand what class we're going to test and to prepare that class to mock static, final etc etc methods (so the methods that are not normally mockable with mockito) as normal methods.
After that you have to do this in your code:
WidgetMarshaller mockMarshaller = mock(WidgetMarshaller.class);
//Here you are doing correcly the mocking of the object
WidgetUploadClient client = new WidgetUploadClient(mockMarshaller);
//Here you have to add this line to create an object that will be spied
client = PowerMockito.spy(client);
//Here you simply spy your class
By the way there's another thing to remember, if you pass
@PrepareForTest(WidgetUploadClient.class)
to the class, you will be able to mock or spy just WidgetUploadClient class, so you have to pass two (or if you want more) parameters to the class using an array as parameter to the annotation, simply write this
@PrepareForTest({WidgetUploadClient.class, WidgetMarshaller.class})
Hope you get it working :D See you
I believe that I am using the APIs correctly but am experiencing by a bug that effects developers trying to use the combination of Robolectric and PowerMock. For reference, the bug can be tracked on Robolectric's issue tracker. The combination of libraries has been broken since at least January 2016 (currently ~6 months ago.)
来源:https://stackoverflow.com/questions/38060683/cannot-mock-spy-final-class-using-powermockito-spy