아래와 같은 코드가 있습니다.
Class A {
public boolean myMethod(someargs) {
MyQueryClass query = new MyQueryClass();
Long id = query.getNextId();
// some more code
}
}
Class MyQueryClass {
....
public Long getNextId() {
//lot of DB code, execute some DB query
return id;
}
}
이제 A.myMethod(someargs)
. 실제 메서드를 건너 뛰고 query.getNextId()
대신 스텁 값을 반환 하고 싶습니다 . 기본적으로 나는 MyQueryClass
.
그래서 내 테스트 케이스에서 다음을 사용했습니다.
MyQueryClass query = PowerMockito.mock(MyQueryClass.class);
PowerMockito.whenNew(MyQueryClass.class).withNoArguments().thenReturn(query);
when(query.getNextId()).thenReturn(1000000L);
boolean b = A.getInstance().myMethod(args);
//asserts
내가 사용 @RunWith(PowerMockRunner.class)
및 @PrepareForTest({MyQueryClass.class})
내 테스트 클래스의 시작이다.
그러나 테스트를 디버깅 할 때 여전히 클래스 의 실제 메서드 getNextId()
를 호출합니다 MyQueryClass
.
내가 여기서 무엇을 놓치고 있습니까? Mockito 및 PowerMockito를 처음 사용하는 사람은 누구나 도울 수 있습니다.
답변
생성자가 호출 되는 @PrepareForTest
클래스를 생성중인 클래스 대신 주석 에 넣어야 합니다 . 새 객체의 Mock 생성을 참조하십시오 .
귀하의 경우 :
✗ @PrepareForTest(MyQueryClass.class)
✓ @PrepareForTest(A.class)
더 일반적 :
✗ @PrepareForTest(NewInstanceClass.class)
✓ @PrepareForTest(ClassThatCreatesTheNewInstance.class)
답변
@TrueDub가 수락 한 답변에서 언급했듯이 생성자가 호출되는 클래스를 @PrepareForTest
.
그러나 이렇게하면 eclemma 및 Sonar에서보고 한 해당 클래스에 대한 커버리지가 해당 클래스에 대해 0이됩니다.
Javassist를 ByteBuddy (# 727)로 대체 할 예정이며이 이전 문제를 해결하는 데 도움이 될 것입니다. 그러나 지금은 JaCoCo On-the-fly 계측과 함께 PowerMock을 사용할 방법이 없습니다. 그리고 IDE에서 코드 커버리지를 얻을 수있는 해결 방법이 없습니다.
따라서 여기서 해결책은 해당 클래스의 인스턴스를 반환 한 다음 정적으로 모의하는 정적 팩토리를 사용하도록 실제 코드를 리팩터링하는 것입니다.
답변
아마도 간단히 사용할 수 있습니다.
Mockito.doReturn(value).when(xxx)