最大恢复追逐次数:
private static final int MAX_FOLLOW_UPS = 20;
处理的业务:
- 实例化StreamAllocation,初始化一个Socket连接对象,获取到输入/输出流()基于Okio
- 开启循环,执行下一个调用链(拦截器),等待返回结果(Response)
- 如果发生错误,判断是否继续请求,否:退出
- 检查响应是否符合要求,是:返回
- 关闭响应结果
- 判断是否达到最大限制数,是:退出
- 检查是否有相同连接,是:释放,重建连接
- 重复以上流程
源码
1 @Override 2 public Response intercept(Chain chain) throws IOException { 3 // 4 Request request = chain.request(); 5 // 1. 初始化一个socket连接对象 6 streamAllocation = new StreamAllocation( 7 client.connectionPool(), createAddress(request.url()), callStackTrace); 8 9 int followUpCount = 0; 10 Response priorResponse = null; 11 while (true) { 12 // 13 if (canceled) { 14 streamAllocation.release(); 15 throw new IOException("Canceled"); 16 } 17 18 Response response = null; 19 boolean releaseConnection = true; 20 try { 21 // 2. 执行下一个拦截器,即BridgeInterceptor 22 response = ((RealInterceptorChain) chain).proceed(request, streamAllocation, null, null); 23 releaseConnection = false; 24 } catch (RouteException e) { 25 // The attempt to connect via a route failed. The request will not have been sent. 26 // 3. 如果有异常,判断是否要恢复 27 if (!recover(e.getLastConnectException(), false, request)) { 28 throw e.getLastConnectException(); 29 } 30 releaseConnection = false; 31 continue; 32 } catch (IOException e) { 33 // An attempt to communicate with a server failed. The request may have been sent. 34 boolean requestSendStarted = !(e instanceof ConnectionShutdownException); 35 if (!recover(e, requestSendStarted, request)) throw e; 36 releaseConnection = false; 37 continue; 38 } finally { 39 // We're throwing an unchecked exception. Release any resources. 40 if (releaseConnection) { 41 streamAllocation.streamFailed(null); 42 streamAllocation.release(); 43 } 44 } 45 46 // Attach the prior response if it exists. Such responses never have a body. 47 if (priorResponse != null) { 48 response = response.newBuilder() 49 .priorResponse(priorResponse.newBuilder() 50 .body(null) 51 .build()) 52 .build(); 53 } 54 // 4. 检查是否符合要求 55 Request followUp = followUpRequest(response); 56 57 if (followUp == null) { 58 if (!forWebSocket) { 59 streamAllocation.release(); 60 } 61 // 返回结果 62 return response; 63 } 64 // 5. 不符合,关闭响应流 65 closeQuietly(response.body()); 66 // 6. 是否超过最大限制 67 if (++followUpCount > MAX_FOLLOW_UPS) { 68 streamAllocation.release(); 69 throw new ProtocolException("Too many follow-up requests: " + followUpCount); 70 } 71 72 if (followUp.body() instanceof UnrepeatableRequestBody) { 73 streamAllocation.release(); 74 throw new HttpRetryException("Cannot retry streamed HTTP body", response.code()); 75 } 76 // 7. 是否有相同的连接 77 if (!sameConnection(response, followUp.url())) { 78 streamAllocation.release(); 79 streamAllocation = new StreamAllocation( 80 client.connectionPool(), createAddress(followUp.url()), callStackTrace); 81 } else if (streamAllocation.codec() != null) { 82 throw new IllegalStateException("Closing the body of " + response 83 + " didn't close its backing stream. Bad interceptor?"); 84 } 85 86 request = followUp; 87 priorResponse = response; 88 } 89 }
初始化连接对象
streamAllocation = new StreamAllocation( client.connectionPool(), createAddress(request.url()), callStackTrace);
注意:此处还没有真正的去建立连接,只是初始化一个连接对象
继续下一个拦截器
上面一步初始化好后,将继续执行下一个连接器BridgeInterceptor,
// 这里有个很重的信息,即会将初始化好的连接对象传递给下一个拦截器,也是贯穿整个请求的连击对象, // 上文我们说过,在拦截器执行过程中,RealInterceptorChain的几个属性字段会一步一步赋值 response = ((RealInterceptorChain) chain).proceed(request, streamAllocation, null, null);
抛出异常
如果抛出异常,将判断是否能够继续连接,以下情况不在,重试:
1 /** 2 * 不在继续连接的情况: 3 * 1. 应用层配置不在连接,默认为true 4 * 2. 请求Request出错不能继续使用 5 * 3. 是否可以恢复的 6 * 3.1、协议错误(ProtocolException) 7 3.2、中断异常(InterruptedIOException) 8 3.3、SSL握手错误(SSLHandshakeException && CertificateException) 9 3.4、certificate pinning错误(SSLPeerUnverifiedException) 10 * 4. 没用更多线路可供选择 11 */ 12 private boolean recover(IOException e, boolean requestSendStarted, Request userRequest) { 13 streamAllocation.streamFailed(e); 14 // 1. 应用层配置不在连接,默认为true 15 // The application layer has forbidden retries. 16 if (!client.retryOnConnectionFailure()) return false; 17 18 // 2. 请求Request出错不能继续使用 19 // We can't send the request body again. 20 if (requestSendStarted && userRequest.body() instanceof UnrepeatableRequestBody) return false; 21 22 // 是否可以恢复的 23 // This exception is fatal. 24 if (!isRecoverable(e, requestSendStarted)) return false; 25 26 // 4. 没用更多线路可供选择 27 // No more routes to attempt. 28 if (!streamAllocation.hasMoreRoutes()) return false; 29 30 // For failure recovery, use the same route selector with a new connection. 31 return true; 32 }