努力弄清楚如何从响应式请求中接收异常并将其冒泡(主体和Http状态代码将是最少的信息)。
我希望使用这里提到的ExchangeFilterFunction,但似乎无法捕获错误。我知道它已注册并通过该方法运行
public class MyExchangeFilterFunction implements ExchangeFilterFunction {
@Override
public Mono<ClientResponse> filter(ClientRequest request, ExchangeFunction next) {
return next.exchange(request).onErrorMap(MyError.class, c -> new RuntimeException("Ow"));
}
}
注册过滤器功能:
WebClient.builder().filter(new MyExchangeFilterFunction())...
转换成错误(因为我不知道如何冒泡):
@Getter
@Setter
@NoArgsConstructor
@AllArgsConstructor
public class MyError extends RuntimeException{
private Long timestamp;
private String path;
private Integer status;
private String error;
private String message;
}
--------更新-------
错误现在与最新代码一起抛出,但无法弄清楚如何将响应主体塞入Flux/Mono错误中。
@Override
public Mono<ClientResponse> filter(ClientRequest request, ExchangeFunction next) {
return next.exchange(request).map(cr -> {
if(cr.statusCode().isError()) {
return ClientResponse.from(cr).body(
Flux.error(new RuntimeException("OW"))
).build();
} else {
return cr;
}
});
}
使异常处理反应式更进一步。
return ClientResponse.from(cr)
.body(cr.body(BodyExtractors.toFlux(AnError.class))
.flatMap(err -> Flux.error(new PassThroughException(err))))
.build();
然后在WebExceptionHandler中:
AnError error;
if(ex instanceof PassThroughException) {
error = ((PassThroughException) ex).getAnError();
} else {
...
}
exchange.getResponse().setStatusCode(HttpStatus.resolve(error.getStatus()));
exchange.getResponse().getHeaders().add(HttpHeaders.CONTENT_TYPE, MediaType.APPLICATION_JSON_UTF8_VALUE);
byte[] bite = null;
try {
bite = objectMapper.writeValueAsBytes(error);
} catch (JsonProcessingException jpe) {
log.error("Unable to write error to client: {}", ex, jpe);
}
return exchange.getResponse().writeWith(Flux.just(exchange.getResponse().bufferFactory().wrap(bite)));