Spring WebFlux WebClient - 如何解决 400 错误请求
Posted
技术标签:
【中文标题】Spring WebFlux WebClient - 如何解决 400 错误请求【英文标题】:Spring WebFlux WebClient - How to resolve 400 Bad Request 【发布时间】:2020-09-02 10:32:42 【问题描述】:我是响应式编程的新手,我正在使用 Spring WebFlux 的 WebClient 向以下 URL 发出 POST 请求,作为我的 Spring Boot 应用程序的一部分,以将现有测验分配给候选人。我无法理解我在构建 WebClient 请求时做错了什么。
终点
https://www.flexiquiz.com/api/v1/users/user_id/quizzes
在我的请求正文中,我需要传递从另一个 API 获得的测验 ID(工作正常)。
"quiz_id": ""
除了传递请求正文之外,我还将 X-API-KEY 作为请求标头的一部分传递。
但是,当我尝试到达终点时,我收到 "message":"400: Bad Request" 错误。
下面是我的代码。
QuizRequest.java
@JsonInclude(JsonInclude.Include.NON_NULL)
@Data
public class QuizRequest
@JsonProperty("quiz_id")
@NotBlank
private String quizId;
public QuizRequest(@NotBlank String quizId)
this.quizId = quizId;
FlexiQuizClient.java
@Service
@Slf4j
public class FlexiQuizClient
private static final String USER_AGENT = "WebClient for FlexiQuiz";
private final WebClient webClient;
@Value("$flexiquiz.baseurl")
private String FLEXIQUIZ_API_BASE_URL;
@Value("$flexiquiz.key")
private String FLEXIQUIZ_API_KEY;
@Autowired
public FlexiQuizClient()
this.webClient = WebClient.builder()
.baseUrl(FLEXIQUIZ_API_BASE_URL)
.defaultHeader(HttpHeaders.USER_AGENT, USER_AGENT)
.filter(logRequest())
.build();
public String assignQuizToCandidate(String userId, QuizRequest quizRequest)
return Objects.requireNonNull(webClient.post()
.uri(FLEXIQUIZ_API_BASE_URL + "/v1/users/" + userId + "/quizzes")
.header("X-API-KEY", FLEXIQUIZ_API_KEY)
.body(Mono.just(quizRequest), QuizRequest.class)
.exchange()
.block())
.bodyToMono(String.class)
.block();
private ExchangeFilterFunction logRequest()
return (clientRequest, next) ->
log.info("Request: ", clientRequest.method(), clientRequest.url());
clientRequest.headers()
.forEach((name, values) -> values.forEach(value -> log.info("=", name, value)));
return next.exchange(clientRequest);
;
在我的资源类(控制器)中,我正在调用 Web 客户端方法,如下所示:
ResponseResource.java
private String assignQuizToCandidate(String userId, QuizRequest quizRequest)
throws ParseException
log.info("Assigning a quiz based on your skill");
String message = quizClient.assignQuizToCandidate(userId, quizRequest);
log.info("message from status: " + message);
JSONParser parser = new JSONParser();
JSONObject json = (JSONObject) parser.parse(message);
return (String) json.get("message");
在另一个方法中,我调用了上面的方法,如下所示。
QuizRequest quizRequest = new QuizRequest(openQuiz.get().getQuizId());
String status = assignQuizToCandidate(userId, quizRequest);
以下是日志:
2020-05-17 10:20:09.938 DEBUG 32600 --- [ctor-http-nio-1] r.n.resources.PooledConnectionProvider : [id: 0x2b404095, L:/192.168.0.106:62197 - R:www.flexiquiz.com/208.117.41.204:443] Channel acquired, now 1 active connections and 0 inactive connections
2020-05-17 10:20:09.938 DEBUG 32600 --- [ctor-http-nio-1] r.netty.http.client.HttpClientConnect : [id: 0x2b404095, L:/192.168.0.106:62197 - R:www.flexiquiz.com/208.117.41.204:443] Handler is being applied: uri=https://www.flexiquiz.com/api/v1/users/userid/quizzes, method=POST
2020-05-17 10:20:09.939 DEBUG 32600 --- [ctor-http-nio-1] r.n.resources.PooledConnectionProvider : [id: 0x2b404095, L:/192.168.0.106:62197 - R:www.flexiquiz.com/208.117.41.204:443] onStateChange(POSTuri=/api/v1/users/userid/quizzes, connection=PooledConnectionchannel=[id: 0x2b404095, L:/192.168.0.106:62197 - R:www.flexiquiz.com/208.117.41.204:443], [request_prepared])
2020-05-17 10:20:09.939 DEBUG 32600 --- [ctor-http-nio-1] o.s.http.codec.json.Jackson2JsonEncoder : [1bbedd72] Encoding [QuizRequest(quizId=quizid)]
2020-05-17 10:20:09.941 DEBUG 32600 --- [ctor-http-nio-1] r.n.resources.PooledConnectionProvider : [id: 0x2b404095, L:/192.168.0.106:62197 - R:www.flexiquiz.com/208.117.41.204:443] onStateChange(POSTuri=/api/v1/users/userid/quizzes, connection=PooledConnectionchannel=[id: 0x2b404095, L:/192.168.0.106:62197 - R:www.flexiquiz.com/208.117.41.204:443], [request_sent])
2020-05-17 10:20:10.189 DEBUG 32600 --- [ctor-http-nio-1] r.n.http.client.HttpClientOperations : [id: 0x2b404095, L:/192.168.0.106:62197 - R:www.flexiquiz.com/208.117.41.204:443] Received response (auto-read:false) : [Cache-Control=private, Content-Type=text/html; charset=utf-8, Server=Microsoft-IIS/10.0, Date=Sun, 17 May 2020 04:50:10 GMT, Content-Length=30]
2020-05-17 10:20:10.189 DEBUG 32600 --- [ctor-http-nio-1] r.n.resources.PooledConnectionProvider : [id: 0x2b404095, L:/192.168.0.106:62197 - R:www.flexiquiz.com/208.117.41.204:443] onStateChange(POSTuri=/api/v1/users/userid/quizzes, connection=PooledConnectionchannel=[id: 0x2b404095, L:/192.168.0.106:62197 - R:www.flexiquiz.com/208.117.41.204:443], [response_received])
2020-05-17 10:20:10.189 DEBUG 32600 --- [ctor-http-nio-1] o.s.w.r.f.client.ExchangeFunctions : [1bbedd72] Response 400 BAD_REQUEST
2020-05-17 10:20:10.189 DEBUG 32600 --- [ctor-http-nio-1] r.n.http.client.HttpClientOperations : [id: 0x2b404095, L:/192.168.0.106:62197 - R:www.flexiquiz.com/208.117.41.204:443] Received last HTTP packet
2020-05-17 10:20:10.189 DEBUG 32600 --- [ctor-http-nio-1] r.n.resources.PooledConnectionProvider : [id: 0x2b404095, L:/192.168.0.106:62197 - R:www.flexiquiz.com/208.117.41.204:443] onStateChange(POSTuri=/api/v1/users/userid/quizzes, connection=PooledConnectionchannel=[id: 0x2b404095, L:/192.168.0.106:62197 - R:www.flexiquiz.com/208.117.41.204:443], [response_completed])
2020-05-17 10:20:10.189 DEBUG 32600 --- [ctor-http-nio-1] r.n.resources.PooledConnectionProvider : [id: 0x2b404095, L:/192.168.0.106:62197 - R:www.flexiquiz.com/208.117.41.204:443] onStateChange(POSTuri=/api/v1/users/userid/quizzes, connection=PooledConnectionchannel=[id: 0x2b404095, L:/192.168.0.106:62197 - R:www.flexiquiz.com/208.117.41.204:443], [disconnecting])
2020-05-17 10:20:10.189 DEBUG 32600 --- [ctor-http-nio-1] r.n.resources.PooledConnectionProvider : [id: 0x2b404095, L:/192.168.0.106:62197 - R:www.flexiquiz.com/208.117.41.204:443] Releasing channel
2020-05-17 10:20:10.189 DEBUG 32600 --- [ctor-http-nio-1] r.n.resources.PooledConnectionProvider : [id: 0x2b404095, L:/192.168.0.106:62197 - R:www.flexiquiz.com/208.117.41.204:443] Channel cleaned, now 0 active connections and 1 inactive connections
2020-05-17 10:20:10.190 DEBUG 32600 --- [ctor-http-nio-1] reactor.netty.channel.FluxReceive : [id: 0x2b404095, L:/192.168.0.106:62197 - R:www.flexiquiz.com/208.117.41.204:443] Subscribing inbound receiver [pending: 1, cancelled:false, inboundDone: true]
2020-05-17 10:20:10.190 DEBUG 32600 --- [ctor-http-nio-1] o.s.core.codec.StringDecoder : [1bbedd72] Decoded ""message":"400: Bad Request""
2020-05-17 10:20:10.190 INFO 32600 --- [nio-8086-exec-6] i.d.ivrauto.resource.ResponseResource : message from status: "message":"400: Bad Request"
2020-05-17 10:20:10.190 INFO 32600 --- [nio-8086-exec-6] i.d.ivrauto.resource.ResponseResource : json.get(message): 400: Bad Request
2020-05-17 10:20:10.190 INFO 32600 --- [nio-8086-exec-6] o.h.e.i.AbstractFlushingEventListener : Processing flush-time cascades
2020-05-17 10:20:10.195 DEBUG 32600 --- [nio-8086-exec-6] o.h.e.i.AbstractFlushingEventListener : Dirty checking collections
以下是我尝试访问的终点。
POST: /v1/users/user_id/quizzes
请求示例
$ curl https://www.flexiquiz.com/api/v1/users/06e3244f-1381-4da4-aa75-996981b42edb/quizzes
-H "X-API-KEY: fcb5f59c-2a2f-44a9-8261-33cbfa97be99"
-d quiz_id="1153af46-9580-4672-af78-f23ce2577a14"
示例响应
"message": "200: OK"
【问题讨论】:
您能否尝试在不使用自定义用户代理标头的情况下发送请求。 @ThomasAndolf 不,它没有帮助。仍然收到 400 Bad Request。 您能否发布您的日志或在调试中运行您的应用程序,以便我们查看实际发出的请求 @ThomasAndolf 用日志编辑了我的问题。 您的服务器端点是什么样的?您提供了客户端代码,但没有提供服务器方法。 【参考方案1】:基于@ThomasAndolf 的回答是我必须做的。
public String assignQuizToCandidate(String userId, String quizId)
final MultiValueMap<String, String> data = new LinkedMultiValueMap<>();
data.add("quiz_id", quizId);
return webClient.post()
.uri(FLEXIQUIZ_API_BASE_URL + "/v1/users/" + userId + "/quizzes")
.header("X-API-KEY", FLEXIQUIZ_API_KEY)
.bodyValue(data)
.retrieve()
.bodyToMono(String.class)
.block();
我将输出作为字符串,因为响应仅包含带有以下消息的字符串:
"message": "200: OK"
【讨论】:
【参考方案2】:您的问题可能是您以错误的格式发送数据。您在正文中以application/json
格式发布数据。
但是,如果您查看请求,它是使用 curl
中的 -d
标志发出的。
来自 curl 文档:
-d,--数据
(HTTP) 将 POST 请求中的指定数据发送到 HTTP 服务器,方式与 当用户填写 HTML 表单并按下提交按钮时,浏览器会执行此操作。这会导致 curl 使用内容类型 application/x-www-form-urlencoded 将数据传递给服务器。 与 -F、--form 比较。
这基本上意味着您需要以FORM
格式发送数据。
Webflux 文档说明如何将数据作为表单请求发送。
Webflux Send FormData
所以你的代码应该看起来像(有点):
public QuizResponse assignQuizToCandidate(String userId, String quizId)
final MultiValueMap<String, String> formData = new LinkedMultiValueMap<>();
formData.add("quiz_id", quizId);
return webClient.post()
.uri(FLEXIQUIZ_API_BASE_URL + "/v1/users/" + userId + "/quizzes")
.header("X-API-KEY", FLEXIQUIZ_API_KEY)
.bodyValue(formData)
.retrive()
.bodyToMono(QuizResponse.class)
.block();
【讨论】:
你是绝对正确的人!我会接受你的回答。另外,我会根据您的回答添加我的解决方案。以上是关于Spring WebFlux WebClient - 如何解决 400 错误请求的主要内容,如果未能解决你的问题,请参考以下文章
Spring WebFlux WebClient 弹性和性能
Spring 5 webflux如何在Webclient上设置超时
在 Spring WebFlux webclient 中设置超时