Programming/테스트

PowerMock으로 Private Method Mocking하기

NavyGuy 2021. 6. 6. 23:09

Overview

Private Method를 테스트 하다보니 Reflection을 사용하여 작성을 하여 번거로움이 있었습니다.
라이브러리의 힘을 빌려 PowerMock 으로 Private Method 를 Mocking하여 유닛테스트를 작성해봅니다.

Gradle Dependencies

아래의 의존성을 추가해줍니다.
powermock-module-junit4
powermock-api-mockito2

testImplementation(
"org.powermock:powermock-module-junit4:2.0.9",
"org.powermock:powermock-api-mockito2:2.0.9"
)

Service Layer

아래의 public 메소드가 있습니다. isNewOrder 는 private 메소드이고, mocking 하여 public 메소드를 Unit 테스트를 해봅니다.

public class CoffeeOrderService {

    private static final String[] ORDER_ID_LIST = {"A10131", "A10132", "A10133"};

    public String getOrderId(String orderId) {
        if(isNewOrder(orderId)){
            orderId = RandomStringUtils.random(5);
        }
        return orderId;
    }

    private boolean isNewOrder(String orderId){
        return Stream.of(ORDER_ID_LIST)
                .anyMatch(orderItem -> orderItem.equals(orderId));
    }

}

Unit Test

Use PowerMock

@RunWith(PowerMockRunner.class)
@PrepareForTest(CoffeeOrderService.class)
public class CoffeeOrderServiceTest {

    private CoffeeOrderService spiedCoffeeOrderService;

    @Before
    public void setUp() {
        spiedCoffeeOrderService = PowerMockito.spy(new CoffeeOrderService());
    }

    @Test
    public void should_getNewOrderId_When_NewOrder() throws Exception {
        String orderId = "A10131";

        when(spiedCoffeeOrderService, "isNewOrder", anyString()).thenReturn(true);
        String actualOrderId = spiedCoffeeOrderService.getOrderId(orderId);

        verifyPrivate(spiedCoffeeOrderService).invoke("isNewOrder", anyString());
        assertNotEquals(orderId, actualOrderId);
    }

    @Test
    public void should_getExistingOrderId_When_OldOrder() throws Exception {
        String orderId = "A10131";

        when(spiedCoffeeOrderService, "isNewOrder", anyString()).thenReturn(false);
        String actualOrderId = spiedCoffeeOrderService.getOrderId(orderId);

        verifyPrivate(spiedCoffeeOrderService).invoke("isNewOrder", anyString());
        assertEquals(orderId, actualOrderId);
    }
 }

PowerMockito.when 메소드로 private method를 mocking 해줍니다.
PowerMockito.verifyPrivate 메소드는 private method가 호출되었는지 검증합니다.

Syntax

when(mock or spy instance, "privateMethodName").thenReturn(//return value);

verifyPrivate(mockedInstance).invoke("privateMethodName");

@PrepareForTest

Failure

처음에 @PrepareForTest 를 붙이지 않았다가 아래와 같은 에러를 받았다.

Misplaced or misused argument matcher detected here:

메소드를 타겟으로 해당 어노테이션을 붙이니 테스트가 깨지고 아래와같은 에러를 받았다.

@PrepareForTest(CoffeeOrderService.class)
    public void should_getNewOrderId_When_NewOrder() throws Exception {
    ...


FAILURE: Build failed with an exception.
filter.includeTestsMatching

Success

@RunWith(PowerMockRunner.class)
@PrepareForTest(CoffeeOrderService.class)
public class CoffeeOrderServiceTest {
...

클래스를 타겟으로 하니 테스트가 정상 작동합니다.

Use Reflection

번외로 Reflection을 이용하여 private method를 직접 테스트 해봅니다.

    @Test
    public void test_isNewOrder_method_byReflection() throws Exception {
        CoffeeOrderService coffeeOrderService = new CoffeeOrderService();

        Method m = coffeeOrderService.getClass().getDeclaredMethod("isNewOrder", String.class);
        m.setAccessible(true);

        // orderIdList : {"A10131", "A10132", "A10133"}
        boolean isNewOrder = (boolean) m.invoke(coffeeOrderService, "Z99999");    
        boolean isNewOrder2 = (boolean) m.invoke(coffeeOrderService, "A10131");

        assertEquals(false, isNewOrder);
        assertEquals(true, isNewOrder2);
    }

 

Reference

Baeldung - powermock-private-method
Software Testing Help - Mocking Private, Static And Void Methods Using Mockito