Подтвердить что ты не робот

Как проверить статический метод void был вызван с помощью power mockito

Поэтому я использую следующее.

Powermock-mockito 1.5.12
Mockito 1.95
junit 4.11

вот мой класс utils

public void InternalUtils {
    public static void sendEmail(String from, String[] to, String msg, String body) {
    }
}

здесь суть

public class InternalService {
       public void processOrder(Order order) {
           if (order.isSuccessful()) {
               InternalUtils.sendEmail(...);
           }
       }
}


@PrepareForTest({InternalUtils.class})
@RunWith(PowerMockRunner.class)
public class InternalService {
   public void verifyEmailSend() {
        mockStatic(Internalutils.class);
        doNothing().when(InternalUtils, "sendEmail", anyString(), any(String.class), anyString(), anyString());
        Order order = mock(Order.class);
        when(order.isSuccessful()).thenReturn(true);
        InternalService is = new InternalService();

        verifyStatic(times(1));
        is.processOrder(order);
   }
}

Вышеуказанный тест терпит неудачу. Предоставляемый режим проверки - нет, но в соответствии с кодом, если заказ успешный, чем электронная почта должна быть отправлена.

4b9b3361

Ответ 1

Если вы насмехаетесь над поведением (с чем-то вроде doNothing()), действительно не нужно звонить на verify*(). Тем не менее, здесь мой удар по повторному написанию вашего тестового метода:

@PrepareForTest({InternalUtils.class})
@RunWith(PowerMockRunner.class)
public class InternalServiceTest { //Note the renaming of the test class.
   public void testProcessOrder() {
        //Variables
        InternalService is = new InternalService();
        Order order = mock(Order.class);

        //Mock Behavior
        when(order.isSuccessful()).thenReturn(true);
        mockStatic(Internalutils.class);
        doNothing().when(InternalUtils.class); //This is the preferred way
                                               //to mock static void methods.
        InternalUtils.sendEmail(anyString(), anyString(), anyString(), anyString());

        //Execute
        is.processOrder(order);            

        //Verify
        verifyStatic(times(1)); //Similar to how you mock static methods
                                //this is how you verify them.
        InternalUtils.sendEmail(anyString(), anyString(), anyString(), anyString());
   }
}

Я сгруппировался в четыре раздела, чтобы лучше выделить, что происходит:

1. Переменные

Я хочу объявить здесь любые аргументы переменных/методов/mock-коллабораторов. Если это что-то используется в нескольких тестах, подумайте о том, чтобы сделать его переменной экземпляра тестового класса.

2. Mock Behavior

Здесь вы определяете поведение всех ваших mocks. Вы устанавливаете возвращаемые значения и ожидания здесь, перед выполнением тестируемого кода. Вообще говоря, если вы установите здесь ложное поведение, вам не нужно будет проверять поведение позже.

3. Выполнить

Ничего особенного здесь; это просто запускает тестируемый код. Мне нравится давать ему свой раздел, чтобы привлечь к нему внимание.

4. Проверка

Это когда вы вызываете любой метод, начинающийся с verify или assert. После того, как тест закончен, вы проверяете, действительно ли произошло то, что вы хотели иметь. Это самая большая ошибка, которую я вижу с помощью вашего тестового метода; вы попытались проверить вызов метода до того, как ему когда-либо была предоставлена ​​возможность запуска. Во-вторых, вы никогда не указали, какой статический метод вы хотите проверить.

Дополнительные примечания

Это в первую очередь личное предпочтение с моей стороны. Существует определенный порядок, который вам нужно делать, но внутри каждой группы есть небольшая комната для маневра. Это помогает мне быстро отделить то, что происходит где.

Я также настоятельно рекомендую ознакомиться с примерами на следующих сайтах, поскольку они очень надежны и могут помочь в большинстве случаев, которые вам понадобятся:

Ответ 2

Вы, вышеприведенный ответ, широко признаны и хорошо задокументированы, я нашел некоторые причины для публикации моего ответа здесь: -

    doNothing().when(InternalUtils.class); //This is the preferred way
                                           //to mock static void methods.
    InternalUtils.sendEmail(anyString(), anyString(), anyString(), anyString());

Здесь я не понимаю, почему мы вызываем InternalUtils.sendEmail самостоятельно. Я объясню в своем коде, почему нам это не нужно.

mockStatic(Internalutils.class);

Итак, мы издеваемся над классом, который хорош. Теперь давайте посмотрим, как нам нужно проверить метод sendEmail (/..../).

@PrepareForTest({InternalService.InternalUtils.class})
@RunWith(PowerMockRunner.class)
public class InternalServiceTest {

    @Mock
    private InternalService.Order order;

    private InternalService internalService;

    @Before
    public void setup() {
        MockitoAnnotations.initMocks(this);
        internalService = new InternalService();
    }

    @Test
    public void processOrder() throws Exception {

        Mockito.when(order.isSuccessful()).thenReturn(true);
        PowerMockito.mockStatic(InternalService.InternalUtils.class);

        internalService.processOrder(order);

        PowerMockito.verifyStatic(times(1));
        InternalService.InternalUtils.sendEmail(anyString(), any(String[].class), anyString(), anyString());
    }

}

Эти две линии - вот где волшебство, Первая строка сообщает инфраструктуре PowerMockito, что ей необходимо проверить класс, который он статически высмеивал. Но какой метод нужно проверить? Вторая строка сообщает, какой метод необходимо проверить.

PowerMockito.verifyStatic(times(1));
InternalService.InternalUtils.sendEmail(anyString(), any(String[].class), anyString(), anyString());

Это код моего класса, sendEmail api дважды.

public class InternalService {

    public void processOrder(Order order) {
        if (order.isSuccessful()) {
            InternalUtils.sendEmail("", new String[1], "", "");
            InternalUtils.sendEmail("", new String[1], "", "");
        }
    }

    public static class InternalUtils{

        public static void sendEmail(String from, String[]  to, String msg, String body){

        }

    }

    public class Order{

        public boolean isSuccessful(){
            return true;
        }

    }

}

Поскольку он дважды вызывается, вам просто нужно изменить проверку (times (2))... что все.