web-dev-qa-db-ja.com

モックされたメソッドに、渡された引数を返させる

次のようなメソッドシグネチャを考えます。

public String myFunction(String abc);

Mockitoはメソッドが受け取ったのと同じ文字列を返すのに役立ちますか?

573

あなたはMockitoで回答を作成することができます。メソッドmyFunctionを持つApplicationという名前のインタフェースがあるとしましょう。

public interface Application {
  public String myFunction(String abc);
}

これがMockitoの答えを使ったテスト方法です。

public void testMyFunction() throws Exception {
  Application mock = mock(Application.class);
  when(mock.myFunction(anyString())).thenAnswer(new Answer<String>() {
    @Override
    public String answer(InvocationOnMock invocation) throws Throwable {
      Object[] args = invocation.getArguments();
      return (String) args[0];
    }
  });

  assertEquals("someString",mock.myFunction("someString"));
  assertEquals("anotherString",mock.myFunction("anotherString"));
}

Mockito 1.9.5およびJava 8以降、ラムダ関数を使用することによってさらに簡単な方法があります。

when(myMock.myFunction(anyString())).thenAnswer(i -> i.getArguments()[0]);

859
Steve

Mockito 1.9.5以降をお持ちの場合は、Answerオブジェクトを作成できる新しい静的メソッドがあります。次のように書く必要があります。

import static org.mockito.Mockito.when;
import static org.mockito.AdditionalAnswers.returnsFirstArg;

when(myMock.myFunction(anyString())).then(returnsFirstArg());

または代わりに

doAnswer(returnsFirstArg()).when(myMock).myFunction(anyString());

returnsFirstArg()メソッドはAdditionalAnswersクラスでは静的であることに注意してください。これはMockito 1.9.5の新機能です。だからあなたは正しい静的インポートが必要です。

487

Java 8では、古いバージョンのMockitoでも1行の回答を作成することができます。

when(myMock.myFunction(anyString()).then(i -> i.getArgumentAt(0, String.class));

もちろん、これはDavid Wallaceによって提案されたAdditionalAnswersを使用するほど有用ではありませんが、引数を「オンザフライ」で変換したい場合には有用かもしれません。

55
Paweł Dyda

私は非常によく似た問題を抱えていました。目標は、オブジェクトを永続化し、名前でそれらを返すことができるサービスを偽造することでした。サービスは次のようになります。

public class RoomService {
    public Room findByName(String roomName) {...}
    public void persist(Room room) {...}
}

サービスモックはルームインスタンスを格納するためにマップを使用します。

RoomService roomService = mock(RoomService.class);
final Map<String, Room> roomMap = new HashMap<String, Room>();

// mock for method persist
doAnswer(new Answer<Void>() {
    @Override
    public Void answer(InvocationOnMock invocation) throws Throwable {
        Object[] arguments = invocation.getArguments();
        if (arguments != null && arguments.length > 0 && arguments[0] != null) {
            Room room = (Room) arguments[0];
            roomMap.put(room.getName(), room);
        }
        return null;
    }
}).when(roomService).persist(any(Room.class));

// mock for method findByName
when(roomService.findByName(anyString())).thenAnswer(new Answer<Room>() {
    @Override
    public Room answer(InvocationOnMock invocation) throws Throwable {
        Object[] arguments = invocation.getArguments();
        if (arguments != null && arguments.length > 0 && arguments[0] != null) {
            String key = (String) arguments[0];
            if (roomMap.containsKey(key)) {
                return roomMap.get(key);
            }
        }
        return null;
    }
});

これでこのモックでテストを実行できます。例えば:

String name = "room";
Room room = new Room(name);
roomService.persist(room);
assertThat(roomService.findByName(name), equalTo(room));
assertNull(roomService.findByName("none"));
37
migu

Java 8では、 Steveの答え になることができます

public void testMyFunction() throws Exception {
    Application mock = mock(Application.class);
    when(mock.myFunction(anyString())).thenAnswer(
    invocation -> {
        Object[] args = invocation.getArguments();
        return args[0];
    });

    assertEquals("someString", mock.myFunction("someString"));
    assertEquals("anotherString", mock.myFunction("anotherString"));
}

編集:さらに短く:

public void testMyFunction() throws Exception {
    Application mock = mock(Application.class);
    when(mock.myFunction(anyString())).thenAnswer(
        invocation -> invocation.getArgument(0));

    assertEquals("someString", mock.myFunction("someString"));
    assertEquals("anotherString", mock.myFunction("anotherString"));
}
29
yiwei

私は似たようなものを使っています(基本的には同じ方法です)。場合によっては、モックオブジェクトが特定の入力に対して事前定義された出力を返すようにすると便利です。それはこのようになります:

private Hashtable<InputObject,  OutputObject> table = new Hashtable<InputObject, OutputObject>();
table.put(input1, ouput1);
table.put(input2, ouput2);

...

when(mockObject.method(any(InputObject.class))).thenAnswer(
       new Answer<OutputObject>()
       {
           @Override
           public OutputObject answer(final InvocationOnMock invocation) throws Throwable
           {
               InputObject input = (InputObject) invocation.getArguments()[0];
               if (table.containsKey(input))
               {
                   return table.get(input);
               }
               else
               {
                   return null; // alternatively, you could throw an exception
               }
           }
       }
       );
5
martin

テストでの実行を保証するためにArgumentCaptorと、引数を評価するためにArgumentCaptorを組み合わせてverify()を使用することができます。

ArgumentCaptor<String> argument = ArgumentCaptor.forClass(String.class);
verify(mock).myFunction(argument.capture());
assertEquals("the expected value here", argument.getValue());

引数の値は、さらなる操作/チェック/その他のために、明らかにargument.getValue()を通してアクセス可能です。

1
fl0w

これはかなり古い質問ですが、私はまだ関連があると思います。また、受け入れられた答えはStringに対してのみ機能します。その間にMockito 2.1があり、いくつかのインポートが変更されたので、私の現在の答えを共有したいと思います。

import static org.mockito.AdditionalAnswers.returnsFirstArg;
import static org.mockito.ArgumentMatchers.any;
import static org.mockito.Mockito.when;

@Mock
private MyClass myClass;

// this will return anything you pass, but it's pretty unrealistic
when(myClass.myFunction(any())).then(returnsFirstArg());
// it is more "life-like" to accept only the right type
when(myClass.myFunction(any(ClassOfArgument.class))).then(returnsFirstArg());

MyClass.myFunctionは次のようになります。

public class MyClass {
    public ClassOfArgument myFunction(ClassOfArgument argument){
        return argument;
    }  
}
1
LazR