/* Android Asynchronous Http Client Copyright (c) 2011 James Smith <james@loopj.com> http://loopj.com Licensed under the Apache License, Version 2.0 (the "License"); you may not use this file except in compliance with the License. You may obtain a copy of the License at http://www.apache.org/licenses/LICENSE-2.0 Unless required by applicable law or agreed to in writing, software distributed under the License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the License for the specific language governing permissions and limitations under the License. */ package com.pyj.http; import java.io.IOException; import java.net.ConnectException; import java.net.SocketException; import java.net.SocketTimeoutException; import java.net.UnknownHostException; import org.apache.http.HttpResponse; import org.apache.http.client.HttpRequestRetryHandler; import org.apache.http.client.methods.HttpUriRequest; import org.apache.http.impl.client.AbstractHttpClient; import org.apache.http.protocol.HttpContext; import com.hupu.games.common.HupuLog; import com.hupu.http.HupuHttpException; class AsyncHttpRequest implements Runnable { private final AbstractHttpClient client; private final HttpContext context; private final HttpUriRequest request; private final AsyncHttpResponseHandler responseHandler; private boolean isBinaryRequest; private int executionCount; private int reqType; public AsyncHttpRequest(AbstractHttpClient client, HttpContext context, HttpUriRequest request, AsyncHttpResponseHandler responseHandler) { this.client = client; this.context = context; this.request = request; this.responseHandler = responseHandler; if(responseHandler instanceof BinaryHttpResponseHandler) { this.isBinaryRequest = true; } } public AsyncHttpRequest(AbstractHttpClient client, HttpContext context, HttpUriRequest request, AsyncHttpResponseHandler responseHandler,int reqType) { this.client = client; this.context = context; this.request = request; this.responseHandler = responseHandler; this.reqType=reqType; if(responseHandler instanceof BinaryHttpResponseHandler) { this.isBinaryRequest = true; } HupuLog.d(request.getURI().toString()); } @Override public void run() { try { if(responseHandler != null){ responseHandler.sendStartMessage(); } makeRequestWithRetries(); if(responseHandler != null) { responseHandler.sendFinishMessage(); } } catch (IOException e) { if(responseHandler != null) { responseHandler.sendFinishMessage(); if(this.isBinaryRequest) { responseHandler.sendFailureMessage(e, (byte[]) null,reqType); } else { responseHandler.sendFailureMessage(e, (String) null,reqType); } } } } private void makeRequest() throws IOException { if(!Thread.currentThread().isInterrupted()) { try { HttpResponse response = client.execute(request, context); if(!Thread.currentThread().isInterrupted()) { if(responseHandler != null) { responseHandler.sendResponseMessage(response,reqType); } } else{ //TODO: should raise InterruptedException? this block is reached whenever the request is cancelled before its response is received } } catch (IOException e) { if(!Thread.currentThread().isInterrupted()) { throw e; } } } } private void makeRequestWithRetries() throws ConnectException { // This is an additional layer of retry logic lifted from droid-fu // See: https://github.com/kaeppler/droid-fu/blob/master/src/main/java/com/github/droidfu/http/BetterHttpRequestBase.java boolean retry = true; IOException cause = null; HttpRequestRetryHandler retryHandler = client.getHttpRequestRetryHandler(); while (retry) { try { makeRequest(); return; } catch (UnknownHostException e) { if(responseHandler != null) { responseHandler.sendFailureMessage(new HupuHttpException("不能连接到服务器!请稍后再试!"), (String)null,reqType); } return; }catch (SocketException e){ // Added to detect host unreachable if(responseHandler != null) { responseHandler.sendFailureMessage(new HupuHttpException("网络连接失败,请稍后再试!"), (String)null,reqType); } return; }catch (SocketTimeoutException e){ if(responseHandler != null) { responseHandler.sendFailureMessage(new HupuHttpException("网络连接超时,请稍后再试!"), (String)null,reqType); } return; } catch (IOException e) { cause = e; retry = retryHandler.retryRequest(cause, ++executionCount, context); } catch (NullPointerException e) { // there's a bug in HttpClient 4.0.x that on some occasions causes // DefaultRequestExecutor to throw an NPE, see // http://code.google.com/p/android/issues/detail?id=5255 cause = new IOException("NPE in HttpClient" + e.getMessage()); retry = retryHandler.retryRequest(cause, ++executionCount, context); } } // no retries left, crap out with exception ConnectException ex = new ConnectException(); ex.initCause(cause); throw ex; } }