Испытание Mockito Spring не проходит, если вы выполняете задачу Gradlew - PullRequest
0 голосов
/ 08 ноября 2019

У меня есть следующий класс для проверки работы моего весеннего приложения.

Это задание обновляет состояние объекта заказа, обращающегося к внешнему API.

Когда я просто запускаю класстест пройден успешно, но когда я выполняю его как часть задачи gradlew (которая запускает несколько интеграционных тестов, включая этот), он завершается неудачно с:

Wanted but not invoked:
client.updateOrder(
    "o11111",
    <Pending order update integration test$$ lambda$ 3 9 9/ 1 6 3 9 3 4 9 4 3 5>
);
-> at OrderUpdateIntegrationTest.shouldSendPendingOrderUpdatesSubmittedOrders(OrderUpdateIntegrationTest.java:77)
Actually, there were zero interactions with this mock.

Класс теста:

@SpringBootTest(webEnvironment = RANDOM_PORT, classes = TestConfig.class)
@ExtendWith(value = {SpringExtension.class, MockitoExtension.class})
@ActiveProfiles("integration-test")
@DirtiesContext
public class OrderUpdateIntegrationTest {

  private static final String ORDER_ID_1 = "o11111";
  private static final String ORDER_ID_2 = "o22222";
  static VerificationWithTimeout FIVE_SECOND_TIMEOUT = timeout(SECONDS.toMillis(5));

  @Autowired
  private TestScheduleConfigurer testScheduleConfigurer;

  @MockBean
  private OrdersClient client;
  @MockBean
  private UpdatePendingOrdersTasklet updateOrderAndSendEmail;
  @MockBean
  private UrlBuilder urlBuilder;

  @Mock
  private ResponseEntity<Order> mock200HttpResponse;
  @Mock
  private ResponseEntity<String> mockEmail200HttpResponse;

  private Order order11111;
  private Order order22222;
  private OrderList ordersList = new OrderList();

  @Test
  public void shouldSendPendingOrderUpdatesSubmittedOrders() throws Exception {
    order11111 = buildOrder(ORDER_ID_1);
    order22222 = buildOrder(ORDER_ID_2);
    ordersList.setItems(Arrays.asList(order11111, order22222));

    when(client.getPendingOrdersObject(anyString())).thenReturn(ordersList);
    when(client.updateOrder(any(String.class), any(Order.class))).thenReturn(mock200HttpResponse);
    when(mock200HttpResponse.getStatusCode()).thenReturn(HttpStatus.OK);
    when(updateOrderAndSendEmail.postSendEmail(any(JSONObject.class))).thenReturn(mockEmail200HttpResponse);
    when(mockEmail200HttpResponse.getStatusCode()).thenReturn(HttpStatus.OK);

    testScheduleConfigurer.triggerPendingOrderUpdateSchedule();

    verify(client, FIVE_SECOND_TIMEOUT).updateOrder(eq(ORDER_ID_1), argThat(argument -> argument.getState().equals(OrderState.PROCESSING)));
    verify(client, FIVE_SECOND_TIMEOUT).updateOrder(eq(ORDER_ID_2), argThat(argument -> argument.getState().equals(OrderState.PROCESSING)));
    verify(urlBuilder, times(2)).buildComponent("emailNotifications");
  }

  private Order buildOrder(String orderId) {
    Order order = new Order();
    order.setId(orderId);
    order.setSiteId("testSite");
    order.setState(OrderState.SUBMITTED);
    order.setLastModifiedDate("2019-10-20T01:02:03.004Z");

    OrderShippingGroup shippingGroup = new OrderShippingGroup();

    OrderAddress emailAddress = new OrderAddress();
    emailaddress.setemail("test@test.test");
    emailAddress.setFirstName("First Name");
    emailAddress.setLastName("Last Name");

    shippingGroup.setShippingAddress(emailAddress);

    order.setShippingGroups(Arrays.asList(shippingGroup));

    return order;
  }
}

Я думаю, что проблема в том, что когда я просто запускаю тест, тег @MockBean создает новый экземпляр OrdersClient для контекста, поэтому, когда вызывается метод when() или verify(), он работает, как и ожидалось, но еслитест интеграции выполняется как задание gradlew, задание может использовать экземпляр, отличный от того, который объявлен как @MockBean.

Есть ли способ использовать один и тот же экземпляр для двух сценариев? Если это предположение верно. Если нет, то что не так?

Спасибо, Педро.

...