openFeign服务间调用保持请求头信息处理
1、注意特殊情况,在定时任务或者内部之间调用,没有request的时候,不要处理直接返回。
2、在GET请求,参数确放在Body里面传递的情况,restTemplate是不认识的,所以这里要转化下处理,然后清空body数据
3、在请求过程中如果出现java.io.IOException: too many bytes written异常,请参考保持请求头造成请求头和content-length不一致
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
|
/** * 解决服务调用丢失请求头的问题 * @author 大仙 * */ @Component public class FeignConfiguration implements RequestInterceptor{ private final Logger logger = LoggerFactory.getLogger(getClass()); @Autowired private ObjectMapper objectMapper; @Override public void apply(RequestTemplate template) { ServletRequestAttributes attributes = (ServletRequestAttributes) RequestContextHolder .getRequestAttributes(); // 注意: RequestContextHolder依赖于 ThreadLocal, 所以在hystrix的隔离策略为THREAD、MQ的消费者、定时任务调用feignClient时,此处应为null if (attributes == null ) { return ; } HttpServletRequest request = attributes.getRequest(); Enumeration<String> headerNames = request.getHeaderNames(); if (headerNames != null ) { while (headerNames.hasMoreElements()) { String name = headerNames.nextElement(); String values = request.getHeader(name); template.header(name, values); } } logger.info( "保持请求头" ); //feign 不支持 GET 方法传 POJO, json body转query if (template.method().equals( "GET" ) && template.requestBody().asBytes() != null ) { try { JsonNode jsonNode = objectMapper.readTree(template.requestBody().asBytes()); Request.Body.empty(); Map<String, Collection<String>> queries = new HashMap<>(); buildQuery(jsonNode, "" , queries); template.queries(queries); } catch (IOException e) { //提示:根据实践项目情况处理此处异常,这里不做扩展。 e.printStackTrace(); } } } /** * 改造 * @param jsonNode * @param path * @param queries */ private void buildQuery(JsonNode jsonNode, String path, Map<String, Collection<String>> queries) { // 叶子节点 if (!jsonNode.isContainerNode()) { if (jsonNode.isNull()) { return ; } Collection<String> values = queries.get(path); if ( null == values) { values = new ArrayList<>(); queries.put(path, values); } values.add(jsonNode.asText()); return ; } // 数组节点 if (jsonNode.isArray()) { Iterator<JsonNode> it = jsonNode.elements(); while (it.hasNext()) { buildQuery(it.next(), path, queries); } } else { Iterator<Map.Entry<String, JsonNode>> it; it = jsonNode.fields(); while (it.hasNext()) { Map.Entry<String, JsonNode> entry = it.next(); if (StringUtils.hasText(path)) { buildQuery(entry.getValue(), path + "." + entry.getKey(), queries); } else { // 根节点 buildQuery(entry.getValue(), entry.getKey(), queries); } } } } } |
保持请求头造成请求头和content-length不一致
Request processin g failed; nested exception is feign.RetryableException: too many bytes written
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
|
2020 - 09 - 08 14 : 07 : 14.718 ERROR 16146 --- [io- 12000 -exec- 5 ] o.a.c.c.C.[.[.[/].[dispatcherServlet] : Servlet.service() for servlet [dispatcherServlet] in context with path [] threw exception [Request processin g failed; nested exception is feign.RetryableException: too many bytes written executing POST http: //pay/wx/create] with root cause java.io.IOException: too many bytes written at sun.net.www.protocol.http.HttpURLConnection$StreamingOutputStream.write(HttpURLConnection.java: 3574 ) ~[na: 1.8 .0_212] at sun.net.www.protocol.http.HttpURLConnection$StreamingOutputStream.write(HttpURLConnection.java: 3565 ) ~[na: 1.8 .0_212] at feign.Client$Default.convertAndSend(Client.java: 181 ) ~[feign-core- 10.4 . 0 .jar!/:na] at feign.Client$Default.execute(Client.java: 77 ) ~[feign-core- 10.4 . 0 .jar!/:na] at org.springframework.cloud.openfeign.ribbon.RetryableFeignLoadBalancer$ 1 .doWithRetry(RetryableFeignLoadBalancer.java: 114 ) ~[spring-cloud-openfeign-core- 2.1 . 5 .RELEASE.jar!/: 2.1 . 5 .RELEASE] at org.springframework.cloud.openfeign.ribbon.RetryableFeignLoadBalancer$ 1 .doWithRetry(RetryableFeignLoadBalancer.java: 94 ) ~[spring-cloud-openfeign-core- 2.1 . 5 .RELEASE.jar!/: 2.1 . 5 .RELEASE] at org.springframework.retry.support.RetryTemplate.doExecute(RetryTemplate.java: 287 ) ~[spring-retry- 1.2 . 5 .RELEASE.jar!/:na] at org.springframework.retry.support.RetryTemplate.execute(RetryTemplate.java: 180 ) ~[spring-retry- 1.2 . 5 .RELEASE.jar!/:na] at org.springframework.cloud.openfeign.ribbon.RetryableFeignLoadBalancer.execute(RetryableFeignLoadBalancer.java: 94 ) ~[spring-cloud-openfeign-core- 2.1 . 5 .RELEASE.jar!/: 2.1 . 5 .RELEASE] at org.springframework.cloud.openfeign.ribbon.RetryableFeignLoadBalancer.execute(RetryableFeignLoadBalancer.java: 54 ) ~[spring-cloud-openfeign-core- 2.1 . 5 .RELEASE.jar!/: 2.1 . 5 .RELEASE] at com.netflix.client.AbstractLoadBalancerAwareClient$ 1 .call(AbstractLoadBalancerAwareClient.java: 104 ) ~[ribbon-loadbalancer- 2.3 . 0 .jar!/: 2.3 . 0 ] at com.netflix.loadbalancer.reactive.LoadBalancerCommand$ 3 $ 1 .call(LoadBalancerCommand.java: 303 ) ~[ribbon-loadbalancer- 2.3 . 0 .jar!/: 2.3 . 0 ] at com.netflix.loadbalancer.reactive.LoadBalancerCommand$ 3 $ 1 .call(LoadBalancerCommand.java: 287 ) ~[ribbon-loadbalancer- 2.3 . 0 .jar!/: 2.3 . 0 ] at rx.internal.util.ScalarSynchronousObservable$ 3 .call(ScalarSynchronousObservable.java: 231 ) ~[rxjava- 1.3 . 8 .jar!/: 1.3 . 8 ] at rx.internal.util.ScalarSynchronousObservable$ 3 .call(ScalarSynchronousObservable.java: 228 ) ~[rxjava- 1.3 . 8 .jar!/: 1.3 . 8 ] at rx.Observable.unsafeSubscribe(Observable.java: 10327 ) ~[rxjava- 1.3 . 8 .jar!/: 1.3 . 8 ] at rx.internal.operators.OnSubscribeConcatMap$ConcatMapSubscriber.drain(OnSubscribeConcatMap.java: 286 ) ~[rxjava- 1.3 . 8 .jar!/: 1.3 . 8 ] at rx.internal.operators.OnSubscribeConcatMap$ConcatMapSubscriber.onNext(OnSubscribeConcatMap.java: 144 ) ~[rxjava- 1.3 . 8 .jar!/: 1.3 . 8 ] at com.netflix.loadbalancer.reactive.LoadBalancerCommand$ 1 .call(LoadBalancerCommand.java: 185 ) ~[ribbon-loadbalancer- 2.3 . 0 .jar!/: 2.3 . 0 ] at com.netflix.loadbalancer.reactive.LoadBalancerCommand$ 1 .call(LoadBalancerCommand.java: 180 ) ~[ribbon-loadbalancer- 2.3 . 0 .jar!/: 2.3 . 0 ] at rx.Observable.unsafeSubscribe(Observable.java: 10327 ) ~[rxjava- 1.3 . 8 .jar!/: 1.3 . 8 ] at rx.internal.operators.OnSubscribeConcatMap.call(OnSubscribeConcatMap.java: 94 ) ~[rxjava- 1.3 . 8 .jar!/: 1.3 . 8 ] at rx.internal.operators.OnSubscribeConcatMap.call(OnSubscribeConcatMap.java: 42 ) ~[rxjava- 1.3 . 8 .jar!/: 1.3 . 8 ] at rx.Observable.unsafeSubscribe(Observable.java: 10327 ) ~[rxjava- 1.3 . 8 .jar!/: 1.3 . 8 ] at rx.internal.operators.OperatorRetryWithPredicate$SourceSubscriber$ 1 .call(OperatorRetryWithPredicate.java: 127 ) ~[rxjava- 1.3 . 8 .jar!/: 1.3 . 8 ] at rx.internal.schedulers.TrampolineScheduler$InnerCurrentThreadScheduler.enqueue(TrampolineScheduler.java: 73 ) ~[rxjava- 1.3 . 8 .jar!/: 1.3 . 8 ] at rx.internal.schedulers.TrampolineScheduler$InnerCurrentThreadScheduler.schedule(TrampolineScheduler.java: 52 ) ~[rxjava- 1.3 . 8 .jar!/: 1.3 . 8 ] at rx.internal.operators.OperatorRetryWithPredicate$SourceSubscriber.onNext(OperatorRetryWithPredicate.java: 79 ) ~[rxjava- 1.3 . 8 .jar!/: 1.3 . 8 ] at rx.internal.operators.OperatorRetryWithPredicate$SourceSubscriber.onNext(OperatorRetryWithPredicate.java: 45 ) ~[rxjava- 1.3 . 8 .jar!/: 1.3 . 8 ] at rx.internal.util.ScalarSynchronousObservable$WeakSingleProducer.request(ScalarSynchronousObservable.java: 276 ) ~[rxjava- 1.3 . 8 .jar!/: 1.3 . 8 ] at rx.Subscriber.setProducer(Subscriber.java: 209 ) ~[rxjava- 1.3 . 8 .jar!/: 1.3 . 8 ] at rx.internal.util.ScalarSynchronousObservable$JustOnSubscribe.call(ScalarSynchronousObservable.java: 138 ) ~[rxjava- 1.3 . 8 .jar!/: 1.3 . 8 ] at rx.internal.util.ScalarSynchronousObservable$JustOnSubscribe.call(ScalarSynchronousObservable.java: 129 ) ~[rxjava- 1.3 . 8 .jar!/: 1.3 . 8 ] at rx.internal.operators.OnSubscribeLift.call(OnSubscribeLift.java: 48 ) ~[rxjava- 1.3 . 8 .jar!/: 1.3 . 8 ] at rx.internal.operators.OnSubscribeLift.call(OnSubscribeLift.java: 30 ) ~[rxjava- 1.3 . 8 .jar!/: 1.3 . 8 |
原因是:body是跟Content-Length 有关系
复制的时候是所有头都复制的,可能导致Content-length长度跟body不一致. 所以只需要判断如果是Content-length就跳过
解决方式:更改复制逻辑
1
2
3
4
5
6
7
8
9
|
while (headerNames.hasMoreElements()) { String name = headerNames.nextElement(); // 跳过 content-length if (name.equals( "content-length" )){ continue ; } String values = request.getHeader(name); template.header(name, values); } |
以上为个人经验,希望能给大家一个参考,也希望大家多多支持服务器之家。
原文链接:https://blog.csdn.net/zhuwei_clark/article/details/108467024