2017-10-26 2 views
1

콜백을 모의하려고 시도했습니다.Guava CacheLoader 콜백 메소드 조롱

public class TestService { 
    private UtilService utilService; 

    private LoadingCache<String, String> cache= CacheBuilder.newBuilder() 
      .build(new CacheLoader<String, String>() { 
       public String load(String key) throws Exception { 
        System.out.println("key = " + key); 
        return getKey(key); 
       } 
      }); 

    public String getkeyFromCache(String key) throws ExecutionException { 
     return cache.get(key); 
    } 

    @VisibleForTesting 
    public String getKey(String key) { 
     return utilService.getKey(key); 
    } 
} 

그리고 이와 같은 테스트 케이스 : 다음은 예제 코드입니다

@RunWith(MockitoJUnitRunner.class) 
public class TestServiceTest { 

    public static final String MYKEY = "Mykey"; 
    @Spy 
    TestService testService=new TestService(); 

    @Before 
    public void before() { 
     MockitoAnnotations.initMocks(this); 
    } 

    @Test 
    public void testCache() throws ExecutionException { 
     doReturn(MYKEY).when(testService).getKey(MYKEY); 
     String result=testService.getkeyFromCache(MYKEY); 
     String result2nd=testService.getkeyFromCache(MYKEY); 
     verify(testService,times(1)).getKey(MYKEY); 
    } 

} 

을하지만 그것이 내가 캐시 기능을 테스트 할

Wanted but not invoked: 
testService.getKey("Mykey"); 
-> at utils.TestServiceTest.testCache(TestServiceTest.java:38) 

However, there were other interactions with this mock: 
-> at utils.TestServiceTest.testCache(TestServiceTest.java:36) 
-> at utils.TestServiceTest.testCache(TestServiceTest.java:37) 

    at utils.TestServiceTest.testCache(TestServiceTest.java:38) 
    at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method) 
    at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:62) 
    at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43) 
    at java.lang.reflect.Method.invoke(Method.java:498) 
    at org.junit.runners.model.FrameworkMethod$1.runReflectiveCall(FrameworkMethod.java:50) 
    at org.junit.internal.runners.model.ReflectiveCallable.run(ReflectiveCallable.java:12) 
    at org.junit.runners.model.FrameworkMethod.invokeExplosively(FrameworkMethod.java:47) 
    at org.junit.internal.runners.statements.InvokeMethod.evaluate(InvokeMethod.java:17) 
    at org.junit.internal.runners.statements.RunBefores.evaluate(RunBefores.java:26) 
    at org.junit.runners.ParentRunner.runLeaf(ParentRunner.java:325) 
    at org.junit.runners.BlockJUnit4ClassRunner.runChild(BlockJUnit4ClassRunner.java:78) 
    at org.junit.runners.BlockJUnit4ClassRunner.runChild(BlockJUnit4ClassRunner.java:57) 
    at org.junit.runners.ParentRunner$3.run(ParentRunner.java:290) 
    at org.junit.runners.ParentRunner$1.schedule(ParentRunner.java:71) 
    at org.junit.runners.ParentRunner.runChildren(ParentRunner.java:288) 
    at org.junit.runners.ParentRunner.access$000(ParentRunner.java:58) 
    at org.junit.runners.ParentRunner$2.evaluate(ParentRunner.java:268) 
    at org.junit.runners.ParentRunner.run(ParentRunner.java:363) 
    at org.mockito.internal.runners.JUnit45AndHigherRunnerImpl.run(JUnit45AndHigherRunnerImpl.java:37) 
    at org.mockito.runners.MockitoJUnitRunner.run(MockitoJUnitRunner.java:62) 
    at org.junit.runner.JUnitCore.run(JUnitCore.java:137) 
    at com.intellij.junit4.JUnit4IdeaTestRunner.startRunnerWithArgs(JUnit4IdeaTestRunner.java:68) 
    at com.intellij.rt.execution.junit.IdeaTestRunner$Repeater.startRunnerWithArgs(IdeaTestRunner.java:47) 
    at com.intellij.rt.execution.junit.JUnitStarter.prepareStreamsAndStart(JUnitStarter.java:242) 
    at com.intellij.rt.execution.junit.JUnitStarter.main(JUnitStarter.java:70) 

를 호출되지 않습니다 같습니다 하지만 UtilService는 아니기 때문에 조롱하고 싶습니다. 이 테스트에서 Answer 또는 Captor를 사용하는 방법을 생각해 보지 못했습니다.

답변

2

당신은 당신이 간첩을 직접 만들었습니다. 따라서 콜백은 실제 메소드에 연결됩니다. Mockito가 객체 생성을 처리하도록 허용하면 문제가 해결됩니다. 나는 당신의 테스트 코드를 리팩토링 한

(나는 mockito의 1.10.19을 사용하고 있습니다) :

@RunWith(MockitoJUnitRunner.class) 
public class TestServiceTest { 

    public static final String MYKEY = "Mykey"; 
    @Spy 
    TestService testService; 

    @Test 
    public void testCache() throws ExecutionException { 
     doReturn(MYKEY).when(testService).getKey(MYKEY); 

     String result=testService.getkeyFromCache(MYKEY); 
     String result2nd=testService.getkeyFromCache(MYKEY); 

     verify(testService).getKey(MYKEY); 
    } 

} 

참고 : 당신이 Mockito 러너를 사용하는 것처럼 - 당신이 Mockito에게 자신을 초기화 할 필요가 없습니다. 또한 시간 (1)은 호출을 확인하기위한 기본값입니다.

+0

고맙습니다. @Admit. 2 가지 문제가있었습니다. 내가 사용하고 있던 버전은 1.9.5이고 서비스의 초기화입니다. 당신이 당신의 mockito 판에 대해 나에게 말했던 좋은 물건 또는 나는 아직도보고있을 것이다. – zpontikas

+0

예, 버전이 중요한 것으로 보입니다. 도움이 된 것을 기쁘게 생각합니다. – Admit

+0

주어진/때/다음 테스트의 서식을 주셔서 감사합니다;) – zpontikas

1

load() 메서드 내에서 getKey() 메서드가 호출되면 mock-proxy가 아닌 TestService 클래스 내부에서 호출됩니다. 그래서

doReturn(MYKEY).when(testService).getKey(MYKEY); 

이 작동하지 않습니다. 그러나 UtilService를 조롱하고해야하며 한 번만 액세스 할 수 있습니다.

+0

답변 주셔서 감사하지만 인정하는 솔루션이 더 좋았습니다. 또한, 당신은 새로운 stackoverflow이기 때문에, 우리는 실제 코드를 게시하지 않고 코드 문제의 예제를 게시하기 때문에 대부분 코드가 잘못되었음을 암시하지 않고 질문에 대답하려고 시도합니다. – zpontikas

+0

@ zpontikas 감사합니다. –