1*6236dae4SAndroid Build Coastguard Worker /***************************************************************************
2*6236dae4SAndroid Build Coastguard Worker * _ _ ____ _
3*6236dae4SAndroid Build Coastguard Worker * Project ___| | | | _ \| |
4*6236dae4SAndroid Build Coastguard Worker * / __| | | | |_) | |
5*6236dae4SAndroid Build Coastguard Worker * | (__| |_| | _ <| |___
6*6236dae4SAndroid Build Coastguard Worker * \___|\___/|_| \_\_____|
7*6236dae4SAndroid Build Coastguard Worker *
8*6236dae4SAndroid Build Coastguard Worker * Copyright (C) Daniel Stenberg, <[email protected]>, et al.
9*6236dae4SAndroid Build Coastguard Worker *
10*6236dae4SAndroid Build Coastguard Worker * This software is licensed as described in the file COPYING, which
11*6236dae4SAndroid Build Coastguard Worker * you should have received as part of this distribution. The terms
12*6236dae4SAndroid Build Coastguard Worker * are also available at https://curl.se/docs/copyright.html.
13*6236dae4SAndroid Build Coastguard Worker *
14*6236dae4SAndroid Build Coastguard Worker * You may opt to use, copy, modify, merge, publish, distribute and/or sell
15*6236dae4SAndroid Build Coastguard Worker * copies of the Software, and permit persons to whom the Software is
16*6236dae4SAndroid Build Coastguard Worker * furnished to do so, under the terms of the COPYING file.
17*6236dae4SAndroid Build Coastguard Worker *
18*6236dae4SAndroid Build Coastguard Worker * This software is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY
19*6236dae4SAndroid Build Coastguard Worker * KIND, either express or implied.
20*6236dae4SAndroid Build Coastguard Worker *
21*6236dae4SAndroid Build Coastguard Worker * SPDX-License-Identifier: curl
22*6236dae4SAndroid Build Coastguard Worker *
23*6236dae4SAndroid Build Coastguard Worker ***************************************************************************/
24*6236dae4SAndroid Build Coastguard Worker
25*6236dae4SAndroid Build Coastguard Worker #include "curl_setup.h"
26*6236dae4SAndroid Build Coastguard Worker
27*6236dae4SAndroid Build Coastguard Worker #if !defined(CURL_DISABLE_PROXY) && !defined(CURL_DISABLE_HTTP)
28*6236dae4SAndroid Build Coastguard Worker
29*6236dae4SAndroid Build Coastguard Worker #include <curl/curl.h>
30*6236dae4SAndroid Build Coastguard Worker #ifdef USE_HYPER
31*6236dae4SAndroid Build Coastguard Worker #include <hyper.h>
32*6236dae4SAndroid Build Coastguard Worker #endif
33*6236dae4SAndroid Build Coastguard Worker #include "urldata.h"
34*6236dae4SAndroid Build Coastguard Worker #include "dynbuf.h"
35*6236dae4SAndroid Build Coastguard Worker #include "sendf.h"
36*6236dae4SAndroid Build Coastguard Worker #include "http.h"
37*6236dae4SAndroid Build Coastguard Worker #include "http1.h"
38*6236dae4SAndroid Build Coastguard Worker #include "http_proxy.h"
39*6236dae4SAndroid Build Coastguard Worker #include "url.h"
40*6236dae4SAndroid Build Coastguard Worker #include "select.h"
41*6236dae4SAndroid Build Coastguard Worker #include "progress.h"
42*6236dae4SAndroid Build Coastguard Worker #include "cfilters.h"
43*6236dae4SAndroid Build Coastguard Worker #include "cf-h1-proxy.h"
44*6236dae4SAndroid Build Coastguard Worker #include "connect.h"
45*6236dae4SAndroid Build Coastguard Worker #include "curl_trc.h"
46*6236dae4SAndroid Build Coastguard Worker #include "curlx.h"
47*6236dae4SAndroid Build Coastguard Worker #include "vtls/vtls.h"
48*6236dae4SAndroid Build Coastguard Worker #include "transfer.h"
49*6236dae4SAndroid Build Coastguard Worker #include "multiif.h"
50*6236dae4SAndroid Build Coastguard Worker
51*6236dae4SAndroid Build Coastguard Worker /* The last 3 #include files should be in this order */
52*6236dae4SAndroid Build Coastguard Worker #include "curl_printf.h"
53*6236dae4SAndroid Build Coastguard Worker #include "curl_memory.h"
54*6236dae4SAndroid Build Coastguard Worker #include "memdebug.h"
55*6236dae4SAndroid Build Coastguard Worker
56*6236dae4SAndroid Build Coastguard Worker
57*6236dae4SAndroid Build Coastguard Worker typedef enum {
58*6236dae4SAndroid Build Coastguard Worker H1_TUNNEL_INIT, /* init/default/no tunnel state */
59*6236dae4SAndroid Build Coastguard Worker H1_TUNNEL_CONNECT, /* CONNECT request is being send */
60*6236dae4SAndroid Build Coastguard Worker H1_TUNNEL_RECEIVE, /* CONNECT answer is being received */
61*6236dae4SAndroid Build Coastguard Worker H1_TUNNEL_RESPONSE, /* CONNECT response received completely */
62*6236dae4SAndroid Build Coastguard Worker H1_TUNNEL_ESTABLISHED,
63*6236dae4SAndroid Build Coastguard Worker H1_TUNNEL_FAILED
64*6236dae4SAndroid Build Coastguard Worker } h1_tunnel_state;
65*6236dae4SAndroid Build Coastguard Worker
66*6236dae4SAndroid Build Coastguard Worker /* struct for HTTP CONNECT tunneling */
67*6236dae4SAndroid Build Coastguard Worker struct h1_tunnel_state {
68*6236dae4SAndroid Build Coastguard Worker struct dynbuf rcvbuf;
69*6236dae4SAndroid Build Coastguard Worker struct dynbuf request_data;
70*6236dae4SAndroid Build Coastguard Worker size_t nsent;
71*6236dae4SAndroid Build Coastguard Worker size_t headerlines;
72*6236dae4SAndroid Build Coastguard Worker struct Curl_chunker ch;
73*6236dae4SAndroid Build Coastguard Worker enum keeponval {
74*6236dae4SAndroid Build Coastguard Worker KEEPON_DONE,
75*6236dae4SAndroid Build Coastguard Worker KEEPON_CONNECT,
76*6236dae4SAndroid Build Coastguard Worker KEEPON_IGNORE
77*6236dae4SAndroid Build Coastguard Worker } keepon;
78*6236dae4SAndroid Build Coastguard Worker curl_off_t cl; /* size of content to read and ignore */
79*6236dae4SAndroid Build Coastguard Worker h1_tunnel_state tunnel_state;
80*6236dae4SAndroid Build Coastguard Worker BIT(chunked_encoding);
81*6236dae4SAndroid Build Coastguard Worker BIT(close_connection);
82*6236dae4SAndroid Build Coastguard Worker };
83*6236dae4SAndroid Build Coastguard Worker
84*6236dae4SAndroid Build Coastguard Worker
tunnel_is_established(struct h1_tunnel_state * ts)85*6236dae4SAndroid Build Coastguard Worker static bool tunnel_is_established(struct h1_tunnel_state *ts)
86*6236dae4SAndroid Build Coastguard Worker {
87*6236dae4SAndroid Build Coastguard Worker return ts && (ts->tunnel_state == H1_TUNNEL_ESTABLISHED);
88*6236dae4SAndroid Build Coastguard Worker }
89*6236dae4SAndroid Build Coastguard Worker
tunnel_is_failed(struct h1_tunnel_state * ts)90*6236dae4SAndroid Build Coastguard Worker static bool tunnel_is_failed(struct h1_tunnel_state *ts)
91*6236dae4SAndroid Build Coastguard Worker {
92*6236dae4SAndroid Build Coastguard Worker return ts && (ts->tunnel_state == H1_TUNNEL_FAILED);
93*6236dae4SAndroid Build Coastguard Worker }
94*6236dae4SAndroid Build Coastguard Worker
tunnel_reinit(struct Curl_cfilter * cf,struct Curl_easy * data,struct h1_tunnel_state * ts)95*6236dae4SAndroid Build Coastguard Worker static CURLcode tunnel_reinit(struct Curl_cfilter *cf,
96*6236dae4SAndroid Build Coastguard Worker struct Curl_easy *data,
97*6236dae4SAndroid Build Coastguard Worker struct h1_tunnel_state *ts)
98*6236dae4SAndroid Build Coastguard Worker {
99*6236dae4SAndroid Build Coastguard Worker (void)data;
100*6236dae4SAndroid Build Coastguard Worker (void)cf;
101*6236dae4SAndroid Build Coastguard Worker DEBUGASSERT(ts);
102*6236dae4SAndroid Build Coastguard Worker Curl_dyn_reset(&ts->rcvbuf);
103*6236dae4SAndroid Build Coastguard Worker Curl_dyn_reset(&ts->request_data);
104*6236dae4SAndroid Build Coastguard Worker ts->tunnel_state = H1_TUNNEL_INIT;
105*6236dae4SAndroid Build Coastguard Worker ts->keepon = KEEPON_CONNECT;
106*6236dae4SAndroid Build Coastguard Worker ts->cl = 0;
107*6236dae4SAndroid Build Coastguard Worker ts->close_connection = FALSE;
108*6236dae4SAndroid Build Coastguard Worker return CURLE_OK;
109*6236dae4SAndroid Build Coastguard Worker }
110*6236dae4SAndroid Build Coastguard Worker
tunnel_init(struct Curl_cfilter * cf,struct Curl_easy * data,struct h1_tunnel_state ** pts)111*6236dae4SAndroid Build Coastguard Worker static CURLcode tunnel_init(struct Curl_cfilter *cf,
112*6236dae4SAndroid Build Coastguard Worker struct Curl_easy *data,
113*6236dae4SAndroid Build Coastguard Worker struct h1_tunnel_state **pts)
114*6236dae4SAndroid Build Coastguard Worker {
115*6236dae4SAndroid Build Coastguard Worker struct h1_tunnel_state *ts;
116*6236dae4SAndroid Build Coastguard Worker
117*6236dae4SAndroid Build Coastguard Worker if(cf->conn->handler->flags & PROTOPT_NOTCPPROXY) {
118*6236dae4SAndroid Build Coastguard Worker failf(data, "%s cannot be done over CONNECT", cf->conn->handler->scheme);
119*6236dae4SAndroid Build Coastguard Worker return CURLE_UNSUPPORTED_PROTOCOL;
120*6236dae4SAndroid Build Coastguard Worker }
121*6236dae4SAndroid Build Coastguard Worker
122*6236dae4SAndroid Build Coastguard Worker ts = calloc(1, sizeof(*ts));
123*6236dae4SAndroid Build Coastguard Worker if(!ts)
124*6236dae4SAndroid Build Coastguard Worker return CURLE_OUT_OF_MEMORY;
125*6236dae4SAndroid Build Coastguard Worker
126*6236dae4SAndroid Build Coastguard Worker infof(data, "allocate connect buffer");
127*6236dae4SAndroid Build Coastguard Worker
128*6236dae4SAndroid Build Coastguard Worker Curl_dyn_init(&ts->rcvbuf, DYN_PROXY_CONNECT_HEADERS);
129*6236dae4SAndroid Build Coastguard Worker Curl_dyn_init(&ts->request_data, DYN_HTTP_REQUEST);
130*6236dae4SAndroid Build Coastguard Worker Curl_httpchunk_init(data, &ts->ch, TRUE);
131*6236dae4SAndroid Build Coastguard Worker
132*6236dae4SAndroid Build Coastguard Worker *pts = ts;
133*6236dae4SAndroid Build Coastguard Worker connkeep(cf->conn, "HTTP proxy CONNECT");
134*6236dae4SAndroid Build Coastguard Worker return tunnel_reinit(cf, data, ts);
135*6236dae4SAndroid Build Coastguard Worker }
136*6236dae4SAndroid Build Coastguard Worker
h1_tunnel_go_state(struct Curl_cfilter * cf,struct h1_tunnel_state * ts,h1_tunnel_state new_state,struct Curl_easy * data)137*6236dae4SAndroid Build Coastguard Worker static void h1_tunnel_go_state(struct Curl_cfilter *cf,
138*6236dae4SAndroid Build Coastguard Worker struct h1_tunnel_state *ts,
139*6236dae4SAndroid Build Coastguard Worker h1_tunnel_state new_state,
140*6236dae4SAndroid Build Coastguard Worker struct Curl_easy *data)
141*6236dae4SAndroid Build Coastguard Worker {
142*6236dae4SAndroid Build Coastguard Worker if(ts->tunnel_state == new_state)
143*6236dae4SAndroid Build Coastguard Worker return;
144*6236dae4SAndroid Build Coastguard Worker /* entering this one */
145*6236dae4SAndroid Build Coastguard Worker switch(new_state) {
146*6236dae4SAndroid Build Coastguard Worker case H1_TUNNEL_INIT:
147*6236dae4SAndroid Build Coastguard Worker CURL_TRC_CF(data, cf, "new tunnel state 'init'");
148*6236dae4SAndroid Build Coastguard Worker tunnel_reinit(cf, data, ts);
149*6236dae4SAndroid Build Coastguard Worker break;
150*6236dae4SAndroid Build Coastguard Worker
151*6236dae4SAndroid Build Coastguard Worker case H1_TUNNEL_CONNECT:
152*6236dae4SAndroid Build Coastguard Worker CURL_TRC_CF(data, cf, "new tunnel state 'connect'");
153*6236dae4SAndroid Build Coastguard Worker ts->tunnel_state = H1_TUNNEL_CONNECT;
154*6236dae4SAndroid Build Coastguard Worker ts->keepon = KEEPON_CONNECT;
155*6236dae4SAndroid Build Coastguard Worker Curl_dyn_reset(&ts->rcvbuf);
156*6236dae4SAndroid Build Coastguard Worker break;
157*6236dae4SAndroid Build Coastguard Worker
158*6236dae4SAndroid Build Coastguard Worker case H1_TUNNEL_RECEIVE:
159*6236dae4SAndroid Build Coastguard Worker CURL_TRC_CF(data, cf, "new tunnel state 'receive'");
160*6236dae4SAndroid Build Coastguard Worker ts->tunnel_state = H1_TUNNEL_RECEIVE;
161*6236dae4SAndroid Build Coastguard Worker break;
162*6236dae4SAndroid Build Coastguard Worker
163*6236dae4SAndroid Build Coastguard Worker case H1_TUNNEL_RESPONSE:
164*6236dae4SAndroid Build Coastguard Worker CURL_TRC_CF(data, cf, "new tunnel state 'response'");
165*6236dae4SAndroid Build Coastguard Worker ts->tunnel_state = H1_TUNNEL_RESPONSE;
166*6236dae4SAndroid Build Coastguard Worker break;
167*6236dae4SAndroid Build Coastguard Worker
168*6236dae4SAndroid Build Coastguard Worker case H1_TUNNEL_ESTABLISHED:
169*6236dae4SAndroid Build Coastguard Worker CURL_TRC_CF(data, cf, "new tunnel state 'established'");
170*6236dae4SAndroid Build Coastguard Worker infof(data, "CONNECT phase completed");
171*6236dae4SAndroid Build Coastguard Worker data->state.authproxy.done = TRUE;
172*6236dae4SAndroid Build Coastguard Worker data->state.authproxy.multipass = FALSE;
173*6236dae4SAndroid Build Coastguard Worker FALLTHROUGH();
174*6236dae4SAndroid Build Coastguard Worker case H1_TUNNEL_FAILED:
175*6236dae4SAndroid Build Coastguard Worker if(new_state == H1_TUNNEL_FAILED)
176*6236dae4SAndroid Build Coastguard Worker CURL_TRC_CF(data, cf, "new tunnel state 'failed'");
177*6236dae4SAndroid Build Coastguard Worker ts->tunnel_state = new_state;
178*6236dae4SAndroid Build Coastguard Worker Curl_dyn_reset(&ts->rcvbuf);
179*6236dae4SAndroid Build Coastguard Worker Curl_dyn_reset(&ts->request_data);
180*6236dae4SAndroid Build Coastguard Worker /* restore the protocol pointer */
181*6236dae4SAndroid Build Coastguard Worker data->info.httpcode = 0; /* clear it as it might've been used for the
182*6236dae4SAndroid Build Coastguard Worker proxy */
183*6236dae4SAndroid Build Coastguard Worker /* If a proxy-authorization header was used for the proxy, then we should
184*6236dae4SAndroid Build Coastguard Worker make sure that it is not accidentally used for the document request
185*6236dae4SAndroid Build Coastguard Worker after we have connected. So let's free and clear it here. */
186*6236dae4SAndroid Build Coastguard Worker Curl_safefree(data->state.aptr.proxyuserpwd);
187*6236dae4SAndroid Build Coastguard Worker #ifdef USE_HYPER
188*6236dae4SAndroid Build Coastguard Worker data->state.hconnect = FALSE;
189*6236dae4SAndroid Build Coastguard Worker #endif
190*6236dae4SAndroid Build Coastguard Worker break;
191*6236dae4SAndroid Build Coastguard Worker }
192*6236dae4SAndroid Build Coastguard Worker }
193*6236dae4SAndroid Build Coastguard Worker
tunnel_free(struct Curl_cfilter * cf,struct Curl_easy * data)194*6236dae4SAndroid Build Coastguard Worker static void tunnel_free(struct Curl_cfilter *cf,
195*6236dae4SAndroid Build Coastguard Worker struct Curl_easy *data)
196*6236dae4SAndroid Build Coastguard Worker {
197*6236dae4SAndroid Build Coastguard Worker if(cf) {
198*6236dae4SAndroid Build Coastguard Worker struct h1_tunnel_state *ts = cf->ctx;
199*6236dae4SAndroid Build Coastguard Worker if(ts) {
200*6236dae4SAndroid Build Coastguard Worker h1_tunnel_go_state(cf, ts, H1_TUNNEL_FAILED, data);
201*6236dae4SAndroid Build Coastguard Worker Curl_dyn_free(&ts->rcvbuf);
202*6236dae4SAndroid Build Coastguard Worker Curl_dyn_free(&ts->request_data);
203*6236dae4SAndroid Build Coastguard Worker Curl_httpchunk_free(data, &ts->ch);
204*6236dae4SAndroid Build Coastguard Worker free(ts);
205*6236dae4SAndroid Build Coastguard Worker cf->ctx = NULL;
206*6236dae4SAndroid Build Coastguard Worker }
207*6236dae4SAndroid Build Coastguard Worker }
208*6236dae4SAndroid Build Coastguard Worker }
209*6236dae4SAndroid Build Coastguard Worker
tunnel_want_send(struct h1_tunnel_state * ts)210*6236dae4SAndroid Build Coastguard Worker static bool tunnel_want_send(struct h1_tunnel_state *ts)
211*6236dae4SAndroid Build Coastguard Worker {
212*6236dae4SAndroid Build Coastguard Worker return (ts->tunnel_state == H1_TUNNEL_CONNECT);
213*6236dae4SAndroid Build Coastguard Worker }
214*6236dae4SAndroid Build Coastguard Worker
215*6236dae4SAndroid Build Coastguard Worker #ifndef USE_HYPER
start_CONNECT(struct Curl_cfilter * cf,struct Curl_easy * data,struct h1_tunnel_state * ts)216*6236dae4SAndroid Build Coastguard Worker static CURLcode start_CONNECT(struct Curl_cfilter *cf,
217*6236dae4SAndroid Build Coastguard Worker struct Curl_easy *data,
218*6236dae4SAndroid Build Coastguard Worker struct h1_tunnel_state *ts)
219*6236dae4SAndroid Build Coastguard Worker {
220*6236dae4SAndroid Build Coastguard Worker struct httpreq *req = NULL;
221*6236dae4SAndroid Build Coastguard Worker int http_minor;
222*6236dae4SAndroid Build Coastguard Worker CURLcode result;
223*6236dae4SAndroid Build Coastguard Worker
224*6236dae4SAndroid Build Coastguard Worker /* This only happens if we have looped here due to authentication
225*6236dae4SAndroid Build Coastguard Worker reasons, and we do not really use the newly cloned URL here
226*6236dae4SAndroid Build Coastguard Worker then. Just free() it. */
227*6236dae4SAndroid Build Coastguard Worker Curl_safefree(data->req.newurl);
228*6236dae4SAndroid Build Coastguard Worker
229*6236dae4SAndroid Build Coastguard Worker result = Curl_http_proxy_create_CONNECT(&req, cf, data, 1);
230*6236dae4SAndroid Build Coastguard Worker if(result)
231*6236dae4SAndroid Build Coastguard Worker goto out;
232*6236dae4SAndroid Build Coastguard Worker
233*6236dae4SAndroid Build Coastguard Worker infof(data, "Establish HTTP proxy tunnel to %s", req->authority);
234*6236dae4SAndroid Build Coastguard Worker
235*6236dae4SAndroid Build Coastguard Worker Curl_dyn_reset(&ts->request_data);
236*6236dae4SAndroid Build Coastguard Worker ts->nsent = 0;
237*6236dae4SAndroid Build Coastguard Worker ts->headerlines = 0;
238*6236dae4SAndroid Build Coastguard Worker http_minor = (cf->conn->http_proxy.proxytype == CURLPROXY_HTTP_1_0) ? 0 : 1;
239*6236dae4SAndroid Build Coastguard Worker
240*6236dae4SAndroid Build Coastguard Worker result = Curl_h1_req_write_head(req, http_minor, &ts->request_data);
241*6236dae4SAndroid Build Coastguard Worker if(!result)
242*6236dae4SAndroid Build Coastguard Worker result = Curl_creader_set_null(data);
243*6236dae4SAndroid Build Coastguard Worker
244*6236dae4SAndroid Build Coastguard Worker out:
245*6236dae4SAndroid Build Coastguard Worker if(result)
246*6236dae4SAndroid Build Coastguard Worker failf(data, "Failed sending CONNECT to proxy");
247*6236dae4SAndroid Build Coastguard Worker if(req)
248*6236dae4SAndroid Build Coastguard Worker Curl_http_req_free(req);
249*6236dae4SAndroid Build Coastguard Worker return result;
250*6236dae4SAndroid Build Coastguard Worker }
251*6236dae4SAndroid Build Coastguard Worker
send_CONNECT(struct Curl_cfilter * cf,struct Curl_easy * data,struct h1_tunnel_state * ts,bool * done)252*6236dae4SAndroid Build Coastguard Worker static CURLcode send_CONNECT(struct Curl_cfilter *cf,
253*6236dae4SAndroid Build Coastguard Worker struct Curl_easy *data,
254*6236dae4SAndroid Build Coastguard Worker struct h1_tunnel_state *ts,
255*6236dae4SAndroid Build Coastguard Worker bool *done)
256*6236dae4SAndroid Build Coastguard Worker {
257*6236dae4SAndroid Build Coastguard Worker char *buf = Curl_dyn_ptr(&ts->request_data);
258*6236dae4SAndroid Build Coastguard Worker size_t request_len = Curl_dyn_len(&ts->request_data);
259*6236dae4SAndroid Build Coastguard Worker size_t blen = request_len;
260*6236dae4SAndroid Build Coastguard Worker CURLcode result = CURLE_OK;
261*6236dae4SAndroid Build Coastguard Worker ssize_t nwritten;
262*6236dae4SAndroid Build Coastguard Worker
263*6236dae4SAndroid Build Coastguard Worker if(blen <= ts->nsent)
264*6236dae4SAndroid Build Coastguard Worker goto out; /* we are done */
265*6236dae4SAndroid Build Coastguard Worker
266*6236dae4SAndroid Build Coastguard Worker blen -= ts->nsent;
267*6236dae4SAndroid Build Coastguard Worker buf += ts->nsent;
268*6236dae4SAndroid Build Coastguard Worker
269*6236dae4SAndroid Build Coastguard Worker nwritten = cf->next->cft->do_send(cf->next, data, buf, blen, FALSE, &result);
270*6236dae4SAndroid Build Coastguard Worker if(nwritten < 0) {
271*6236dae4SAndroid Build Coastguard Worker if(result == CURLE_AGAIN) {
272*6236dae4SAndroid Build Coastguard Worker result = CURLE_OK;
273*6236dae4SAndroid Build Coastguard Worker }
274*6236dae4SAndroid Build Coastguard Worker goto out;
275*6236dae4SAndroid Build Coastguard Worker }
276*6236dae4SAndroid Build Coastguard Worker
277*6236dae4SAndroid Build Coastguard Worker DEBUGASSERT(blen >= (size_t)nwritten);
278*6236dae4SAndroid Build Coastguard Worker ts->nsent += (size_t)nwritten;
279*6236dae4SAndroid Build Coastguard Worker Curl_debug(data, CURLINFO_HEADER_OUT, buf, (size_t)nwritten);
280*6236dae4SAndroid Build Coastguard Worker
281*6236dae4SAndroid Build Coastguard Worker out:
282*6236dae4SAndroid Build Coastguard Worker if(result)
283*6236dae4SAndroid Build Coastguard Worker failf(data, "Failed sending CONNECT to proxy");
284*6236dae4SAndroid Build Coastguard Worker *done = (!result && (ts->nsent >= request_len));
285*6236dae4SAndroid Build Coastguard Worker return result;
286*6236dae4SAndroid Build Coastguard Worker }
287*6236dae4SAndroid Build Coastguard Worker
on_resp_header(struct Curl_cfilter * cf,struct Curl_easy * data,struct h1_tunnel_state * ts,const char * header)288*6236dae4SAndroid Build Coastguard Worker static CURLcode on_resp_header(struct Curl_cfilter *cf,
289*6236dae4SAndroid Build Coastguard Worker struct Curl_easy *data,
290*6236dae4SAndroid Build Coastguard Worker struct h1_tunnel_state *ts,
291*6236dae4SAndroid Build Coastguard Worker const char *header)
292*6236dae4SAndroid Build Coastguard Worker {
293*6236dae4SAndroid Build Coastguard Worker CURLcode result = CURLE_OK;
294*6236dae4SAndroid Build Coastguard Worker struct SingleRequest *k = &data->req;
295*6236dae4SAndroid Build Coastguard Worker (void)cf;
296*6236dae4SAndroid Build Coastguard Worker
297*6236dae4SAndroid Build Coastguard Worker if((checkprefix("WWW-Authenticate:", header) &&
298*6236dae4SAndroid Build Coastguard Worker (401 == k->httpcode)) ||
299*6236dae4SAndroid Build Coastguard Worker (checkprefix("Proxy-authenticate:", header) &&
300*6236dae4SAndroid Build Coastguard Worker (407 == k->httpcode))) {
301*6236dae4SAndroid Build Coastguard Worker
302*6236dae4SAndroid Build Coastguard Worker bool proxy = (k->httpcode == 407);
303*6236dae4SAndroid Build Coastguard Worker char *auth = Curl_copy_header_value(header);
304*6236dae4SAndroid Build Coastguard Worker if(!auth)
305*6236dae4SAndroid Build Coastguard Worker return CURLE_OUT_OF_MEMORY;
306*6236dae4SAndroid Build Coastguard Worker
307*6236dae4SAndroid Build Coastguard Worker CURL_TRC_CF(data, cf, "CONNECT: fwd auth header '%s'", header);
308*6236dae4SAndroid Build Coastguard Worker result = Curl_http_input_auth(data, proxy, auth);
309*6236dae4SAndroid Build Coastguard Worker
310*6236dae4SAndroid Build Coastguard Worker free(auth);
311*6236dae4SAndroid Build Coastguard Worker
312*6236dae4SAndroid Build Coastguard Worker if(result)
313*6236dae4SAndroid Build Coastguard Worker return result;
314*6236dae4SAndroid Build Coastguard Worker }
315*6236dae4SAndroid Build Coastguard Worker else if(checkprefix("Content-Length:", header)) {
316*6236dae4SAndroid Build Coastguard Worker if(k->httpcode/100 == 2) {
317*6236dae4SAndroid Build Coastguard Worker /* A client MUST ignore any Content-Length or Transfer-Encoding
318*6236dae4SAndroid Build Coastguard Worker header fields received in a successful response to CONNECT.
319*6236dae4SAndroid Build Coastguard Worker "Successful" described as: 2xx (Successful). RFC 7231 4.3.6 */
320*6236dae4SAndroid Build Coastguard Worker infof(data, "Ignoring Content-Length in CONNECT %03d response",
321*6236dae4SAndroid Build Coastguard Worker k->httpcode);
322*6236dae4SAndroid Build Coastguard Worker }
323*6236dae4SAndroid Build Coastguard Worker else {
324*6236dae4SAndroid Build Coastguard Worker (void)curlx_strtoofft(header + strlen("Content-Length:"),
325*6236dae4SAndroid Build Coastguard Worker NULL, 10, &ts->cl);
326*6236dae4SAndroid Build Coastguard Worker }
327*6236dae4SAndroid Build Coastguard Worker }
328*6236dae4SAndroid Build Coastguard Worker else if(Curl_compareheader(header,
329*6236dae4SAndroid Build Coastguard Worker STRCONST("Connection:"), STRCONST("close")))
330*6236dae4SAndroid Build Coastguard Worker ts->close_connection = TRUE;
331*6236dae4SAndroid Build Coastguard Worker else if(checkprefix("Transfer-Encoding:", header)) {
332*6236dae4SAndroid Build Coastguard Worker if(k->httpcode/100 == 2) {
333*6236dae4SAndroid Build Coastguard Worker /* A client MUST ignore any Content-Length or Transfer-Encoding
334*6236dae4SAndroid Build Coastguard Worker header fields received in a successful response to CONNECT.
335*6236dae4SAndroid Build Coastguard Worker "Successful" described as: 2xx (Successful). RFC 7231 4.3.6 */
336*6236dae4SAndroid Build Coastguard Worker infof(data, "Ignoring Transfer-Encoding in "
337*6236dae4SAndroid Build Coastguard Worker "CONNECT %03d response", k->httpcode);
338*6236dae4SAndroid Build Coastguard Worker }
339*6236dae4SAndroid Build Coastguard Worker else if(Curl_compareheader(header,
340*6236dae4SAndroid Build Coastguard Worker STRCONST("Transfer-Encoding:"),
341*6236dae4SAndroid Build Coastguard Worker STRCONST("chunked"))) {
342*6236dae4SAndroid Build Coastguard Worker infof(data, "CONNECT responded chunked");
343*6236dae4SAndroid Build Coastguard Worker ts->chunked_encoding = TRUE;
344*6236dae4SAndroid Build Coastguard Worker /* reset our chunky engine */
345*6236dae4SAndroid Build Coastguard Worker Curl_httpchunk_reset(data, &ts->ch, TRUE);
346*6236dae4SAndroid Build Coastguard Worker }
347*6236dae4SAndroid Build Coastguard Worker }
348*6236dae4SAndroid Build Coastguard Worker else if(Curl_compareheader(header,
349*6236dae4SAndroid Build Coastguard Worker STRCONST("Proxy-Connection:"),
350*6236dae4SAndroid Build Coastguard Worker STRCONST("close")))
351*6236dae4SAndroid Build Coastguard Worker ts->close_connection = TRUE;
352*6236dae4SAndroid Build Coastguard Worker else if(!strncmp(header, "HTTP/1.", 7) &&
353*6236dae4SAndroid Build Coastguard Worker ((header[7] == '0') || (header[7] == '1')) &&
354*6236dae4SAndroid Build Coastguard Worker (header[8] == ' ') &&
355*6236dae4SAndroid Build Coastguard Worker ISDIGIT(header[9]) && ISDIGIT(header[10]) && ISDIGIT(header[11]) &&
356*6236dae4SAndroid Build Coastguard Worker !ISDIGIT(header[12])) {
357*6236dae4SAndroid Build Coastguard Worker /* store the HTTP code from the proxy */
358*6236dae4SAndroid Build Coastguard Worker data->info.httpproxycode = k->httpcode = (header[9] - '0') * 100 +
359*6236dae4SAndroid Build Coastguard Worker (header[10] - '0') * 10 + (header[11] - '0');
360*6236dae4SAndroid Build Coastguard Worker }
361*6236dae4SAndroid Build Coastguard Worker return result;
362*6236dae4SAndroid Build Coastguard Worker }
363*6236dae4SAndroid Build Coastguard Worker
recv_CONNECT_resp(struct Curl_cfilter * cf,struct Curl_easy * data,struct h1_tunnel_state * ts,bool * done)364*6236dae4SAndroid Build Coastguard Worker static CURLcode recv_CONNECT_resp(struct Curl_cfilter *cf,
365*6236dae4SAndroid Build Coastguard Worker struct Curl_easy *data,
366*6236dae4SAndroid Build Coastguard Worker struct h1_tunnel_state *ts,
367*6236dae4SAndroid Build Coastguard Worker bool *done)
368*6236dae4SAndroid Build Coastguard Worker {
369*6236dae4SAndroid Build Coastguard Worker CURLcode result = CURLE_OK;
370*6236dae4SAndroid Build Coastguard Worker struct SingleRequest *k = &data->req;
371*6236dae4SAndroid Build Coastguard Worker char *linep;
372*6236dae4SAndroid Build Coastguard Worker size_t line_len;
373*6236dae4SAndroid Build Coastguard Worker int error, writetype;
374*6236dae4SAndroid Build Coastguard Worker
375*6236dae4SAndroid Build Coastguard Worker #define SELECT_OK 0
376*6236dae4SAndroid Build Coastguard Worker #define SELECT_ERROR 1
377*6236dae4SAndroid Build Coastguard Worker
378*6236dae4SAndroid Build Coastguard Worker error = SELECT_OK;
379*6236dae4SAndroid Build Coastguard Worker *done = FALSE;
380*6236dae4SAndroid Build Coastguard Worker
381*6236dae4SAndroid Build Coastguard Worker if(!Curl_conn_data_pending(data, cf->sockindex))
382*6236dae4SAndroid Build Coastguard Worker return CURLE_OK;
383*6236dae4SAndroid Build Coastguard Worker
384*6236dae4SAndroid Build Coastguard Worker while(ts->keepon) {
385*6236dae4SAndroid Build Coastguard Worker ssize_t nread;
386*6236dae4SAndroid Build Coastguard Worker char byte;
387*6236dae4SAndroid Build Coastguard Worker
388*6236dae4SAndroid Build Coastguard Worker /* Read one byte at a time to avoid a race condition. Wait at most one
389*6236dae4SAndroid Build Coastguard Worker second before looping to ensure continuous pgrsUpdates. */
390*6236dae4SAndroid Build Coastguard Worker result = Curl_conn_recv(data, cf->sockindex, &byte, 1, &nread);
391*6236dae4SAndroid Build Coastguard Worker if(result == CURLE_AGAIN)
392*6236dae4SAndroid Build Coastguard Worker /* socket buffer drained, return */
393*6236dae4SAndroid Build Coastguard Worker return CURLE_OK;
394*6236dae4SAndroid Build Coastguard Worker
395*6236dae4SAndroid Build Coastguard Worker if(Curl_pgrsUpdate(data))
396*6236dae4SAndroid Build Coastguard Worker return CURLE_ABORTED_BY_CALLBACK;
397*6236dae4SAndroid Build Coastguard Worker
398*6236dae4SAndroid Build Coastguard Worker if(result) {
399*6236dae4SAndroid Build Coastguard Worker ts->keepon = KEEPON_DONE;
400*6236dae4SAndroid Build Coastguard Worker break;
401*6236dae4SAndroid Build Coastguard Worker }
402*6236dae4SAndroid Build Coastguard Worker
403*6236dae4SAndroid Build Coastguard Worker if(nread <= 0) {
404*6236dae4SAndroid Build Coastguard Worker if(data->set.proxyauth && data->state.authproxy.avail &&
405*6236dae4SAndroid Build Coastguard Worker data->state.aptr.proxyuserpwd) {
406*6236dae4SAndroid Build Coastguard Worker /* proxy auth was requested and there was proxy auth available,
407*6236dae4SAndroid Build Coastguard Worker then deem this as "mere" proxy disconnect */
408*6236dae4SAndroid Build Coastguard Worker ts->close_connection = TRUE;
409*6236dae4SAndroid Build Coastguard Worker infof(data, "Proxy CONNECT connection closed");
410*6236dae4SAndroid Build Coastguard Worker }
411*6236dae4SAndroid Build Coastguard Worker else {
412*6236dae4SAndroid Build Coastguard Worker error = SELECT_ERROR;
413*6236dae4SAndroid Build Coastguard Worker failf(data, "Proxy CONNECT aborted");
414*6236dae4SAndroid Build Coastguard Worker }
415*6236dae4SAndroid Build Coastguard Worker ts->keepon = KEEPON_DONE;
416*6236dae4SAndroid Build Coastguard Worker break;
417*6236dae4SAndroid Build Coastguard Worker }
418*6236dae4SAndroid Build Coastguard Worker
419*6236dae4SAndroid Build Coastguard Worker if(ts->keepon == KEEPON_IGNORE) {
420*6236dae4SAndroid Build Coastguard Worker /* This means we are currently ignoring a response-body */
421*6236dae4SAndroid Build Coastguard Worker
422*6236dae4SAndroid Build Coastguard Worker if(ts->cl) {
423*6236dae4SAndroid Build Coastguard Worker /* A Content-Length based body: simply count down the counter
424*6236dae4SAndroid Build Coastguard Worker and make sure to break out of the loop when we are done! */
425*6236dae4SAndroid Build Coastguard Worker ts->cl--;
426*6236dae4SAndroid Build Coastguard Worker if(ts->cl <= 0) {
427*6236dae4SAndroid Build Coastguard Worker ts->keepon = KEEPON_DONE;
428*6236dae4SAndroid Build Coastguard Worker break;
429*6236dae4SAndroid Build Coastguard Worker }
430*6236dae4SAndroid Build Coastguard Worker }
431*6236dae4SAndroid Build Coastguard Worker else if(ts->chunked_encoding) {
432*6236dae4SAndroid Build Coastguard Worker /* chunked-encoded body, so we need to do the chunked dance
433*6236dae4SAndroid Build Coastguard Worker properly to know when the end of the body is reached */
434*6236dae4SAndroid Build Coastguard Worker size_t consumed = 0;
435*6236dae4SAndroid Build Coastguard Worker
436*6236dae4SAndroid Build Coastguard Worker /* now parse the chunked piece of data so that we can
437*6236dae4SAndroid Build Coastguard Worker properly tell when the stream ends */
438*6236dae4SAndroid Build Coastguard Worker result = Curl_httpchunk_read(data, &ts->ch, &byte, 1, &consumed);
439*6236dae4SAndroid Build Coastguard Worker if(result)
440*6236dae4SAndroid Build Coastguard Worker return result;
441*6236dae4SAndroid Build Coastguard Worker if(Curl_httpchunk_is_done(data, &ts->ch)) {
442*6236dae4SAndroid Build Coastguard Worker /* we are done reading chunks! */
443*6236dae4SAndroid Build Coastguard Worker infof(data, "chunk reading DONE");
444*6236dae4SAndroid Build Coastguard Worker ts->keepon = KEEPON_DONE;
445*6236dae4SAndroid Build Coastguard Worker }
446*6236dae4SAndroid Build Coastguard Worker }
447*6236dae4SAndroid Build Coastguard Worker continue;
448*6236dae4SAndroid Build Coastguard Worker }
449*6236dae4SAndroid Build Coastguard Worker
450*6236dae4SAndroid Build Coastguard Worker if(Curl_dyn_addn(&ts->rcvbuf, &byte, 1)) {
451*6236dae4SAndroid Build Coastguard Worker failf(data, "CONNECT response too large");
452*6236dae4SAndroid Build Coastguard Worker return CURLE_RECV_ERROR;
453*6236dae4SAndroid Build Coastguard Worker }
454*6236dae4SAndroid Build Coastguard Worker
455*6236dae4SAndroid Build Coastguard Worker /* if this is not the end of a header line then continue */
456*6236dae4SAndroid Build Coastguard Worker if(byte != 0x0a)
457*6236dae4SAndroid Build Coastguard Worker continue;
458*6236dae4SAndroid Build Coastguard Worker
459*6236dae4SAndroid Build Coastguard Worker ts->headerlines++;
460*6236dae4SAndroid Build Coastguard Worker linep = Curl_dyn_ptr(&ts->rcvbuf);
461*6236dae4SAndroid Build Coastguard Worker line_len = Curl_dyn_len(&ts->rcvbuf); /* amount of bytes in this line */
462*6236dae4SAndroid Build Coastguard Worker
463*6236dae4SAndroid Build Coastguard Worker /* output debug if that is requested */
464*6236dae4SAndroid Build Coastguard Worker Curl_debug(data, CURLINFO_HEADER_IN, linep, line_len);
465*6236dae4SAndroid Build Coastguard Worker
466*6236dae4SAndroid Build Coastguard Worker /* send the header to the callback */
467*6236dae4SAndroid Build Coastguard Worker writetype = CLIENTWRITE_HEADER | CLIENTWRITE_CONNECT |
468*6236dae4SAndroid Build Coastguard Worker (ts->headerlines == 1 ? CLIENTWRITE_STATUS : 0);
469*6236dae4SAndroid Build Coastguard Worker result = Curl_client_write(data, writetype, linep, line_len);
470*6236dae4SAndroid Build Coastguard Worker if(result)
471*6236dae4SAndroid Build Coastguard Worker return result;
472*6236dae4SAndroid Build Coastguard Worker
473*6236dae4SAndroid Build Coastguard Worker result = Curl_bump_headersize(data, line_len, TRUE);
474*6236dae4SAndroid Build Coastguard Worker if(result)
475*6236dae4SAndroid Build Coastguard Worker return result;
476*6236dae4SAndroid Build Coastguard Worker
477*6236dae4SAndroid Build Coastguard Worker /* Newlines are CRLF, so the CR is ignored as the line is not
478*6236dae4SAndroid Build Coastguard Worker really terminated until the LF comes. Treat a following CR
479*6236dae4SAndroid Build Coastguard Worker as end-of-headers as well.*/
480*6236dae4SAndroid Build Coastguard Worker
481*6236dae4SAndroid Build Coastguard Worker if(('\r' == linep[0]) ||
482*6236dae4SAndroid Build Coastguard Worker ('\n' == linep[0])) {
483*6236dae4SAndroid Build Coastguard Worker /* end of response-headers from the proxy */
484*6236dae4SAndroid Build Coastguard Worker
485*6236dae4SAndroid Build Coastguard Worker if((407 == k->httpcode) && !data->state.authproblem) {
486*6236dae4SAndroid Build Coastguard Worker /* If we get a 407 response code with content length
487*6236dae4SAndroid Build Coastguard Worker when we have no auth problem, we must ignore the
488*6236dae4SAndroid Build Coastguard Worker whole response-body */
489*6236dae4SAndroid Build Coastguard Worker ts->keepon = KEEPON_IGNORE;
490*6236dae4SAndroid Build Coastguard Worker
491*6236dae4SAndroid Build Coastguard Worker if(ts->cl) {
492*6236dae4SAndroid Build Coastguard Worker infof(data, "Ignore %" FMT_OFF_T " bytes of response-body", ts->cl);
493*6236dae4SAndroid Build Coastguard Worker }
494*6236dae4SAndroid Build Coastguard Worker else if(ts->chunked_encoding) {
495*6236dae4SAndroid Build Coastguard Worker infof(data, "Ignore chunked response-body");
496*6236dae4SAndroid Build Coastguard Worker }
497*6236dae4SAndroid Build Coastguard Worker else {
498*6236dae4SAndroid Build Coastguard Worker /* without content-length or chunked encoding, we
499*6236dae4SAndroid Build Coastguard Worker cannot keep the connection alive since the close is
500*6236dae4SAndroid Build Coastguard Worker the end signal so we bail out at once instead */
501*6236dae4SAndroid Build Coastguard Worker CURL_TRC_CF(data, cf, "CONNECT: no content-length or chunked");
502*6236dae4SAndroid Build Coastguard Worker ts->keepon = KEEPON_DONE;
503*6236dae4SAndroid Build Coastguard Worker }
504*6236dae4SAndroid Build Coastguard Worker }
505*6236dae4SAndroid Build Coastguard Worker else {
506*6236dae4SAndroid Build Coastguard Worker ts->keepon = KEEPON_DONE;
507*6236dae4SAndroid Build Coastguard Worker }
508*6236dae4SAndroid Build Coastguard Worker
509*6236dae4SAndroid Build Coastguard Worker DEBUGASSERT(ts->keepon == KEEPON_IGNORE
510*6236dae4SAndroid Build Coastguard Worker || ts->keepon == KEEPON_DONE);
511*6236dae4SAndroid Build Coastguard Worker continue;
512*6236dae4SAndroid Build Coastguard Worker }
513*6236dae4SAndroid Build Coastguard Worker
514*6236dae4SAndroid Build Coastguard Worker result = on_resp_header(cf, data, ts, linep);
515*6236dae4SAndroid Build Coastguard Worker if(result)
516*6236dae4SAndroid Build Coastguard Worker return result;
517*6236dae4SAndroid Build Coastguard Worker
518*6236dae4SAndroid Build Coastguard Worker Curl_dyn_reset(&ts->rcvbuf);
519*6236dae4SAndroid Build Coastguard Worker } /* while there is buffer left and loop is requested */
520*6236dae4SAndroid Build Coastguard Worker
521*6236dae4SAndroid Build Coastguard Worker if(error)
522*6236dae4SAndroid Build Coastguard Worker result = CURLE_RECV_ERROR;
523*6236dae4SAndroid Build Coastguard Worker *done = (ts->keepon == KEEPON_DONE);
524*6236dae4SAndroid Build Coastguard Worker if(!result && *done && data->info.httpproxycode/100 != 2) {
525*6236dae4SAndroid Build Coastguard Worker /* Deal with the possibly already received authenticate
526*6236dae4SAndroid Build Coastguard Worker headers. 'newurl' is set to a new URL if we must loop. */
527*6236dae4SAndroid Build Coastguard Worker result = Curl_http_auth_act(data);
528*6236dae4SAndroid Build Coastguard Worker }
529*6236dae4SAndroid Build Coastguard Worker return result;
530*6236dae4SAndroid Build Coastguard Worker }
531*6236dae4SAndroid Build Coastguard Worker
532*6236dae4SAndroid Build Coastguard Worker #else /* USE_HYPER */
533*6236dae4SAndroid Build Coastguard Worker
CONNECT_host(struct Curl_cfilter * cf,struct Curl_easy * data,char ** pauthority,char ** phost_header)534*6236dae4SAndroid Build Coastguard Worker static CURLcode CONNECT_host(struct Curl_cfilter *cf,
535*6236dae4SAndroid Build Coastguard Worker struct Curl_easy *data,
536*6236dae4SAndroid Build Coastguard Worker char **pauthority,
537*6236dae4SAndroid Build Coastguard Worker char **phost_header)
538*6236dae4SAndroid Build Coastguard Worker {
539*6236dae4SAndroid Build Coastguard Worker const char *hostname;
540*6236dae4SAndroid Build Coastguard Worker int port;
541*6236dae4SAndroid Build Coastguard Worker bool ipv6_ip;
542*6236dae4SAndroid Build Coastguard Worker CURLcode result;
543*6236dae4SAndroid Build Coastguard Worker char *authority; /* for CONNECT, the destination host + port */
544*6236dae4SAndroid Build Coastguard Worker char *host_header = NULL; /* Host: authority */
545*6236dae4SAndroid Build Coastguard Worker
546*6236dae4SAndroid Build Coastguard Worker result = Curl_http_proxy_get_destination(cf, &hostname, &port, &ipv6_ip);
547*6236dae4SAndroid Build Coastguard Worker if(result)
548*6236dae4SAndroid Build Coastguard Worker return result;
549*6236dae4SAndroid Build Coastguard Worker
550*6236dae4SAndroid Build Coastguard Worker authority = aprintf("%s%s%s:%d", ipv6_ip ? "[":"", hostname,
551*6236dae4SAndroid Build Coastguard Worker ipv6_ip ? "]" : "", port);
552*6236dae4SAndroid Build Coastguard Worker if(!authority)
553*6236dae4SAndroid Build Coastguard Worker return CURLE_OUT_OF_MEMORY;
554*6236dae4SAndroid Build Coastguard Worker
555*6236dae4SAndroid Build Coastguard Worker /* If user is not overriding the Host header later */
556*6236dae4SAndroid Build Coastguard Worker if(!Curl_checkProxyheaders(data, cf->conn, STRCONST("Host"))) {
557*6236dae4SAndroid Build Coastguard Worker host_header = aprintf("Host: %s\r\n", authority);
558*6236dae4SAndroid Build Coastguard Worker if(!host_header) {
559*6236dae4SAndroid Build Coastguard Worker free(authority);
560*6236dae4SAndroid Build Coastguard Worker return CURLE_OUT_OF_MEMORY;
561*6236dae4SAndroid Build Coastguard Worker }
562*6236dae4SAndroid Build Coastguard Worker }
563*6236dae4SAndroid Build Coastguard Worker *pauthority = authority;
564*6236dae4SAndroid Build Coastguard Worker *phost_header = host_header;
565*6236dae4SAndroid Build Coastguard Worker return CURLE_OK;
566*6236dae4SAndroid Build Coastguard Worker }
567*6236dae4SAndroid Build Coastguard Worker
568*6236dae4SAndroid Build Coastguard Worker /* The Hyper version of CONNECT */
start_CONNECT(struct Curl_cfilter * cf,struct Curl_easy * data,struct h1_tunnel_state * ts)569*6236dae4SAndroid Build Coastguard Worker static CURLcode start_CONNECT(struct Curl_cfilter *cf,
570*6236dae4SAndroid Build Coastguard Worker struct Curl_easy *data,
571*6236dae4SAndroid Build Coastguard Worker struct h1_tunnel_state *ts)
572*6236dae4SAndroid Build Coastguard Worker {
573*6236dae4SAndroid Build Coastguard Worker struct connectdata *conn = cf->conn;
574*6236dae4SAndroid Build Coastguard Worker struct hyptransfer *h = &data->hyp;
575*6236dae4SAndroid Build Coastguard Worker curl_socket_t tunnelsocket = Curl_conn_cf_get_socket(cf, data);
576*6236dae4SAndroid Build Coastguard Worker hyper_io *io = NULL;
577*6236dae4SAndroid Build Coastguard Worker hyper_request *req = NULL;
578*6236dae4SAndroid Build Coastguard Worker hyper_headers *headers = NULL;
579*6236dae4SAndroid Build Coastguard Worker hyper_clientconn_options *options = NULL;
580*6236dae4SAndroid Build Coastguard Worker hyper_task *handshake = NULL;
581*6236dae4SAndroid Build Coastguard Worker hyper_task *task = NULL; /* for the handshake */
582*6236dae4SAndroid Build Coastguard Worker hyper_clientconn *client = NULL;
583*6236dae4SAndroid Build Coastguard Worker hyper_task *sendtask = NULL; /* for the send */
584*6236dae4SAndroid Build Coastguard Worker char *authority = NULL; /* for CONNECT */
585*6236dae4SAndroid Build Coastguard Worker char *host_header = NULL; /* Host: */
586*6236dae4SAndroid Build Coastguard Worker CURLcode result = CURLE_OUT_OF_MEMORY;
587*6236dae4SAndroid Build Coastguard Worker (void)ts;
588*6236dae4SAndroid Build Coastguard Worker
589*6236dae4SAndroid Build Coastguard Worker io = hyper_io_new();
590*6236dae4SAndroid Build Coastguard Worker if(!io) {
591*6236dae4SAndroid Build Coastguard Worker failf(data, "Couldn't create hyper IO");
592*6236dae4SAndroid Build Coastguard Worker result = CURLE_OUT_OF_MEMORY;
593*6236dae4SAndroid Build Coastguard Worker goto error;
594*6236dae4SAndroid Build Coastguard Worker }
595*6236dae4SAndroid Build Coastguard Worker /* tell Hyper how to read/write network data */
596*6236dae4SAndroid Build Coastguard Worker h->io_ctx.data = data;
597*6236dae4SAndroid Build Coastguard Worker h->io_ctx.sockindex = cf->sockindex;
598*6236dae4SAndroid Build Coastguard Worker hyper_io_set_userdata(io, &h->io_ctx);
599*6236dae4SAndroid Build Coastguard Worker hyper_io_set_read(io, Curl_hyper_recv);
600*6236dae4SAndroid Build Coastguard Worker hyper_io_set_write(io, Curl_hyper_send);
601*6236dae4SAndroid Build Coastguard Worker conn->sockfd = tunnelsocket;
602*6236dae4SAndroid Build Coastguard Worker
603*6236dae4SAndroid Build Coastguard Worker data->state.hconnect = TRUE;
604*6236dae4SAndroid Build Coastguard Worker
605*6236dae4SAndroid Build Coastguard Worker /* create an executor to poll futures */
606*6236dae4SAndroid Build Coastguard Worker if(!h->exec) {
607*6236dae4SAndroid Build Coastguard Worker h->exec = hyper_executor_new();
608*6236dae4SAndroid Build Coastguard Worker if(!h->exec) {
609*6236dae4SAndroid Build Coastguard Worker failf(data, "Couldn't create hyper executor");
610*6236dae4SAndroid Build Coastguard Worker result = CURLE_OUT_OF_MEMORY;
611*6236dae4SAndroid Build Coastguard Worker goto error;
612*6236dae4SAndroid Build Coastguard Worker }
613*6236dae4SAndroid Build Coastguard Worker }
614*6236dae4SAndroid Build Coastguard Worker
615*6236dae4SAndroid Build Coastguard Worker options = hyper_clientconn_options_new();
616*6236dae4SAndroid Build Coastguard Worker if(!options) {
617*6236dae4SAndroid Build Coastguard Worker failf(data, "Couldn't create hyper client options");
618*6236dae4SAndroid Build Coastguard Worker result = CURLE_OUT_OF_MEMORY;
619*6236dae4SAndroid Build Coastguard Worker goto error;
620*6236dae4SAndroid Build Coastguard Worker }
621*6236dae4SAndroid Build Coastguard Worker hyper_clientconn_options_set_preserve_header_case(options, 1);
622*6236dae4SAndroid Build Coastguard Worker hyper_clientconn_options_set_preserve_header_order(options, 1);
623*6236dae4SAndroid Build Coastguard Worker
624*6236dae4SAndroid Build Coastguard Worker hyper_clientconn_options_exec(options, h->exec);
625*6236dae4SAndroid Build Coastguard Worker
626*6236dae4SAndroid Build Coastguard Worker /* "Both the `io` and the `options` are consumed in this function
627*6236dae4SAndroid Build Coastguard Worker call" */
628*6236dae4SAndroid Build Coastguard Worker handshake = hyper_clientconn_handshake(io, options);
629*6236dae4SAndroid Build Coastguard Worker if(!handshake) {
630*6236dae4SAndroid Build Coastguard Worker failf(data, "Couldn't create hyper client handshake");
631*6236dae4SAndroid Build Coastguard Worker result = CURLE_OUT_OF_MEMORY;
632*6236dae4SAndroid Build Coastguard Worker goto error;
633*6236dae4SAndroid Build Coastguard Worker }
634*6236dae4SAndroid Build Coastguard Worker io = NULL;
635*6236dae4SAndroid Build Coastguard Worker options = NULL;
636*6236dae4SAndroid Build Coastguard Worker
637*6236dae4SAndroid Build Coastguard Worker if(HYPERE_OK != hyper_executor_push(h->exec, handshake)) {
638*6236dae4SAndroid Build Coastguard Worker failf(data, "Couldn't hyper_executor_push the handshake");
639*6236dae4SAndroid Build Coastguard Worker result = CURLE_OUT_OF_MEMORY;
640*6236dae4SAndroid Build Coastguard Worker goto error;
641*6236dae4SAndroid Build Coastguard Worker }
642*6236dae4SAndroid Build Coastguard Worker handshake = NULL; /* ownership passed on */
643*6236dae4SAndroid Build Coastguard Worker
644*6236dae4SAndroid Build Coastguard Worker task = hyper_executor_poll(h->exec);
645*6236dae4SAndroid Build Coastguard Worker if(!task) {
646*6236dae4SAndroid Build Coastguard Worker failf(data, "Couldn't hyper_executor_poll the handshake");
647*6236dae4SAndroid Build Coastguard Worker result = CURLE_OUT_OF_MEMORY;
648*6236dae4SAndroid Build Coastguard Worker goto error;
649*6236dae4SAndroid Build Coastguard Worker }
650*6236dae4SAndroid Build Coastguard Worker
651*6236dae4SAndroid Build Coastguard Worker client = hyper_task_value(task);
652*6236dae4SAndroid Build Coastguard Worker hyper_task_free(task);
653*6236dae4SAndroid Build Coastguard Worker
654*6236dae4SAndroid Build Coastguard Worker req = hyper_request_new();
655*6236dae4SAndroid Build Coastguard Worker if(!req) {
656*6236dae4SAndroid Build Coastguard Worker failf(data, "Couldn't hyper_request_new");
657*6236dae4SAndroid Build Coastguard Worker result = CURLE_OUT_OF_MEMORY;
658*6236dae4SAndroid Build Coastguard Worker goto error;
659*6236dae4SAndroid Build Coastguard Worker }
660*6236dae4SAndroid Build Coastguard Worker if(hyper_request_set_method(req, (uint8_t *)"CONNECT",
661*6236dae4SAndroid Build Coastguard Worker strlen("CONNECT"))) {
662*6236dae4SAndroid Build Coastguard Worker failf(data, "error setting method");
663*6236dae4SAndroid Build Coastguard Worker result = CURLE_OUT_OF_MEMORY;
664*6236dae4SAndroid Build Coastguard Worker goto error;
665*6236dae4SAndroid Build Coastguard Worker }
666*6236dae4SAndroid Build Coastguard Worker
667*6236dae4SAndroid Build Coastguard Worker /* This only happens if we have looped here due to authentication
668*6236dae4SAndroid Build Coastguard Worker reasons, and we do not really use the newly cloned URL here
669*6236dae4SAndroid Build Coastguard Worker then. Just free() it. */
670*6236dae4SAndroid Build Coastguard Worker Curl_safefree(data->req.newurl);
671*6236dae4SAndroid Build Coastguard Worker
672*6236dae4SAndroid Build Coastguard Worker result = CONNECT_host(cf, data, &authority, &host_header);
673*6236dae4SAndroid Build Coastguard Worker if(result)
674*6236dae4SAndroid Build Coastguard Worker goto error;
675*6236dae4SAndroid Build Coastguard Worker
676*6236dae4SAndroid Build Coastguard Worker infof(data, "Establish HTTP proxy tunnel to %s", authority);
677*6236dae4SAndroid Build Coastguard Worker
678*6236dae4SAndroid Build Coastguard Worker if(hyper_request_set_uri(req, (uint8_t *)authority,
679*6236dae4SAndroid Build Coastguard Worker strlen(authority))) {
680*6236dae4SAndroid Build Coastguard Worker failf(data, "error setting path");
681*6236dae4SAndroid Build Coastguard Worker result = CURLE_OUT_OF_MEMORY;
682*6236dae4SAndroid Build Coastguard Worker goto error;
683*6236dae4SAndroid Build Coastguard Worker }
684*6236dae4SAndroid Build Coastguard Worker if(data->set.verbose) {
685*6236dae4SAndroid Build Coastguard Worker char *se = aprintf("CONNECT %s HTTP/1.1\r\n", authority);
686*6236dae4SAndroid Build Coastguard Worker if(!se) {
687*6236dae4SAndroid Build Coastguard Worker result = CURLE_OUT_OF_MEMORY;
688*6236dae4SAndroid Build Coastguard Worker goto error;
689*6236dae4SAndroid Build Coastguard Worker }
690*6236dae4SAndroid Build Coastguard Worker Curl_debug(data, CURLINFO_HEADER_OUT, se, strlen(se));
691*6236dae4SAndroid Build Coastguard Worker free(se);
692*6236dae4SAndroid Build Coastguard Worker }
693*6236dae4SAndroid Build Coastguard Worker /* Setup the proxy-authorization header, if any */
694*6236dae4SAndroid Build Coastguard Worker result = Curl_http_output_auth(data, conn, "CONNECT", HTTPREQ_GET,
695*6236dae4SAndroid Build Coastguard Worker authority, TRUE);
696*6236dae4SAndroid Build Coastguard Worker if(result)
697*6236dae4SAndroid Build Coastguard Worker goto error;
698*6236dae4SAndroid Build Coastguard Worker Curl_safefree(authority);
699*6236dae4SAndroid Build Coastguard Worker
700*6236dae4SAndroid Build Coastguard Worker /* default is 1.1 */
701*6236dae4SAndroid Build Coastguard Worker if((conn->http_proxy.proxytype == CURLPROXY_HTTP_1_0) &&
702*6236dae4SAndroid Build Coastguard Worker (HYPERE_OK != hyper_request_set_version(req,
703*6236dae4SAndroid Build Coastguard Worker HYPER_HTTP_VERSION_1_0))) {
704*6236dae4SAndroid Build Coastguard Worker failf(data, "error setting HTTP version");
705*6236dae4SAndroid Build Coastguard Worker result = CURLE_OUT_OF_MEMORY;
706*6236dae4SAndroid Build Coastguard Worker goto error;
707*6236dae4SAndroid Build Coastguard Worker }
708*6236dae4SAndroid Build Coastguard Worker
709*6236dae4SAndroid Build Coastguard Worker headers = hyper_request_headers(req);
710*6236dae4SAndroid Build Coastguard Worker if(!headers) {
711*6236dae4SAndroid Build Coastguard Worker failf(data, "hyper_request_headers");
712*6236dae4SAndroid Build Coastguard Worker result = CURLE_OUT_OF_MEMORY;
713*6236dae4SAndroid Build Coastguard Worker goto error;
714*6236dae4SAndroid Build Coastguard Worker }
715*6236dae4SAndroid Build Coastguard Worker if(host_header) {
716*6236dae4SAndroid Build Coastguard Worker result = Curl_hyper_header(data, headers, host_header);
717*6236dae4SAndroid Build Coastguard Worker if(result)
718*6236dae4SAndroid Build Coastguard Worker goto error;
719*6236dae4SAndroid Build Coastguard Worker Curl_safefree(host_header);
720*6236dae4SAndroid Build Coastguard Worker }
721*6236dae4SAndroid Build Coastguard Worker
722*6236dae4SAndroid Build Coastguard Worker if(data->state.aptr.proxyuserpwd) {
723*6236dae4SAndroid Build Coastguard Worker result = Curl_hyper_header(data, headers,
724*6236dae4SAndroid Build Coastguard Worker data->state.aptr.proxyuserpwd);
725*6236dae4SAndroid Build Coastguard Worker if(result)
726*6236dae4SAndroid Build Coastguard Worker goto error;
727*6236dae4SAndroid Build Coastguard Worker }
728*6236dae4SAndroid Build Coastguard Worker
729*6236dae4SAndroid Build Coastguard Worker if(!Curl_checkProxyheaders(data, conn, STRCONST("User-Agent")) &&
730*6236dae4SAndroid Build Coastguard Worker data->set.str[STRING_USERAGENT] && *data->set.str[STRING_USERAGENT]) {
731*6236dae4SAndroid Build Coastguard Worker struct dynbuf ua;
732*6236dae4SAndroid Build Coastguard Worker Curl_dyn_init(&ua, DYN_HTTP_REQUEST);
733*6236dae4SAndroid Build Coastguard Worker result = Curl_dyn_addf(&ua, "User-Agent: %s\r\n",
734*6236dae4SAndroid Build Coastguard Worker data->set.str[STRING_USERAGENT]);
735*6236dae4SAndroid Build Coastguard Worker if(result)
736*6236dae4SAndroid Build Coastguard Worker goto error;
737*6236dae4SAndroid Build Coastguard Worker result = Curl_hyper_header(data, headers, Curl_dyn_ptr(&ua));
738*6236dae4SAndroid Build Coastguard Worker if(result)
739*6236dae4SAndroid Build Coastguard Worker goto error;
740*6236dae4SAndroid Build Coastguard Worker Curl_dyn_free(&ua);
741*6236dae4SAndroid Build Coastguard Worker }
742*6236dae4SAndroid Build Coastguard Worker
743*6236dae4SAndroid Build Coastguard Worker if(!Curl_checkProxyheaders(data, conn, STRCONST("Proxy-Connection"))) {
744*6236dae4SAndroid Build Coastguard Worker result = Curl_hyper_header(data, headers,
745*6236dae4SAndroid Build Coastguard Worker "Proxy-Connection: Keep-Alive");
746*6236dae4SAndroid Build Coastguard Worker if(result)
747*6236dae4SAndroid Build Coastguard Worker goto error;
748*6236dae4SAndroid Build Coastguard Worker }
749*6236dae4SAndroid Build Coastguard Worker
750*6236dae4SAndroid Build Coastguard Worker result = Curl_add_custom_headers(data, TRUE, headers);
751*6236dae4SAndroid Build Coastguard Worker if(result)
752*6236dae4SAndroid Build Coastguard Worker goto error;
753*6236dae4SAndroid Build Coastguard Worker
754*6236dae4SAndroid Build Coastguard Worker result = Curl_creader_set_null(data);
755*6236dae4SAndroid Build Coastguard Worker if(result)
756*6236dae4SAndroid Build Coastguard Worker goto error;
757*6236dae4SAndroid Build Coastguard Worker
758*6236dae4SAndroid Build Coastguard Worker sendtask = hyper_clientconn_send(client, req);
759*6236dae4SAndroid Build Coastguard Worker if(!sendtask) {
760*6236dae4SAndroid Build Coastguard Worker failf(data, "hyper_clientconn_send");
761*6236dae4SAndroid Build Coastguard Worker result = CURLE_OUT_OF_MEMORY;
762*6236dae4SAndroid Build Coastguard Worker goto error;
763*6236dae4SAndroid Build Coastguard Worker }
764*6236dae4SAndroid Build Coastguard Worker req = NULL;
765*6236dae4SAndroid Build Coastguard Worker
766*6236dae4SAndroid Build Coastguard Worker if(HYPERE_OK != hyper_executor_push(h->exec, sendtask)) {
767*6236dae4SAndroid Build Coastguard Worker failf(data, "Couldn't hyper_executor_push the send");
768*6236dae4SAndroid Build Coastguard Worker result = CURLE_OUT_OF_MEMORY;
769*6236dae4SAndroid Build Coastguard Worker goto error;
770*6236dae4SAndroid Build Coastguard Worker }
771*6236dae4SAndroid Build Coastguard Worker sendtask = NULL; /* ownership passed on */
772*6236dae4SAndroid Build Coastguard Worker
773*6236dae4SAndroid Build Coastguard Worker hyper_clientconn_free(client);
774*6236dae4SAndroid Build Coastguard Worker client = NULL;
775*6236dae4SAndroid Build Coastguard Worker
776*6236dae4SAndroid Build Coastguard Worker error:
777*6236dae4SAndroid Build Coastguard Worker free(host_header);
778*6236dae4SAndroid Build Coastguard Worker free(authority);
779*6236dae4SAndroid Build Coastguard Worker if(io)
780*6236dae4SAndroid Build Coastguard Worker hyper_io_free(io);
781*6236dae4SAndroid Build Coastguard Worker if(options)
782*6236dae4SAndroid Build Coastguard Worker hyper_clientconn_options_free(options);
783*6236dae4SAndroid Build Coastguard Worker if(handshake)
784*6236dae4SAndroid Build Coastguard Worker hyper_task_free(handshake);
785*6236dae4SAndroid Build Coastguard Worker if(client)
786*6236dae4SAndroid Build Coastguard Worker hyper_clientconn_free(client);
787*6236dae4SAndroid Build Coastguard Worker if(req)
788*6236dae4SAndroid Build Coastguard Worker hyper_request_free(req);
789*6236dae4SAndroid Build Coastguard Worker
790*6236dae4SAndroid Build Coastguard Worker return result;
791*6236dae4SAndroid Build Coastguard Worker }
792*6236dae4SAndroid Build Coastguard Worker
send_CONNECT(struct Curl_cfilter * cf,struct Curl_easy * data,struct h1_tunnel_state * ts,bool * done)793*6236dae4SAndroid Build Coastguard Worker static CURLcode send_CONNECT(struct Curl_cfilter *cf,
794*6236dae4SAndroid Build Coastguard Worker struct Curl_easy *data,
795*6236dae4SAndroid Build Coastguard Worker struct h1_tunnel_state *ts,
796*6236dae4SAndroid Build Coastguard Worker bool *done)
797*6236dae4SAndroid Build Coastguard Worker {
798*6236dae4SAndroid Build Coastguard Worker struct hyptransfer *h = &data->hyp;
799*6236dae4SAndroid Build Coastguard Worker struct connectdata *conn = cf->conn;
800*6236dae4SAndroid Build Coastguard Worker hyper_task *task = NULL;
801*6236dae4SAndroid Build Coastguard Worker hyper_error *hypererr = NULL;
802*6236dae4SAndroid Build Coastguard Worker CURLcode result = CURLE_OK;
803*6236dae4SAndroid Build Coastguard Worker
804*6236dae4SAndroid Build Coastguard Worker (void)ts;
805*6236dae4SAndroid Build Coastguard Worker (void)conn;
806*6236dae4SAndroid Build Coastguard Worker do {
807*6236dae4SAndroid Build Coastguard Worker task = hyper_executor_poll(h->exec);
808*6236dae4SAndroid Build Coastguard Worker if(task) {
809*6236dae4SAndroid Build Coastguard Worker bool error = hyper_task_type(task) == HYPER_TASK_ERROR;
810*6236dae4SAndroid Build Coastguard Worker if(error)
811*6236dae4SAndroid Build Coastguard Worker hypererr = hyper_task_value(task);
812*6236dae4SAndroid Build Coastguard Worker hyper_task_free(task);
813*6236dae4SAndroid Build Coastguard Worker if(error) {
814*6236dae4SAndroid Build Coastguard Worker /* this could probably use a better error code? */
815*6236dae4SAndroid Build Coastguard Worker result = CURLE_OUT_OF_MEMORY;
816*6236dae4SAndroid Build Coastguard Worker goto error;
817*6236dae4SAndroid Build Coastguard Worker }
818*6236dae4SAndroid Build Coastguard Worker }
819*6236dae4SAndroid Build Coastguard Worker } while(task);
820*6236dae4SAndroid Build Coastguard Worker error:
821*6236dae4SAndroid Build Coastguard Worker *done = (result == CURLE_OK);
822*6236dae4SAndroid Build Coastguard Worker if(hypererr) {
823*6236dae4SAndroid Build Coastguard Worker uint8_t errbuf[256];
824*6236dae4SAndroid Build Coastguard Worker size_t errlen = hyper_error_print(hypererr, errbuf, sizeof(errbuf));
825*6236dae4SAndroid Build Coastguard Worker failf(data, "Hyper: %.*s", (int)errlen, errbuf);
826*6236dae4SAndroid Build Coastguard Worker hyper_error_free(hypererr);
827*6236dae4SAndroid Build Coastguard Worker }
828*6236dae4SAndroid Build Coastguard Worker return result;
829*6236dae4SAndroid Build Coastguard Worker }
830*6236dae4SAndroid Build Coastguard Worker
recv_CONNECT_resp(struct Curl_cfilter * cf,struct Curl_easy * data,struct h1_tunnel_state * ts,bool * done)831*6236dae4SAndroid Build Coastguard Worker static CURLcode recv_CONNECT_resp(struct Curl_cfilter *cf,
832*6236dae4SAndroid Build Coastguard Worker struct Curl_easy *data,
833*6236dae4SAndroid Build Coastguard Worker struct h1_tunnel_state *ts,
834*6236dae4SAndroid Build Coastguard Worker bool *done)
835*6236dae4SAndroid Build Coastguard Worker {
836*6236dae4SAndroid Build Coastguard Worker struct hyptransfer *h = &data->hyp;
837*6236dae4SAndroid Build Coastguard Worker CURLcode result;
838*6236dae4SAndroid Build Coastguard Worker int didwhat;
839*6236dae4SAndroid Build Coastguard Worker
840*6236dae4SAndroid Build Coastguard Worker (void)ts;
841*6236dae4SAndroid Build Coastguard Worker result = Curl_hyper_stream(data, cf->conn, &didwhat,
842*6236dae4SAndroid Build Coastguard Worker CURL_CSELECT_IN | CURL_CSELECT_OUT);
843*6236dae4SAndroid Build Coastguard Worker *done = data->req.done;
844*6236dae4SAndroid Build Coastguard Worker if(result || !*done)
845*6236dae4SAndroid Build Coastguard Worker return result;
846*6236dae4SAndroid Build Coastguard Worker if(h->exec) {
847*6236dae4SAndroid Build Coastguard Worker hyper_executor_free(h->exec);
848*6236dae4SAndroid Build Coastguard Worker h->exec = NULL;
849*6236dae4SAndroid Build Coastguard Worker }
850*6236dae4SAndroid Build Coastguard Worker if(h->read_waker) {
851*6236dae4SAndroid Build Coastguard Worker hyper_waker_free(h->read_waker);
852*6236dae4SAndroid Build Coastguard Worker h->read_waker = NULL;
853*6236dae4SAndroid Build Coastguard Worker }
854*6236dae4SAndroid Build Coastguard Worker if(h->write_waker) {
855*6236dae4SAndroid Build Coastguard Worker hyper_waker_free(h->write_waker);
856*6236dae4SAndroid Build Coastguard Worker h->write_waker = NULL;
857*6236dae4SAndroid Build Coastguard Worker }
858*6236dae4SAndroid Build Coastguard Worker return result;
859*6236dae4SAndroid Build Coastguard Worker }
860*6236dae4SAndroid Build Coastguard Worker
861*6236dae4SAndroid Build Coastguard Worker #endif /* USE_HYPER */
862*6236dae4SAndroid Build Coastguard Worker
H1_CONNECT(struct Curl_cfilter * cf,struct Curl_easy * data,struct h1_tunnel_state * ts)863*6236dae4SAndroid Build Coastguard Worker static CURLcode H1_CONNECT(struct Curl_cfilter *cf,
864*6236dae4SAndroid Build Coastguard Worker struct Curl_easy *data,
865*6236dae4SAndroid Build Coastguard Worker struct h1_tunnel_state *ts)
866*6236dae4SAndroid Build Coastguard Worker {
867*6236dae4SAndroid Build Coastguard Worker struct connectdata *conn = cf->conn;
868*6236dae4SAndroid Build Coastguard Worker CURLcode result;
869*6236dae4SAndroid Build Coastguard Worker bool done;
870*6236dae4SAndroid Build Coastguard Worker
871*6236dae4SAndroid Build Coastguard Worker if(tunnel_is_established(ts))
872*6236dae4SAndroid Build Coastguard Worker return CURLE_OK;
873*6236dae4SAndroid Build Coastguard Worker if(tunnel_is_failed(ts))
874*6236dae4SAndroid Build Coastguard Worker return CURLE_RECV_ERROR; /* Need a cfilter close and new bootstrap */
875*6236dae4SAndroid Build Coastguard Worker
876*6236dae4SAndroid Build Coastguard Worker do {
877*6236dae4SAndroid Build Coastguard Worker timediff_t check;
878*6236dae4SAndroid Build Coastguard Worker
879*6236dae4SAndroid Build Coastguard Worker check = Curl_timeleft(data, NULL, TRUE);
880*6236dae4SAndroid Build Coastguard Worker if(check <= 0) {
881*6236dae4SAndroid Build Coastguard Worker failf(data, "Proxy CONNECT aborted due to timeout");
882*6236dae4SAndroid Build Coastguard Worker result = CURLE_OPERATION_TIMEDOUT;
883*6236dae4SAndroid Build Coastguard Worker goto out;
884*6236dae4SAndroid Build Coastguard Worker }
885*6236dae4SAndroid Build Coastguard Worker
886*6236dae4SAndroid Build Coastguard Worker switch(ts->tunnel_state) {
887*6236dae4SAndroid Build Coastguard Worker case H1_TUNNEL_INIT:
888*6236dae4SAndroid Build Coastguard Worker /* Prepare the CONNECT request and make a first attempt to send. */
889*6236dae4SAndroid Build Coastguard Worker CURL_TRC_CF(data, cf, "CONNECT start");
890*6236dae4SAndroid Build Coastguard Worker result = start_CONNECT(cf, data, ts);
891*6236dae4SAndroid Build Coastguard Worker if(result)
892*6236dae4SAndroid Build Coastguard Worker goto out;
893*6236dae4SAndroid Build Coastguard Worker h1_tunnel_go_state(cf, ts, H1_TUNNEL_CONNECT, data);
894*6236dae4SAndroid Build Coastguard Worker FALLTHROUGH();
895*6236dae4SAndroid Build Coastguard Worker
896*6236dae4SAndroid Build Coastguard Worker case H1_TUNNEL_CONNECT:
897*6236dae4SAndroid Build Coastguard Worker /* see that the request is completely sent */
898*6236dae4SAndroid Build Coastguard Worker CURL_TRC_CF(data, cf, "CONNECT send");
899*6236dae4SAndroid Build Coastguard Worker result = send_CONNECT(cf, data, ts, &done);
900*6236dae4SAndroid Build Coastguard Worker if(result || !done)
901*6236dae4SAndroid Build Coastguard Worker goto out;
902*6236dae4SAndroid Build Coastguard Worker h1_tunnel_go_state(cf, ts, H1_TUNNEL_RECEIVE, data);
903*6236dae4SAndroid Build Coastguard Worker FALLTHROUGH();
904*6236dae4SAndroid Build Coastguard Worker
905*6236dae4SAndroid Build Coastguard Worker case H1_TUNNEL_RECEIVE:
906*6236dae4SAndroid Build Coastguard Worker /* read what is there */
907*6236dae4SAndroid Build Coastguard Worker CURL_TRC_CF(data, cf, "CONNECT receive");
908*6236dae4SAndroid Build Coastguard Worker result = recv_CONNECT_resp(cf, data, ts, &done);
909*6236dae4SAndroid Build Coastguard Worker if(Curl_pgrsUpdate(data)) {
910*6236dae4SAndroid Build Coastguard Worker result = CURLE_ABORTED_BY_CALLBACK;
911*6236dae4SAndroid Build Coastguard Worker goto out;
912*6236dae4SAndroid Build Coastguard Worker }
913*6236dae4SAndroid Build Coastguard Worker /* error or not complete yet. return for more multi-multi */
914*6236dae4SAndroid Build Coastguard Worker if(result || !done)
915*6236dae4SAndroid Build Coastguard Worker goto out;
916*6236dae4SAndroid Build Coastguard Worker /* got it */
917*6236dae4SAndroid Build Coastguard Worker h1_tunnel_go_state(cf, ts, H1_TUNNEL_RESPONSE, data);
918*6236dae4SAndroid Build Coastguard Worker FALLTHROUGH();
919*6236dae4SAndroid Build Coastguard Worker
920*6236dae4SAndroid Build Coastguard Worker case H1_TUNNEL_RESPONSE:
921*6236dae4SAndroid Build Coastguard Worker CURL_TRC_CF(data, cf, "CONNECT response");
922*6236dae4SAndroid Build Coastguard Worker if(data->req.newurl) {
923*6236dae4SAndroid Build Coastguard Worker /* not the "final" response, we need to do a follow up request.
924*6236dae4SAndroid Build Coastguard Worker * If the other side indicated a connection close, or if someone
925*6236dae4SAndroid Build Coastguard Worker * else told us to close this connection, do so now.
926*6236dae4SAndroid Build Coastguard Worker */
927*6236dae4SAndroid Build Coastguard Worker Curl_req_soft_reset(&data->req, data);
928*6236dae4SAndroid Build Coastguard Worker if(ts->close_connection || conn->bits.close) {
929*6236dae4SAndroid Build Coastguard Worker /* Close this filter and the sub-chain, re-connect the
930*6236dae4SAndroid Build Coastguard Worker * sub-chain and continue. Closing this filter will
931*6236dae4SAndroid Build Coastguard Worker * reset our tunnel state. To avoid recursion, we return
932*6236dae4SAndroid Build Coastguard Worker * and expect to be called again.
933*6236dae4SAndroid Build Coastguard Worker */
934*6236dae4SAndroid Build Coastguard Worker CURL_TRC_CF(data, cf, "CONNECT need to close+open");
935*6236dae4SAndroid Build Coastguard Worker infof(data, "Connect me again please");
936*6236dae4SAndroid Build Coastguard Worker Curl_conn_cf_close(cf, data);
937*6236dae4SAndroid Build Coastguard Worker connkeep(conn, "HTTP proxy CONNECT");
938*6236dae4SAndroid Build Coastguard Worker result = Curl_conn_cf_connect(cf->next, data, FALSE, &done);
939*6236dae4SAndroid Build Coastguard Worker goto out;
940*6236dae4SAndroid Build Coastguard Worker }
941*6236dae4SAndroid Build Coastguard Worker else {
942*6236dae4SAndroid Build Coastguard Worker /* staying on this connection, reset state */
943*6236dae4SAndroid Build Coastguard Worker h1_tunnel_go_state(cf, ts, H1_TUNNEL_INIT, data);
944*6236dae4SAndroid Build Coastguard Worker }
945*6236dae4SAndroid Build Coastguard Worker }
946*6236dae4SAndroid Build Coastguard Worker break;
947*6236dae4SAndroid Build Coastguard Worker
948*6236dae4SAndroid Build Coastguard Worker default:
949*6236dae4SAndroid Build Coastguard Worker break;
950*6236dae4SAndroid Build Coastguard Worker }
951*6236dae4SAndroid Build Coastguard Worker
952*6236dae4SAndroid Build Coastguard Worker } while(data->req.newurl);
953*6236dae4SAndroid Build Coastguard Worker
954*6236dae4SAndroid Build Coastguard Worker DEBUGASSERT(ts->tunnel_state == H1_TUNNEL_RESPONSE);
955*6236dae4SAndroid Build Coastguard Worker if(data->info.httpproxycode/100 != 2) {
956*6236dae4SAndroid Build Coastguard Worker /* a non-2xx response and we have no next URL to try. */
957*6236dae4SAndroid Build Coastguard Worker Curl_safefree(data->req.newurl);
958*6236dae4SAndroid Build Coastguard Worker /* failure, close this connection to avoid reuse */
959*6236dae4SAndroid Build Coastguard Worker streamclose(conn, "proxy CONNECT failure");
960*6236dae4SAndroid Build Coastguard Worker h1_tunnel_go_state(cf, ts, H1_TUNNEL_FAILED, data);
961*6236dae4SAndroid Build Coastguard Worker failf(data, "CONNECT tunnel failed, response %d", data->req.httpcode);
962*6236dae4SAndroid Build Coastguard Worker return CURLE_RECV_ERROR;
963*6236dae4SAndroid Build Coastguard Worker }
964*6236dae4SAndroid Build Coastguard Worker /* 2xx response, SUCCESS! */
965*6236dae4SAndroid Build Coastguard Worker h1_tunnel_go_state(cf, ts, H1_TUNNEL_ESTABLISHED, data);
966*6236dae4SAndroid Build Coastguard Worker infof(data, "CONNECT tunnel established, response %d",
967*6236dae4SAndroid Build Coastguard Worker data->info.httpproxycode);
968*6236dae4SAndroid Build Coastguard Worker result = CURLE_OK;
969*6236dae4SAndroid Build Coastguard Worker
970*6236dae4SAndroid Build Coastguard Worker out:
971*6236dae4SAndroid Build Coastguard Worker if(result)
972*6236dae4SAndroid Build Coastguard Worker h1_tunnel_go_state(cf, ts, H1_TUNNEL_FAILED, data);
973*6236dae4SAndroid Build Coastguard Worker return result;
974*6236dae4SAndroid Build Coastguard Worker }
975*6236dae4SAndroid Build Coastguard Worker
cf_h1_proxy_connect(struct Curl_cfilter * cf,struct Curl_easy * data,bool blocking,bool * done)976*6236dae4SAndroid Build Coastguard Worker static CURLcode cf_h1_proxy_connect(struct Curl_cfilter *cf,
977*6236dae4SAndroid Build Coastguard Worker struct Curl_easy *data,
978*6236dae4SAndroid Build Coastguard Worker bool blocking, bool *done)
979*6236dae4SAndroid Build Coastguard Worker {
980*6236dae4SAndroid Build Coastguard Worker CURLcode result;
981*6236dae4SAndroid Build Coastguard Worker struct h1_tunnel_state *ts = cf->ctx;
982*6236dae4SAndroid Build Coastguard Worker
983*6236dae4SAndroid Build Coastguard Worker if(cf->connected) {
984*6236dae4SAndroid Build Coastguard Worker *done = TRUE;
985*6236dae4SAndroid Build Coastguard Worker return CURLE_OK;
986*6236dae4SAndroid Build Coastguard Worker }
987*6236dae4SAndroid Build Coastguard Worker
988*6236dae4SAndroid Build Coastguard Worker CURL_TRC_CF(data, cf, "connect");
989*6236dae4SAndroid Build Coastguard Worker result = cf->next->cft->do_connect(cf->next, data, blocking, done);
990*6236dae4SAndroid Build Coastguard Worker if(result || !*done)
991*6236dae4SAndroid Build Coastguard Worker return result;
992*6236dae4SAndroid Build Coastguard Worker
993*6236dae4SAndroid Build Coastguard Worker *done = FALSE;
994*6236dae4SAndroid Build Coastguard Worker if(!ts) {
995*6236dae4SAndroid Build Coastguard Worker result = tunnel_init(cf, data, &ts);
996*6236dae4SAndroid Build Coastguard Worker if(result)
997*6236dae4SAndroid Build Coastguard Worker return result;
998*6236dae4SAndroid Build Coastguard Worker cf->ctx = ts;
999*6236dae4SAndroid Build Coastguard Worker }
1000*6236dae4SAndroid Build Coastguard Worker
1001*6236dae4SAndroid Build Coastguard Worker /* TODO: can we do blocking? */
1002*6236dae4SAndroid Build Coastguard Worker /* We want "seamless" operations through HTTP proxy tunnel */
1003*6236dae4SAndroid Build Coastguard Worker
1004*6236dae4SAndroid Build Coastguard Worker result = H1_CONNECT(cf, data, ts);
1005*6236dae4SAndroid Build Coastguard Worker if(result)
1006*6236dae4SAndroid Build Coastguard Worker goto out;
1007*6236dae4SAndroid Build Coastguard Worker Curl_safefree(data->state.aptr.proxyuserpwd);
1008*6236dae4SAndroid Build Coastguard Worker
1009*6236dae4SAndroid Build Coastguard Worker out:
1010*6236dae4SAndroid Build Coastguard Worker *done = (result == CURLE_OK) && tunnel_is_established(cf->ctx);
1011*6236dae4SAndroid Build Coastguard Worker if(*done) {
1012*6236dae4SAndroid Build Coastguard Worker cf->connected = TRUE;
1013*6236dae4SAndroid Build Coastguard Worker /* The real request will follow the CONNECT, reset request partially */
1014*6236dae4SAndroid Build Coastguard Worker Curl_req_soft_reset(&data->req, data);
1015*6236dae4SAndroid Build Coastguard Worker Curl_client_reset(data);
1016*6236dae4SAndroid Build Coastguard Worker Curl_pgrsSetUploadCounter(data, 0);
1017*6236dae4SAndroid Build Coastguard Worker Curl_pgrsSetDownloadCounter(data, 0);
1018*6236dae4SAndroid Build Coastguard Worker
1019*6236dae4SAndroid Build Coastguard Worker tunnel_free(cf, data);
1020*6236dae4SAndroid Build Coastguard Worker }
1021*6236dae4SAndroid Build Coastguard Worker return result;
1022*6236dae4SAndroid Build Coastguard Worker }
1023*6236dae4SAndroid Build Coastguard Worker
cf_h1_proxy_adjust_pollset(struct Curl_cfilter * cf,struct Curl_easy * data,struct easy_pollset * ps)1024*6236dae4SAndroid Build Coastguard Worker static void cf_h1_proxy_adjust_pollset(struct Curl_cfilter *cf,
1025*6236dae4SAndroid Build Coastguard Worker struct Curl_easy *data,
1026*6236dae4SAndroid Build Coastguard Worker struct easy_pollset *ps)
1027*6236dae4SAndroid Build Coastguard Worker {
1028*6236dae4SAndroid Build Coastguard Worker struct h1_tunnel_state *ts = cf->ctx;
1029*6236dae4SAndroid Build Coastguard Worker
1030*6236dae4SAndroid Build Coastguard Worker if(!cf->connected) {
1031*6236dae4SAndroid Build Coastguard Worker /* If we are not connected, but the filter "below" is
1032*6236dae4SAndroid Build Coastguard Worker * and not waiting on something, we are tunneling. */
1033*6236dae4SAndroid Build Coastguard Worker curl_socket_t sock = Curl_conn_cf_get_socket(cf, data);
1034*6236dae4SAndroid Build Coastguard Worker if(ts) {
1035*6236dae4SAndroid Build Coastguard Worker /* when we have sent a CONNECT to a proxy, we should rather either
1036*6236dae4SAndroid Build Coastguard Worker wait for the socket to become readable to be able to get the
1037*6236dae4SAndroid Build Coastguard Worker response headers or if we are still sending the request, wait
1038*6236dae4SAndroid Build Coastguard Worker for write. */
1039*6236dae4SAndroid Build Coastguard Worker if(tunnel_want_send(ts))
1040*6236dae4SAndroid Build Coastguard Worker Curl_pollset_set_out_only(data, ps, sock);
1041*6236dae4SAndroid Build Coastguard Worker else
1042*6236dae4SAndroid Build Coastguard Worker Curl_pollset_set_in_only(data, ps, sock);
1043*6236dae4SAndroid Build Coastguard Worker }
1044*6236dae4SAndroid Build Coastguard Worker else
1045*6236dae4SAndroid Build Coastguard Worker Curl_pollset_set_out_only(data, ps, sock);
1046*6236dae4SAndroid Build Coastguard Worker }
1047*6236dae4SAndroid Build Coastguard Worker }
1048*6236dae4SAndroid Build Coastguard Worker
cf_h1_proxy_destroy(struct Curl_cfilter * cf,struct Curl_easy * data)1049*6236dae4SAndroid Build Coastguard Worker static void cf_h1_proxy_destroy(struct Curl_cfilter *cf,
1050*6236dae4SAndroid Build Coastguard Worker struct Curl_easy *data)
1051*6236dae4SAndroid Build Coastguard Worker {
1052*6236dae4SAndroid Build Coastguard Worker CURL_TRC_CF(data, cf, "destroy");
1053*6236dae4SAndroid Build Coastguard Worker tunnel_free(cf, data);
1054*6236dae4SAndroid Build Coastguard Worker }
1055*6236dae4SAndroid Build Coastguard Worker
cf_h1_proxy_close(struct Curl_cfilter * cf,struct Curl_easy * data)1056*6236dae4SAndroid Build Coastguard Worker static void cf_h1_proxy_close(struct Curl_cfilter *cf,
1057*6236dae4SAndroid Build Coastguard Worker struct Curl_easy *data)
1058*6236dae4SAndroid Build Coastguard Worker {
1059*6236dae4SAndroid Build Coastguard Worker CURL_TRC_CF(data, cf, "close");
1060*6236dae4SAndroid Build Coastguard Worker if(cf) {
1061*6236dae4SAndroid Build Coastguard Worker cf->connected = FALSE;
1062*6236dae4SAndroid Build Coastguard Worker if(cf->ctx) {
1063*6236dae4SAndroid Build Coastguard Worker h1_tunnel_go_state(cf, cf->ctx, H1_TUNNEL_INIT, data);
1064*6236dae4SAndroid Build Coastguard Worker }
1065*6236dae4SAndroid Build Coastguard Worker if(cf->next)
1066*6236dae4SAndroid Build Coastguard Worker cf->next->cft->do_close(cf->next, data);
1067*6236dae4SAndroid Build Coastguard Worker }
1068*6236dae4SAndroid Build Coastguard Worker }
1069*6236dae4SAndroid Build Coastguard Worker
1070*6236dae4SAndroid Build Coastguard Worker
1071*6236dae4SAndroid Build Coastguard Worker struct Curl_cftype Curl_cft_h1_proxy = {
1072*6236dae4SAndroid Build Coastguard Worker "H1-PROXY",
1073*6236dae4SAndroid Build Coastguard Worker CF_TYPE_IP_CONNECT|CF_TYPE_PROXY,
1074*6236dae4SAndroid Build Coastguard Worker 0,
1075*6236dae4SAndroid Build Coastguard Worker cf_h1_proxy_destroy,
1076*6236dae4SAndroid Build Coastguard Worker cf_h1_proxy_connect,
1077*6236dae4SAndroid Build Coastguard Worker cf_h1_proxy_close,
1078*6236dae4SAndroid Build Coastguard Worker Curl_cf_def_shutdown,
1079*6236dae4SAndroid Build Coastguard Worker Curl_cf_http_proxy_get_host,
1080*6236dae4SAndroid Build Coastguard Worker cf_h1_proxy_adjust_pollset,
1081*6236dae4SAndroid Build Coastguard Worker Curl_cf_def_data_pending,
1082*6236dae4SAndroid Build Coastguard Worker Curl_cf_def_send,
1083*6236dae4SAndroid Build Coastguard Worker Curl_cf_def_recv,
1084*6236dae4SAndroid Build Coastguard Worker Curl_cf_def_cntrl,
1085*6236dae4SAndroid Build Coastguard Worker Curl_cf_def_conn_is_alive,
1086*6236dae4SAndroid Build Coastguard Worker Curl_cf_def_conn_keep_alive,
1087*6236dae4SAndroid Build Coastguard Worker Curl_cf_def_query,
1088*6236dae4SAndroid Build Coastguard Worker };
1089*6236dae4SAndroid Build Coastguard Worker
Curl_cf_h1_proxy_insert_after(struct Curl_cfilter * cf_at,struct Curl_easy * data)1090*6236dae4SAndroid Build Coastguard Worker CURLcode Curl_cf_h1_proxy_insert_after(struct Curl_cfilter *cf_at,
1091*6236dae4SAndroid Build Coastguard Worker struct Curl_easy *data)
1092*6236dae4SAndroid Build Coastguard Worker {
1093*6236dae4SAndroid Build Coastguard Worker struct Curl_cfilter *cf;
1094*6236dae4SAndroid Build Coastguard Worker CURLcode result;
1095*6236dae4SAndroid Build Coastguard Worker
1096*6236dae4SAndroid Build Coastguard Worker (void)data;
1097*6236dae4SAndroid Build Coastguard Worker result = Curl_cf_create(&cf, &Curl_cft_h1_proxy, NULL);
1098*6236dae4SAndroid Build Coastguard Worker if(!result)
1099*6236dae4SAndroid Build Coastguard Worker Curl_conn_cf_insert_after(cf_at, cf);
1100*6236dae4SAndroid Build Coastguard Worker return result;
1101*6236dae4SAndroid Build Coastguard Worker }
1102*6236dae4SAndroid Build Coastguard Worker
1103*6236dae4SAndroid Build Coastguard Worker #endif /* !CURL_DISABLE_PROXY && ! CURL_DISABLE_HTTP */
1104