Subversion Repositories SmartDukaan

Rev

Details | Last modification | View Log | RSS feed

Rev Author Line No. Line
21478 rajender 1
/*
2
 * Copyright (C) 2011 The Android Open Source Project
3
 *
4
 * Licensed under the Apache License, Version 2.0 (the "License");
5
 * you may not use this file except in compliance with the License.
6
 * You may obtain a copy of the License at
7
 *
8
 *      http://www.apache.org/licenses/LICENSE-2.0
9
 *
10
 * Unless required by applicable law or agreed to in writing, software
11
 * distributed under the License is distributed on an "AS IS" BASIS,
12
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13
 * See the License for the specific language governing permissions and
14
 * limitations under the License.
15
 */
16
 
17
package com.android.volley.toolbox;
18
 
19
import com.android.volley.AuthFailureError;
20
import com.android.volley.Request;
21
import com.android.volley.Request.Method;
22
 
23
import org.apache.http.Header;
24
import org.apache.http.HttpEntity;
25
import org.apache.http.HttpResponse;
26
import org.apache.http.HttpStatus;
27
import org.apache.http.ProtocolVersion;
28
import org.apache.http.StatusLine;
29
import org.apache.http.entity.BasicHttpEntity;
30
import org.apache.http.message.BasicHeader;
31
import org.apache.http.message.BasicHttpResponse;
32
import org.apache.http.message.BasicStatusLine;
33
 
34
import java.io.DataOutputStream;
35
import java.io.IOException;
36
import java.io.InputStream;
37
import java.net.HttpURLConnection;
38
import java.net.URL;
39
import java.util.HashMap;
40
import java.util.List;
41
import java.util.Map;
42
import java.util.Map.Entry;
43
 
44
import javax.net.ssl.HttpsURLConnection;
45
import javax.net.ssl.SSLSocketFactory;
46
 
47
/**
48
 * An {@link HttpStack} based on {@link HttpURLConnection}.
49
 */
