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
If you love us? You can donate to us via Paypal or buy me a coffee so we can maintain and grow! Thank you!
Donate Us With