私はこれについてグーグルで調べましたが、関連するものは見つかりませんでした。私はこのようなものを持っています:
Object obj = getObject();
Mockeable mock= Mockito.mock(Mockeable.class);
Mockito.when(mock.mymethod(obj )).thenReturn(null);
Testeable testableObj = new Testeable();
testableObj.setMockeable(mock);
command.runtestmethod();
ここで、mymethod(Object o)
内で呼び出されるruntestmethod()
が、他のオブジェクトではなく、オブジェクトo
で呼び出されたことを確認します。しかし、たとえば、次のように、検証に関係なく、常にテストに合格します。
Mockito.verify(mock.mymethod(Mockito.eq(obj)));
または
Mockito.verify(mock.mymethod(Mockito.eq(null)));
または
Mockito.verify(mock.mymethod(Mockito.eq("something_else")));
私はいつもテストに合格しています。どうすればその検証を達成できますか(可能な場合)?
ありがとうございました。
ArgumentMatcher
の代替は ArgumentCaptor
です。
公式の例:
ArgumentCaptor<Person> argument = ArgumentCaptor.forClass(Person.class);
verify(mock).doSomething(argument.capture());
assertEquals("John", argument.getValue().getName());
キャプターは @ Captor アノテーションを使用して定義することもできます:
@Captor ArgumentCaptor<Person> captor;
//... MockitoAnnotations.initMocks(this);
@Test public void test() {
//...
verify(mock).doSomething(captor.capture());
assertEquals("John", captor.getValue().getName());
}
オブジェクトの.equalsメソッドを利用して論理的な等価性を実現しようとしていますか? Mockitoに含まれているargThatマッチャーを利用してこれを行うことができます
import static org.mockito.Matchers.argThat
次に、各オブジェクトの.equalsメソッドを遅延させる独自の引数マッチャーを実装できます
private class ObjectEqualityArgumentMatcher<T> extends ArgumentMatcher<T> {
T thisObject;
public ObjectEqualityArgumentMatcher(T thisObject) {
this.thisObject = thisObject;
}
@Override
public boolean matches(Object argument) {
return thisObject.equals(argument);
}
}
コードを使用して、更新して読むことができます...
Object obj = getObject();
Mockeable mock= Mockito.mock(Mockeable.class);
Mockito.when(mock.mymethod(obj)).thenReturn(null);
Testeable obj = new Testeable();
obj.setMockeable(mock);
command.runtestmethod();
verify(mock).mymethod(argThat(new ObjectEqualityArgumentMatcher<Object>(obj)));
正確な平等(メモリ内の同じオブジェクト)を求めている場合は、
verify(mock).mymethod(obj);
これにより、一度呼び出されたことが検証されます。
eq
マッチャーは必要ありません。.verify(mock)
の外にある必要があります。これで、何も検証せずに(メソッド呼び出しを行わずに)メソッド呼び出しの結果で検証を開始しています。したがって、すべてのテストに合格しています。コードは次のようになります。
Mockito.verify(mock).mymethod(obj);
Mockito.verify(mock).mymethod(null);
Mockito.verify(mock).mymethod("something_else");
もう1つの方法は、再定義する代わりにorg.mockito.internal.matchers.Equals.Equalsメソッドを使用することです。
verify(myMock).myMethod((inputObject)Mockito.argThat(new Equals(inputObjectWanted)));
この方法でMockito.verifyを使用しました
@UnitTest
public class JUnitServiceTest
{
@Mock
private MyCustomService myCustomService;
@Test
public void testVerifyMethod()
{
Mockito.verify(myCustomService, Mockito.never()).mymethod(parameters); // method will never call (an alternative can be pick to use times(0))
Mockito.verify(myCustomService, Mockito.times(2)).mymethod(parameters); // method will call for 2 times
Mockito.verify(myCustomService, Mockito.atLeastOnce()).mymethod(parameters); // method will call atleast 1 time
Mockito.verify(myCustomService, Mockito.atLeast(2)).mymethod(parameters); // method will call atleast 2 times
Mockito.verify(myCustomService, Mockito.atMost(3)).mymethod(parameters); // method will call at most 3 times
Mockito.verify(myCustomService, Mockito.only()).mymethod(parameters); // no other method called except this
}
}
モック可能クラスのequalsメソッドをチェックしましたか?これが常にtrueを返す場合、または同じインスタンスに対して同じインスタンスをテストし、equalメソッドが上書きされない場合(したがって、参照に対するチェックのみ)、trueを返します。
これが、引数の検証に失敗する方法です。
import static org.mockito.ArgumentMatchers.argThat;
import static org.mockito.Mockito.verify;
...
verify(mock).mymethod(argThat(
(x)->false
));
Same()マッチャーで試してみましたか?次のように:
verify(mockObj).someMethod(same(specificInstance));
同じ問題がありました。 eq()マッチャーとrefEq()マッチャーで試しましたが、常に誤検知がありました。 same()マッチャーを使用すると、引数が異なるインスタンスである場合にテストが失敗し、引数が同じインスタンスになったときに合格しました。