В статье речь пойдет об интеграции веб-приложений, написанных с помощью Spring и работающих по HTTP. Название Spring Cloud Contract, на мой взгляд, вводит в заблуждение, так как не имеет ничего общего с cloud.


Речь пойдет об API контрактах.


Для юнит-тестирования контроллеров достаточно часто используются mockMCV или RestAssured. Для моков на стороне фронтэнда используются моск-серверы, например Wiremock или Pact. Но зачастую, юнит-тесты пишут одни люди, а моки другие.


Это может привести к пролбемам при интеграции.


Например, сервер при отсутствии данных может возвращать 204 NO_CONTENT, а клиент может ожидать 200 OK и пустой json.


Тут не важно, кто из них прав. Проблема в том, что кто-то сделал ошибку и она будет найдена не раньше этапа интеграции.


Вот эту проблему и призван решить spring cloud contract.


Что такое spring cloud contract


Это файл, в котором на yaml или groovyDSL диалекте описано, как должны выглядеть запрос и ответ. По умолчанию все контракты лежат в папке /src/test/resources/contracts/*.


Для примера протестируем простейший GET-endpoint


@GetMapping("/bets/{userId}")
public ResponseEntity<List<Bet>> getBets(@PathVariable("userId") String userId) {
   List<Bet> bets = service.getByUserId(userId);
   if (bets.isEmpty()) {
       return ResponseEntity.noContent().build();
   }
   return ResponseEntity.ok(bets);
}

Опишем контракт


org.springframework.cloud.contract.spec.Contract.make {
   request {
       method 'GET'
       urlPath '/bets/2'
   }
   response {
       status 200
       headers {
           header('Content-Type', 'application/json')
       }
       body('''
         {
           "sport":"football",
           "amount": 1
         }
         '''
       )
   }
}

Далее из этого файла с помощью maven или gradle плагина генерируются юнит-тесты и json’ы для wiremock.


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


{
 "id" : "df8f7b73-c242-4664-add3-7214ac6356ff",
 "request" : {
   "urlPath" : "/bets/2",
   "method" : "GET"
 },
 "response" : {
   "status" : 200,
   "body" : "{\"amount\":1,\"sport\":\"football\"}",
   "headers" : {
     "Content-Type" : "application/json"
   },
   "transformers" : [ "response-template" ]
 },
 "uuid" : "df8f7b73-c242-4664-add3-7214ac6356ff"
}

Wiremock можно запустить локально, надо только скачать jar отсюда. По умолчанию json-моки надо положить папку mappings.


$java -jar wiremock-standalone-2.18.0.jar 

Ниже показан сгеренированный тест. По умолчанию использована библиотека RestAssured, но могут быть использоаваны mockMVC или spockframework.


public class UserControllerTest extends ContractBae {

  @Test
  public void validate_get_200() throws Exception {
     // given:
        MockMvcRequestSpecification request = given();

     // when:
        ResponseOptions response = given().spec(request)
              .get("/bets/2");

     // then:
        assertThat(response.statusCode()).isEqualTo(200);
        assertThat(response.header("Content-Type")).isEqualTo("application/json");
     // and:
        DocumentContext parsedJson = JsonPath.parse(response.getBody().asString());
        assertThatJson(parsedJson).field("['amount']").isEqualTo(1);
        assertThatJson(parsedJson).field("['sport']").isEqualTo("football");
  }
}

Следует отметить, что все сгенерированные классы наследуют какой-нибудь базовый класс (базовых классов может быть несколько), в котором инициализируются все необходимые параметры для тестов. Путь к базовому классу описывается в настройках плагина.


Для данного примера базовый класс может выглядеть так:


@RunWith(SpringRunner.class)
@SpringBootTest(webEnvironment = SpringBootTest.WebEnvironment.RANDOM_PORT, classes = SccDemoApplication.class)
public abstract class ContractBae {
   @LocalServerPort
   int port;
   @Autowired
   protected WebApplicationContext webApplicationContext;

   @Before
   public void configureRestAssured() {
       RestAssured.port = port;
       MockMvc mockMvc = MockMvcBuilders.webAppContextSetup(webApplicationContext)
               .build();
       RestAssuredMockMvc.mockMvc(mockMvc);
   }
}

В итоге получаем, что и тесты и моки получены из одного источника. Если юнит-тесты проходят и условный фронтэнд работает на моках, то и с интеграцией проблем не будет.


Но это еще не все
Моки может использовать не только фронтэнд, но само приложение для интеграции с другим приложением. Спринг умеет запускать моск-сервер, надо только сгенерировать jar с моками и передать путь к нему аннотации @AutoConfigureStubRunner


Допустим что наш контроллер делает HTTP к другому приложению:


@GetMapping("/bets/{userId}")
public ResponseEntity<List<Bet>> getBets(@PathVariable("userId") String userId) {
   if(!isUsetExists(userId)) {
       return ResponseEntity.notFound().build();
   }
   List<Bet> bets = service.getByUserId(userId);
   if (bets.isEmpty()) {
       return ResponseEntity.noContent().build();
   }
   return ResponseEntity.ok(bets);
}

private boolean isUsetExists(String userId) {
   try {
       restTemplate.getForObject("/exists/" + userId, Void.class);
       return true;
   } catch (HttpStatusCodeException ignore) {
       return false;
   }
}

Тогда надо просто описать пусть к jar с моками в базовом классе


@RunWith(SpringRunner.class)
@SpringBootTest(webEnvironment = SpringBootTest.WebEnvironment.RANDOM_PORT, classes = SccDemoApplication.class)
@AutoConfigureStubRunner(ids = {"me.dehasi.contracts.demo:sub-service-stubs:+:stubs:8090"}, stubsMode = StubRunnerProperties.StubsMode.LOCAL)
public abstract class ContractBase {

Т.к. это тесты контроллера, то из этих же тестов можно сгенерировать ascii-doc сниппеты (полноценная статья про rest-docs уже есть на хабре) .


Что мы имеем
Получается, что у нас есть один источник API контракта, который описан на человекопонятном языке, и из него мы генерируем юнит-тесты (теоретически еще и документацию), и из него же моки. Данный подход снижает риски ошибок интеграции между веб-приложениями.


Примеры можно посмотреть на официальном сайте например.


Примеры кода в статье взяты из простейшего проекта тут.

Комментарии (0)