Java Spring云契约与webflux路由

Java Spring云契约与webflux路由,java,spring,spring-boot,spring-webflux,spring-cloud-contract,Java,Spring,Spring Boot,Spring Webflux,Spring Cloud Contract,我正在尝试使用SpringCloudContract和使用Spring5路由的rest服务,但它不起作用。 我在客户端,试图在junit测试中使用存根运行程序。 如果我使用i classic@RestController和flux,它工作得很好,但是如果我尝试使用RouterFunction更改控制器,它就不工作了,我得到了一个404。 这是我的示例代码 pom.xml <parent> <groupId>org.springframework.boot&

我正在尝试使用SpringCloudContract和使用Spring5路由的rest服务,但它不起作用。 我在客户端,试图在junit测试中使用存根运行程序。 如果我使用i classic@RestController和flux,它工作得很好,但是如果我尝试使用RouterFunction更改控制器,它就不工作了,我得到了一个404。 这是我的示例代码

pom.xml

<parent>
        <groupId>org.springframework.boot</groupId>
        <artifactId>spring-boot-starter-parent</artifactId>
        <version>2.0.5.RELEASE</version>
        <relativePath/> <!-- lookup parent from repository -->
    </parent>
<dependencies>
...
   <dependency>
               <groupId>org.springframework.cloud</groupId>
               <artifactId>spring-cloud-starter-contract-stub-runner</artifactId>
               <scope>test</scope>
   </dependency>
</dependencies>
即使删除@autoconfiguresubrunner注释,我也会遇到同样的问题。如果我只添加存根运行程序依赖项,我会发现这个行为有问题。
我还尝试使用最新版本的SpringBoot和SpringCloud合约,但我遇到了同样的问题。有人能帮我吗?

Spring Cloud Contract存根运行程序只需在给定(或随机)端口上启动WireMock服务器。Stub Runner不会发生与
WebTestClient
相关的任何事情。换句话说,您很可能错误配置了
WebTestClient

让我们试着确保你没有误用这个项目。如果您让服务A通过WebClient调用服务B,那么服务B应该定义合同,并从合同中创建测试和跨度。然后在服务A端,您将使用Spring Cloud Contract存根运行程序启动服务B的存根。无论您使用什么(RestTemplate、WebClient等),您都将向我们为您启动的WireMock服务器发送HTTP调用

如何将Spring云契约存根运行程序与WebTestClient一起使用的示例(摘自:)


我删除了@AutoConfigureStubRunner,它可以很好地处理routes。好的,但是你是如何使用这个stubrunner的?我需要使用存根来进行消费者测试,我想这就是我所做的。我在producer端创建一个groovy契约,运行mvn install命令生成存根。在我的consumer中,我有一个服务,它通过一个特定的url调用生产者。我将存根运行程序配置为在此url上运行并运行测试。我错在哪里?我有一个404,即使我删除了存根运行程序自动配置注释,但我的Dependency中有存根运行程序启动程序,我已经看到了这个示例。但是我在消费者方面,而不是生产者方面。我在样本中添加了一个通过测试来更新答案。我将尝试做一些其他测试。但我的webclient配置是我在第一篇文章中看到的。如果我创建了一个只调用消费者的测试,那么它工作得很好。当我添加存根运行程序依赖项(不向测试添加配置)时,测试失败。谢谢你的回复
@Configuration
@EnableWebFlux
public class Routing {

    @Autowired
    private TestLoginController loginController;

    @Bean
    public HttpHandler routerFunction() {
        return WebHttpHandlerBuilder
                .webHandler(RouterFunctions.toWebHandler(createRouterFunction()))
                .build();
    }

    private RouterFunction<ServerResponse> createRouterFunction() {
        return route(POST("/testlogin"), loginController::testLogin);
    }
}
@Component
public class TestLoginController {

    @Autowired
    private TestLoginService testLoginService;

    public Mono<ServerResponse> testLogin(ServerRequest request) {
        return Mono.just(request)
                   .flatMap(req -> ServerResponse.ok()
                                                 .body(testLoginService.testLogin(request.bodyToMono(LoginRequest.class)), LoginResponse.class)
                           );
    }
}
@RunWith(SpringRunner.class)
@SpringBootTest(webEnvironment = SpringBootTest.WebEnvironment.RANDOM_PORT)
@AutoConfigureStubRunner(ids = {"groupId:artifactId:+:stubs:8090"},
        stubsMode = StubRunnerProperties.StubsMode.LOCAL)
