web-dev-qa-db-ja.com

Mockito-実際のオブジェクトをスパイして元のメソッドを呼び出す

次のコードを想像してください:

_List list = .....
List spy = spy(list);
doThrow(new NullpointerException()).when(spy).get(0);
_

doThrow(....)list.get(0)を実行します-これはまったく意味がありません。モックの動作を定義し、ここでメソッドを呼び出さないようにしたいのですが、何か不足していますか?

編集:リストはCGLIBによって装飾されています。 CGLIBプロキシを削除すると、Mockitoは期待どおりに動作します。 CGLIBプロキシを使用するときにこのような問題を解決する方法はありますか?

20
Maciej Miklas
_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スパイに基づいて:doThrow(new NullPointerException("my test nullpointer")).when(spyMs1).get();

「TEST 4」は、すでにCGLIBで装飾されているため、予期される例外をスローしません。CGLIb呼び出しが実行されていることもコンソールで確認できます:スパイオブジェクトではなく_GLIB decorated GET call_。 SpringのAOPをCGLIBプロキシで使用すると、同じ効果が得られます。

8
Maciej Miklas
Mockito.doThrow(new NullpointerException()).when(spy).get(0);

ここでの問題は、部分的なモックを作成しようとしているため、テストクラスに注釈を付ける必要があることです。

@PrepareForTest(List.class)

これは機能する場合と機能しない場合があります。例外処理をテストするコードを見ると、部分的にモックされたオブジェクトではなく、完全にモックされたオブジェクトに対して常にそれを行ってきました。また、部分的なモックを作成するときにPowerMockitoを広範囲に使用しているため、ライブラリが必要な機能を実行する可能性があります。

0
user785262