import static org.mockito.Mockito.doThrow;
import static org.mockito.Mockito.spy;
import java.lang.reflect.Method;
import org.junit.Test;
import net.sf.cglib.proxy.Enhancer;
import net.sf.cglib.proxy.MethodInterceptor;
import net.sf.cglib.proxy.MethodProxy;
public class MockitoSpyTest {
@Test
public void execTest() {
System.out.println("*** TEST 1 ***");
System.out.println("Test on unmodified object");
MySet ms = new MySetImpl();
ms.set("test value");
System.out.println("Set contains: " + ms.get());
// decorate ms1 with easymock
System.out.println("\n*** TEST 2 ***");
MySet spyMs = spy(ms);
doThrow(new NullPointerException("my test nullpointer")).when(spyMs).get();
System.out.println("Test decorated object with SPY");
spyMs.set("test value");
try {
System.out.println("Set contains: " + spyMs.get());
} catch (NullPointerException e) {
System.out.println("NullPointerException - as expected");
}
// Enhance call with CGLIB
System.out.println("\n*** TEST 3 ***");
System.out.println("Test on CGLIB decorated object");
Enhancer enc = new Enhancer();
enc.setSuperclass(MySetImpl.class);
enc.setInterfaces(new Class[] { MySet.class });
enc.setCallback(new MethodInterceptor() {
@Override
public Object intercept(Object obj, Method method, Object[] args, MethodProxy proxy) throws Throwable {
if ("get".equals(method.getName())) {
System.out.println("CGLIB decorated GET call");
}
return proxy.invokeSuper(obj, args);
}
});
MySet ms1 = (MySet) enc.create();
ms1.set("test value");
System.out.println("Set contains: " + ms1.get());
// decorate ms1 with easymock
System.out.println("\n*** TEST 4 ***");
System.out.println("Test on CGLIB decorated object with SPY");
MySet spyMs1 = spy(ms1);
doThrow(new NullPointerException("my test nullpointer")).when(spyMs1).get();
spyMs1.set("test value");
System.out.println("Set contains: " + spyMs1.get());
}
public interface MySet {
void set(String val);
String get();
}
public static class MySetImpl implements MySet {
String val;
public void set(String val) {
this.val = val;
System.out.println("Original SET call:" + val);
}
public String get() {
System.out.println("Original GET call:" + val);
return val;
}
}
}
例の出力を生成します。
*** TEST 1 ***
Test on unmodified object
Original SET call:test value
Original GET call:test value
Set contains: test value
*** TEST 2 ***
Test decorated object with SPY
Original SET call:test value
NullPointerException - as expected
*** TEST 3 ***
Test on CGLIB decorated object
Original SET call:test value
CGLIB decorated GET call
Original GET call:test value
Set contains: test value
*** TEST 4 ***
Test on CGLIB decorated object with SPY
CGLIB decorated GET call
Original GET call:test value
Original SET call:test value
CGLIB decorated GET call
Original GET call:test value
Set contains: test value
今TEST 2
とTEST 4
NullPointerException
get
上のコールスローする必要があります - mockitoスパイに基づく: "TEST 4が" スロー期待しないdoThrow(new NullPointerException("my test nullpointer")).when(spyMs1).get();
を既にCGLIBで装飾されているため例外です - CGLIb呼び出しが実行されていることをコンソールで確認することもできます:GLIB decorated GET call
とspyオブジェクトを呼び出さないでください。 CGLIBプロキシでSpring AOPを使用する場合、同じ効果が得られます。
これは実際にはビルダーパターンを使用するmockitoの構文です。 (そう多くの他の模造フレームワークのように) –
私がMockitoを知る限り、あなたのコードは正しいです。ドキュメンテーションでは、 'doThrow'はvoidメソッドのためのものだと言います。これが問題かどうかを確認するには 'clear()'を使います。しかし、なぜdoThrowが 'doReturn'のような非voidメソッドでも動作してはならないのは本当に想像できません。 – flyx
あなたがここで何を求めているか分かりません。はい、doThrow()をvoid以外のメソッドでも使用できます。ドキュメントがvoidメソッドを記述する理由は、スタブスローの2番目の構文があり、voidメソッドでは機能せず、ドキュメント内でdoThrowする前に提示されるためです。あなたの例のget(0)は、実際のリストではなく、スパイでのみ呼び出されます。スパイは、それがdoThrowのコンテキスト内にあることを知っているので、REAL get(0)を呼び出す代わりに、代わりにそれをスタブします。それはあなたが求めていることですか?もしそうなら、私はこのコメントを答えに変えるでしょう。 –