19066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project/* 29066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * Copyright (C) 2007 The Android Open Source Project 39066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * 49066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * Licensed under the Apache License, Version 2.0 (the "License"); 59066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * you may not use this file except in compliance with the License. 69066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * You may obtain a copy of the License at 79066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * 89066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * http://www.apache.org/licenses/LICENSE-2.0 99066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * 109066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * Unless required by applicable law or agreed to in writing, software 119066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * distributed under the License is distributed on an "AS IS" BASIS, 129066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 139066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * See the License for the specific language governing permissions and 149066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * limitations under the License. 159066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project */ 169066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 179066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectpackage android.net.http; 189066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 199066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectimport android.content.Context; 209066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectimport android.os.SystemClock; 219066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 229066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectimport java.io.IOException; 239066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectimport java.net.UnknownHostException; 249066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectimport java.util.LinkedList; 259066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 269066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectimport javax.net.ssl.SSLHandshakeException; 279066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 289066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectimport org.apache.http.ConnectionReuseStrategy; 299066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectimport org.apache.http.HttpEntity; 309066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectimport org.apache.http.HttpException; 319066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectimport org.apache.http.HttpHost; 329066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectimport org.apache.http.HttpVersion; 339066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectimport org.apache.http.ParseException; 349066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectimport org.apache.http.ProtocolVersion; 359066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectimport org.apache.http.protocol.ExecutionContext; 369066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectimport org.apache.http.protocol.HttpContext; 379066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectimport org.apache.http.protocol.BasicHttpContext; 389066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 399066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project/** 409066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * {@hide} 419066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project */ 429066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectabstract class Connection { 439066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 449066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project /** 459066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * Allow a TCP connection 60 idle seconds before erroring out 469066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project */ 479066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project static final int SOCKET_TIMEOUT = 60000; 489066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 499066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project private static final int SEND = 0; 509066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project private static final int READ = 1; 519066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project private static final int DRAIN = 2; 529066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project private static final int DONE = 3; 539066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project private static final String[] states = {"SEND", "READ", "DRAIN", "DONE"}; 549066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 559066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project Context mContext; 569066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 579066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project /** The low level connection */ 589066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project protected AndroidHttpClientConnection mHttpClientConnection = null; 599066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 609066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project /** 619066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * The server SSL certificate associated with this connection 629066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * (null if the connection is not secure) 639066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * It would be nice to store the whole certificate chain, but 649066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * we want to keep things as light-weight as possible 659066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project */ 669066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project protected SslCertificate mCertificate = null; 679066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 689066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project /** 699066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * The host this connection is connected to. If using proxy, 709066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * this is set to the proxy address 719066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project */ 729066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project HttpHost mHost; 739066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 749066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project /** true if the connection can be reused for sending more requests */ 759066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project private boolean mCanPersist; 769066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 779066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project /** context required by ConnectionReuseStrategy. */ 789066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project private HttpContext mHttpContext; 799066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 809066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project /** set when cancelled */ 819066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project private static int STATE_NORMAL = 0; 829066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project private static int STATE_CANCEL_REQUESTED = 1; 839066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project private int mActive = STATE_NORMAL; 849066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 859066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project /** The number of times to try to re-connect (if connect fails). */ 869066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project private final static int RETRY_REQUEST_LIMIT = 2; 879066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 889066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project private static final int MIN_PIPE = 2; 899066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project private static final int MAX_PIPE = 3; 909066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 919066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project /** 929066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * Doesn't seem to exist anymore in the new HTTP client, so copied here. 939066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project */ 949066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project private static final String HTTP_CONNECTION = "http.connection"; 959066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 969066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project RequestFeeder mRequestFeeder; 979066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 989066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project /** 999066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * Buffer for feeding response blocks to webkit. One block per 1009066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * connection reduces memory churn. 1019066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project */ 1029066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project private byte[] mBuf; 1039066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 1049066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project protected Connection(Context context, HttpHost host, 1059066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project RequestFeeder requestFeeder) { 1069066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project mContext = context; 1079066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project mHost = host; 1089066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project mRequestFeeder = requestFeeder; 1099066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 1109066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project mCanPersist = false; 1119066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project mHttpContext = new BasicHttpContext(null); 1129066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 1139066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 1149066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project HttpHost getHost() { 1159066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return mHost; 1169066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 1179066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 1189066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project /** 1199066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * connection factory: returns an HTTP or HTTPS connection as 1209066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * necessary 1219066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project */ 1229066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project static Connection getConnection( 12386806ce11a89260147d7c2efa2c192b711d923dbPatrick Scott Context context, HttpHost host, HttpHost proxy, 1249066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project RequestFeeder requestFeeder) { 1259066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 1269066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (host.getSchemeName().equals("http")) { 12786806ce11a89260147d7c2efa2c192b711d923dbPatrick Scott return new HttpConnection(context, host, requestFeeder); 1289066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 1299066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 1309066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project // Otherwise, default to https 13186806ce11a89260147d7c2efa2c192b711d923dbPatrick Scott return new HttpsConnection(context, host, proxy, requestFeeder); 1329066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 1339066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 1349066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project /** 1359066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * @return The server SSL certificate associated with this 1369066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * connection (null if the connection is not secure) 1379066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project */ 1389066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project /* package */ SslCertificate getCertificate() { 1399066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return mCertificate; 1409066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 1419066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 1429066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project /** 1439066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * Close current network connection 1449066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * Note: this runs in non-network thread 1459066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project */ 1469066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project void cancel() { 1479066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project mActive = STATE_CANCEL_REQUESTED; 1489066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project closeConnection(); 1499066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (HttpLog.LOGV) HttpLog.v( 1509066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project "Connection.cancel(): connection closed " + mHost); 1519066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 1529066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 1539066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project /** 1549066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * Process requests in queue 1559066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * pipelines requests 1569066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project */ 1579066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project void processRequests(Request firstRequest) { 1589066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project Request req = null; 1599066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project boolean empty; 1609066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project int error = EventHandler.OK; 1619066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project Exception exception = null; 1629066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 1639066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project LinkedList<Request> pipe = new LinkedList<Request>(); 1649066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 1659066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project int minPipe = MIN_PIPE, maxPipe = MAX_PIPE; 1669066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project int state = SEND; 1679066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 1689066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project while (state != DONE) { 1699066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (HttpLog.LOGV) HttpLog.v( 1709066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project states[state] + " pipe " + pipe.size()); 1719066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 1729066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project /* If a request was cancelled, give other cancel requests 1739066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project some time to go through so we don't uselessly restart 1749066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project connections */ 1759066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (mActive == STATE_CANCEL_REQUESTED) { 1769066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project try { 1779066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project Thread.sleep(100); 1789066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } catch (InterruptedException x) { /* ignore */ } 1799066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project mActive = STATE_NORMAL; 1809066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 1819066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 1829066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project switch (state) { 1839066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project case SEND: { 1849066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (pipe.size() == maxPipe) { 1859066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project state = READ; 1869066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project break; 1879066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 1889066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project /* get a request */ 1899066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (firstRequest == null) { 1909066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project req = mRequestFeeder.getRequest(mHost); 1919066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } else { 1929066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project req = firstRequest; 1939066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project firstRequest = null; 1949066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 1959066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (req == null) { 1969066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project state = DRAIN; 1979066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project break; 1989066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 1999066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project req.setConnection(this); 2009066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 2019066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project /* Don't work on cancelled requests. */ 2029066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (req.mCancelled) { 2039066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (HttpLog.LOGV) HttpLog.v( 2049066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project "processRequests(): skipping cancelled request " 2059066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project + req); 2069066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project req.complete(); 2079066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project break; 2089066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 2099066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 2109066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (mHttpClientConnection == null || 2119066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project !mHttpClientConnection.isOpen()) { 2129066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project /* If this call fails, the address is bad or 2139066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project the net is down. Punt for now. 2149066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 2159066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project FIXME: blow out entire queue here on 2169066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project connection failure if net up? */ 2179066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 2189066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (!openHttpConnection(req)) { 2199066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project state = DONE; 2209066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project break; 2219066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 2229066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 2239066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 224dba8cb76371960457e91b31fa396478f809a5a34Brian Carlstrom /* we have a connection, let the event handler 225dba8cb76371960457e91b31fa396478f809a5a34Brian Carlstrom * know of any associated certificate, 226dba8cb76371960457e91b31fa396478f809a5a34Brian Carlstrom * potentially none. 227dba8cb76371960457e91b31fa396478f809a5a34Brian Carlstrom */ 228dba8cb76371960457e91b31fa396478f809a5a34Brian Carlstrom req.mEventHandler.certificate(mCertificate); 229dba8cb76371960457e91b31fa396478f809a5a34Brian Carlstrom 2309066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project try { 2319066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project /* FIXME: don't increment failure count if old 2329066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project connection? There should not be a penalty for 2339066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project attempting to reuse an old connection */ 2349066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project req.sendRequest(mHttpClientConnection); 2359066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } catch (HttpException e) { 2369066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project exception = e; 2379066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project error = EventHandler.ERROR; 2389066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } catch (IOException e) { 2399066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project exception = e; 2409066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project error = EventHandler.ERROR_IO; 2419066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } catch (IllegalStateException e) { 2429066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project exception = e; 2439066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project error = EventHandler.ERROR_IO; 2449066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 2459066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (exception != null) { 2469066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (httpFailure(req, error, exception) && 2479066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project !req.mCancelled) { 2489066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project /* retry request if not permanent failure 2499066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project or cancelled */ 2509066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project pipe.addLast(req); 2519066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 2529066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project exception = null; 253aaebc86386c8bb44c25dd06fe573e52ef6b60fbePatrick Scott state = clearPipe(pipe) ? DONE : SEND; 2549066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project minPipe = maxPipe = 1; 2559066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project break; 2569066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 2579066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 2589066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project pipe.addLast(req); 2599066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (!mCanPersist) state = READ; 2609066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project break; 2619066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 2629066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 2639066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project case DRAIN: 2649066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project case READ: { 2659066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project empty = !mRequestFeeder.haveRequest(mHost); 2669066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project int pipeSize = pipe.size(); 2679066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (state != DRAIN && pipeSize < minPipe && 2689066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project !empty && mCanPersist) { 2699066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project state = SEND; 2709066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project break; 2719066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } else if (pipeSize == 0) { 2729066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project /* Done if no other work to do */ 2739066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project state = empty ? DONE : SEND; 2749066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project break; 2759066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 2769066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 2779066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project req = (Request)pipe.removeFirst(); 2789066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (HttpLog.LOGV) HttpLog.v( 2799066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project "processRequests() reading " + req); 2809066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 2819066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project try { 2829066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project req.readResponse(mHttpClientConnection); 2839066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } catch (ParseException e) { 2849066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project exception = e; 2859066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project error = EventHandler.ERROR_IO; 2869066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } catch (IOException e) { 2879066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project exception = e; 2889066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project error = EventHandler.ERROR_IO; 2899066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } catch (IllegalStateException e) { 2909066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project exception = e; 2919066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project error = EventHandler.ERROR_IO; 2929066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 2939066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (exception != null) { 2949066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (httpFailure(req, error, exception) && 2959066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project !req.mCancelled) { 2969066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project /* retry request if not permanent failure 2979066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project or cancelled */ 2989066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project req.reset(); 2999066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project pipe.addFirst(req); 3009066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 3019066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project exception = null; 3029066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project mCanPersist = false; 3039066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 3049066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (!mCanPersist) { 3059066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (HttpLog.LOGV) HttpLog.v( 3069066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project "processRequests(): no persist, closing " + 3079066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project mHost); 3089066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 3099066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project closeConnection(); 3109066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 3119066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project mHttpContext.removeAttribute(HTTP_CONNECTION); 3129066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project clearPipe(pipe); 3139066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project minPipe = maxPipe = 1; 314aaebc86386c8bb44c25dd06fe573e52ef6b60fbePatrick Scott state = SEND; 3159066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 3169066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project break; 3179066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 3189066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 3199066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 3209066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 3219066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 3229066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project /** 3239066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * After a send/receive failure, any pipelined requests must be 3249066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * cleared back to the mRequest queue 3259066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * @return true if mRequests is empty after pipe cleared 3269066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project */ 3279066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project private boolean clearPipe(LinkedList<Request> pipe) { 3289066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project boolean empty = true; 3299066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (HttpLog.LOGV) HttpLog.v( 3309066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project "Connection.clearPipe(): clearing pipe " + pipe.size()); 3319066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project synchronized (mRequestFeeder) { 3329066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project Request tReq; 3339066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project while (!pipe.isEmpty()) { 3349066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project tReq = (Request)pipe.removeLast(); 3359066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (HttpLog.LOGV) HttpLog.v( 3369066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project "clearPipe() adding back " + mHost + " " + tReq); 3379066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project mRequestFeeder.requeueRequest(tReq); 3389066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project empty = false; 3399066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 34086806ce11a89260147d7c2efa2c192b711d923dbPatrick Scott if (empty) empty = !mRequestFeeder.haveRequest(mHost); 3419066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 3429066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return empty; 3439066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 3449066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 3459066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project /** 3469066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * @return true on success 3479066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project */ 3489066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project private boolean openHttpConnection(Request req) { 3499066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 3509066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project long now = SystemClock.uptimeMillis(); 3519066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project int error = EventHandler.OK; 3529066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project Exception exception = null; 3539066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 3549066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project try { 3559066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project // reset the certificate to null before opening a connection 3569066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project mCertificate = null; 3579066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project mHttpClientConnection = openConnection(req); 3589066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (mHttpClientConnection != null) { 3599066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project mHttpClientConnection.setSocketTimeout(SOCKET_TIMEOUT); 3609066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project mHttpContext.setAttribute(HTTP_CONNECTION, 3619066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project mHttpClientConnection); 3629066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } else { 3639066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project // we tried to do SSL tunneling, failed, 3649066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project // and need to drop the request; 3659066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project // we have already informed the handler 3669066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project req.mFailCount = RETRY_REQUEST_LIMIT; 3679066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return false; 3689066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 3699066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } catch (UnknownHostException e) { 3709066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (HttpLog.LOGV) HttpLog.v("Failed to open connection"); 3719066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project error = EventHandler.ERROR_LOOKUP; 3729066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project exception = e; 3739066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } catch (IllegalArgumentException e) { 3749066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (HttpLog.LOGV) HttpLog.v("Illegal argument exception"); 3759066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project error = EventHandler.ERROR_CONNECT; 3769066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project req.mFailCount = RETRY_REQUEST_LIMIT; 3779066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project exception = e; 3789066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } catch (SSLConnectionClosedByUserException e) { 3799066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project // hack: if we have an SSL connection failure, 3809066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project // we don't want to reconnect 3819066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project req.mFailCount = RETRY_REQUEST_LIMIT; 3829066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project // no error message 3839066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return false; 3849066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } catch (SSLHandshakeException e) { 3859066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project // hack: if we have an SSL connection failure, 3869066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project // we don't want to reconnect 3879066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project req.mFailCount = RETRY_REQUEST_LIMIT; 3889066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (HttpLog.LOGV) HttpLog.v( 3899066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project "SSL exception performing handshake"); 3909066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project error = EventHandler.ERROR_FAILED_SSL_HANDSHAKE; 3919066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project exception = e; 3929066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } catch (IOException e) { 3939066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project error = EventHandler.ERROR_CONNECT; 3949066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project exception = e; 3959066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 3969066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 3979066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (HttpLog.LOGV) { 3989066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project long now2 = SystemClock.uptimeMillis(); 3999066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project HttpLog.v("Connection.openHttpConnection() " + 4009066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project (now2 - now) + " " + mHost); 4019066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 4029066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 4039066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (error == EventHandler.OK) { 4049066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return true; 4059066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } else { 406aaebc86386c8bb44c25dd06fe573e52ef6b60fbePatrick Scott if (req.mFailCount < RETRY_REQUEST_LIMIT) { 4079066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project // requeue 4089066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project mRequestFeeder.requeueRequest(req); 4099066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project req.mFailCount++; 4109066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } else { 4119066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project httpFailure(req, error, exception); 4129066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 4139066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return error == EventHandler.OK; 4149066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 4159066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 4169066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 4179066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project /** 4189066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * Helper. Calls the mEventHandler's error() method only if 4199066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * request failed permanently. Increments mFailcount on failure. 4209066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * 4219066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * Increments failcount only if the network is believed to be 4229066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * connected 4239066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * 4249066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * @return true if request can be retried (less than 4259066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * RETRY_REQUEST_LIMIT failures have occurred). 4269066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project */ 4279066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project private boolean httpFailure(Request req, int errorId, Exception e) { 4289066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project boolean ret = true; 4299066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 4309066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project // e.printStackTrace(); 4319066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (HttpLog.LOGV) HttpLog.v( 4329066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project "httpFailure() ******* " + e + " count " + req.mFailCount + 433aaebc86386c8bb44c25dd06fe573e52ef6b60fbePatrick Scott " " + mHost + " " + req.getUri()); 4349066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 435aaebc86386c8bb44c25dd06fe573e52ef6b60fbePatrick Scott if (++req.mFailCount >= RETRY_REQUEST_LIMIT) { 4369066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project ret = false; 4379066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project String error; 4389066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (errorId < 0) { 43983d4a23c280bdcaf6c301651b76ddc6fbf08949cIain Merrick error = ErrorStrings.getString(errorId, mContext); 4409066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } else { 4419066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project Throwable cause = e.getCause(); 4429066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project error = cause != null ? cause.toString() : e.getMessage(); 4439066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 4449066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project req.mEventHandler.error(errorId, error); 4459066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project req.complete(); 4469066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 4479066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 4489066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project closeConnection(); 4499066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project mHttpContext.removeAttribute(HTTP_CONNECTION); 4509066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 4519066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return ret; 4529066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 4539066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 4549066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project HttpContext getHttpContext() { 4559066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return mHttpContext; 4569066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 4579066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 4589066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project /** 4599066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * Use same logic as ConnectionReuseStrategy 4609066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * @see ConnectionReuseStrategy 4619066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project */ 4629066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project private boolean keepAlive(HttpEntity entity, 4639066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project ProtocolVersion ver, int connType, final HttpContext context) { 4649066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project org.apache.http.HttpConnection conn = (org.apache.http.HttpConnection) 4659066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project context.getAttribute(ExecutionContext.HTTP_CONNECTION); 4669066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 4679066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (conn != null && !conn.isOpen()) 4689066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return false; 4699066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project // do NOT check for stale connection, that is an expensive operation 4709066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 4719066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (entity != null) { 4729066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (entity.getContentLength() < 0) { 4739066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (!entity.isChunked() || ver.lessEquals(HttpVersion.HTTP_1_0)) { 4749066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project // if the content length is not known and is not chunk 4759066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project // encoded, the connection cannot be reused 4769066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return false; 4779066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 4789066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 4799066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 4809066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project // Check for 'Connection' directive 4819066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (connType == Headers.CONN_CLOSE) { 4829066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return false; 4839066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } else if (connType == Headers.CONN_KEEP_ALIVE) { 4849066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return true; 4859066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 4869066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project // Resorting to protocol version default close connection policy 4879066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return !ver.lessEquals(HttpVersion.HTTP_1_0); 4889066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 4899066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 4909066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project void setCanPersist(HttpEntity entity, ProtocolVersion ver, int connType) { 4919066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project mCanPersist = keepAlive(entity, ver, connType, mHttpContext); 4929066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 4939066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 4949066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project void setCanPersist(boolean canPersist) { 4959066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project mCanPersist = canPersist; 4969066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 4979066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 4989066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project boolean getCanPersist() { 4999066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return mCanPersist; 5009066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 5019066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 5029066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project /** typically http or https... set by subclass */ 5039066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project abstract String getScheme(); 5049066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project abstract void closeConnection(); 5059066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project abstract AndroidHttpClientConnection openConnection(Request req) throws IOException; 5069066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 5079066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project /** 5089066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * Prints request queue to log, for debugging. 5099066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * returns request count 5109066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project */ 5119066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project public synchronized String toString() { 5129066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return mHost.toString(); 5139066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 5149066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 5159066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project byte[] getBuf() { 5169066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (mBuf == null) mBuf = new byte[8192]; 5179066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return mBuf; 5189066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 5199066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 5209066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project} 521