Как смоделировать Spring RestTemplate с помощью mockito? - PullRequest
0 голосов
/ 31 августа 2018

На самом деле я пытаюсь сделать UT ниже метода:

    @Override
public Reservation findReservation(Long id) {
    RestTemplate restTemplate = new RestTemplate();
    Reservation reservation = restTemplate.getForObject(RESERVATION_REST_URL+id, Reservation.class);
    return reservation;
}

Я делаю UT, как показано ниже:

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

@Test
public void testFindReservation(){
    Reservation reservation = new Reservation();
    reservation.setId(1000l);
    reservation.setCheckiIn(true);
    reservation.setNumberOfBags(10);

 when(restTemplate.getForObject("http://localhost:8080/flightreservation/reserva 
 tions/1000", Reservation.class))
            .thenReturn(reservation);

   Reservation res =  reservationRestClient.findReservation(1000l);

   Assert.assertNotNull(res);
}

Но это выдает ошибку. Насколько я понимаю, это не издевается должным образом, так или иначе RestTemplate пытается вызвать реальный API, а не издеваться.

ResourceAccessException: I/O error on GET request for http://localhost:8080/flightreservation/reserva 
     tions/1000

Класс обслуживания: -

    @Service
public class ReservationRestServiceImpl implements ReservationRestService {
    private static final String RESERVATION_REST_URL = "http://localhost:8080/flightreservation/reservations/";
    private final RestTemplate restTemplate;

    public ReservationRestServiceImpl(RestTemplateBuilder restTemplateBuilder) {
        this.restTemplate = restTemplateBuilder.build();
    }

    @Override
    public Reservation fetchReservationByPnr(Long id) {

        Reservation reservation = restTemplate.getForObject(RESERVATION_REST_URL+id, Reservation.class);
        return reservation;
    }
}

И тестовый файл как ниже, после Mocking RestTemplateBuilder в тестовом файле, он дает NLP.

    @RunWith(SpringRunner.class)
//@AutoConfigureWebClient
@SpringBootTest(classes = { WebcheckinApplication.class })
public class WebcheckinApplicationTests {


    private ReservationRestServiceImpl reservationRestServiceImpl;

    @Mock
    private RestTemplateBuilder restTemplateBuilder;

    @Before
    public void setUp() throws Exception {
        reservationRestServiceImpl = new ReservationRestServiceImpl();
        MockitoAnnotations.initMocks(this);
    }

    @Test
    public void contextLoads() {
    }


    @Test
    public void testFindReservation(){
        Reservation reservation = new Reservation();
        reservation.setId(1000l);
        reservation.setCheckiIn(true);
        reservation.setNumberOfBags(10);

        RestTemplateBuilder builder = mock(RestTemplateBuilder.class);
        reservationRestServiceImpl.setBuilder(builder);
        reservationRestServiceImpl.init();

        when(builder.build().getForObject("http://localhost:8080/flightreservation/reservations/1000", Reservation.class))
                .thenReturn(reservation);

       Reservation res =  reservationRestServiceImpl.fetchReservationByPnr(1000l);

       Assert.assertNotNull(res);

        //assertEquals("{message : 'under construction'}", result);
    }
}

Ответы [ 2 ]

0 голосов
/ 31 августа 2018

@ Сандип Тивари ... сделайте так, чтобы ваш тестовый набор был настроен так ...

@Autowired
    private MockRestServiceServer server;

    @Autowired
    private RestTemplate restTemplate;

add these following in @Before method ....
------------------------------------
server.expect(ExpectedCount.manyTimes(), requestTo(URL)).andRespond(withSuccess(detailsString, MediaType.APPLICATION_JSON));
0 голосов
/ 31 августа 2018

Я не уверен, какую версию Spring вы используете. Однако в Spring рекомендуется использовать шаблон RestTemplateBuilder вместо непосредственного создания RestTemplate

https://docs.spring.io/spring-boot/docs/current/reference/html/boot-features-resttemplate.html

RestTemplateBuilder будет внедрен в сервис, а остальная часть шаблона построена из него. В вашем тестовом примере вы можете добавить Mocked RestTemplateBuilder, который, в свою очередь, может вернуть поддельный RestTemplate.

Итак, если у вас есть возможность реорганизовать исходный сервис, у вас может быть код, который гораздо более тестируем.

- Подробнее -

Так я бы настроил класс ...

public class SomeService {

    private RestTemplateBuilder builder;
    private RestTemplate restTemplate;

    @Autowired
    public void setBuilder(RestTemplateBuilder builder) {
        this.builder = builder;
    }

    @PostConstruct
    public void init() {
        restTemplate = builder.build();
    }

    public Object fetchReservationByPnr(Long id) {
        return restTemplate.getForObject("someUrl"+id, Object.class);
    }
}

Затем в своем тесте вы можете просто создать службу, назначить Mocked RestTemplateBuilder и вызвать init. Вам не понадобится Spring Runner или запустить тест в Spring Context. Достаточно стандартного бегуна JUnit.

Я не полностью выполнил ваш тест, но это работает для меня

import org.junit.Test;
import org.mockito.Mockito;
import org.springframework.boot.web.client.RestTemplateBuilder;
import org.springframework.web.client.RestTemplate;

public class SomeServiceTest {

    @Test
    public void testMe() {
        RestTemplateBuilder mockedBuilder = Mockito.mock(RestTemplateBuilder.class);
        RestTemplate mockedRestTemplate = Mockito.mock(RestTemplate.class);
        Mockito.when(mockedBuilder.build()).thenReturn(mockedRestTemplate);

        SomeService someService = new SomeService();
        someService.setBuilder(mockedBuilder);
        someService.init();

        Mockito.verify(mockedBuilder).build();
    }
}

Для вашего теста просто добавьте дополнительные макеты к макету RestTemplate

...