ChatGPT解决这个技术问题 Extra ChatGPT

throw checked Exceptions from mocks with Mockito

I'm trying to have one of my mocked objects throw a checked Exception when a particular method is called. I'm trying the following.

@Test(expectedExceptions = SomeException.class)
public void throwCheckedException() {
    List<String> list = mock(List.class);
    when(list.get(0)).thenThrow(new SomeException());
    String test = list.get(0);
}

public class SomeException extends Exception {
}

However, that produces the following error.

org.testng.TestException: 
Expected exception com.testing.MockitoCheckedExceptions$SomeException but got org.mockito.exceptions.base.MockitoException: 
Checked exception is invalid for this method!
Invalid: com.testing.MockitoCheckedExceptions$SomeException

Looking at the Mockito documentation, they only use RuntimeException, is it not possible to throw checked Exceptions from a mock object with Mockito?


a
ahmednabil88

Check the Java API for List.
The get(int index) method is declared to throw only the IndexOutOfBoundException which extends RuntimeException.
You are trying to tell Mockito to throw an exception SomeException() that is not valid to be thrown by that particular method call.

To clarify further.
The List interface does not provide for a checked Exception to be thrown from the get(int index) method and that is why Mockito is failing.
When you create the mocked List, Mockito will use the definition of List.class to creates its mock.

The behavior you are specifying with the when(list.get(0)).thenThrow(new SomeException()) doesn't match the method signature in List API, because get(int index) method does not throw SomeException() so Mockito fails.

If you really want to do this, then have Mockito throw a new RuntimeException() or even better throw a new ArrayIndexOutOfBoundsException() since the API specifies that that is the only valid Exception to be thrown.


While my real code wasn't actually using List, your answer applies for that method call as well. I was mocking the wrong method. Thank you.
extra: Mocktio will not complain if you doThrow an a method without any throwables, but you will also get this exception
For Kotliners: Kotlin doesn't have checked exceptions, so you cannot normally declare (in the function signature) that the function throws an exception. However, you can annotate the function with Throws annotation to make the compiler generate the same bytecode as declaring throws in the equivalent Java code. See [here] (kotlinlang.org/api/latest/jvm/stdlib/kotlin.jvm/-throws/…) for more details.
This check is enforced since the release of Mockito 2.11.0 (see 2.10.3).
Does the same logic apply for Scala as well?
D
David Rawson

A workaround is to use a willAnswer() method.

For example the following works (and doesn't throw a MockitoException but actually throws a checked Exception as required here) using BDDMockito:

given(someObj.someMethod(stringArg1)).willAnswer( invocation -> { throw new Exception("abc msg"); });

The equivalent for plain Mockito would to use the doAnswer method


or use willAnswer( invocation -> { throw new Exception("abc msg"); }).given(someObj).someMethod(stringArg1); when the method returns void.
or use when(someObj.someMethod(stringArg1)).thenAnswer(invocation -> { throw new Exception("abc msg"); });
Great workaround, thanks! For Kotliners who want to (1) use that seamlessly as an extension function and (2) be able to pass several args like willThrow() normally allows, I've written a Gist
s
spottedmahn

There is the solution with Kotlin :

given(myObject.myCall()).willAnswer {
    throw IOException("Ooops")
}

Where given comes from

import org.mockito.BDDMockito.given


This will work with Java as well .thenAnswer((t) -> { throw new IOException(); });
D
David Moles

Note that in general, Mockito does allow throwing checked exceptions so long as the exception is declared in the message signature. For instance, given

class BarException extends Exception {
  // this is a checked exception
}

interface Foo {
  Bar frob() throws BarException
}

it's legal to write:

Foo foo = mock(Foo.class);
when(foo.frob()).thenThrow(BarException.class)

However, if you throw a checked exception not declared in the method signature, e.g.

class QuxException extends Exception {
  // a different checked exception
}

Foo foo = mock(Foo.class);
when(foo.frob()).thenThrow(QuxException.class)

Mockito will fail at runtime with the somewhat misleading, generic message:

Checked exception is invalid for this method!
Invalid: QuxException

This may lead you to believe that checked exceptions in general are unsupported, but in fact Mockito is only trying to tell you that this checked exception isn't valid for this method.


A
Alok Gupta

This works for me in Kotlin:

when(list.get(0)).thenThrow(new ArrayIndexOutOfBoundsException());

Note : Throw any defined exception other than Exception()


Just what I was looking for, can throw any Exception other than Exception