50
public class HurlStack implements HttpStack {
51
 
52
    private static final String HEADER_CONTENT_TYPE = "Content-Type";
53
 
54
    /**
55
     * An interface for transforming URLs before use.
56
     */
57
    public interface UrlRewriter {
58
        /**
59
         * Returns a URL to use instead of the provided one, or null to indicate
60
         * this URL should not be used at all.
61
         */
62
        String rewriteUrl(String originalUrl);
63
    }
64
 
65
    private final UrlRewriter mUrlRewriter;
66
    private final SSLSocketFactory mSslSocketFactory;
67
 
68
    public HurlStack() {
69
        this(null);
70
    }
71
 
72
    /**
73
     * @param urlRewriter Rewriter to use for request URLs
74
     */
75
    public HurlStack(UrlRewriter urlRewriter) {
76
        this(urlRewriter, null);
77
    }
78
 
79
    /**
80
     * @param urlRewriter Rewriter to use for request URLs
81
     * @param sslSocketFactory SSL factory to use for HTTPS connections
82
     */
83
    public HurlStack(UrlRewriter urlRewriter, SSLSocketFactory sslSocketFactory) {
84
        mUrlRewriter = urlRewriter;
85
        mSslSocketFactory = sslSocketFactory;
86
    }
87
 
88
    @Override
89
    public HttpResponse performRequest(Request<?> request, Map<String, String> additionalHeaders)
90
            throws IOException, AuthFailureError {
91
        String url = request.getUrl();
92
        HashMap<String, String> map = new HashMap<String, String>();
93
        map.putAll(request.getHeaders());
94
        map.putAll(additionalHeaders);
95
        if (mUrlRewriter != null) {
96
            String rewritten = mUrlRewriter.rewriteUrl(url);
97
            if (rewritten == null) {
98
                throw new IOException("URL blocked by rewriter: " + url);
99
            }
100
            url = rewritten;
101
        }
102
        URL parsedUrl = new URL(url);
103
        HttpURLConnection connection = openConnection(parsedUrl, request);
104
        for (String headerName : map.keySet()) {
105
            connection.addRequestProperty(headerName, map.get(headerName));
106
        }
107
        setConnectionParametersForRequest(connection, request);
108
        // Initialize HttpResponse with data from the HttpURLConnection.
109
        ProtocolVersion protocolVersion = new ProtocolVersion("HTTP", 1, 1);
110
        int responseCode = connection.getResponseCode();
111
        if (responseCode == -1) {
112
            // -1 is returned by getResponseCode() if the response code could not be retrieved.
113
            // Signal to the caller that something was wrong with the connection.
114
            throw new IOException("Could not retrieve response code from HttpUrlConnection.");
115
        }
116
        StatusLine responseStatus = new BasicStatusLine(protocolVersion,
117
                connection.getResponseCode(), connection.getResponseMessage());
118
        BasicHttpResponse response = new BasicHttpResponse(responseStatus);
119
        if (hasResponseBody(request.getMethod(), responseStatus.getStatusCode())) {
120
            response.setEntity(entityFromConnection(connection));
121
        }
122
        for (Entry<String, List<String>> header : connection.getHeaderFields().entrySet()) {
123
            if (header.getKey() != null) {
124
                Header h = new BasicHeader(header.getKey(), header.getValue().get(0));
125
                response.addHeader(h);
126
            }
127
        }
128
        return response;
129
    }
130
 
131
    /**
132
     * Checks if a response message contains a body.
133
     * @see <a href="https://tools.ietf.org/html/rfc7230#section-3.3">RFC 7230 section 3.3</a>
134
     * @param requestMethod request method
135
     * @param responseCode response status code
136
     * @return whether the response has a body
137
     */
138
    private static boolean hasResponseBody(int requestMethod, int responseCode) {
139
        return requestMethod != Request.Method.HEAD
140
            && !(HttpStatus.SC_CONTINUE <= responseCode && responseCode < HttpStatus.SC_OK)
141
            && responseCode != HttpStatus.SC_NO_CONTENT
142
            && responseCode != HttpStatus.SC_NOT_MODIFIED;
143
    }
144
 
145
    /**
146
     * Initializes an {@link HttpEntity} from the given {@link HttpURLConnection}.
147
     * @param connection
148
     * @return an HttpEntity populated with data from <code>connection</code>.
149
     */
150
    private static HttpEntity entityFromConnection(HttpURLConnection connection) {
151
        BasicHttpEntity entity = new BasicHttpEntity();
152
        InputStream inputStream;
153
        try {
154
            inputStream = connection.getInputStream();
155
        } catch (IOException ioe) {
156
            inputStream = connection.getErrorStream();
157
        }
158
        entity.setContent(inputStream);
159
        entity.setContentLength(connection.getContentLength());
160
        entity.setContentEncoding(connection.getContentEncoding());
161
        entity.setContentType(connection.getContentType());
162
        return entity;
163
    }
164
 
165
    /**
166
     * Create an {@link HttpURLConnection} for the specified {@code url}.
167
     */
168
    protected HttpURLConnection createConnection(URL url) throws IOException {
169
        return (HttpURLConnection) url.openConnection();
170
    }
171
 
172
    /**
173
     * Opens an {@link HttpURLConnection} with parameters.
174
     * @param url
175
     * @return an open connection
176
     * @throws IOException
177
     */
178
    private HttpURLConnection openConnection(URL url, Request<?> request) throws IOException {
179
        HttpURLConnection connection = createConnection(url);
180
 
181
        int timeoutMs = request.getTimeoutMs();
182
        connection.setConnectTimeout(timeoutMs);
183
        connection.setReadTimeout(timeoutMs);
184
        connection.setUseCaches(false);
185
        connection.setDoInput(true);
186
 
187
        // use caller-provided custom SslSocketFactory, if any, for HTTPS
188
        if ("https".equals(url.getProtocol()) && mSslSocketFactory != null) {
189
            ((HttpsURLConnection)connection).setSSLSocketFactory(mSslSocketFactory);
190
        }
191
 
192
        return connection;
193
    }
194
 
195
    @SuppressWarnings("deprecation")
196
    /* package */ static void setConnectionParametersForRequest(HttpURLConnection connection,
197
            Request<?> request) throws IOException, AuthFailureError {
198
        switch (request.getMethod()) {
199
            case Method.DEPRECATED_GET_OR_POST:
200
                // This is the deprecated way that needs to be handled for backwards compatibility.
201
                // If the request's post body is null, then the assumption is that the request is
202
                // GET.  Otherwise, it is assumed that the request is a POST.
203
                byte[] postBody = request.getPostBody();
204
                if (postBody != null) {
205
                    // Prepare output. There is no need to set Content-Length explicitly,
206
                    // since this is handled by HttpURLConnection using the size of the prepared
207
                    // output stream.
208
                    connection.setDoOutput(true);
209
                    connection.setRequestMethod("POST");
210
                    connection.addRequestProperty(HEADER_CONTENT_TYPE,
211
                            request.getPostBodyContentType());
212
                    DataOutputStream out = new DataOutputStream(connection.getOutputStream());
213
                    out.write(postBody);
214
                    out.close();
215
                }
216
                break;
217
            case Method.GET:
218
                // Not necessary to set the request method because connection defaults to GET but
219
                // being explicit here.
220
                connection.setRequestMethod("GET");
221
                break;
222
            case Method.DELETE:
223
                connection.setRequestMethod("DELETE");
224
                break;
225
            case Method.POST:
226
                connection.setRequestMethod("POST");
227
                addBodyIfExists(connection, request);
228
                break;
229
            case Method.PUT:
230
                connection.setRequestMethod("PUT");
231
                addBodyIfExists(connection, request);
232
                break;
233
            case Method.HEAD:
234
                connection.setRequestMethod("HEAD");
235
                break;
236
            case Method.OPTIONS:
237
                connection.setRequestMethod("OPTIONS");
238
                break;
239
            case Method.TRACE:
240
                connection.setRequestMethod("TRACE");
241
                break;
242
            case Method.PATCH:
243
                connection.setRequestMethod("PATCH");
244
                addBodyIfExists(connection, request);
245
                break;
246
            default:
247
                throw new IllegalStateException("Unknown method type.");
248
        }
249
    }
250
 
251
    private static void addBodyIfExists(HttpURLConnection connection, Request<?> request)
252
            throws IOException, AuthFailureError {
253
        byte[] body = request.getBody();
254
        if (body != null) {
255
            connection.setDoOutput(true);
256
            connection.addRequestProperty(HEADER_CONTENT_TYPE, request.getBodyContentType());
257
            DataOutputStream out = new DataOutputStream(connection.getOutputStream());
258
            out.write(body);
259
            out.close();
260
        }
261
    }
262
}