public class DemoApplicationTests {

    @LocalServerPort
    private int port;


    @Test
    public void contextLoads() throws Exception {
        LoginRequest request = new LoginRequest();

        WebTestClient
                .bindToServer()
                .baseUrl("http://localhost:" + port)
                .build()
                .post()
                .uri("testlogin").accept(MediaType.APPLICATION_JSON)
                .body(BodyInserters.fromObject(request))
                .exchange()
                .expectStatus().isOk()
                .expectBody()
                ....
    }

}
package com.example;
 import java.util.Objects;
 import org.junit.Test;
import org.junit.runner.RunWith;
 import org.springframework.boot.test.autoconfigure.json.AutoConfigureJsonTesters;
import org.springframework.boot.test.autoconfigure.web.servlet.AutoConfigureMockMvc;
import org.springframework.boot.test.context.SpringBootTest;
import org.springframework.boot.test.context.SpringBootTest.WebEnvironment;
import org.springframework.cloud.contract.stubrunner.spring.AutoConfigureStubRunner;
import org.springframework.cloud.contract.stubrunner.spring.StubRunnerPort;
import org.springframework.cloud.contract.stubrunner.spring.StubRunnerProperties;
import org.springframework.test.annotation.DirtiesContext;
import org.springframework.test.context.junit4.SpringRunner;
import org.springframework.test.web.reactive.server.WebTestClient;
 /**
 * @author Marcin Grzejszczak
 */
@RunWith(SpringRunner.class)
@SpringBootTest(webEnvironment = WebEnvironment.MOCK)
@AutoConfigureMockMvc
@AutoConfigureJsonTesters
//remove::start[]
@AutoConfigureStubRunner(stubsMode = StubRunnerProperties.StubsMode.LOCAL, ids = "com.example:beer-api-producer-webflux")
//remove::end[]
@DirtiesContext
//@org.junit.Ignore
public class BeerControllerWebClientTest extends AbstractTest {
    //remove::start[]
    @StubRunnerPort("beer-api-producer-webflux") int producerPort;
    //remove::end[]
    @Test public void should_give_me_a_beer_when_im_old_enough() throws Exception {
        //remove::start[]
        WebTestClient.bindToServer()
                .build()
                .post()
                .uri("http://localhost:" + producerPort + "/check")
                .syncBody(new WebClientPerson("marcin", 22))
                .header("Content-Type", "application/json")
                .exchange()
                .expectStatus().is2xxSuccessful()
                .expectBody(WebClientResponse.class)
                .isEqualTo(new WebClientResponse(WebClientResponseStatus.OK));
        //remove::end[]
    }
    @Test public void should_reject_a_beer_when_im_too_young() throws Exception {
        //remove::start[]
        WebTestClient.bindToServer()
                .build()
                .post()
                .uri("http://localhost:" + producerPort + "/check")
                .syncBody(new WebClientPerson("marcin", 17))
                .header("Content-Type", "application/json")
                .exchange()
                .expectStatus().is2xxSuccessful()
                .expectBody(WebClientResponse.class)
                .isEqualTo(new WebClientResponse(WebClientResponseStatus.NOT_OK));
        //remove::end[]
    }
}
 class WebClientPerson {
    public String name;
    public int age;
    public WebClientPerson(String name, int age) {
        this.name = name;
        this.age = age;
    }
    public WebClientPerson() {
    }
}
 class WebClientResponse {
    public WebClientResponseStatus status;
    WebClientResponse(WebClientResponseStatus status) {
        this.status = status;
    }
    WebClientResponse() {
    }
    @Override
    public boolean equals(Object o) {
        if (this == o) return true;
        if (o == null || getClass() != o.getClass()) return false;
        WebClientResponse that = (WebClientResponse) o;
        return status == that.status;
    }
    @Override
    public int hashCode() {
        return Objects.hash(status);
    }
}
 enum WebClientResponseStatus {
    OK, NOT_OK
}