prepare v4.2.5
[m6w6/ext-http] / src / php_http_client_curl.c
1 /*
2 +--------------------------------------------------------------------+
3 | PECL :: http |
4 +--------------------------------------------------------------------+
5 | Redistribution and use in source and binary forms, with or without |
6 | modification, are permitted provided that the conditions mentioned |
7 | in the accompanying LICENSE file are met. |
8 +--------------------------------------------------------------------+
9 | Copyright (c) 2004-2014, Michael Wallner <mike@php.net> |
10 +--------------------------------------------------------------------+
11 */
12
13 #include "php_http_api.h"
14 #include "php_http_client.h"
15 #include "php_http_client_curl_event.h"
16 #include "php_http_client_curl_user.h"
17
18 #if PHP_HTTP_HAVE_LIBCURL
19
20 #define DEBUG_COOKIES 0
21
22 #if PHP_HTTP_HAVE_LIBCURL_OPENSSL
23 # include <openssl/ssl.h>
24 #endif
25 #if PHP_HTTP_HAVE_LIBCURL_GNUTLS
26 # include <gnutls/gnutls.h>
27 #endif
28
29 typedef struct php_http_client_curl_handler {
30 CURL *handle;
31 php_resource_factory_t *rf;
32 php_http_client_t *client;
33 php_http_client_progress_state_t progress;
34 php_http_client_enqueue_t queue;
35
36 struct {
37 php_http_buffer_t headers;
38 php_http_message_body_t *body;
39 } response;
40
41 struct {
42 HashTable cache;
43
44 struct curl_slist *proxyheaders;
45 struct curl_slist *headers;
46 struct curl_slist *resolve;
47 php_http_buffer_t cookies;
48 php_http_buffer_t ranges;
49
50 struct {
51 uint32_t count;
52 double delay;
53 } retry;
54
55 long redirects;
56 unsigned range_request:1;
57 unsigned encode_cookies:1;
58
59 } options;
60
61 } php_http_client_curl_handler_t;
62
63 typedef struct php_http_curle_storage {
64 char *url;
65 char *cookiestore;
66 CURLcode errorcode;
67 char errorbuffer[0x100];
68 } php_http_curle_storage_t;
69
70 static inline php_http_curle_storage_t *php_http_curle_get_storage(CURL *ch) {
71 php_http_curle_storage_t *st = NULL;
72
73 curl_easy_getinfo(ch, CURLINFO_PRIVATE, &st);
74
75 if (!st) {
76 st = pecalloc(1, sizeof(*st), 1);
77 curl_easy_setopt(ch, CURLOPT_PRIVATE, st);
78 curl_easy_setopt(ch, CURLOPT_ERRORBUFFER, st->errorbuffer);
79 }
80
81 return st;
82 }
83
84 static void *php_http_curle_ctor(void *opaque, void *init_arg)
85 {
86 void *ch;
87
88 if ((ch = curl_easy_init())) {
89 php_http_curle_get_storage(ch);
90 return ch;
91 }
92 return NULL;
93 }
94
95 static void *php_http_curle_copy(void *opaque, void *handle)
96 {
97 void *ch;
98
99 if ((ch = curl_easy_duphandle(handle))) {
100 curl_easy_reset(ch);
101 php_http_curle_get_storage(ch);
102 return ch;
103 }
104 return NULL;
105 }
106
107 static void php_http_curle_dtor(void *opaque, void *handle)
108 {
109 php_http_curle_storage_t *st = php_http_curle_get_storage(handle);
110
111 curl_easy_cleanup(handle);
112
113 if (st) {
114 if (st->url) {
115 pefree(st->url, 1);
116 }
117 if (st->cookiestore) {
118 pefree(st->cookiestore, 1);
119 }
120 pefree(st, 1);
121 }
122 }
123
124 static php_resource_factory_ops_t php_http_curle_resource_factory_ops = {
125 php_http_curle_ctor,
126 php_http_curle_copy,
127 php_http_curle_dtor
128 };
129
130 static void *php_http_curlm_ctor(void *opaque, void *init_arg)
131 {
132 php_http_client_curl_handle_t *curl = calloc(1, sizeof(*curl));
133
134 if (!(curl->multi = curl_multi_init())) {
135 free(curl);
136 return NULL;
137 }
138 if (!(curl->share = curl_share_init())) {
139 curl_multi_cleanup(curl->multi);
140 free(curl);
141 return NULL;
142 }
143 curl_share_setopt(curl->share, CURLSHOPT_SHARE, CURL_LOCK_DATA_COOKIE);
144 curl_share_setopt(curl->share, CURLSHOPT_SHARE, CURL_LOCK_DATA_SSL_SESSION);
145 return curl;
146 }
147
148 static void php_http_curlm_dtor(void *opaque, void *handle)
149 {
150 php_http_client_curl_handle_t *curl = handle;
151
152 curl_share_cleanup(curl->share);
153 curl_multi_cleanup(curl->multi);
154 free(handle);
155 }
156
157 static php_resource_factory_ops_t php_http_curlm_resource_factory_ops = {
158 php_http_curlm_ctor,
159 NULL,
160 php_http_curlm_dtor
161 };
162
163 /* curl callbacks */
164
165 static size_t php_http_curle_read_callback(void *data, size_t len, size_t n, void *ctx)
166 {
167 php_stream *s = php_http_message_body_stream(ctx);
168
169 if (s) {
170 return php_stream_read(s, data, len * n);
171 }
172 return 0;
173 }
174
175 #if PHP_HTTP_CURL_VERSION(7,32,0)
176 static int php_http_curle_xferinfo_callback(void *ctx, curl_off_t dltotal, curl_off_t dlnow, curl_off_t ultotal, curl_off_t ulnow)
177 #else
178 static int php_http_curle_progress_callback(void *ctx, double dltotal, double dlnow, double ultotal, double ulnow)
179 #endif
180 {
181 php_http_client_curl_handler_t *h = ctx;
182
183 if (h->progress.dl.total != dltotal
184 || h->progress.dl.now != dlnow
185 || h->progress.ul.total != ultotal
186 || h->progress.ul.now != ulnow
187 ) {
188 h->progress.dl.total = dltotal;
189 h->progress.dl.now = dlnow;
190 h->progress.ul.total = ultotal;
191 h->progress.ul.now = ulnow;
192 }
193
194 if (h->client->callback.progress.func) {
195 h->client->callback.progress.func(h->client->callback.progress.arg, h->client, &h->queue, &h->progress);
196 }
197
198 return 0;
199 }
200
201 static int php_http_curle_seek_callback(void *userdata, curl_off_t offset, int origin)
202 {
203 php_http_message_body_t *body = userdata;
204
205 if (!body) {
206 return 1;
207 }
208 if (0 == php_stream_seek(php_http_message_body_stream(body), offset, origin)) {
209 return 0;
210 }
211 return 2;
212 }
213
214 static int php_http_curle_raw_callback(CURL *ch, curl_infotype type, char *data, size_t length, void *ctx)
215 {
216 php_http_client_curl_handler_t *h = ctx;
217 unsigned utype = PHP_HTTP_CLIENT_DEBUG_INFO;
218
219 /* catch progress */
220 switch (type) {
221 case CURLINFO_TEXT:
222 if (data[0] == '-') {
223 goto text;
224 } else if (php_memnstr(data, ZEND_STRL("Adding handle:"), data + length)) {
225 h->progress.info = "setup";
226 } else if (php_memnstr(data, ZEND_STRL("addHandle"), data + length)) {
227 h->progress.info = "setup";
228 } else if (php_memnstr(data, ZEND_STRL("About to connect"), data + length)) {
229 h->progress.info = "resolve";
230 } else if (php_memnstr(data, ZEND_STRL("Trying"), data + length)) {
231 h->progress.info = "connect";
232 } else if (php_memnstr(data, ZEND_STRL("Found bundle for host"), data + length)) {
233 h->progress.info = "connect";
234 } else if (php_memnstr(data, ZEND_STRL("Connected"), data + length)) {
235 h->progress.info = "connected";
236 } else if (php_memnstr(data, ZEND_STRL("Re-using existing connection!"), data + length)) {
237 h->progress.info = "connected";
238 } else if (php_memnstr(data, ZEND_STRL("blacklisted"), data + length)) {
239 h->progress.info = "blacklist check";
240 } else if (php_memnstr(data, ZEND_STRL("TLS"), data + length)) {
241 h->progress.info = "ssl negotiation";
242 } else if (php_memnstr(data, ZEND_STRL("SSL"), data + length)) {
243 h->progress.info = "ssl negotiation";
244 } else if (php_memnstr(data, ZEND_STRL("certificate"), data + length)) {
245 h->progress.info = "ssl negotiation";
246 } else if (php_memnstr(data, ZEND_STRL("ALPN"), data + length)) {
247 h->progress.info = "alpn";
248 } else if (php_memnstr(data, ZEND_STRL("NPN"), data + length)) {
249 h->progress.info = "npn";
250 } else if (php_memnstr(data, ZEND_STRL("upload"), data + length)) {
251 h->progress.info = "uploaded";
252 } else if (php_memnstr(data, ZEND_STRL("left intact"), data + length)) {
253 h->progress.info = "not disconnected";
254 } else if (php_memnstr(data, ZEND_STRL("closed"), data + length)) {
255 h->progress.info = "disconnected";
256 } else if (php_memnstr(data, ZEND_STRL("Issue another request"), data + length)) {
257 h->progress.info = "redirect";
258 } else if (php_memnstr(data, ZEND_STRL("Operation timed out"), data + length)) {
259 h->progress.info = "timeout";
260 } else {
261 text:;
262 #if 0
263 h->progress.info = data;
264 data[length - 1] = '\0';
265 #endif
266 }
267 if (h->client->callback.progress.func) {
268 h->client->callback.progress.func(h->client->callback.progress.arg, h->client, &h->queue, &h->progress);
269 }
270 break;
271
272 case CURLINFO_HEADER_OUT:
273 utype |= PHP_HTTP_CLIENT_DEBUG_HEADER;
274 goto data_out;
275
276 case CURLINFO_SSL_DATA_OUT:
277 utype |= PHP_HTTP_CLIENT_DEBUG_SSL;
278 goto data_out;
279
280 case CURLINFO_DATA_OUT:
281 data_out:
282 utype |= PHP_HTTP_CLIENT_DEBUG_OUT;
283 h->progress.info = "send";
284 break;
285
286 case CURLINFO_HEADER_IN:
287 utype |= PHP_HTTP_CLIENT_DEBUG_HEADER;
288 goto data_in;
289
290 case CURLINFO_SSL_DATA_IN:
291 utype |= PHP_HTTP_CLIENT_DEBUG_SSL;
292 goto data_in;
293
294 case CURLINFO_DATA_IN:
295 data_in:
296 utype |= PHP_HTTP_CLIENT_DEBUG_IN;
297 h->progress.info = "receive";
298 break;
299
300 default:
301 break;
302 }
303
304 if (h->client->callback.debug.func) {
305 h->client->callback.debug.func(h->client->callback.debug.arg, h->client, &h->queue, utype, data, length);
306 }
307
308 #if 0
309 /* debug */
310 _dpf(type, data, length);
311 #endif
312
313 return 0;
314 }
315
316 static size_t php_http_curle_header_callback(char *data, size_t n, size_t l, void *arg)
317 {
318 php_http_client_curl_handler_t *h = arg;
319
320 return php_http_buffer_append(&h->response.headers, data, n * l);
321 }
322
323 static size_t php_http_curle_body_callback(char *data, size_t n, size_t l, void *arg)
324 {
325 php_http_client_curl_handler_t *h = arg;
326
327 return php_http_message_body_append(h->response.body, data, n*l);
328 }
329
330 static ZEND_RESULT_CODE php_http_curle_get_info(CURL *ch, HashTable *info)
331 {
332 char *c = NULL;
333 long l = 0;
334 double d = 0;
335 curl_off_t o = 0;
336 struct curl_slist *s = NULL, *p = NULL;
337 zval tmp;
338
339 ZVAL_NULL(&tmp);
340
341 /* BEGIN::CURLINFO */
342 if (CURLE_OK == curl_easy_getinfo(ch, CURLINFO_EFFECTIVE_URL, &c)) {
343 ZVAL_STRING(&tmp, STR_PTR(c));
344 zend_hash_str_update(info, "effective_url", lenof("effective_url"), &tmp);
345 }
346 if (CURLE_OK == curl_easy_getinfo(ch, CURLINFO_RESPONSE_CODE, &l)) {
347 ZVAL_LONG(&tmp, l);
348 zend_hash_str_update(info, "response_code", lenof("response_code"), &tmp);
349 }
350 if (CURLE_OK == curl_easy_getinfo(ch, CURLINFO_TOTAL_TIME, &d)) {
351 ZVAL_DOUBLE(&tmp, d);
352 zend_hash_str_update(info, "total_time", lenof("total_time"), &tmp);
353 }
354 if (CURLE_OK == curl_easy_getinfo(ch, CURLINFO_NAMELOOKUP_TIME, &d)) {
355 ZVAL_DOUBLE(&tmp, d);
356 zend_hash_str_update(info, "namelookup_time", lenof("namelookup_time"), &tmp);
357 }
358 if (CURLE_OK == curl_easy_getinfo(ch, CURLINFO_CONNECT_TIME, &d)) {
359 ZVAL_DOUBLE(&tmp, d);
360 zend_hash_str_update(info, "connect_time", lenof("connect_time"), &tmp);
361 }
362 if (CURLE_OK == curl_easy_getinfo(ch, CURLINFO_PRETRANSFER_TIME, &d)) {
363 ZVAL_DOUBLE(&tmp, d);
364 zend_hash_str_update(info, "pretransfer_time", lenof("pretransfer_time"), &tmp);
365 }
366 if (CURLE_OK == curl_easy_getinfo(ch, CURLINFO_SIZE_UPLOAD, &d)) {
367 ZVAL_DOUBLE(&tmp, d);
368 zend_hash_str_update(info, "size_upload", lenof("size_upload"), &tmp);
369 }
370 if (CURLE_OK == curl_easy_getinfo(ch, CURLINFO_SIZE_UPLOAD_T, &o)) {
371 ZVAL_LONG(&tmp, o);
372 zend_hash_str_update(info, "size_upload_t", lenof("size_upload_t"), &tmp);
373 }
374 if (CURLE_OK == curl_easy_getinfo(ch, CURLINFO_SIZE_DOWNLOAD, &d)) {
375 ZVAL_DOUBLE(&tmp, d);
376 zend_hash_str_update(info, "size_download", lenof("size_download"), &tmp);
377 }
378 if (CURLE_OK == curl_easy_getinfo(ch, CURLINFO_SIZE_DOWNLOAD_T, &o)) {
379 ZVAL_LONG(&tmp, o);
380 zend_hash_str_update(info, "size_download_t", lenof("size_download_t"), &tmp);
381 }
382 if (CURLE_OK == curl_easy_getinfo(ch, CURLINFO_SPEED_DOWNLOAD, &d)) {
383 ZVAL_DOUBLE(&tmp, d);
384 zend_hash_str_update(info, "speed_download", lenof("speed_download"), &tmp);
385 }
386 if (CURLE_OK == curl_easy_getinfo(ch, CURLINFO_SPEED_DOWNLOAD_T, &o)) {
387 ZVAL_LONG(&tmp, o);
388 zend_hash_str_update(info, "speed_download_t", lenof("speed_download_t"), &tmp);
389 }
390 if (CURLE_OK == curl_easy_getinfo(ch, CURLINFO_SPEED_UPLOAD, &d)) {
391 ZVAL_DOUBLE(&tmp, d);
392 zend_hash_str_update(info, "speed_upload", lenof("speed_upload"), &tmp);
393 }
394 if (CURLE_OK == curl_easy_getinfo(ch, CURLINFO_SPEED_UPLOAD_T, &o)) {
395 ZVAL_LONG(&tmp, o);
396 zend_hash_str_update(info, "speed_upload_t", lenof("speed_upload_t"), &tmp);
397 }
398 if (CURLE_OK == curl_easy_getinfo(ch, CURLINFO_HEADER_SIZE, &l)) {
399 ZVAL_LONG(&tmp, l);
400 zend_hash_str_update(info, "header_size", lenof("header_size"), &tmp);
401 }
402 if (CURLE_OK == curl_easy_getinfo(ch, CURLINFO_REQUEST_SIZE, &l)) {
403 ZVAL_LONG(&tmp, l);
404 zend_hash_str_update(info, "request_size", lenof("request_size"), &tmp);
405 }
406 if (CURLE_OK == curl_easy_getinfo(ch, CURLINFO_SSL_VERIFYRESULT, &l)) {
407 ZVAL_LONG(&tmp, l);
408 zend_hash_str_update(info, "ssl_verifyresult", lenof("ssl_verifyresult"), &tmp);
409 }
410 if (CURLE_OK == curl_easy_getinfo(ch, CURLINFO_FILETIME, &l)) {
411 ZVAL_LONG(&tmp, l);
412 zend_hash_str_update(info, "filetime", lenof("filetime"), &tmp);
413 }
414 if (CURLE_OK == curl_easy_getinfo(ch, CURLINFO_FILETIME_T, &o)) {
415 ZVAL_LONG(&tmp, o);
416 zend_hash_str_update(info, "filetime_t", lenof("filetime_t"), &tmp);
417 }
418 if (CURLE_OK == curl_easy_getinfo(ch, CURLINFO_CONTENT_LENGTH_DOWNLOAD, &d)) {
419 ZVAL_DOUBLE(&tmp, d);
420 zend_hash_str_update(info, "content_length_download", lenof("content_length_download"), &tmp);
421 }
422 if (CURLE_OK == curl_easy_getinfo(ch, CURLINFO_CONTENT_LENGTH_DOWNLOAD_T, &o)) {
423 ZVAL_LONG(&tmp, o);
424 zend_hash_str_update(info, "content_length_download_t", lenof("content_length_download_t"), &tmp);
425 }
426 if (CURLE_OK == curl_easy_getinfo(ch, CURLINFO_CONTENT_LENGTH_UPLOAD, &d)) {
427 ZVAL_DOUBLE(&tmp, d);
428 zend_hash_str_update(info, "content_length_upload", lenof("content_length_upload"), &tmp);
429 }
430 if (CURLE_OK == curl_easy_getinfo(ch, CURLINFO_CONTENT_LENGTH_UPLOAD_T, &o)) {
431 ZVAL_LONG(&tmp, o);
432 zend_hash_str_update(info, "content_length_upload_t", lenof("content_length_upload_t"), &tmp);
433 }
434 if (CURLE_OK == curl_easy_getinfo(ch, CURLINFO_STARTTRANSFER_TIME, &d)) {
435 ZVAL_DOUBLE(&tmp, d);
436 zend_hash_str_update(info, "starttransfer_time", lenof("starttransfer_time"), &tmp);
437 }
438 if (CURLE_OK == curl_easy_getinfo(ch, CURLINFO_CONTENT_TYPE, &c)) {
439 ZVAL_STRING(&tmp, STR_PTR(c));
440 zend_hash_str_update(info, "content_type", lenof("content_type"), &tmp);
441 }
442 if (CURLE_OK == curl_easy_getinfo(ch, CURLINFO_REDIRECT_TIME, &d)) {
443 ZVAL_DOUBLE(&tmp, d);
444 zend_hash_str_update(info, "redirect_time", lenof("redirect_time"), &tmp);
445 }
446 if (CURLE_OK == curl_easy_getinfo(ch, CURLINFO_REDIRECT_COUNT, &l)) {
447 ZVAL_LONG(&tmp, l);
448 zend_hash_str_update(info, "redirect_count", lenof("redirect_count"), &tmp);
449 }
450 if (CURLE_OK == curl_easy_getinfo(ch, CURLINFO_HTTP_CONNECTCODE, &l)) {
451 ZVAL_LONG(&tmp, l);
452 zend_hash_str_update(info, "connect_code", lenof("connect_code"), &tmp);
453 }
454 if (CURLE_OK == curl_easy_getinfo(ch, CURLINFO_HTTPAUTH_AVAIL, &l)) {
455 ZVAL_LONG(&tmp, l);
456 zend_hash_str_update(info, "httpauth_avail", lenof("httpauth_avail"), &tmp);
457 }
458 if (CURLE_OK == curl_easy_getinfo(ch, CURLINFO_PROXYAUTH_AVAIL, &l)) {
459 ZVAL_LONG(&tmp, l);
460 zend_hash_str_update(info, "proxyauth_avail", lenof("proxyauth_avail"), &tmp);
461 }
462 if (CURLE_OK == curl_easy_getinfo(ch, CURLINFO_OS_ERRNO, &l)) {
463 ZVAL_LONG(&tmp, l);
464 zend_hash_str_update(info, "os_errno", lenof("os_errno"), &tmp);
465 }
466 if (CURLE_OK == curl_easy_getinfo(ch, CURLINFO_NUM_CONNECTS, &l)) {
467 ZVAL_LONG(&tmp, l);
468 zend_hash_str_update(info, "num_connects", lenof("num_connects"), &tmp);
469 }
470 if (CURLE_OK == curl_easy_getinfo(ch, CURLINFO_SSL_ENGINES, &s)) {
471 array_init(&tmp);
472 for (p = s; p; p = p->next) {
473 if (p->data) {
474 add_next_index_string(&tmp, p->data);
475 }
476 }
477 zend_hash_str_update(info, "ssl_engines", lenof("ssl_engines"), &tmp);
478 curl_slist_free_all(s);
479 }
480 if (CURLE_OK == curl_easy_getinfo(ch, CURLINFO_REDIRECT_URL, &c)) {
481 ZVAL_STRING(&tmp, STR_PTR(c));
482 zend_hash_str_update(info, "redirect_url", lenof("redirect_url"), &tmp);
483 }
484 #if PHP_HTTP_CURL_VERSION(7,19,0)
485 if (CURLE_OK == curl_easy_getinfo(ch, CURLINFO_PRIMARY_IP, &c)) {
486 ZVAL_STRING(&tmp, STR_PTR(c));
487 zend_hash_str_update(info, "primary_ip", lenof("primary_ip"), &tmp);
488 }
489 #endif
490 #if PHP_HTTP_CURL_VERSION(7,19,0)
491 if (CURLE_OK == curl_easy_getinfo(ch, CURLINFO_APPCONNECT_TIME, &d)) {
492 ZVAL_DOUBLE(&tmp, d);
493 zend_hash_str_update(info, "appconnect_time", lenof("appconnect_time"), &tmp);
494 }
495 #endif
496 #if PHP_HTTP_CURL_VERSION(7,19,4)
497 if (CURLE_OK == curl_easy_getinfo(ch, CURLINFO_CONDITION_UNMET, &l)) {
498 ZVAL_LONG(&tmp, l);
499 zend_hash_str_update(info, "condition_unmet", lenof("condition_unmet"), &tmp);
500 }
501 #endif
502 #if PHP_HTTP_CURL_VERSION(7,21,0)
503 if (CURLE_OK == curl_easy_getinfo(ch, CURLINFO_PRIMARY_PORT, &l)) {
504 ZVAL_LONG(&tmp, l);
505 zend_hash_str_update(info, "primary_port", lenof("primary_port"), &tmp);
506 }
507 #endif
508 #if PHP_HTTP_CURL_VERSION(7,21,0)
509 if (CURLE_OK == curl_easy_getinfo(ch, CURLINFO_LOCAL_IP, &c)) {
510 ZVAL_STRING(&tmp, STR_PTR(c));
511 zend_hash_str_update(info, "local_ip", lenof("local_ip"), &tmp);
512 }
513 #endif
514 #if PHP_HTTP_CURL_VERSION(7,21,0)
515 if (CURLE_OK == curl_easy_getinfo(ch, CURLINFO_LOCAL_PORT, &l)) {
516 ZVAL_LONG(&tmp, l);
517 zend_hash_str_update(info, "local_port", lenof("local_port"), &tmp);
518 }
519 #endif
520 #if PHP_HTTP_CURL_VERSION(7,50,0)
521 if (CURLE_OK == curl_easy_getinfo(ch, CURLINFO_HTTP_VERSION, &l)) {
522 ZVAL_LONG(&tmp, l);
523 zend_hash_str_update(info, "http_version", lenof("http_version"), &tmp);
524 }
525 #endif
526 #if PHP_HTTP_CURL_VERSION(7,52,0)
527 if (CURLE_OK == curl_easy_getinfo(ch, CURLINFO_PROXY_SSL_VERIFYRESULT, &l)) {
528 ZVAL_LONG(&tmp, l);
529 zend_hash_str_update(info, "proxy_ssl_verifyresult", lenof("proxy_ssl_verifyresult"), &tmp);
530 }
531 #endif
532 #if PHP_HTTP_CURL_VERSION(7,52,0)
533 if (CURLE_OK == curl_easy_getinfo(ch, CURLINFO_PROTOCOL, &l)) {
534 ZVAL_LONG(&tmp, l);
535 zend_hash_str_update(info, "protocol", lenof("protocol"), &tmp);
536 }
537 #endif
538 #if PHP_HTTP_CURL_VERSION(7,52,0)
539 if (CURLE_OK == curl_easy_getinfo(ch, CURLINFO_SCHEME, &c)) {
540 ZVAL_STRING(&tmp, STR_PTR(c));
541 zend_hash_str_update(info, "scheme", lenof("scheme"), &tmp);
542 }
543 #endif
544 if (CURLE_OK == curl_easy_getinfo(ch, CURLINFO_TOTAL_TIME_T, &o)) {
545 ZVAL_LONG(&tmp, o);
546 zend_hash_str_update(info, "total_time_t", lenof("total_time_t"), &tmp);
547 }
548 if (CURLE_OK == curl_easy_getinfo(ch, CURLINFO_NAMELOOKUP_TIME_T, &o)) {
549 ZVAL_LONG(&tmp, o);
550 zend_hash_str_update(info, "namelookup_time_t", lenof("namelookup_time_t"), &tmp);
551 }
552 if (CURLE_OK == curl_easy_getinfo(ch, CURLINFO_CONNECT_TIME_T, &o)) {
553 ZVAL_LONG(&tmp, o);
554 zend_hash_str_update(info, "connect_time_t", lenof("connect_time_t"), &tmp);
555 }
556 if (CURLE_OK == curl_easy_getinfo(ch, CURLINFO_PRETRANSFER_TIME_T, &o)) {
557 ZVAL_LONG(&tmp, o);
558 zend_hash_str_update(info, "pretransfer_time_t", lenof("pretransfer_time_t"), &tmp);
559 }
560 if (CURLE_OK == curl_easy_getinfo(ch, CURLINFO_STARTTRANSFER_TIME_T, &o)) {
561 ZVAL_LONG(&tmp, o);
562 zend_hash_str_update(info, "starttransfer_time_t", lenof("starttransfer_time_t"), &tmp);
563 }
564 if (CURLE_OK == curl_easy_getinfo(ch, CURLINFO_REDIRECT_TIME_T, &o)) {
565 ZVAL_LONG(&tmp, o);
566 zend_hash_str_update(info, "redirect_time_t", lenof("redirect_time_t"), &tmp);
567 }
568 if (CURLE_OK == curl_easy_getinfo(ch, CURLINFO_APPCONNECT_TIME_T, &o)) {
569 ZVAL_LONG(&tmp, o);
570 zend_hash_str_update(info, "appconnect_time_t", lenof("appconnect_time_t"), &tmp);
571 }
572 #if PHP_HTTP_CURL_VERSION(7,66,0)
573 if (CURLE_OK == curl_easy_getinfo(ch, CURLINFO_RETRY_AFTER, &o)) {
574 ZVAL_LONG(&tmp, o);
575 zend_hash_str_update(info, "retry_after", lenof("retry_after"), &tmp);
576 }
577 #endif
578 #if PHP_HTTP_CURL_VERSION(7,72,0)
579 if (CURLE_OK == curl_easy_getinfo(ch, CURLINFO_EFFECTIVE_METHOD, &c)) {
580 ZVAL_STRING(&tmp, STR_PTR(c));
581 zend_hash_str_update(info, "effective_method", lenof("effective_method"), &tmp);
582 }
583 #endif
584 #if PHP_HTTP_CURL_VERSION(7,73,0)
585 if (CURLE_OK == curl_easy_getinfo(ch, CURLINFO_PROXY_ERROR, &l)) {
586 ZVAL_LONG(&tmp, l);
587 zend_hash_str_update(info, "proxy_error", lenof("proxy_error"), &tmp);
588 }
589 #endif
590 #if PHP_HTTP_CURL_VERSION(7,76,0)
591 if (CURLE_OK == curl_easy_getinfo(ch, CURLINFO_REFERER, &c)) {
592 ZVAL_STRING(&tmp, STR_PTR(c));
593 zend_hash_str_update(info, "referer", lenof("referer"), &tmp);
594 }
595 #endif
596 #if PHP_HTTP_CURL_VERSION(7,84,0)
597 if (CURLE_OK == curl_easy_getinfo(ch, CURLINFO_CAINFO, &c)) {
598 ZVAL_STRING(&tmp, STR_PTR(c));
599 zend_hash_str_update(info, "cainfo", lenof("cainfo"), &tmp);
600 }
601 #endif
602 #if PHP_HTTP_CURL_VERSION(7,84,0)
603 if (CURLE_OK == curl_easy_getinfo(ch, CURLINFO_CAPATH, &c)) {
604 ZVAL_STRING(&tmp, STR_PTR(c));
605 zend_hash_str_update(info, "capath", lenof("capath"), &tmp);
606 }
607 #endif
608
609 /* END::CURLINFO */
610
611 #if PHP_HTTP_CURL_VERSION(7,34,0)
612 {
613 zval ti_array, subarray;
614 struct curl_tlssessioninfo *ti;
615
616 #if PHP_HTTP_CURL_VERSION(7,48,0)
617 if (CURLE_OK == curl_easy_getinfo(ch, CURLINFO_TLS_SSL_PTR, &ti)) {
618 #else
619 if (CURLE_OK == curl_easy_getinfo(ch, CURLINFO_TLS_SESSION, &ti)) {
620 #endif
621 char *backend;
622
623 ZVAL_NULL(&subarray);
624 array_init(&ti_array);
625
626 switch (ti->backend) {
627 case CURLSSLBACKEND_NONE:
628 backend = "none";
629 break;
630 case CURLSSLBACKEND_OPENSSL:
631 backend = "openssl";
632 #if PHP_HTTP_HAVE_LIBCURL_OPENSSL
633 {
634 #if PHP_HTTP_CURL_VERSION(7,48,0)
635 SSL *ssl = ti->internals;
636 SSL_CTX *ctx = ssl ? SSL_get_SSL_CTX(ssl) : NULL;
637 #else
638 SSL_CTX *ctx = ti->internals;
639 #endif
640
641 array_init(&subarray);
642 if (ctx) {
643 add_assoc_long_ex(&subarray, ZEND_STRL("number"), SSL_CTX_sess_number(ctx));
644 add_assoc_long_ex(&subarray, ZEND_STRL("connect"), SSL_CTX_sess_connect(ctx));
645 add_assoc_long_ex(&subarray, ZEND_STRL("connect_good"), SSL_CTX_sess_connect_good(ctx));
646 add_assoc_long_ex(&subarray, ZEND_STRL("connect_renegotiate"), SSL_CTX_sess_connect_renegotiate(ctx));
647 add_assoc_long_ex(&subarray, ZEND_STRL("hits"), SSL_CTX_sess_hits(ctx));
648 add_assoc_long_ex(&subarray, ZEND_STRL("cache_full"), SSL_CTX_sess_cache_full(ctx));
649 }
650 }
651 #endif
652 break;
653 case CURLSSLBACKEND_GNUTLS:
654 backend = "gnutls";
655 #if PHP_HTTP_HAVE_LIBCURL_GNUTLS
656 {
657 gnutls_session_t sess = ti->internals;
658 char *desc;
659
660 array_init(&subarray);
661 if (sess) {
662 if ((desc = gnutls_session_get_desc(sess))) {
663 add_assoc_string_ex(&subarray, ZEND_STRL("desc"), desc);
664 gnutls_free(desc);
665 }
666 add_assoc_bool_ex(&subarray, ZEND_STRL("resumed"), gnutls_session_is_resumed(sess));
667 }
668 }
669 #endif
670 break;
671 case CURLSSLBACKEND_NSS:
672 backend = "nss";
673 break;
674 #if !PHP_HTTP_CURL_VERSION(7,39,0)
675 case CURLSSLBACKEND_QSOSSL:
676 backend = "qsossl";
677 break;
678 #else
679 case CURLSSLBACKEND_GSKIT:
680 backend = "gskit";
681 break;
682 #endif
683 case CURLSSLBACKEND_POLARSSL:
684 backend = "polarssl";
685 break;
686 case CURLSSLBACKEND_CYASSL:
687 backend = "cyassl";
688 break;
689 case CURLSSLBACKEND_SCHANNEL:
690 backend = "schannel";
691 break;
692 case CURLSSLBACKEND_DARWINSSL:
693 backend = "darwinssl";
694 break;
695 default:
696 backend = "unknown";
697 }
698 add_assoc_string_ex(&ti_array, ZEND_STRL("backend"), backend);
699 add_assoc_zval_ex(&ti_array, ZEND_STRL("internals"), &subarray);
700 zend_hash_str_update(info, "tls_session", lenof("tls_session"), &ti_array);
701 }
702 }
703 #endif
704
705 #if (PHP_HTTP_CURL_VERSION(7,19,1) && PHP_HTTP_HAVE_LIBCURL_OPENSSL) || \
706 (PHP_HTTP_CURL_VERSION(7,34,0) && PHP_HTTP_HAVE_LIBCURL_NSS) || \
707 (PHP_HTTP_CURL_VERSION(7,39,0) && PHP_HTTP_HAVE_LIBCURL_GSKIT) || \
708 (PHP_HTTP_CURL_VERSION(7,42,0) && PHP_HTTP_HAVE_LIBCURL_GNUTLS) || \
709 (PHP_HTTP_CURL_VERSION(7,79,0) && PHP_HTTP_HAVE_LIBCURL_SECURETRANSPORT)
710 {
711 int i;
712 zval ci_array, subarray;
713 struct curl_certinfo *ci;
714 char *colon, *keyname;
715
716 if (CURLE_OK == curl_easy_getinfo(ch, CURLINFO_CERTINFO, &ci)) {
717 array_init(&ci_array);
718
719 for (i = 0; i < ci->num_of_certs; ++i) {
720 s = ci->certinfo[i];
721
722 array_init(&subarray);
723 for (p = s; p; p = p->next) {
724 if (p->data) {
725 if ((colon = strchr(p->data, ':'))) {
726 keyname = estrndup(p->data, colon - p->data);
727 add_assoc_string_ex(&subarray, keyname, colon - p->data, colon + 1);
728 efree(keyname);
729 } else {
730 add_next_index_string(&subarray, p->data);
731 }
732 }
733 }
734 add_next_index_zval(&ci_array, &subarray);
735 }
736 zend_hash_str_update(info, "certinfo", lenof("certinfo"), &ci_array);
737 }
738 }
739 #endif
740 {
741 php_http_curle_storage_t *st = php_http_curle_get_storage(ch);
742
743 ZVAL_LONG(&tmp, st->errorcode);
744 zend_hash_str_update(info, "curlcode", lenof("curlcode"), &tmp);
745 ZVAL_STRING(&tmp, st->errorbuffer);
746 zend_hash_str_update(info, "error", lenof("error"), &tmp);
747 }
748
749 return SUCCESS;
750 }
751
752 static int compare_queue(php_http_client_enqueue_t *e, void *handle)
753 {
754 return handle == ((php_http_client_curl_handler_t *) e->opaque)->handle;
755 }
756
757 static php_http_message_t *php_http_curlm_responseparser(php_http_client_curl_handler_t *h)
758 {
759 php_http_message_t *response;
760 php_http_header_parser_t parser;
761 zval *zh, tmp;
762
763 response = php_http_message_init(NULL, 0, h->response.body);
764 php_http_header_parser_init(&parser);
765 while (h->response.headers.used) {
766 php_http_header_parser_state_t st = php_http_header_parser_parse(&parser,
767 &h->response.headers, PHP_HTTP_HEADER_PARSER_CLEANUP, &response->hdrs,
768 (php_http_info_callback_t) php_http_message_info_callback, (void *) &response);
769 if (PHP_HTTP_HEADER_PARSER_STATE_FAILURE == st) {
770 break;
771 }
772 }
773 php_http_header_parser_dtor(&parser);
774
775 /* move body to right message */
776 if (response->body != h->response.body) {
777 php_http_message_t *ptr = response;
778
779 while (ptr->parent) {
780 ptr = ptr->parent;
781 }
782 php_http_message_body_free(&response->body);
783 response->body = ptr->body;
784 ptr->body = NULL;
785 }
786 php_http_message_body_addref(h->response.body);
787
788 /* let's update the response headers */
789 if ((zh = php_http_message_header(response, ZEND_STRL("Content-Length")))) {
790 ZVAL_COPY(&tmp, zh);
791 zend_hash_str_update(&response->hdrs, "X-Original-Content-Length", lenof("X-Original-Content-Length"), &tmp);
792 }
793 if ((zh = php_http_message_header(response, ZEND_STRL("Transfer-Encoding")))) {
794 ZVAL_COPY(&tmp, zh);
795 zend_hash_str_del(&response->hdrs, "Transfer-Encoding", lenof("Transfer-Encoding"));
796 zend_hash_str_update(&response->hdrs, "X-Original-Transfer-Encoding", lenof("X-Original-Transfer-Encoding"), &tmp);
797 }
798 if ((zh = php_http_message_header(response, ZEND_STRL("Content-Range")))) {
799 ZVAL_COPY(&tmp, zh);
800 zend_hash_str_del(&response->hdrs, "Content-Range", lenof("Content-Range"));
801 zend_hash_str_update(&response->hdrs, "X-Original-Content-Range", lenof("X-Original-Content-Range"), &tmp);
802 }
803 if ((zh = php_http_message_header(response, ZEND_STRL("Content-Encoding")))) {
804 ZVAL_COPY(&tmp, zh);
805 zend_hash_str_del(&response->hdrs, "Content-Encoding", lenof("Content-Encoding"));
806 zend_hash_str_update(&response->hdrs, "X-Original-Content-Encoding", lenof("X-Original-Content-Encoding"), &tmp);
807 }
808 php_http_message_update_headers(response);
809
810 return response;
811 }
812
813 void php_http_client_curl_responsehandler(php_http_client_t *context)
814 {
815 int err_count = 0, remaining = 0;
816 php_http_curle_storage_t *st, *err = NULL;
817 php_http_client_enqueue_t *enqueue;
818 php_http_client_curl_t *curl = context->ctx;
819
820 do {
821 CURLMsg *msg = curl_multi_info_read(curl->handle->multi, &remaining);
822
823 if (msg && CURLMSG_DONE == msg->msg) {
824 if (CURLE_OK != msg->data.result) {
825 st = php_http_curle_get_storage(msg->easy_handle);
826 st->errorcode = msg->data.result;
827
828 /* defer the warnings/exceptions, so the callback is still called for this request */
829 if (!err) {
830 err = ecalloc(remaining + 1, sizeof(*err));
831 }
832 memcpy(&err[err_count], st, sizeof(*st));
833 if (st->url) {
834 err[err_count].url = estrdup(st->url);
835 }
836 err_count++;
837 }
838
839 if ((enqueue = php_http_client_enqueued(context, msg->easy_handle, compare_queue))) {
840 php_http_client_curl_handler_t *handler = enqueue->opaque;
841 php_http_message_t *response = php_http_curlm_responseparser(handler);
842
843 if (response) {
844 context->callback.response.func(context->callback.response.arg, context, &handler->queue, &response);
845 php_http_message_free(&response);
846 }
847 }
848 }
849 } while (remaining);
850
851 if (err_count) {
852 int i = 0;
853
854 do {
855 php_error_docref(NULL, E_WARNING, "%s; %s (%s)", curl_easy_strerror(err[i].errorcode), err[i].errorbuffer, STR_PTR(err[i].url));
856 if (err[i].url) {
857 efree(err[i].url);
858 }
859 } while (++i < err_count);
860
861 efree(err);
862 }
863 }
864
865 void php_http_client_curl_loop(php_http_client_t *client, curl_socket_t s, int curl_action)
866 {
867 CURLMcode rc;
868 php_http_client_curl_t *curl = client->ctx;
869
870 #if DBG_EVENTS
871 fprintf(stderr, "H");
872 #endif
873
874 do {
875 rc = curl_multi_socket_action(curl->handle->multi, s, curl_action, &curl->unfinished);
876 } while (CURLM_CALL_MULTI_PERFORM == rc);
877
878 if (CURLM_OK != rc) {
879 php_error_docref(NULL, E_WARNING, "%s", curl_multi_strerror(rc));
880 }
881
882 php_http_client_curl_responsehandler(client);
883 }
884
885 /* curl options */
886
887 static php_http_options_t php_http_curle_options, php_http_curlm_options;
888
889 #define PHP_HTTP_CURLE_OPTION_CHECK_STRLEN 0x0001
890 #define PHP_HTTP_CURLE_OPTION_CHECK_BASEDIR 0x0002
891 #define PHP_HTTP_CURLE_OPTION_TRANSFORM_MS 0x0004
892 #define PHP_HTTP_CURLE_OPTION_IGNORE_RC 0x0008
893
894 static ZEND_RESULT_CODE php_http_curle_option_set_ssl_verifyhost(php_http_option_t *opt, zval *val, void *userdata)
895 {
896 php_http_client_curl_handler_t *curl = userdata;
897 CURL *ch = curl->handle;
898
899 if (CURLE_OK != curl_easy_setopt(ch, opt->option, Z_TYPE_P(val) == IS_TRUE ? 2 : 0)) {
900 return FAILURE;
901 }
902 return SUCCESS;
903 }
904
905 static ZEND_RESULT_CODE php_http_curle_option_set_cookiesession(php_http_option_t *opt, zval *val, void *userdata)
906 {
907 php_http_client_curl_handler_t *curl = userdata;
908 CURL *ch = curl->handle;
909
910 if (CURLE_OK != curl_easy_setopt(ch, CURLOPT_COOKIESESSION, (long) (Z_TYPE_P(val) == IS_TRUE))) {
911 return FAILURE;
912 }
913 if (Z_TYPE_P(val) == IS_TRUE) {
914 #if DEBUG_COOKIES
915 fprintf(stderr, "CURLOPT_COOKIELIST: SESS\n");
916 #endif
917 if (CURLE_OK != curl_easy_setopt(ch, CURLOPT_COOKIELIST, "SESS")) {
918 return FAILURE;
919 }
920 }
921
922 return SUCCESS;
923 }
924
925 static ZEND_RESULT_CODE php_http_curle_option_set_cookiestore(php_http_option_t *opt, zval *val, void *userdata)
926 {
927 php_http_client_curl_handler_t *curl = userdata;
928 CURL *ch = curl->handle;
929 php_http_curle_storage_t *storage = php_http_curle_get_storage(curl->handle);
930
931 if (storage->cookiestore) {
932 pefree(storage->cookiestore, 1);
933 }
934 if (val && Z_TYPE_P(val) == IS_STRING && Z_STRLEN_P(val)) {
935 storage->cookiestore = pestrndup(Z_STRVAL_P(val), Z_STRLEN_P(val), 1);
936 } else {
937 storage->cookiestore = NULL;
938 }
939
940 #if DEBUG_COOKIES
941 fprintf(stderr, "CURLOPT_COOKIEFILE: %s\n", storage->cookiestore);
942 #endif
943 // does NOT enable ch->data.cookies until transfer; adds to ch->stsate.cookielist
944 if (CURLE_OK != curl_easy_setopt(ch, CURLOPT_COOKIEFILE, storage->cookiestore ? storage->cookiestore : "")) {
945 return FAILURE;
946 }
947 #if DEBUG_COOKIES
948 fprintf(stderr, "CURLOPT_COOKIEJAR: %s\n", storage->cookiestore);
949 #endif
950 // enables ch->data.cookies
951 if (CURLE_OK != curl_easy_setopt(ch, CURLOPT_COOKIEJAR, storage->cookiestore)) {
952 return FAILURE;
953 }
954
955 return SUCCESS;
956 }
957
958 static ZEND_RESULT_CODE php_http_curle_option_set_cookies(php_http_option_t *opt, zval *val, void *userdata)
959 {
960 php_http_client_curl_handler_t *curl = userdata;
961 CURL *ch = curl->handle;
962
963 if (val && Z_TYPE_P(val) != IS_NULL) {
964 HashTable *ht = HASH_OF(val);
965
966 if (curl->options.encode_cookies) {
967 if (SUCCESS == php_http_url_encode_hash_ex(ht, &curl->options.cookies, ZEND_STRL(";"), ZEND_STRL("="), NULL, 0)) {
968 php_http_buffer_fix(&curl->options.cookies);
969 if (CURLE_OK != curl_easy_setopt(ch, CURLOPT_COOKIE, curl->options.cookies.data)) {
970 return FAILURE;
971 }
972 } else {
973 return FAILURE;
974 }
975 } else {
976 php_http_arrkey_t cookie_key;
977 zval *cookie_val;
978
979 ZEND_HASH_FOREACH_KEY_VAL(ht, cookie_key.h, cookie_key.key, cookie_val)
980 {
981 zend_string *zs = zval_get_string(cookie_val);
982
983 php_http_arrkey_stringify(&cookie_key, NULL);
984 php_http_buffer_appendf(&curl->options.cookies, "%s=%s; ", cookie_key.key->val, zs->val);
985 php_http_arrkey_dtor(&cookie_key);
986
987 zend_string_release(zs);
988 }
989 ZEND_HASH_FOREACH_END();
990
991 php_http_buffer_fix(&curl->options.cookies);
992 if (curl->options.cookies.used) {
993 if (CURLE_OK != curl_easy_setopt(ch, CURLOPT_COOKIE, curl->options.cookies.data)) {
994 return FAILURE;
995 }
996 }
997 }
998 } else {
999 php_http_buffer_reset(&curl->options.cookies);
1000 if (CURLE_OK != curl_easy_setopt(ch, CURLOPT_COOKIE, NULL)) {
1001 return FAILURE;
1002 }
1003 }
1004 return SUCCESS;
1005 }
1006
1007 static ZEND_RESULT_CODE php_http_curle_option_set_encodecookies(php_http_option_t *opt, zval *val, void *userdata)
1008 {
1009 php_http_client_curl_handler_t *curl = userdata;
1010
1011 curl->options.encode_cookies = Z_TYPE_P(val) == IS_TRUE;
1012 return SUCCESS;
1013 }
1014
1015 static ZEND_RESULT_CODE php_http_curle_option_set_lastmodified(php_http_option_t *opt, zval *val, void *userdata)
1016 {
1017 php_http_client_curl_handler_t *curl = userdata;
1018 CURL *ch = curl->handle;
1019
1020 if (Z_LVAL_P(val)) {
1021 if (Z_LVAL_P(val) > 0) {
1022 if (CURLE_OK != curl_easy_setopt(ch, CURLOPT_TIMEVALUE, Z_LVAL_P(val))) {
1023 return FAILURE;
1024 }
1025 } else {
1026 if (CURLE_OK != curl_easy_setopt(ch, CURLOPT_TIMEVALUE, (long) sapi_get_request_time() + Z_LVAL_P(val))) {
1027 return FAILURE;
1028 }
1029 }
1030 if (CURLE_OK != curl_easy_setopt(ch, CURLOPT_TIMECONDITION, (long) (curl->options.range_request ? CURL_TIMECOND_IFUNMODSINCE : CURL_TIMECOND_IFMODSINCE))) {
1031 return FAILURE;
1032 }
1033 } else {
1034 if ( CURLE_OK != curl_easy_setopt(ch, CURLOPT_TIMEVALUE, 0)
1035 || CURLE_OK != curl_easy_setopt(ch, CURLOPT_TIMECONDITION, 0)
1036 ) {
1037 return FAILURE;
1038 }
1039 }
1040 return SUCCESS;
1041 }
1042
1043 #if PHP_HTTP_CURL_VERSION(7,64,1)
1044 static ZEND_RESULT_CODE php_http_curle_option_set_altsvc_ctrl(php_http_option_t *opt, zval *val, void *userdata)
1045 {
1046 php_http_client_curl_handler_t *curl = userdata;
1047 CURL *ch = curl->handle;
1048
1049 if (Z_LVAL_P(val)) {
1050 if (CURLE_OK != curl_easy_setopt(ch, CURLOPT_ALTSVC_CTRL, Z_LVAL_P(val))) {
1051 return FAILURE;
1052 }
1053 }
1054 return SUCCESS;
1055 }
1056 #endif
1057
1058 static ZEND_RESULT_CODE php_http_curle_option_set_compress(php_http_option_t *opt, zval *val, void *userdata)
1059 {
1060 php_http_client_curl_handler_t *curl = userdata;
1061 CURL *ch = curl->handle;
1062
1063 #if !PHP_HTTP_CURL_VERSION(7,21,6)
1064 # define CURLOPT_ACCEPT_ENCODING CURLOPT_ENCODING
1065 #endif
1066 if (CURLE_OK != curl_easy_setopt(ch, CURLOPT_ACCEPT_ENCODING, Z_TYPE_P(val) == IS_TRUE ? "" : NULL)) {
1067 return FAILURE;
1068 }
1069 return SUCCESS;
1070 }
1071
1072 static ZEND_RESULT_CODE php_http_curle_option_set_etag(php_http_option_t *opt, zval *val, void *userdata)
1073 {
1074 php_http_client_curl_handler_t *curl = userdata;
1075 php_http_buffer_t header;
1076
1077 if (val && Z_TYPE_P(val) == IS_STRING && Z_STRLEN_P(val)) {
1078 zend_bool is_quoted = !((Z_STRVAL_P(val)[0] != '"') || (Z_STRVAL_P(val)[Z_STRLEN_P(val)-1] != '"'));
1079 php_http_buffer_init(&header);
1080 php_http_buffer_appendf(&header, is_quoted?"%s: %s":"%s: \"%s\"", curl->options.range_request?"If-Match":"If-None-Match", Z_STRVAL_P(val));
1081 php_http_buffer_fix(&header);
1082 curl->options.headers = curl_slist_append(curl->options.headers, header.data);
1083 php_http_buffer_dtor(&header);
1084 }
1085 return SUCCESS;
1086 }
1087
1088 static ZEND_RESULT_CODE php_http_curle_option_set_range(php_http_option_t *opt, zval *val, void *userdata)
1089 {
1090 php_http_client_curl_handler_t *curl = userdata;
1091 CURL *ch = curl->handle;
1092
1093 php_http_buffer_reset(&curl->options.ranges);
1094
1095 if (val && Z_TYPE_P(val) != IS_NULL) {
1096 zval *rr, *rb, *re;
1097 HashTable *ht = HASH_OF(val);
1098
1099 ZEND_HASH_FOREACH_VAL(ht, rr)
1100 {
1101 if (Z_TYPE_P(rr) == IS_ARRAY) {
1102 if (2 == php_http_array_list(Z_ARRVAL_P(rr), 2, &rb, &re)) {
1103 zend_long rbl = zval_get_long(rb), rel = zval_get_long(re);
1104
1105 if (rbl >= 0) {
1106 if (rel > 0) {
1107 php_http_buffer_appendf(&curl->options.ranges, "%ld-%ld,", rbl, rel);
1108 } else {
1109 php_http_buffer_appendf(&curl->options.ranges, "%ld-", rbl);
1110 }
1111 } else if (rel > 0) {
1112 php_http_buffer_appendf(&curl->options.ranges, "-%ld", rel);
1113 }
1114 }
1115 }
1116 }
1117 ZEND_HASH_FOREACH_END();
1118
1119 if (curl->options.ranges.used) {
1120 curl->options.range_request = 1;
1121 /* ditch last comma */
1122 curl->options.ranges.data[curl->options.ranges.used - 1] = '\0';
1123 }
1124 }
1125
1126 if (CURLE_OK != curl_easy_setopt(ch, CURLOPT_RANGE, curl->options.ranges.data)) {
1127 return FAILURE;
1128 }
1129 return SUCCESS;
1130 }
1131
1132 static ZEND_RESULT_CODE php_http_curle_option_set_resume(php_http_option_t *opt, zval *val, void *userdata)
1133 {
1134 php_http_client_curl_handler_t *curl = userdata;
1135 CURL *ch = curl->handle;
1136
1137 if (Z_LVAL_P(val) > 0) {
1138 curl->options.range_request = 1;
1139 }
1140 if (CURLE_OK != curl_easy_setopt(ch, CURLOPT_RESUME_FROM, Z_LVAL_P(val))) {
1141 return FAILURE;
1142 }
1143 return SUCCESS;
1144 }
1145
1146 static ZEND_RESULT_CODE php_http_curle_option_set_retrydelay(php_http_option_t *opt, zval *val, void *userdata)
1147 {
1148 php_http_client_curl_handler_t *curl = userdata;
1149
1150 curl->options.retry.delay = Z_DVAL_P(val);
1151 return SUCCESS;
1152 }
1153
1154 static ZEND_RESULT_CODE php_http_curle_option_set_retrycount(php_http_option_t *opt, zval *val, void *userdata)
1155 {
1156 php_http_client_curl_handler_t *curl = userdata;
1157
1158 curl->options.retry.count = Z_LVAL_P(val);
1159 return SUCCESS;
1160 }
1161
1162 static ZEND_RESULT_CODE php_http_curle_option_set_redirect(php_http_option_t *opt, zval *val, void *userdata)
1163 {
1164 php_http_client_curl_handler_t *curl = userdata;
1165 CURL *ch = curl->handle;
1166
1167 if ( CURLE_OK != curl_easy_setopt(ch, CURLOPT_FOLLOWLOCATION, Z_LVAL_P(val) ? 1L : 0L)
1168 || CURLE_OK != curl_easy_setopt(ch, CURLOPT_MAXREDIRS, curl->options.redirects = Z_LVAL_P(val))
1169 ) {
1170 return FAILURE;
1171 }
1172 return SUCCESS;
1173 }
1174
1175 static ZEND_RESULT_CODE php_http_curle_option_set_portrange(php_http_option_t *opt, zval *val, void *userdata)
1176 {
1177 php_http_client_curl_handler_t *curl = userdata;
1178 CURL *ch = curl->handle;
1179 long localport = 0, localportrange = 0;
1180
1181 if (val && Z_TYPE_P(val) != IS_NULL) {
1182 zval *zps, *zpe;
1183
1184 switch (php_http_array_list(Z_ARRVAL_P(val), 2, &zps, &zpe)) {
1185 case 2:
1186 localportrange = labs(zval_get_long(zps)-zval_get_long(zpe))+1L;
1187 /* no break */
1188 case 1:
1189 localport = (zval_get_long(zpe) > 0) ? MIN(zval_get_long(zps), zval_get_long(zpe)) : zval_get_long(zps);
1190 break;
1191 default:
1192 break;
1193 }
1194 }
1195 if ( CURLE_OK != curl_easy_setopt(ch, CURLOPT_LOCALPORT, localport)
1196 || CURLE_OK != curl_easy_setopt(ch, CURLOPT_LOCALPORTRANGE, localportrange)
1197 ) {
1198 return FAILURE;
1199 }
1200 return SUCCESS;
1201 }
1202
1203 #if PHP_HTTP_CURL_VERSION(7,37,0)
1204 static ZEND_RESULT_CODE php_http_curle_option_set_proxyheader(php_http_option_t *opt, zval *val, void *userdata)
1205 {
1206 php_http_client_curl_handler_t *curl = userdata;
1207
1208 if (val && Z_TYPE_P(val) != IS_NULL) {
1209 php_http_arrkey_t header_key;
1210 zval *header_val;
1211 php_http_buffer_t header;
1212
1213 php_http_buffer_init(&header);
1214 ZEND_HASH_FOREACH_KEY_VAL(Z_ARRVAL_P(val), header_key.h, header_key.key, header_val)
1215 {
1216 if (header_key.key) {
1217 zend_string *zs = zval_get_string(header_val);
1218
1219 php_http_buffer_appendf(&header, "%s: %s", header_key.key->val, zs->val);
1220 zend_string_release(zs);
1221
1222 php_http_buffer_fix(&header);
1223 curl->options.proxyheaders = curl_slist_append(curl->options.proxyheaders, header.data);
1224 php_http_buffer_reset(&header);
1225
1226 }
1227 }
1228 ZEND_HASH_FOREACH_END();
1229 php_http_buffer_dtor(&header);
1230 }
1231 if (CURLE_OK != curl_easy_setopt(curl->handle, CURLOPT_PROXYHEADER, curl->options.proxyheaders)) {
1232 return FAILURE;
1233 }
1234 if (CURLE_OK != curl_easy_setopt(curl->handle, CURLOPT_HEADEROPT, CURLHEADER_SEPARATE)) {
1235 curl_easy_setopt(curl->handle, CURLOPT_PROXYHEADER, NULL);
1236 return FAILURE;
1237 }
1238 return SUCCESS;
1239 }
1240 #endif
1241
1242 #if PHP_HTTP_CURL_VERSION(7,21,3)
1243 static ZEND_RESULT_CODE php_http_curle_option_set_resolve(php_http_option_t *opt, zval *val, void *userdata)
1244 {
1245 php_http_client_curl_handler_t *curl = userdata;
1246 CURL *ch = curl->handle;
1247
1248 if (val && Z_TYPE_P(val) != IS_NULL) {
1249 HashTable *ht = HASH_OF(val);
1250 zval *data;
1251
1252 ZEND_HASH_FOREACH_VAL(ht, data)
1253 {
1254 zend_string *zs = zval_get_string(data);
1255 curl->options.resolve = curl_slist_append(curl->options.resolve, zs->val);
1256 zend_string_release(zs);
1257 }
1258 ZEND_HASH_FOREACH_END();
1259
1260 if (CURLE_OK != curl_easy_setopt(ch, CURLOPT_RESOLVE, curl->options.resolve)) {
1261 return FAILURE;
1262 }
1263 } else {
1264 if (CURLE_OK != curl_easy_setopt(ch, CURLOPT_RESOLVE, NULL)) {
1265 return FAILURE;
1266 }
1267 }
1268 return SUCCESS;
1269 }
1270 #endif
1271
1272 #if PHP_HTTP_CURL_VERSION(7,21,4) && PHP_HTTP_HAVE_LIBCURL_TLSAUTH_TYPE
1273 static ZEND_RESULT_CODE php_http_curle_option_set_ssl_tlsauthtype(php_http_option_t *opt, zval *val, void *userdata)
1274 {
1275 php_http_client_curl_handler_t *curl = userdata;
1276 CURL *ch = curl->handle;
1277
1278 if (val && Z_LVAL_P(val)) {
1279 switch (Z_LVAL_P(val)) {
1280 case CURL_TLSAUTH_NONE:
1281 break;
1282 case CURL_TLSAUTH_SRP:
1283 if (CURLE_OK == curl_easy_setopt(ch, opt->option, "SRP")) {
1284 return SUCCESS;
1285 }
1286 /* no break */
1287 default:
1288 return FAILURE;
1289 }
1290 }
1291 if (CURLE_OK != curl_easy_setopt(ch, opt->option, "NONE")) {
1292 return FAILURE;
1293 }
1294 return SUCCESS;
1295 }
1296 #endif
1297
1298 static void php_http_curle_options_init(php_http_options_t *registry)
1299 {
1300 php_http_option_t *opt;
1301
1302 /* url options */
1303 #if PHP_HTTP_CURL_VERSION(7,42,0)
1304 php_http_option_register(registry, ZEND_STRL("path_as_is"), CURLOPT_PATH_AS_IS, _IS_BOOL);
1305 #endif
1306
1307 /* proxy */
1308 php_http_option_register(registry, ZEND_STRL("proxyhost"), CURLOPT_PROXY, IS_STRING);
1309 php_http_option_register(registry, ZEND_STRL("proxytype"), CURLOPT_PROXYTYPE, IS_LONG);
1310 php_http_option_register(registry, ZEND_STRL("proxyport"), CURLOPT_PROXYPORT, IS_LONG);
1311 if ((opt = php_http_option_register(registry, ZEND_STRL("proxyauth"), CURLOPT_PROXYUSERPWD, IS_STRING))) {
1312 opt->flags |= PHP_HTTP_CURLE_OPTION_CHECK_STRLEN;
1313 }
1314 if ((opt = php_http_option_register(registry, ZEND_STRL("proxyauthtype"), CURLOPT_PROXYAUTH, IS_LONG))) {
1315 Z_LVAL(opt->defval) = CURLAUTH_ANYSAFE;
1316 }
1317 php_http_option_register(registry, ZEND_STRL("proxytunnel"), CURLOPT_HTTPPROXYTUNNEL, _IS_BOOL);
1318 #if PHP_HTTP_CURL_VERSION(7,19,4)
1319 php_http_option_register(registry, ZEND_STRL("noproxy"), CURLOPT_NOPROXY, IS_STRING);
1320 #endif
1321 #if PHP_HTTP_CURL_VERSION(7,55,0)
1322 php_http_option_register(registry, ZEND_STRL("socks5_auth"), CURLOPT_SOCKS5_AUTH, IS_LONG);
1323 #endif
1324
1325 #if PHP_HTTP_CURL_VERSION(7,37,0)
1326 if ((opt = php_http_option_register(registry, ZEND_STRL("proxyheader"), CURLOPT_PROXYHEADER, IS_ARRAY))) {
1327 opt->setter = php_http_curle_option_set_proxyheader;
1328 }
1329 #endif
1330 #if PHP_HTTP_CURL_VERSION(7,43,0)
1331 if (PHP_HTTP_CURL_FEATURE(CURL_VERSION_GSSAPI)
1332 && (opt = php_http_option_register(registry, ZEND_STRL("proxy_service_name"), CURLOPT_PROXY_SERVICE_NAME, IS_STRING))
1333 ) {
1334 opt->flags |= PHP_HTTP_CURLE_OPTION_CHECK_STRLEN;
1335 }
1336 #endif
1337 #if PHP_HTTP_CURL_VERSION(7,60,0)
1338 php_http_option_register(registry, ZEND_STRL("haproxy_protocol"), CURLOPT_HAPROXYPROTOCOL, _IS_BOOL);
1339 #endif
1340
1341 /* unix sockets */
1342 #if PHP_HTTP_CURL_VERSION(7,40,0)
1343 if (PHP_HTTP_CURL_FEATURE(CURL_VERSION_UNIX_SOCKETS)) {
1344 if ((opt = php_http_option_register(registry, ZEND_STRL("unix_socket_path"), CURLOPT_UNIX_SOCKET_PATH, IS_STRING))) {
1345 opt->flags |= PHP_HTTP_CURLE_OPTION_CHECK_STRLEN;
1346 opt->flags |= PHP_HTTP_CURLE_OPTION_CHECK_BASEDIR;
1347 }
1348 #if PHP_HTTP_CURL_VERSION(7,53,0)
1349 if ((opt = php_http_option_register(registry, ZEND_STRL("abstract_unix_socket"), CURLOPT_ABSTRACT_UNIX_SOCKET, IS_STRING))) {
1350 opt->flags |= PHP_HTTP_CURLE_OPTION_CHECK_STRLEN;
1351 }
1352 #endif
1353 }
1354 #endif
1355
1356 /* dns */
1357 if ((opt = php_http_option_register(registry, ZEND_STRL("dns_cache_timeout"), CURLOPT_DNS_CACHE_TIMEOUT, IS_LONG))) {
1358 Z_LVAL(opt->defval) = 60;
1359 }
1360 php_http_option_register(registry, ZEND_STRL("ipresolve"), CURLOPT_IPRESOLVE, IS_LONG);
1361 #if PHP_HTTP_CURL_VERSION(7,21,3)
1362 if ((opt = php_http_option_register(registry, ZEND_STRL("resolve"), CURLOPT_RESOLVE, IS_ARRAY))) {
1363 opt->setter = php_http_curle_option_set_resolve;
1364 }
1365 #endif
1366 #if PHP_HTTP_HAVE_LIBCURL_ARES
1367 # if PHP_HTTP_CURL_VERSION(7,24,0)
1368 if ((opt = php_http_option_register(registry, ZEND_STRL("dns_servers"), CURLOPT_DNS_SERVERS, IS_STRING))) {
1369 opt->flags |= PHP_HTTP_CURLE_OPTION_CHECK_STRLEN;
1370 }
1371 # endif
1372 # if PHP_HTTP_CURL_VERSION(7,33,0)
1373 if ((opt = php_http_option_register(registry, ZEND_STRL("dns_interface"), CURLOPT_DNS_INTERFACE, IS_STRING))) {
1374 opt->flags |= PHP_HTTP_CURLE_OPTION_CHECK_STRLEN;
1375 }
1376 if ((opt = php_http_option_register(registry, ZEND_STRL("dns_local_ip4"), CURLOPT_DNS_LOCAL_IP4, IS_STRING))) {
1377 opt->flags |= PHP_HTTP_CURLE_OPTION_CHECK_STRLEN;
1378 }
1379 if ((opt = php_http_option_register(registry, ZEND_STRL("dns_local_ip6"), CURLOPT_DNS_LOCAL_IP6, IS_STRING))) {
1380 opt->flags |= PHP_HTTP_CURLE_OPTION_CHECK_STRLEN;
1381 }
1382 # endif
1383 #endif
1384 #if PHP_HTTP_CURL_VERSION(7,60,0)
1385 php_http_option_register(registry, ZEND_STRL("dns_shuffle_addresses"), CURLOPT_DNS_SHUFFLE_ADDRESSES, _IS_BOOL);
1386 #endif
1387 #if PHP_HTTP_CURL_VERSION(7,62,0)
1388 php_http_option_register(registry, ZEND_STRL("doh_url"), CURLOPT_DOH_URL, IS_STRING);
1389 #endif
1390
1391
1392 /* limits */
1393 php_http_option_register(registry, ZEND_STRL("low_speed_limit"), CURLOPT_LOW_SPEED_LIMIT, IS_LONG);
1394 php_http_option_register(registry, ZEND_STRL("low_speed_time"), CURLOPT_LOW_SPEED_TIME, IS_LONG);
1395
1396 /* LSF weirdance
1397 php_http_option_register(registry, ZEND_STRL("max_send_speed"), CURLOPT_MAX_SEND_SPEED_LARGE, IS_LONG);
1398 php_http_option_register(registry, ZEND_STRL("max_recv_speed"), CURLOPT_MAX_RECV_SPEED_LARGE, IS_LONG);
1399 */
1400
1401 /* connection handling */
1402 /* crashes
1403 if ((opt = php_http_option_register(registry, ZEND_STRL("maxconnects"), CURLOPT_MAXCONNECTS, IS_LONG))) {
1404 Z_LVAL(opt->defval) = 5;
1405 }
1406 */
1407 php_http_option_register(registry, ZEND_STRL("fresh_connect"), CURLOPT_FRESH_CONNECT, _IS_BOOL);
1408 php_http_option_register(registry, ZEND_STRL("forbid_reuse"), CURLOPT_FORBID_REUSE, _IS_BOOL);
1409 #if PHP_HTTP_CURL_VERSION(7,65,0)
1410 if ((opt = php_http_option_register(registry, ZEND_STRL("maxage_conn"), CURLOPT_MAXAGE_CONN, IS_LONG))) {
1411 ZVAL_LONG(&opt->defval, 118);
1412 }
1413 #endif
1414
1415 /* outgoing interface */
1416 php_http_option_register(registry, ZEND_STRL("interface"), CURLOPT_INTERFACE, IS_STRING);
1417 if ((opt = php_http_option_register(registry, ZEND_STRL("portrange"), CURLOPT_LOCALPORT, IS_ARRAY))) {
1418 opt->setter = php_http_curle_option_set_portrange;
1419 }
1420
1421 /* another endpoint port */
1422 php_http_option_register(registry, ZEND_STRL("port"), CURLOPT_PORT, IS_LONG);
1423
1424 /* RFC4007 zone_id */
1425 #if PHP_HTTP_CURL_VERSION(7,19,0)
1426 php_http_option_register(registry, ZEND_STRL("address_scope"), CURLOPT_ADDRESS_SCOPE, IS_LONG);
1427 #endif
1428
1429 /* auth */
1430 if ((opt = php_http_option_register(registry, ZEND_STRL("httpauth"), CURLOPT_USERPWD, IS_STRING))) {
1431 opt->flags |= PHP_HTTP_CURLE_OPTION_CHECK_STRLEN;
1432 }
1433 if ((opt = php_http_option_register(registry, ZEND_STRL("httpauthtype"), CURLOPT_HTTPAUTH, IS_LONG))) {
1434 Z_LVAL(opt->defval) = CURLAUTH_ANYSAFE;
1435 }
1436 #if PHP_HTTP_CURL_VERSION(7,43,0)
1437 if (PHP_HTTP_CURL_FEATURE(CURL_VERSION_SSPI) || PHP_HTTP_CURL_FEATURE(CURL_VERSION_GSSAPI))
1438 if ((opt = php_http_option_register(registry, ZEND_STRL("service_name"), CURLOPT_SERVICE_NAME, IS_STRING))) {
1439 opt->flags |= PHP_HTTP_CURLE_OPTION_CHECK_STRLEN;
1440 }
1441 #endif
1442 #if PHP_HTTP_CURL_VERSION(7,61,0)
1443 if ((opt = php_http_option_register(registry, ZEND_STRL("xoauth2_bearer"), CURLOPT_XOAUTH2_BEARER, IS_STRING))) {
1444 opt->flags |= PHP_HTTP_CURLE_OPTION_CHECK_STRLEN;
1445 }
1446 #endif
1447 #if PHP_HTTP_CURL_VERSION(7,75,0)
1448 php_http_option_register(registry, ZEND_STRL("aws_sigv4"), CURLOPT_AWS_SIGV4, IS_STRING);
1449 #endif
1450
1451 /* redirects */
1452 if ((opt = php_http_option_register(registry, ZEND_STRL("redirect"), CURLOPT_FOLLOWLOCATION, IS_LONG))) {
1453 opt->setter = php_http_curle_option_set_redirect;
1454 }
1455 php_http_option_register(registry, ZEND_STRL("unrestricted_auth"), CURLOPT_UNRESTRICTED_AUTH, _IS_BOOL);
1456 #if PHP_HTTP_CURL_VERSION(7,19,1)
1457 php_http_option_register(registry, ZEND_STRL("postredir"), CURLOPT_POSTREDIR, IS_LONG);
1458 #endif
1459
1460 /* retries */
1461 if ((opt = php_http_option_register(registry, ZEND_STRL("retrycount"), 0, IS_LONG))) {
1462 opt->setter = php_http_curle_option_set_retrycount;
1463 }
1464 if ((opt = php_http_option_register(registry, ZEND_STRL("retrydelay"), 0, IS_DOUBLE))) {
1465 opt->setter = php_http_curle_option_set_retrydelay;
1466 }
1467
1468 /* referer */
1469 if ((opt = php_http_option_register(registry, ZEND_STRL("referer"), CURLOPT_REFERER, IS_STRING))) {
1470 opt->flags |= PHP_HTTP_CURLE_OPTION_CHECK_STRLEN;
1471 }
1472 if ((opt = php_http_option_register(registry, ZEND_STRL("autoreferer"), CURLOPT_AUTOREFERER, _IS_BOOL))) {
1473 ZVAL_BOOL(&opt->defval, 1);
1474 }
1475
1476 /* useragent */
1477 if ((opt = php_http_option_register(registry, ZEND_STRL("useragent"), CURLOPT_USERAGENT, IS_STRING))) {
1478 /* don't check strlen, to allow sending no useragent at all */
1479 ZVAL_PSTRING(&opt->defval,
1480 "PECL_HTTP/" PHP_PECL_HTTP_VERSION " "
1481 "PHP/" PHP_VERSION " "
1482 "libcurl/" LIBCURL_VERSION);
1483 }
1484
1485 /* resume */
1486 if ((opt = php_http_option_register(registry, ZEND_STRL("resume"), CURLOPT_RESUME_FROM, IS_LONG))) {
1487 opt->setter = php_http_curle_option_set_resume;
1488 }
1489 /* ranges */
1490 if ((opt = php_http_option_register(registry, ZEND_STRL("range"), CURLOPT_RANGE, IS_ARRAY))) {
1491 opt->setter = php_http_curle_option_set_range;
1492 }
1493
1494 /* etag */
1495 if ((opt = php_http_option_register(registry, ZEND_STRL("etag"), 0, IS_STRING))) {
1496 opt->flags |= PHP_HTTP_CURLE_OPTION_CHECK_STRLEN;
1497 opt->setter = php_http_curle_option_set_etag;
1498 }
1499
1500 /* compression */
1501 if ((opt = php_http_option_register(registry, ZEND_STRL("compress"), 0, _IS_BOOL))) {
1502 opt->setter = php_http_curle_option_set_compress;
1503 }
1504
1505 /* lastmodified */
1506 if ((opt = php_http_option_register(registry, ZEND_STRL("lastmodified"), 0, IS_LONG))) {
1507 opt->setter = php_http_curle_option_set_lastmodified;
1508 }
1509
1510 /* cookies */
1511 if ((opt = php_http_option_register(registry, ZEND_STRL("encodecookies"), 0, _IS_BOOL))) {
1512 opt->setter = php_http_curle_option_set_encodecookies;
1513 ZVAL_BOOL(&opt->defval, 1);
1514 }
1515 if ((opt = php_http_option_register(registry, ZEND_STRL("cookies"), 0, IS_ARRAY))) {
1516 opt->setter = php_http_curle_option_set_cookies;
1517 }
1518
1519 /* cookiesession, don't load session cookies from cookiestore */
1520 if ((opt = php_http_option_register(registry, ZEND_STRL("cookiesession"), CURLOPT_COOKIESESSION, _IS_BOOL))) {
1521 opt->setter = php_http_curle_option_set_cookiesession;
1522 }
1523 /* cookiestore, read initial cookies from that file and store cookies back into that file */
1524 if ((opt = php_http_option_register(registry, ZEND_STRL("cookiestore"), 0, IS_STRING))) {
1525 opt->flags |= PHP_HTTP_CURLE_OPTION_CHECK_STRLEN;
1526 opt->flags |= PHP_HTTP_CURLE_OPTION_CHECK_BASEDIR;
1527 opt->setter = php_http_curle_option_set_cookiestore;
1528 }
1529
1530 /* maxfilesize */
1531 php_http_option_register(registry, ZEND_STRL("maxfilesize"), CURLOPT_MAXFILESIZE, IS_LONG);
1532
1533 /* http protocol version */
1534 php_http_option_register(registry, ZEND_STRL("protocol"), CURLOPT_HTTP_VERSION, IS_LONG);
1535 #if PHP_HTTP_CURL_VERSION(7,64,0)
1536 php_http_option_register(registry, ZEND_STRL("http09_allowed"), CURLOPT_HTTP09_ALLOWED, _IS_BOOL);
1537 #endif
1538
1539 /* timeouts */
1540 if ((opt = php_http_option_register(registry, ZEND_STRL("timeout"), CURLOPT_TIMEOUT_MS, IS_DOUBLE))) {
1541 opt->flags |= PHP_HTTP_CURLE_OPTION_TRANSFORM_MS;
1542 }
1543 if ((opt = php_http_option_register(registry, ZEND_STRL("connecttimeout"), CURLOPT_CONNECTTIMEOUT_MS, IS_DOUBLE))) {
1544 opt->flags |= PHP_HTTP_CURLE_OPTION_TRANSFORM_MS;
1545 Z_DVAL(opt->defval) = 3;
1546 }
1547 #if PHP_HTTP_CURL_VERSION(7,36,0)
1548 if ((opt = php_http_option_register(registry, ZEND_STRL("expect_100_timeout"), CURLOPT_EXPECT_100_TIMEOUT_MS, IS_DOUBLE))) {
1549 opt->flags |= PHP_HTTP_CURLE_OPTION_TRANSFORM_MS;
1550 Z_DVAL(opt->defval) = 1;
1551 }
1552 #endif
1553
1554 /* tcp */
1555 php_http_option_register(registry, ZEND_STRL("tcp_nodelay"), CURLOPT_TCP_NODELAY, _IS_BOOL);
1556 #if PHP_HTTP_CURL_VERSION(7,25,0)
1557 php_http_option_register(registry, ZEND_STRL("tcp_keepalive"), CURLOPT_TCP_KEEPALIVE, _IS_BOOL);
1558 if ((opt = php_http_option_register(registry, ZEND_STRL("tcp_keepidle"), CURLOPT_TCP_KEEPIDLE, IS_LONG))) {
1559 Z_LVAL(opt->defval) = 60;
1560 }
1561 if ((opt = php_http_option_register(registry, ZEND_STRL("tcp_keepintvl"), CURLOPT_TCP_KEEPINTVL, IS_LONG))) {
1562 Z_LVAL(opt->defval) = 60;
1563 }
1564 #endif
1565 #if PHP_HTTP_CURL_VERSION(7,49,0)
1566 # if defined(linux) || defined(__APPLE__)
1567 /* CURLOPT_TCP_FASTOPEN is not supported (yet) on Windows */
1568 php_http_option_register(registry, ZEND_STRL("tcp_fastopen"), CURLOPT_TCP_FASTOPEN, _IS_BOOL);
1569 # endif
1570 #endif
1571
1572 /* ssl */
1573 if (PHP_HTTP_CURL_FEATURE(CURL_VERSION_SSL)) {
1574 php_http_option_t *ssl_opt, *proxy_opt;
1575
1576 if ((ssl_opt = php_http_option_register(registry, ZEND_STRL("ssl"), 0, IS_ARRAY))) {
1577 php_http_options_t *ssl_registry = &ssl_opt->suboptions;
1578
1579 if ((opt = php_http_option_register(ssl_registry, ZEND_STRL("cert"), CURLOPT_SSLCERT, IS_STRING))) {
1580 opt->flags |= PHP_HTTP_CURLE_OPTION_CHECK_STRLEN;
1581 opt->flags |= PHP_HTTP_CURLE_OPTION_CHECK_BASEDIR;
1582 }
1583 if ((opt = php_http_option_register(ssl_registry, ZEND_STRL("certtype"), CURLOPT_SSLCERTTYPE, IS_STRING))) {
1584 opt->flags |= PHP_HTTP_CURLE_OPTION_CHECK_STRLEN;
1585 ZVAL_PSTRING(&opt->defval, "PEM");
1586 }
1587 if ((opt = php_http_option_register(ssl_registry, ZEND_STRL("key"), CURLOPT_SSLKEY, IS_STRING))) {
1588 opt->flags |= PHP_HTTP_CURLE_OPTION_CHECK_STRLEN;
1589 opt->flags |= PHP_HTTP_CURLE_OPTION_CHECK_BASEDIR;
1590 }
1591 if ((opt = php_http_option_register(ssl_registry, ZEND_STRL("keytype"), CURLOPT_SSLKEYTYPE, IS_STRING))) {
1592 opt->flags |= PHP_HTTP_CURLE_OPTION_CHECK_STRLEN;
1593 ZVAL_PSTRING(&opt->defval, "PEM");
1594 }
1595 if ((opt = php_http_option_register(ssl_registry, ZEND_STRL("keypasswd"), CURLOPT_SSLKEYPASSWD, IS_STRING))) {
1596 opt->flags |= PHP_HTTP_CURLE_OPTION_CHECK_STRLEN;
1597 }
1598 php_http_option_register(ssl_registry, ZEND_STRL("engine"), CURLOPT_SSLENGINE, IS_STRING);
1599 php_http_option_register(ssl_registry, ZEND_STRL("version"), CURLOPT_SSLVERSION, IS_LONG);
1600 if ((opt = php_http_option_register(ssl_registry, ZEND_STRL("verifypeer"), CURLOPT_SSL_VERIFYPEER, _IS_BOOL))) {
1601 ZVAL_BOOL(&opt->defval, 1);
1602 }
1603 if ((opt = php_http_option_register(ssl_registry, ZEND_STRL("verifyhost"), CURLOPT_SSL_VERIFYHOST, _IS_BOOL))) {
1604 ZVAL_BOOL(&opt->defval, 1);
1605 opt->setter = php_http_curle_option_set_ssl_verifyhost;
1606 }
1607 #if PHP_HTTP_CURL_VERSION(7,41,0) && (PHP_HTTP_HAVE_LIBCURL_OPENSSL || PHP_HTTP_HAVE_LIBCURL_NSS || PHP_HTTP_HAVE_LIBCURL_GNUTLS)
1608 php_http_option_register(ssl_registry, ZEND_STRL("verifystatus"), CURLOPT_SSL_VERIFYSTATUS, _IS_BOOL);
1609 #endif
1610 php_http_option_register(ssl_registry, ZEND_STRL("cipher_list"), CURLOPT_SSL_CIPHER_LIST, IS_STRING);
1611 #if PHP_HTTP_HAVE_LIBCURL_CAINFO
1612 if ((opt = php_http_option_register(ssl_registry, ZEND_STRL("cainfo"), CURLOPT_CAINFO, IS_STRING))) {
1613 opt->flags |= PHP_HTTP_CURLE_OPTION_CHECK_STRLEN;
1614 opt->flags |= PHP_HTTP_CURLE_OPTION_CHECK_BASEDIR;
1615 # ifdef PHP_HTTP_CAINFO
1616 ZVAL_PSTRING(&opt->defval, PHP_HTTP_CAINFO);
1617 # endif
1618 }
1619 #endif
1620 #if PHP_HTTP_HAVE_LIBCURL_CAPATH
1621 if ((opt = php_http_option_register(ssl_registry, ZEND_STRL("capath"), CURLOPT_CAPATH, IS_STRING))) {
1622 opt->flags |= PHP_HTTP_CURLE_OPTION_CHECK_STRLEN;
1623 opt->flags |= PHP_HTTP_CURLE_OPTION_CHECK_BASEDIR;
1624 # ifdef PHP_HTTP_CAPATH
1625 ZVAL_PSTRING(&opt->defval, PHP_HTTP_CAPATH);
1626 # endif
1627 }
1628 #endif
1629 if ((opt = php_http_option_register(ssl_registry, ZEND_STRL("random_file"), CURLOPT_RANDOM_FILE, IS_STRING))) {
1630 opt->flags |= PHP_HTTP_CURLE_OPTION_CHECK_STRLEN;
1631 opt->flags |= PHP_HTTP_CURLE_OPTION_CHECK_BASEDIR;
1632 }
1633 if ((opt = php_http_option_register(ssl_registry, ZEND_STRL("egdsocket"), CURLOPT_EGDSOCKET, IS_STRING))) {
1634 opt->flags |= PHP_HTTP_CURLE_OPTION_CHECK_STRLEN;
1635 opt->flags |= PHP_HTTP_CURLE_OPTION_CHECK_BASEDIR;
1636 }
1637 #if PHP_HTTP_CURL_VERSION(7,19,0)
1638 if ((opt = php_http_option_register(ssl_registry, ZEND_STRL("issuercert"), CURLOPT_ISSUERCERT, IS_STRING))) {
1639 opt->flags |= PHP_HTTP_CURLE_OPTION_CHECK_STRLEN;
1640 opt->flags |= PHP_HTTP_CURLE_OPTION_CHECK_BASEDIR;
1641 }
1642 # if PHP_HTTP_HAVE_LIBCURL_OPENSSL
1643 if ((opt = php_http_option_register(ssl_registry, ZEND_STRL("crlfile"), CURLOPT_CRLFILE, IS_STRING))) {
1644 opt->flags |= PHP_HTTP_CURLE_OPTION_CHECK_STRLEN;
1645 opt->flags |= PHP_HTTP_CURLE_OPTION_CHECK_BASEDIR;
1646 }
1647 # endif
1648 #endif
1649 #if (PHP_HTTP_CURL_VERSION(7,19,1) && PHP_HTTP_HAVE_LIBCURL_OPENSSL) || (PHP_HTTP_CURL_VERSION(7,34,0) && PHP_HTTP_HAVE_LIBCURL_NSS) || (PHP_HTTP_CURL_VERSION(7,42,0) && defined(PHP_HTTP_HAVE_LIBCURL_GNUTLS)) || (PHP_HTTP_CURL_VERSION(7,39,0) && defined(PHP_HTTP_HAVE_LIBCURL_GSKIT))
1650 if ((opt = php_http_option_register(ssl_registry, ZEND_STRL("certinfo"), CURLOPT_CERTINFO, _IS_BOOL))) {
1651 ZVAL_FALSE(&opt->defval);
1652 }
1653 #endif
1654 #if PHP_HTTP_CURL_VERSION(7,36,0)
1655 if ((opt = php_http_option_register(ssl_registry, ZEND_STRL("enable_npn"), CURLOPT_SSL_ENABLE_NPN, _IS_BOOL))) {
1656 ZVAL_BOOL(&opt->defval, 1);
1657 }
1658 if ((opt = php_http_option_register(ssl_registry, ZEND_STRL("enable_alpn"), CURLOPT_SSL_ENABLE_ALPN, _IS_BOOL))) {
1659 ZVAL_BOOL(&opt->defval, 1);
1660 }
1661 #endif
1662 #if PHP_HTTP_CURL_VERSION(7,39,0)
1663 /* FIXME: see http://curl.haxx.se/libcurl/c/CURLOPT_PINNEDPUBLICKEY.html#AVAILABILITY */
1664 if ((opt = php_http_option_register(ssl_registry, ZEND_STRL("pinned_publickey"), CURLOPT_PINNEDPUBLICKEY, IS_STRING))) {
1665 opt->flags |= PHP_HTTP_CURLE_OPTION_CHECK_STRLEN;
1666 opt->flags |= PHP_HTTP_CURLE_OPTION_CHECK_BASEDIR;
1667 }
1668 #endif
1669 #if PHP_HTTP_CURL_VERSION(7,21,4) && PHP_HTTP_HAVE_LIBCURL_TLSAUTH_TYPE
1670 if ((opt = php_http_option_register(ssl_registry, ZEND_STRL("tlsauthtype"), CURLOPT_TLSAUTH_TYPE, IS_LONG))) {
1671 opt->setter = php_http_curle_option_set_ssl_tlsauthtype;
1672 }
1673 if ((opt = php_http_option_register(ssl_registry, ZEND_STRL("tlsauthuser"), CURLOPT_TLSAUTH_USERNAME, IS_STRING))) {
1674 opt->flags |= PHP_HTTP_CURLE_OPTION_CHECK_STRLEN;
1675 }
1676 if ((opt = php_http_option_register(ssl_registry, ZEND_STRL("tlsauthpass"), CURLOPT_TLSAUTH_PASSWORD, IS_STRING))) {
1677 opt->flags |= PHP_HTTP_CURLE_OPTION_CHECK_STRLEN;
1678 }
1679 #endif
1680 #if PHP_HTTP_CURL_VERSION(7,42,0) && (PHP_HTTP_HAVE_LIBCURL_NSS || PHP_HTTP_HAVE_LIBCURL_SECURETRANSPORT)
1681 if ((opt = php_http_option_register(ssl_registry, ZEND_STRL("falsestart"), CURLOPT_SSL_FALSESTART, _IS_BOOL))) {
1682 opt->flags |= PHP_HTTP_CURLE_OPTION_IGNORE_RC;
1683 }
1684 #endif
1685 #if PHP_HTTP_CURL_VERSION(7,61,0) && PHP_HTTP_HAVE_LIBCURL_TLS13_CIPHERS
1686 if ((opt = php_http_option_register(ssl_registry, ZEND_STRL("tls13_ciphers"), CURLOPT_TLS13_CIPHERS, IS_STRING))) {
1687 opt->flags |= PHP_HTTP_CURLE_OPTION_CHECK_STRLEN;
1688 }
1689 #endif
1690 }
1691
1692 #if PHP_HTTP_CURL_VERSION(7,52,0)
1693 /* proxy_ssl */
1694 if ((proxy_opt = php_http_option_register(registry, ZEND_STRL("proxy_ssl"), 0, IS_ARRAY))) {
1695 php_http_options_t *proxy_registry = &proxy_opt->suboptions;
1696
1697 if ((opt = php_http_option_register(proxy_registry, ZEND_STRL("cert"), CURLOPT_PROXY_SSLCERT, IS_STRING))) {
1698 opt->flags |= PHP_HTTP_CURLE_OPTION_CHECK_STRLEN;
1699 opt->flags |= PHP_HTTP_CURLE_OPTION_CHECK_BASEDIR;
1700 }
1701 if ((opt = php_http_option_register(proxy_registry, ZEND_STRL("certtype"), CURLOPT_PROXY_SSLCERTTYPE, IS_STRING))) {
1702 opt->flags |= PHP_HTTP_CURLE_OPTION_CHECK_STRLEN;
1703 ZVAL_PSTRING(&opt->defval, "PEM");
1704 }
1705 if ((opt = php_http_option_register(proxy_registry, ZEND_STRL("key"), CURLOPT_PROXY_SSLKEY, IS_STRING))) {
1706 opt->flags |= PHP_HTTP_CURLE_OPTION_CHECK_STRLEN;
1707 opt->flags |= PHP_HTTP_CURLE_OPTION_CHECK_BASEDIR;
1708 }
1709 if ((opt = php_http_option_register(proxy_registry, ZEND_STRL("keytype"), CURLOPT_PROXY_SSLKEYTYPE, IS_STRING))) {
1710 opt->flags |= PHP_HTTP_CURLE_OPTION_CHECK_STRLEN;
1711 ZVAL_PSTRING(&opt->defval, "PEM");
1712 }
1713 if ((opt = php_http_option_register(proxy_registry, ZEND_STRL("keypasswd"), CURLOPT_PROXY_KEYPASSWD, IS_STRING))) {
1714 opt->flags |= PHP_HTTP_CURLE_OPTION_CHECK_STRLEN;
1715 }
1716 php_http_option_register(proxy_registry, ZEND_STRL("version"), CURLOPT_PROXY_SSLVERSION, IS_LONG);
1717 if ((opt = php_http_option_register(proxy_registry, ZEND_STRL("verifypeer"), CURLOPT_PROXY_SSL_VERIFYPEER, _IS_BOOL))) {
1718 ZVAL_BOOL(&opt->defval, 1);
1719 }
1720 if ((opt = php_http_option_register(proxy_registry, ZEND_STRL("verifyhost"), CURLOPT_PROXY_SSL_VERIFYHOST, _IS_BOOL))) {
1721 ZVAL_BOOL(&opt->defval, 1);
1722 opt->setter = php_http_curle_option_set_ssl_verifyhost;
1723 }
1724 php_http_option_register(proxy_registry, ZEND_STRL("cipher_list"), CURLOPT_PROXY_SSL_CIPHER_LIST, IS_STRING);
1725 # if PHP_HTTP_CURL_VERSION(7,71,0)
1726 if ((opt = php_http_option_register(proxy_registry, ZEND_STRL("issuercert"), CURLOPT_PROXY_ISSUERCERT, IS_STRING))) {
1727 opt->flags |= PHP_HTTP_CURLE_OPTION_CHECK_STRLEN;
1728 opt->flags |= PHP_HTTP_CURLE_OPTION_CHECK_BASEDIR;
1729 }
1730 # endif
1731 # if PHP_HTTP_HAVE_LIBCURL_OPENSSL
1732 if ((opt = php_http_option_register(proxy_registry, ZEND_STRL("crlfile"), CURLOPT_PROXY_CRLFILE, IS_STRING))) {
1733 opt->flags |= PHP_HTTP_CURLE_OPTION_CHECK_STRLEN;
1734 opt->flags |= PHP_HTTP_CURLE_OPTION_CHECK_BASEDIR;
1735 }
1736 # endif
1737 # if PHP_HTTP_HAVE_LIBCURL_CAINFO
1738 if ((opt = php_http_option_register(proxy_registry, ZEND_STRL("cainfo"), CURLOPT_PROXY_CAINFO, IS_STRING))) {
1739 opt->flags |= PHP_HTTP_CURLE_OPTION_CHECK_STRLEN;
1740 opt->flags |= PHP_HTTP_CURLE_OPTION_CHECK_BASEDIR;
1741 # ifdef PHP_HTTP_CAINFO
1742 ZVAL_PSTRING(&opt->defval, PHP_HTTP_CAINFO);
1743 # endif
1744 }
1745 # endif
1746 # if PHP_HTTP_HAVE_LIBCURL_CAPATH
1747 if ((opt = php_http_option_register(proxy_registry, ZEND_STRL("capath"), CURLOPT_PROXY_CAPATH, IS_STRING))) {
1748 opt->flags |= PHP_HTTP_CURLE_OPTION_CHECK_STRLEN;
1749 opt->flags |= PHP_HTTP_CURLE_OPTION_CHECK_BASEDIR;
1750 # ifdef PHP_HTTP_CAPATH
1751 ZVAL_PSTRING(&opt->defval, PHP_HTTP_CAPATH);
1752 # endif
1753 }
1754 # endif
1755
1756 # if PHP_HTTP_HAVE_LIBCURL_TLSAUTH_TYPE
1757 if ((opt = php_http_option_register(proxy_registry, ZEND_STRL("tlsauthtype"), CURLOPT_PROXY_TLSAUTH_TYPE, IS_LONG))) {
1758 opt->setter = php_http_curle_option_set_ssl_tlsauthtype;
1759 }
1760 if ((opt = php_http_option_register(proxy_registry, ZEND_STRL("tlsauthuser"), CURLOPT_PROXY_TLSAUTH_USERNAME, IS_STRING))) {
1761 opt->flags |= PHP_HTTP_CURLE_OPTION_CHECK_STRLEN;
1762 }
1763 if ((opt = php_http_option_register(proxy_registry, ZEND_STRL("tlsauthpass"), CURLOPT_PROXY_TLSAUTH_PASSWORD, IS_STRING))) {
1764 opt->flags |= PHP_HTTP_CURLE_OPTION_CHECK_STRLEN;
1765 }
1766 # endif
1767 # if PHP_HTTP_CURL_VERSION(7,59,0)
1768 /* FIXME: see http://curl.haxx.se/libcurl/c/CURLOPT_PINNEDPUBLICKEY.html#AVAILABILITY */
1769 if ((opt = php_http_option_register(proxy_registry, ZEND_STRL("pinned_publickey"), CURLOPT_PROXY_PINNEDPUBLICKEY, IS_STRING))) {
1770 opt->flags |= PHP_HTTP_CURLE_OPTION_CHECK_STRLEN;
1771 opt->flags |= PHP_HTTP_CURLE_OPTION_CHECK_BASEDIR;
1772 }
1773 # endif
1774 # if PHP_HTTP_CURL_VERSION(7,61,0) && PHP_HTTP_HAVE_LIBCURL_TLS13_CIPHERS
1775 if ((opt = php_http_option_register(proxy_registry, ZEND_STRL("tls13_ciphers"), CURLOPT_PROXY_TLS13_CIPHERS, IS_STRING))) {
1776 opt->flags |= PHP_HTTP_CURLE_OPTION_CHECK_STRLEN;
1777 }
1778 # endif
1779 }
1780 #endif
1781 }
1782
1783 #if PHP_HTTP_CURL_VERSION(7,64,1)
1784 # if !PHP_HTTP_HAVE_LIBCURL_ALT_SVC
1785 if (PHP_HTTP_CURL_FEATURE(CURL_VERSION_ALTSVC)) {
1786 # endif
1787 if ((opt = php_http_option_register(registry, ZEND_STRL("altsvc_ctrl"), CURLOPT_ALTSVC_CTRL, IS_LONG))) {
1788 opt->setter = php_http_curle_option_set_altsvc_ctrl;
1789 }
1790 if ((opt = php_http_option_register(registry, ZEND_STRL("altsvc"), CURLOPT_ALTSVC, IS_STRING))) {
1791 opt->flags |= PHP_HTTP_CURLE_OPTION_CHECK_BASEDIR;
1792 }
1793 # if !PHP_HTTP_HAVE_LIBCURL_ALT_SVC
1794 }
1795 # endif
1796 #endif
1797 #if PHP_HTTP_CURL_VERSION(7,74,0)
1798 # if !PHP_HTTP_HAVE_LIBCURL_HSTS
1799 if (PHP_HTTP_CURL_FEATURE(CURL_VERSION_HSTS)) {
1800 # endif
1801 php_http_option_register(registry, ZEND_STRL("hsts_ctrl"), CURLOPT_HSTS_CTRL, IS_LONG);
1802 if ((opt = php_http_option_register(registry, ZEND_STRL("hsts"), CURLOPT_HSTS, IS_STRING))) {
1803 opt->flags |= PHP_HTTP_CURLE_OPTION_CHECK_BASEDIR;
1804 }
1805 # if !PHP_HTTP_HAVE_LIBCURL_HSTS
1806 }
1807 # endif
1808 #endif
1809 }
1810
1811 static zval *php_http_curle_get_option(php_http_option_t *opt, HashTable *options, void *userdata)
1812 {
1813 php_http_client_curl_handler_t *curl = userdata;
1814 zval *option;
1815
1816 if ((option = php_http_option_get(opt, options, NULL))) {
1817 zval zopt;
1818
1819 ZVAL_DUP(&zopt, option);
1820 convert_to_explicit_type(&zopt, opt->type);
1821 zend_hash_update(&curl->options.cache, opt->name, &zopt);
1822 return zend_hash_find(&curl->options.cache, opt->name);
1823 }
1824 return option;
1825 }
1826
1827 static ZEND_RESULT_CODE php_http_curle_set_option(php_http_option_t *opt, zval *val, void *userdata)
1828 {
1829 php_http_client_curl_handler_t *curl = userdata;
1830 CURL *ch = curl->handle;
1831 zval tmp;
1832 CURLcode rc = CURLE_UNKNOWN_OPTION;
1833 ZEND_RESULT_CODE rv = SUCCESS;
1834
1835 if (!val) {
1836 val = &opt->defval;
1837 }
1838
1839 switch (opt->type) {
1840 case _IS_BOOL:
1841 if (opt->setter) {
1842 rv = opt->setter(opt, val, curl);
1843 } else if (CURLE_OK != curl_easy_setopt(ch, opt->option, (long) (Z_TYPE_P(val) == IS_TRUE))) {
1844 rv = FAILURE;
1845 }
1846 break;
1847
1848 case IS_LONG:
1849 if (opt->setter) {
1850 rv = opt->setter(opt, val, curl);
1851 } else if (CURLE_OK != curl_easy_setopt(ch, opt->option, Z_LVAL_P(val))) {
1852 rv = FAILURE;
1853 }
1854 break;
1855
1856 case IS_STRING:
1857 if (opt->setter) {
1858 rv = opt->setter(opt, val, curl);
1859 } else if (!val || Z_TYPE_P(val) == IS_NULL) {
1860 if (CURLE_OK != (rc = curl_easy_setopt(ch, opt->option, NULL))) {
1861 rv = FAILURE;
1862 }
1863 } else if ((opt->flags & PHP_HTTP_CURLE_OPTION_CHECK_STRLEN) && !Z_STRLEN_P(val)) {
1864 if (CURLE_OK != (rc = curl_easy_setopt(ch, opt->option, NULL))) {
1865 rv = FAILURE;
1866 }
1867 } else if ((opt->flags & PHP_HTTP_CURLE_OPTION_CHECK_BASEDIR) && Z_STRVAL_P(val) && SUCCESS != php_check_open_basedir(Z_STRVAL_P(val))) {
1868 if (CURLE_OK != (rc = curl_easy_setopt(ch, opt->option, NULL))) {
1869 rv = FAILURE;
1870 }
1871 } else if (CURLE_OK != (rc = curl_easy_setopt(ch, opt->option, Z_STRVAL_P(val)))) {
1872 rv = FAILURE;
1873 }
1874 break;
1875
1876 case IS_DOUBLE:
1877 if (opt->flags & PHP_HTTP_CURLE_OPTION_TRANSFORM_MS) {
1878 tmp = *val;
1879 Z_DVAL(tmp) *= 1000;
1880 val = &tmp;
1881 }
1882 if (opt->setter) {
1883 rv = opt->setter(opt, val, curl);
1884 } else if (CURLE_OK != curl_easy_setopt(ch, opt->option, (long) Z_DVAL_P(val))) {
1885 rv = FAILURE;
1886 }
1887 break;
1888
1889 case IS_ARRAY:
1890 if (opt->setter) {
1891 rv = opt->setter(opt, val, curl);
1892 } else if (Z_TYPE_P(val) != IS_NULL) {
1893 rv = php_http_options_apply(&opt->suboptions, Z_ARRVAL_P(val), curl);
1894 }
1895 break;
1896
1897 default:
1898 if (opt->setter) {
1899 rv = opt->setter(opt, val, curl);
1900 } else {
1901 rv = FAILURE;
1902 }
1903 break;
1904 }
1905 if (rv != SUCCESS) {
1906 if (opt->flags & PHP_HTTP_CURLE_OPTION_IGNORE_RC) {
1907 rv = SUCCESS;
1908 } else {
1909 php_error_docref(NULL, E_NOTICE, "Could not set option %s (%s)", opt->name->val, curl_easy_strerror(rc));
1910 }
1911 }
1912 return rv;
1913 }
1914
1915 #if PHP_HTTP_CURL_VERSION(7,30,0) && !PHP_HTTP_CURL_VERSION(7,62,0)
1916 static ZEND_RESULT_CODE php_http_curlm_option_set_pipelining_bl(php_http_option_t *opt, zval *value, void *userdata)
1917 {
1918 php_http_client_t *client = userdata;
1919 php_http_client_curl_t *curl = client->ctx;
1920 CURLM *ch = curl->handle->multi;
1921 HashTable tmp_ht;
1922 char **bl = NULL;
1923
1924 /* array of char *, ending with a NULL */
1925 if (value && Z_TYPE_P(value) != IS_NULL) {
1926 zval *entry;
1927 HashTable *ht = HASH_OF(value);
1928 int c = zend_hash_num_elements(ht);
1929 char **ptr = ecalloc(c + 1, sizeof(char *));
1930
1931 bl = ptr;
1932
1933 zend_hash_init(&tmp_ht, c, NULL, ZVAL_PTR_DTOR, 0);
1934 array_join(ht, &tmp_ht, 0, ARRAY_JOIN_STRINGIFY);
1935
1936 ZEND_HASH_FOREACH_VAL(&tmp_ht, entry)
1937 {
1938 *ptr++ = Z_STRVAL_P(entry);
1939 }
1940 ZEND_HASH_FOREACH_END();
1941 }
1942
1943 if (CURLM_OK != curl_multi_setopt(ch, opt->option, bl)) {
1944 if (bl) {
1945 efree(bl);
1946 zend_hash_destroy(&tmp_ht);
1947 }
1948 return FAILURE;
1949 }
1950
1951 if (bl) {
1952 efree(bl);
1953 zend_hash_destroy(&tmp_ht);
1954 }
1955 return SUCCESS;
1956 }
1957 #endif
1958
1959 static inline ZEND_RESULT_CODE php_http_curlm_use_eventloop(php_http_client_t *h, php_http_client_curl_ops_t *ev_ops, zval *init_data)
1960 {
1961 php_http_client_curl_t *curl = h->ctx;
1962 void *ev_ctx;
1963
1964 if (ev_ops) {
1965 if (!(ev_ctx = ev_ops->init(h, init_data))) {
1966 return FAILURE;
1967 }
1968 curl->ev_ctx = ev_ctx;
1969 curl->ev_ops = ev_ops;
1970 } else {
1971 if (curl->ev_ops) {
1972 if (curl->ev_ctx) {
1973 curl->ev_ops->dtor(&curl->ev_ctx);
1974 }
1975 curl->ev_ops = NULL;
1976 }
1977 }
1978
1979 return SUCCESS;
1980 }
1981
1982 static ZEND_RESULT_CODE php_http_curlm_option_set_use_eventloop(php_http_option_t *opt, zval *value, void *userdata)
1983 {
1984 php_http_client_t *client = userdata;
1985 php_http_client_curl_ops_t *ev_ops = NULL;
1986
1987 if (value && Z_TYPE_P(value) == IS_OBJECT && instanceof_function(Z_OBJCE_P(value), php_http_client_curl_user_get_class_entry())) {
1988 ev_ops = php_http_client_curl_user_ops_get();
1989 #if PHP_HTTP_HAVE_LIBEVENT
1990 } else if (value && zend_is_true(value)) {
1991 ev_ops = php_http_client_curl_event_ops_get();
1992 #endif
1993 }
1994
1995 return php_http_curlm_use_eventloop(client, ev_ops, value);
1996 }
1997
1998 static ZEND_RESULT_CODE php_http_curlm_option_set_share_cookies(php_http_option_t *opt, zval *value, void *userdata)
1999 {
2000 php_http_client_t *client = userdata;
2001 php_http_client_curl_t *curl = client->ctx;
2002 CURLSHcode rc;
2003
2004 if (Z_TYPE_P(value) == IS_TRUE) {
2005 rc = curl_share_setopt(curl->handle->share, CURLSHOPT_SHARE, CURL_LOCK_DATA_COOKIE);
2006 } else {
2007 rc = curl_share_setopt(curl->handle->share, CURLSHOPT_UNSHARE, CURL_LOCK_DATA_COOKIE);
2008 }
2009
2010 if (CURLSHE_OK != rc) {
2011 php_error_docref(NULL, E_NOTICE, "Could not set option %s (%s)", opt->name->val, curl_share_strerror(rc));
2012 return FAILURE;
2013 }
2014 return SUCCESS;
2015 }
2016
2017 #if PHP_HTTP_HAVE_LIBCURL_SHARE_SSL
2018 static ZEND_RESULT_CODE php_http_curlm_option_set_share_ssl(php_http_option_t *opt, zval *value, void *userdata)
2019 {
2020 php_http_client_t *client = userdata;
2021 php_http_client_curl_t *curl = client->ctx;
2022 CURLSHcode rc;
2023
2024 if (Z_TYPE_P(value) == IS_TRUE) {
2025 rc = curl_share_setopt(curl->handle->share, CURLSHOPT_SHARE, CURL_LOCK_DATA_SSL_SESSION);
2026 } else {
2027 rc = curl_share_setopt(curl->handle->share, CURLSHOPT_UNSHARE, CURL_LOCK_DATA_SSL_SESSION);
2028 }
2029
2030 if (CURLSHE_OK != rc) {
2031 php_error_docref(NULL, E_NOTICE, "Could not set option %s (%s)", opt->name->val, curl_share_strerror(rc));
2032 return FAILURE;
2033 }
2034 return SUCCESS;
2035 }
2036 #endif
2037
2038 static void php_http_curlm_options_init(php_http_options_t *registry)
2039 {
2040 php_http_option_t *opt;
2041
2042 /* set size of connection cache */
2043 if ((opt = php_http_option_register(registry, ZEND_STRL("maxconnects"), CURLMOPT_MAXCONNECTS, IS_LONG))) {
2044 /* -1 == default, 0 == unlimited */
2045 ZVAL_LONG(&opt->defval, -1);
2046 }
2047 /* set max number of connections to a single host */
2048 #if PHP_HTTP_CURL_VERSION(7,30,0)
2049 php_http_option_register(registry, ZEND_STRL("max_host_connections"), CURLMOPT_MAX_HOST_CONNECTIONS, IS_LONG);
2050 #endif
2051 /* max simultaneously open connections */
2052 #if PHP_HTTP_CURL_VERSION(7,30,0)
2053 php_http_option_register(registry, ZEND_STRL("max_total_connections"), CURLMOPT_MAX_TOTAL_CONNECTIONS, IS_LONG);
2054 #endif
2055 #if PHP_HTTP_CURL_VERSION(7,67,0)
2056 if ((opt = php_http_option_register(registry, ZEND_STRL("max_concurrent_streams"), CURLMOPT_MAX_CONCURRENT_STREAMS, IS_LONG))) {
2057 ZVAL_LONG(&opt->defval, 100);
2058 }
2059 #endif
2060
2061 #if !PHP_HTTP_CURL_VERSION(7,62,0)
2062 /* enable/disable HTTP pipelining */
2063 php_http_option_register(registry, ZEND_STRL("pipelining"), CURLMOPT_PIPELINING, _IS_BOOL);
2064 # if PHP_HTTP_CURL_VERSION(7,30,0)
2065 /* maximum number of requests in a pipeline */
2066 if ((opt = php_http_option_register(registry, ZEND_STRL("max_pipeline_length"), CURLMOPT_MAX_PIPELINE_LENGTH, IS_LONG))) {
2067 ZVAL_LONG(&opt->defval, 5);
2068 }
2069 /* chunk length threshold for pipelining */
2070 php_http_option_register(registry, ZEND_STRL("chunk_length_penalty_size"), CURLMOPT_CHUNK_LENGTH_PENALTY_SIZE, IS_LONG);
2071 /* size threshold for pipelining penalty */
2072 php_http_option_register(registry, ZEND_STRL("content_length_penalty_size"), CURLMOPT_CONTENT_LENGTH_PENALTY_SIZE, IS_LONG);
2073 /* pipelining server blacklist */
2074 if ((opt = php_http_option_register(registry, ZEND_STRL("pipelining_server_bl"), CURLMOPT_PIPELINING_SERVER_BL, IS_ARRAY))) {
2075 opt->setter = php_http_curlm_option_set_pipelining_bl;
2076 }
2077 /* pipelining host blacklist */
2078 if ((opt = php_http_option_register(registry, ZEND_STRL("pipelining_site_bl"), CURLMOPT_PIPELINING_SITE_BL, IS_ARRAY))) {
2079 opt->setter = php_http_curlm_option_set_pipelining_bl;
2080 }
2081 # endif
2082 #endif
2083 /* events */
2084 if ((opt = php_http_option_register(registry, ZEND_STRL("use_eventloop"), 0, 0))) {
2085 opt->setter = php_http_curlm_option_set_use_eventloop;
2086 }
2087 /* share */
2088 if ((opt = php_http_option_register(registry, ZEND_STRL("share_cookies"), 0, _IS_BOOL))) {
2089 opt->setter = php_http_curlm_option_set_share_cookies;
2090 ZVAL_TRUE(&opt->defval);
2091 }
2092 #if PHP_HTTP_HAVE_LIBCURL_SHARE_SSL
2093 if ((opt = php_http_option_register(registry, ZEND_STRL("share_ssl"), 0, _IS_BOOL))) {
2094 opt->setter = php_http_curlm_option_set_share_ssl;
2095 ZVAL_TRUE(&opt->defval);
2096 }
2097 #endif
2098 }
2099
2100 static ZEND_RESULT_CODE php_http_curlm_set_option(php_http_option_t *opt, zval *val, void *userdata)
2101 {
2102 php_http_client_t *client = userdata;
2103 php_http_client_curl_t *curl = client->ctx;
2104 CURLM *ch = curl->handle->multi;
2105 zval zopt, *orig = val;
2106 CURLMcode rc = CURLM_UNKNOWN_OPTION;
2107 ZEND_RESULT_CODE rv = SUCCESS;
2108
2109 if (!val) {
2110 val = &opt->defval;
2111 } else if (opt->type && Z_TYPE_P(val) != opt->type && !(Z_TYPE_P(val) == IS_NULL && opt->type == IS_ARRAY)) {
2112 ZVAL_DUP(&zopt, val);
2113 convert_to_explicit_type(&zopt, opt->type);
2114
2115 val = &zopt;
2116 }
2117
2118 if (opt->setter) {
2119 rv = opt->setter(opt, val, client);
2120 } else {
2121 switch (opt->type) {
2122 case _IS_BOOL:
2123 if (CURLM_OK != (rc = curl_multi_setopt(ch, opt->option, (long) zend_is_true(val)))) {
2124 rv = FAILURE;
2125 php_error_docref(NULL, E_NOTICE, "Could not set option %s (%s)", opt->name->val, curl_multi_strerror(rc));
2126 }
2127 break;
2128 case IS_LONG:
2129 if (CURLM_OK != (rc = curl_multi_setopt(ch, opt->option, Z_LVAL_P(val)))) {
2130 rv = FAILURE;
2131 php_error_docref(NULL, E_NOTICE, "Could not set option %s (%s)", opt->name->val, curl_multi_strerror(rc));
2132 }
2133 break;
2134 default:
2135 rv = FAILURE;
2136 php_error_docref(NULL, E_NOTICE, "Could not set option %s", opt->name->val);
2137 break;
2138 }
2139 }
2140
2141 if (val && val != orig && val != &opt->defval) {
2142 zval_ptr_dtor(val);
2143 }
2144
2145 return rv;
2146 }
2147
2148 /* client ops */
2149
2150 static ZEND_RESULT_CODE php_http_client_curl_handler_reset(php_http_client_curl_handler_t *handler)
2151 {
2152 php_http_client_curl_t *curl = handler->client->ctx;
2153 CURL *ch = handler->handle;
2154 php_http_curle_storage_t *st;
2155
2156 if ((st = php_http_curle_get_storage(ch))) {
2157 if (st->url) {
2158 pefree(st->url, 1);
2159 st->url = NULL;
2160 }
2161 if (st->cookiestore) {
2162 pefree(st->cookiestore, 1);
2163 st->cookiestore = NULL;
2164 }
2165 st->errorbuffer[0] = '\0';
2166 st->errorcode = 0;
2167 }
2168
2169 curl_easy_setopt(ch, CURLOPT_URL, NULL);
2170 curl_easy_setopt(ch, CURLOPT_CUSTOMREQUEST, NULL);
2171 curl_easy_setopt(ch, CURLOPT_HTTPGET, 1L);
2172 curl_easy_setopt(ch, CURLOPT_NOBODY, 0L);
2173 /* libcurl < 7.19.6 does not clear auth info with USERPWD set to NULL */
2174 #if PHP_HTTP_CURL_VERSION(7,19,1)
2175 curl_easy_setopt(ch, CURLOPT_PROXYUSERNAME, NULL);
2176 curl_easy_setopt(ch, CURLOPT_PROXYPASSWORD, NULL);
2177 curl_easy_setopt(ch, CURLOPT_USERNAME, NULL);
2178 curl_easy_setopt(ch, CURLOPT_PASSWORD, NULL);
2179 #endif
2180
2181 #if PHP_HTTP_CURL_VERSION(7,21,3)
2182 if (handler->options.resolve) {
2183 curl_slist_free_all(handler->options.resolve);
2184 handler->options.resolve = NULL;
2185 }
2186 #endif
2187 handler->options.retry.count = 0;
2188 handler->options.retry.delay = 0;
2189 handler->options.redirects = 0;
2190 handler->options.encode_cookies = 1;
2191
2192 if (handler->options.headers) {
2193 curl_slist_free_all(handler->options.headers);
2194 handler->options.headers = NULL;
2195 }
2196 if (handler->options.proxyheaders) {
2197 curl_slist_free_all(handler->options.proxyheaders);
2198 handler->options.proxyheaders = NULL;
2199 }
2200
2201 php_http_buffer_reset(&handler->options.cookies);
2202 php_http_buffer_reset(&handler->options.ranges);
2203
2204 if (php_http_message_body_size(handler->response.body)) {
2205 php_http_message_body_free(&handler->response.body);
2206 handler->response.body = php_http_message_body_init(NULL, NULL);
2207 }
2208 php_http_buffer_reset(&handler->response.headers);
2209
2210 #if ZTS
2211 curl_easy_setopt(ch, CURLOPT_NOSIGNAL, 1L);
2212 #endif
2213 curl_easy_setopt(ch, CURLOPT_HEADER, 0L);
2214 curl_easy_setopt(ch, CURLOPT_FILETIME, 1L);
2215 curl_easy_setopt(ch, CURLOPT_AUTOREFERER, 1L);
2216 curl_easy_setopt(ch, CURLOPT_VERBOSE, 1L);
2217 curl_easy_setopt(ch, CURLOPT_NOPROGRESS, 0L);
2218 curl_easy_setopt(ch, CURLOPT_HEADERFUNCTION, php_http_curle_header_callback);
2219 curl_easy_setopt(ch, CURLOPT_WRITEFUNCTION, php_http_curle_body_callback);
2220 curl_easy_setopt(ch, CURLOPT_DEBUGFUNCTION, php_http_curle_raw_callback);
2221 curl_easy_setopt(ch, CURLOPT_READFUNCTION, php_http_curle_read_callback);
2222 curl_easy_setopt(ch, CURLOPT_SEEKFUNCTION, php_http_curle_seek_callback);
2223 #if PHP_HTTP_CURL_VERSION(7,32,0)
2224 curl_easy_setopt(ch, CURLOPT_XFERINFOFUNCTION, php_http_curle_xferinfo_callback);
2225 curl_easy_setopt(ch, CURLOPT_XFERINFODATA, handler);
2226 #else
2227 curl_easy_setopt(ch, CURLOPT_PROGRESSFUNCTION, php_http_curle_progress_callback);
2228 curl_easy_setopt(ch, CURLOPT_PROGRESSDATA, handler);
2229 #endif
2230 curl_easy_setopt(ch, CURLOPT_DEBUGDATA, handler);
2231 curl_easy_setopt(ch, CURLOPT_WRITEDATA, handler);
2232 curl_easy_setopt(ch, CURLOPT_HEADERDATA, handler);
2233 #if DEBUG_COOKIES
2234 fprintf(stderr, "CURLOPT_SHARE: %p\n", curl->handle->share);
2235 #endif
2236 curl_easy_setopt(ch, CURLOPT_SHARE, curl->handle->share);
2237
2238 return SUCCESS;
2239 }
2240
2241 static php_http_client_curl_handler_t *php_http_client_curl_handler_init(php_http_client_t *h, php_resource_factory_t *rf)
2242 {
2243 void *handle;
2244 php_http_client_curl_handler_t *handler;
2245
2246 if (!(handle = php_resource_factory_handle_ctor(rf, NULL))) {
2247 php_error_docref(NULL, E_WARNING, "Failed to initialize curl handle");
2248 return NULL;
2249 }
2250
2251 handler = ecalloc(1, sizeof(*handler));
2252 handler->rf = rf;
2253 handler->client = h;
2254 handler->handle = handle;
2255 handler->response.body = php_http_message_body_init(NULL, NULL);
2256 php_http_buffer_init(&handler->response.headers);
2257 php_http_buffer_init(&handler->options.cookies);
2258 php_http_buffer_init(&handler->options.ranges);
2259 zend_hash_init(&handler->options.cache, 0, NULL, ZVAL_PTR_DTOR, 0);
2260
2261 php_http_client_curl_handler_reset(handler);
2262
2263 return handler;
2264 }
2265
2266
2267 static ZEND_RESULT_CODE php_http_client_curl_handler_prepare(php_http_client_curl_handler_t *curl, php_http_client_enqueue_t *enqueue)
2268 {
2269 size_t body_size;
2270 php_http_message_t *msg = enqueue->request;
2271 php_http_curle_storage_t *storage = php_http_curle_get_storage(curl->handle);
2272
2273 /* request url */
2274 if (!PHP_HTTP_INFO(msg).request.url) {
2275 php_error_docref(NULL, E_WARNING, "Cannot request empty URL");
2276 return FAILURE;
2277 }
2278 storage->errorbuffer[0] = '\0';
2279 if (storage->url) {
2280 pefree(storage->url, 1);
2281 }
2282 php_http_url_to_string(PHP_HTTP_INFO(msg).request.url, &storage->url, NULL, 1);
2283 curl_easy_setopt(curl->handle, CURLOPT_URL, storage->url);
2284
2285 /* apply options */
2286 if (SUCCESS != php_http_options_apply(&php_http_curle_options, enqueue->options, curl)) {
2287 return FAILURE;
2288 }
2289
2290 /* request headers */
2291 php_http_message_update_headers(msg);
2292 if (zend_hash_num_elements(&msg->hdrs)) {
2293 php_http_arrkey_t header_key;
2294 zval *header_val;
2295 zend_string *header_str;
2296 php_http_buffer_t header;
2297 #if !PHP_HTTP_CURL_VERSION(7,23,0)
2298 zval *ct = zend_hash_str_find(&msg->hdrs, ZEND_STRL("Content-Length"));
2299 #endif
2300
2301 php_http_buffer_init(&header);
2302 ZEND_HASH_FOREACH_KEY_VAL(&msg->hdrs, header_key.h, header_key.key, header_val)
2303 {
2304 if (header_key.key) {
2305 #if !PHP_HTTP_CURL_VERSION(7,23,0)
2306 /* avoid duplicate content-length header */
2307 if (ct && ct == header_val) {
2308 continue;
2309 }
2310 #endif
2311 header_str = zval_get_string(header_val);
2312 php_http_buffer_appendf(&header, "%s: %s", header_key.key->val, header_str->val);
2313 php_http_buffer_fix(&header);
2314 curl->options.headers = curl_slist_append(curl->options.headers, header.data);
2315 php_http_buffer_reset(&header);
2316 zend_string_release(header_str);
2317 }
2318 }
2319 ZEND_HASH_FOREACH_END();
2320 php_http_buffer_dtor(&header);
2321 }
2322 curl_easy_setopt(curl->handle, CURLOPT_HTTPHEADER, curl->options.headers);
2323
2324 /* attach request body */
2325 if ((body_size = php_http_message_body_size(msg->body))) {
2326 /* RFC2616, section 4.3 (para. 4) states that »a message-body MUST NOT be included in a request if the
2327 * specification of the request method (section 5.1.1) does not allow sending an entity-body in request.«
2328 * Following the clause in section 5.1.1 (para. 2) that request methods »MUST be implemented with the
2329 * same semantics as those specified in section 9« reveal that not any single defined HTTP/1.1 method
2330 * does not allow a request body.
2331 */
2332 php_stream_rewind(php_http_message_body_stream(msg->body));
2333 curl_easy_setopt(curl->handle, CURLOPT_SEEKDATA, msg->body);
2334 curl_easy_setopt(curl->handle, CURLOPT_READDATA, msg->body);
2335 curl_easy_setopt(curl->handle, CURLOPT_INFILESIZE, body_size);
2336 curl_easy_setopt(curl->handle, CURLOPT_POSTFIELDSIZE, body_size);
2337 curl_easy_setopt(curl->handle, CURLOPT_POST, 1L);
2338 } else {
2339 curl_easy_setopt(curl->handle, CURLOPT_SEEKDATA, NULL);
2340 curl_easy_setopt(curl->handle, CURLOPT_READDATA, NULL);
2341 curl_easy_setopt(curl->handle, CURLOPT_INFILESIZE, 0L);
2342 curl_easy_setopt(curl->handle, CURLOPT_POSTFIELDSIZE, 0L);
2343 }
2344
2345 /*
2346 * Always use CUSTOMREQUEST, else curl won't send any request body for GET etc.
2347 * See e.g. bug #69313.
2348 *
2349 * Here's what curl does:
2350 * - CURLOPT_HTTPGET: ignore request body
2351 * - CURLOPT_UPLOAD: set "Expect: 100-continue" header
2352 * - CURLOPT_POST: set "Content-Type: application/x-www-form-urlencoded" header
2353 * Now select the least bad.
2354 *
2355 * See also https://tools.ietf.org/html/rfc7231#section-5.1.1
2356 */
2357 if (PHP_HTTP_INFO(msg).request.method) {
2358 switch(php_http_select_str(PHP_HTTP_INFO(msg).request.method, 2, "HEAD", "PUT")) {
2359 case 0:
2360 curl_easy_setopt(curl->handle, CURLOPT_NOBODY, 1L);
2361 break;
2362 case 1:
2363 curl_easy_setopt(curl->handle, CURLOPT_UPLOAD, 1L);
2364 break;
2365 default:
2366 curl_easy_setopt(curl->handle, CURLOPT_CUSTOMREQUEST, PHP_HTTP_INFO(msg).request.method);
2367 }
2368 } else {
2369 php_error_docref(NULL, E_WARNING, "Cannot use empty request method");
2370 return FAILURE;
2371 }
2372
2373 return SUCCESS;
2374 }
2375
2376 static void php_http_client_curl_handler_clear(php_http_client_curl_handler_t *handler)
2377 {
2378 curl_easy_setopt(handler->handle, CURLOPT_NOPROGRESS, 1L);
2379 #if PHP_HTTP_CURL_VERSION(7,32,0)
2380 curl_easy_setopt(handler->handle, CURLOPT_XFERINFOFUNCTION, NULL);
2381 #else
2382 curl_easy_setopt(handler->handle, CURLOPT_PROGRESSFUNCTION, NULL);
2383 #endif
2384 curl_easy_setopt(handler->handle, CURLOPT_VERBOSE, 0L);
2385 curl_easy_setopt(handler->handle, CURLOPT_DEBUGFUNCTION, NULL);
2386 /* see gh issue #84 */
2387 #if DEBUG_COOKIES
2388 fprintf(stderr, "CURLOPT_COOKIELIST: FLUSH\n");
2389 fprintf(stderr, "CURLOPT_SHARE: (null)\n");
2390 #endif
2391 curl_easy_setopt(handler->handle, CURLOPT_COOKIELIST, "FLUSH");
2392 curl_easy_setopt(handler->handle, CURLOPT_SHARE, NULL);
2393 #if PHP_HTTP_CURL_VERSION(7,63,0) && !PHP_HTTP_CURL_VERSION(7,65,0)
2394 {
2395 php_http_curle_storage_t *st = php_http_curle_get_storage(handler->handle);
2396 curl_easy_setopt(handler->handle, CURLOPT_COOKIEJAR, st ? st->cookiestore : NULL);
2397 }
2398 #endif
2399 }
2400
2401 static void php_http_client_curl_handler_dtor(php_http_client_curl_handler_t *handler)
2402 {
2403 php_http_client_curl_handler_clear(handler);
2404
2405 php_resource_factory_handle_dtor(handler->rf, handler->handle);
2406 php_resource_factory_free(&handler->rf);
2407
2408 php_http_message_body_free(&handler->response.body);
2409 php_http_buffer_dtor(&handler->response.headers);
2410 php_http_buffer_dtor(&handler->options.ranges);
2411 php_http_buffer_dtor(&handler->options.cookies);
2412 zend_hash_destroy(&handler->options.cache);
2413
2414 #if PHP_HTTP_CURL_VERSION(7,21,3)
2415 if (handler->options.resolve) {
2416 curl_slist_free_all(handler->options.resolve);
2417 handler->options.resolve = NULL;
2418 }
2419 #endif
2420
2421 if (handler->options.headers) {
2422 curl_slist_free_all(handler->options.headers);
2423 handler->options.headers = NULL;
2424 }
2425
2426 if (handler->options.proxyheaders) {
2427 curl_slist_free_all(handler->options.proxyheaders);
2428 handler->options.proxyheaders = NULL;
2429 }
2430
2431 efree(handler);
2432 }
2433
2434 static php_http_client_t *php_http_client_curl_init(php_http_client_t *h, void *handle)
2435 {
2436 php_http_client_curl_t *curl;
2437
2438 if (!handle && !(handle = php_resource_factory_handle_ctor(h->rf, NULL))) {
2439 php_error_docref(NULL, E_WARNING, "Failed to initialize curl handle");
2440 return NULL;
2441 }
2442
2443 curl = ecalloc(1, sizeof(*curl));
2444 curl->handle = handle;
2445 curl->unfinished = 0;
2446 h->ctx = curl;
2447
2448 return h;
2449 }
2450
2451 static void php_http_client_curl_dtor(php_http_client_t *h)
2452 {
2453 php_http_client_curl_t *curl = h->ctx;
2454
2455 if (curl->ev_ops) {
2456 curl->ev_ops->dtor(&curl->ev_ctx);
2457 curl->ev_ops = NULL;
2458 }
2459 curl->unfinished = 0;
2460
2461 php_resource_factory_handle_dtor(h->rf, curl->handle);
2462
2463 efree(curl);
2464 h->ctx = NULL;
2465 }
2466
2467 static void queue_dtor(php_http_client_enqueue_t *e)
2468 {
2469 php_http_client_curl_handler_t *handler = e->opaque;
2470
2471 if (handler->queue.dtor) {
2472 e->opaque = handler->queue.opaque;
2473 handler->queue.dtor(e);
2474 }
2475 php_http_client_curl_handler_dtor(handler);
2476 }
2477
2478 static void retire_ch(php_persistent_handle_factory_t *f, void **handle)
2479 {
2480 CURL *ch = *handle;
2481 /* erase all cookies */
2482 if (ch) {
2483 curl_easy_reset(ch);
2484 curl_easy_setopt(ch, CURLOPT_COOKIELIST, "ALL");
2485 curl_easy_setopt(ch, CURLOPT_COOKIEFILE, NULL);
2486 }
2487 }
2488
2489 static php_resource_factory_t *create_rf(php_http_client_t *h, php_http_client_enqueue_t *enqueue)
2490 {
2491 php_persistent_handle_factory_t *pf = NULL;
2492 php_resource_factory_t *rf = NULL;
2493 php_http_url_t *url = enqueue->request->http.info.request.url;
2494
2495 if (!url || (!url->host && !url->path)) {
2496 php_error_docref(NULL, E_WARNING, "Cannot request empty URL");
2497 return NULL;
2498 }
2499
2500 /* only if the client itself is setup for persistence */
2501 if (php_resource_factory_is_persistent(h->rf)) {
2502 zend_string *id;
2503 char *id_str = NULL;
2504 size_t id_len;
2505 int port = url->port ? url->port : 80;
2506 zval *zport;
2507 php_persistent_handle_factory_t *phf = h->rf->data;
2508
2509 if ((zport = zend_hash_str_find(enqueue->options, ZEND_STRL("port")))) {
2510 zend_long lport = zval_get_long(zport);
2511
2512 if (lport > 0) {
2513 port = lport;
2514 }
2515 }
2516
2517 id_len = spprintf(&id_str, 0, "%.*s:%s:%d", (int) phf->ident->len, phf->ident->val, STR_PTR(url->host), port);
2518 id = php_http_cs2zs(id_str, id_len);
2519 pf = php_persistent_handle_concede(NULL, PHP_HTTP_G->client.curl.driver.request_name, id, NULL, retire_ch);
2520 zend_string_release(id);
2521 }
2522
2523 if (pf) {
2524 rf = php_persistent_handle_resource_factory_init(NULL, pf);
2525 } else {
2526 rf = php_resource_factory_init(NULL, &php_http_curle_resource_factory_ops, NULL, NULL);
2527 }
2528
2529 return rf;
2530 }
2531
2532 static ZEND_RESULT_CODE php_http_client_curl_enqueue(php_http_client_t *h, php_http_client_enqueue_t *enqueue)
2533 {
2534 CURLMcode rs;
2535 php_http_client_curl_t *curl = h->ctx;
2536 php_http_client_curl_handler_t *handler;
2537 php_http_client_progress_state_t *progress;
2538 php_resource_factory_t *rf;
2539
2540 rf = create_rf(h, enqueue);
2541 if (!rf) {
2542 return FAILURE;
2543 }
2544
2545 handler = php_http_client_curl_handler_init(h, rf);
2546 if (!handler) {
2547 return FAILURE;
2548 }
2549
2550 if (SUCCESS != php_http_client_curl_handler_prepare(handler, enqueue)) {
2551 php_http_client_curl_handler_dtor(handler);
2552 return FAILURE;
2553 }
2554
2555 handler->queue = *enqueue;
2556 enqueue->opaque = handler;
2557 enqueue->dtor = queue_dtor;
2558
2559 if (CURLM_OK != (rs = curl_multi_add_handle(curl->handle->multi, handler->handle))) {
2560 php_http_client_curl_handler_dtor(handler);
2561 php_error_docref(NULL, E_WARNING, "Could not enqueue request: %s", curl_multi_strerror(rs));
2562 return FAILURE;
2563 }
2564
2565 zend_llist_add_element(&h->requests, enqueue);
2566 ++curl->unfinished;
2567
2568 if (h->callback.progress.func && SUCCESS == php_http_client_getopt(h, PHP_HTTP_CLIENT_OPT_PROGRESS_INFO, enqueue->request, &progress)) {
2569 progress->info = "start";
2570 h->callback.progress.func(h->callback.progress.arg, h, &handler->queue, progress);
2571 progress->started = 1;
2572 }
2573
2574 return SUCCESS;
2575 }
2576
2577 static ZEND_RESULT_CODE php_http_client_curl_requeue(php_http_client_t *h, php_http_client_enqueue_t *enqueue)
2578 {
2579 CURLMcode rs;
2580 php_http_client_curl_t *curl = h->ctx;
2581 php_http_client_curl_handler_t *handler = enqueue->opaque;
2582 php_http_client_progress_state_t *progress;
2583
2584 if (SUCCESS != php_http_client_curl_handler_reset(handler)) {
2585 return FAILURE;
2586 }
2587
2588 if (SUCCESS != php_http_client_curl_handler_prepare(handler, enqueue)) {
2589 return FAILURE;
2590 }
2591
2592 if (CURLM_OK != (rs = curl_multi_remove_handle(curl->handle->multi, handler->handle))) {
2593 php_error_docref(NULL, E_WARNING, "Could not dequeue request: %s", curl_multi_strerror(rs));
2594 return FAILURE;
2595 }
2596
2597 if (CURLM_OK != (rs = curl_multi_add_handle(curl->handle->multi, handler->handle))) {
2598 zend_llist_del_element(&h->requests, handler->handle, (int (*)(void *, void *)) compare_queue);
2599 php_error_docref(NULL, E_WARNING, "Could not enqueue request: %s", curl_multi_strerror(rs));
2600 return FAILURE;
2601 }
2602
2603 ++curl->unfinished;
2604
2605 if (h->callback.progress.func && SUCCESS == php_http_client_getopt(h, PHP_HTTP_CLIENT_OPT_PROGRESS_INFO, enqueue->request, &progress)) {
2606 progress->info = "start";
2607 h->callback.progress.func(h->callback.progress.arg, h, &handler->queue, progress);
2608 progress->started = 1;
2609 }
2610
2611 return SUCCESS;
2612 }
2613
2614 static ZEND_RESULT_CODE php_http_client_curl_dequeue(php_http_client_t *h, php_http_client_enqueue_t *enqueue)
2615 {
2616 CURLMcode rs;
2617 php_http_client_curl_t *curl = h->ctx;
2618 php_http_client_curl_handler_t *handler = enqueue->opaque;
2619
2620 if (h->callback.depth && !CG(unclean_shutdown)) {
2621 php_error_docref(NULL, E_WARNING, "Could not dequeue request while executing callbacks");
2622 return FAILURE;
2623 }
2624
2625 php_http_client_curl_handler_clear(handler);
2626 if (CURLM_OK == (rs = curl_multi_remove_handle(curl->handle->multi, handler->handle))) {
2627 zend_llist_del_element(&h->requests, handler->handle, (int (*)(void *, void *)) compare_queue);
2628 return SUCCESS;
2629 } else {
2630 php_error_docref(NULL, E_WARNING, "Could not dequeue request: %s", curl_multi_strerror(rs));
2631 }
2632
2633 return FAILURE;
2634 }
2635
2636 static void php_http_client_curl_reset(php_http_client_t *h)
2637 {
2638 zend_llist_element *next_el, *this_el;
2639
2640 for (this_el = h->requests.head; this_el; this_el = next_el) {
2641 next_el = this_el->next;
2642 php_http_client_curl_dequeue(h, (void *) this_el->data);
2643 }
2644 }
2645
2646 #if PHP_WIN32
2647 # define SELECT_ERROR SOCKET_ERROR
2648 #else
2649 # define SELECT_ERROR -1
2650 #endif
2651
2652 static ZEND_RESULT_CODE php_http_client_curl_wait(php_http_client_t *h, struct timeval *custom_timeout)
2653 {
2654 int MAX;
2655 fd_set R, W, E;
2656 struct timeval timeout;
2657 php_http_client_curl_t *curl = h->ctx;
2658
2659 if (curl->ev_ops) {
2660 return curl->ev_ops->wait(curl->ev_ctx, custom_timeout);
2661 }
2662
2663 FD_ZERO(&R);
2664 FD_ZERO(&W);
2665 FD_ZERO(&E);
2666
2667 if (CURLM_OK == curl_multi_fdset(curl->handle->multi, &R, &W, &E, &MAX)) {
2668 if (custom_timeout && timerisset(custom_timeout)) {
2669 timeout = *custom_timeout;
2670 } else {
2671 php_http_client_curl_get_timeout(curl, 1000, &timeout);
2672 }
2673
2674 if (MAX == -1) {
2675 php_http_sleep((double) timeout.tv_sec + (double) (timeout.tv_usec / PHP_HTTP_MCROSEC));
2676 return SUCCESS;
2677 } else if (SELECT_ERROR != select(MAX + 1, &R, &W, &E, &timeout)) {
2678 return SUCCESS;
2679 }
2680 }
2681 return FAILURE;
2682 }
2683
2684 static int php_http_client_curl_once(php_http_client_t *h)
2685 {
2686 php_http_client_curl_t *curl = h->ctx;
2687
2688 if (!h->callback.depth) {
2689 if (curl->ev_ops) {
2690 curl->ev_ops->once(curl->ev_ctx);
2691 } else {
2692 while (CURLM_CALL_MULTI_PERFORM == curl_multi_perform(curl->handle->multi, &curl->unfinished));
2693 }
2694
2695 php_http_client_curl_responsehandler(h);
2696 }
2697
2698 return curl->unfinished;
2699 }
2700
2701 static ZEND_RESULT_CODE php_http_client_curl_exec(php_http_client_t *h)
2702 {
2703 php_http_client_curl_t *curl = h->ctx;
2704
2705 if (!h->callback.depth) {
2706 if (curl->ev_ops) {
2707 return curl->ev_ops->exec(curl->ev_ctx);
2708 }
2709
2710 while (php_http_client_curl_once(h) && !EG(exception)) {
2711 if (SUCCESS != php_http_client_curl_wait(h, NULL)) {
2712 #if PHP_WIN32
2713 /* see http://msdn.microsoft.com/library/en-us/winsock/winsock/windows_sockets_error_codes_2.asp */
2714 php_error_docref(NULL, E_WARNING, "WinSock error: %d", WSAGetLastError());
2715 #else
2716 php_error_docref(NULL, E_WARNING, "%s", strerror(errno));
2717 #endif
2718 return FAILURE;
2719 }
2720 }
2721 }
2722
2723 return SUCCESS;
2724 }
2725
2726 static ZEND_RESULT_CODE php_http_client_curl_setopt(php_http_client_t *h, php_http_client_setopt_opt_t opt, void *arg)
2727 {
2728 php_http_client_curl_t *curl = h->ctx;
2729
2730 (void) curl;
2731
2732 switch (opt) {
2733 case PHP_HTTP_CLIENT_OPT_CONFIGURATION:
2734 return php_http_options_apply(&php_http_curlm_options, (HashTable *) arg, h);
2735 break;
2736 #if !PHP_HTTP_CURL_VERSION(7,62,0)
2737 case PHP_HTTP_CLIENT_OPT_ENABLE_PIPELINING:
2738 if (CURLM_OK != curl_multi_setopt(curl->handle->multi, CURLMOPT_PIPELINING, (long) *((zend_bool *) arg))) {
2739 return FAILURE;
2740 }
2741 break;
2742 #endif
2743 case PHP_HTTP_CLIENT_OPT_USE_EVENTS:
2744 #if PHP_HTTP_HAVE_LIBEVENT
2745 return php_http_curlm_use_eventloop(h, (*(zend_bool *) arg)
2746 ? php_http_client_curl_event_ops_get()
2747 : NULL, NULL);
2748 break;
2749 #endif
2750
2751 default:
2752 return FAILURE;
2753 }
2754 return SUCCESS;
2755 }
2756
2757 static int apply_available_options(zval *pDest, int num_args, va_list args, zend_hash_key *hash_key)
2758 {
2759 php_http_option_t *opt = Z_PTR_P(pDest);
2760 HashTable *ht;
2761 zval entry;
2762 int c;
2763
2764 ht = va_arg(args, HashTable*);
2765
2766 if ((c = zend_hash_num_elements(&opt->suboptions.options))) {
2767 array_init_size(&entry, c);
2768 zend_hash_apply_with_arguments(&opt->suboptions.options, apply_available_options, 1, Z_ARRVAL(entry));
2769 } else {
2770 /* catch deliberate NULL options */
2771 if (Z_TYPE(opt->defval) == IS_STRING && !Z_STRVAL(opt->defval)) {
2772 ZVAL_NULL(&entry);
2773 } else {
2774 ZVAL_ZVAL(&entry, &opt->defval, 1, 0);
2775 }
2776 }
2777
2778 if (hash_key->key) {
2779 zend_hash_update(ht, hash_key->key, &entry);
2780 } else {
2781 zend_hash_index_update(ht, hash_key->h, &entry);
2782 }
2783
2784 return ZEND_HASH_APPLY_KEEP;
2785 }
2786
2787 static ZEND_RESULT_CODE php_http_client_curl_getopt(php_http_client_t *h, php_http_client_getopt_opt_t opt, void *arg, void **res)
2788 {
2789 php_http_client_enqueue_t *enqueue;
2790
2791 switch (opt) {
2792 case PHP_HTTP_CLIENT_OPT_PROGRESS_INFO:
2793 if ((enqueue = php_http_client_enqueued(h, arg, NULL))) {
2794 php_http_client_curl_handler_t *handler = enqueue->opaque;
2795
2796 *((php_http_client_progress_state_t **) res) = &handler->progress;
2797 return SUCCESS;
2798 }
2799 break;
2800
2801 case PHP_HTTP_CLIENT_OPT_TRANSFER_INFO:
2802 if ((enqueue = php_http_client_enqueued(h, arg, NULL))) {
2803 php_http_client_curl_handler_t *handler = enqueue->opaque;
2804
2805 php_http_curle_get_info(handler->handle, *(HashTable **) res);
2806 return SUCCESS;
2807 }
2808 break;
2809
2810 case PHP_HTTP_CLIENT_OPT_AVAILABLE_OPTIONS:
2811 zend_hash_apply_with_arguments(&php_http_curle_options.options, apply_available_options, 1, *(HashTable **) res);
2812 break;
2813
2814 case PHP_HTTP_CLIENT_OPT_AVAILABLE_CONFIGURATION:
2815 zend_hash_apply_with_arguments(&php_http_curlm_options.options, apply_available_options, 1, *(HashTable **) res);
2816 break;
2817
2818 default:
2819 break;
2820 }
2821
2822 return FAILURE;
2823 }
2824
2825 static php_http_client_ops_t php_http_client_curl_ops = {
2826 &php_http_curlm_resource_factory_ops,
2827 php_http_client_curl_init,
2828 NULL /* copy */,
2829 php_http_client_curl_dtor,
2830 php_http_client_curl_reset,
2831 php_http_client_curl_exec,
2832 php_http_client_curl_wait,
2833 php_http_client_curl_once,
2834 php_http_client_curl_enqueue,
2835 php_http_client_curl_dequeue,
2836 php_http_client_curl_requeue,
2837 php_http_client_curl_setopt,
2838 php_http_client_curl_getopt
2839 };
2840
2841 php_http_client_ops_t *php_http_client_curl_get_ops(void)
2842 {
2843 return &php_http_client_curl_ops;
2844 }
2845
2846 #define REGISTER_NS_STRING_OR_NULL_CONSTANT(ns, name, str, flags) \
2847 do { \
2848 if ((str) != NULL) { \
2849 REGISTER_NS_STRING_CONSTANT(ns, name, str, flags); \
2850 } else { \
2851 REGISTER_NS_NULL_CONSTANT(ns, name, flags); \
2852 } \
2853 } while (0)
2854
2855 PHP_MINIT_FUNCTION(http_client_curl)
2856 {
2857 curl_version_info_data *info;
2858 php_http_options_t *options;
2859
2860 PHP_HTTP_G->client.curl.driver.driver_name = zend_string_init(ZEND_STRL("curl"), 1);
2861 PHP_HTTP_G->client.curl.driver.client_name = zend_string_init(ZEND_STRL("http\\Client\\Curl"), 1);
2862 PHP_HTTP_G->client.curl.driver.request_name = zend_string_init(ZEND_STRL("http\\Client\\Curl\\Request"), 1);
2863 PHP_HTTP_G->client.curl.driver.client_ops = &php_http_client_curl_ops;
2864
2865 if (SUCCESS != php_http_client_driver_add(&PHP_HTTP_G->client.curl.driver)) {
2866 return FAILURE;
2867 }
2868
2869 if (SUCCESS != php_persistent_handle_provide(PHP_HTTP_G->client.curl.driver.client_name, &php_http_curlm_resource_factory_ops, NULL, NULL)) {
2870 return FAILURE;
2871 }
2872 if (SUCCESS != php_persistent_handle_provide(PHP_HTTP_G->client.curl.driver.request_name, &php_http_curle_resource_factory_ops, NULL, NULL)) {
2873 return FAILURE;
2874 }
2875
2876 if ((options = php_http_options_init(&php_http_curle_options, 1))) {
2877 options->getter = php_http_curle_get_option;
2878 options->setter = php_http_curle_set_option;
2879
2880 php_http_curle_options_init(options);
2881 }
2882 if ((options = php_http_options_init(&php_http_curlm_options, 1))) {
2883 options->getter = php_http_option_get;
2884 options->setter = php_http_curlm_set_option;
2885
2886 php_http_curlm_options_init(options);
2887 }
2888
2889 if ((info = curl_version_info(CURLVERSION_NOW))) {
2890 char tmp_ver[0x20], *tmp_ptr, *tmp_end;
2891 #define tmp_ver_init() do {\
2892 tmp_ver[0] = 0; \
2893 tmp_ptr = &tmp_ver[0]; \
2894 tmp_end = &tmp_ver[sizeof(tmp_ver) - 1]; \
2895 } while (0)
2896
2897 /*
2898 * Feature constants
2899 */
2900 REGISTER_NS_LONG_CONSTANT("http\\Client\\Curl", "FEATURES", info->features, CONST_CS|CONST_PERSISTENT);
2901
2902 REGISTER_NS_LONG_CONSTANT("http\\Client\\Curl\\Features", "IPV6", CURL_VERSION_IPV6, CONST_CS|CONST_PERSISTENT);
2903 REGISTER_NS_LONG_CONSTANT("http\\Client\\Curl\\Features", "KERBEROS4", CURL_VERSION_KERBEROS4, CONST_CS|CONST_PERSISTENT);
2904 REGISTER_NS_LONG_CONSTANT("http\\Client\\Curl\\Features", "SSL", CURL_VERSION_SSL, CONST_CS|CONST_PERSISTENT);
2905 REGISTER_NS_LONG_CONSTANT("http\\Client\\Curl\\Features", "LIBZ", CURL_VERSION_LIBZ, CONST_CS|CONST_PERSISTENT);
2906 REGISTER_NS_LONG_CONSTANT("http\\Client\\Curl\\Features", "NTLM", CURL_VERSION_NTLM, CONST_CS|CONST_PERSISTENT);
2907 REGISTER_NS_LONG_CONSTANT("http\\Client\\Curl\\Features", "GSSNEGOTIATE", CURL_VERSION_GSSNEGOTIATE, CONST_CS|CONST_PERSISTENT);
2908 REGISTER_NS_LONG_CONSTANT("http\\Client\\Curl\\Features", "ASYNCHDNS", CURL_VERSION_ASYNCHDNS, CONST_CS|CONST_PERSISTENT);
2909 REGISTER_NS_LONG_CONSTANT("http\\Client\\Curl\\Features", "SPNEGO", CURL_VERSION_SPNEGO, CONST_CS|CONST_PERSISTENT);
2910 REGISTER_NS_LONG_CONSTANT("http\\Client\\Curl\\Features", "LARGEFILE", CURL_VERSION_LARGEFILE, CONST_CS|CONST_PERSISTENT);
2911 REGISTER_NS_LONG_CONSTANT("http\\Client\\Curl\\Features", "IDN", CURL_VERSION_IDN, CONST_CS|CONST_PERSISTENT);
2912 REGISTER_NS_LONG_CONSTANT("http\\Client\\Curl\\Features", "SSPI", CURL_VERSION_SSPI, CONST_CS|CONST_PERSISTENT);
2913 #if PHP_HTTP_CURL_VERSION(7,21,4)
2914 REGISTER_NS_LONG_CONSTANT("http\\Client\\Curl\\Features", "TLSAUTH_SRP", CURL_VERSION_TLSAUTH_SRP, CONST_CS|CONST_PERSISTENT);
2915 #endif
2916 #if PHP_HTTP_CURL_VERSION(7,22,0)
2917 REGISTER_NS_LONG_CONSTANT("http\\Client\\Curl\\Features", "NTLM_WB", CURL_VERSION_NTLM_WB, CONST_CS|CONST_PERSISTENT);
2918 #endif
2919 #if PHP_HTTP_CURL_VERSION(7,33,0)
2920 REGISTER_NS_LONG_CONSTANT("http\\Client\\Curl\\Features", "HTTP2", CURL_VERSION_HTTP2, CONST_CS|CONST_PERSISTENT);
2921 #endif
2922 #if PHP_HTTP_CURL_VERSION(7,38,0)
2923 REGISTER_NS_LONG_CONSTANT("http\\Client\\Curl\\Features", "GSSAPI", CURL_VERSION_GSSAPI, CONST_CS|CONST_PERSISTENT);
2924 #endif
2925 #if PHP_HTTP_CURL_VERSION(7,40,0)
2926 REGISTER_NS_LONG_CONSTANT("http\\Client\\Curl\\Features", "KERBEROS5", CURL_VERSION_KERBEROS5, CONST_CS|CONST_PERSISTENT);
2927 REGISTER_NS_LONG_CONSTANT("http\\Client\\Curl\\Features", "UNIX_SOCKETS", CURL_VERSION_UNIX_SOCKETS, CONST_CS|CONST_PERSISTENT);
2928 #endif
2929 #if PHP_HTTP_CURL_VERSION(7,47,0)
2930 REGISTER_NS_LONG_CONSTANT("http\\Client\\Curl\\Features", "PSL", CURL_VERSION_PSL, CONST_CS|CONST_PERSISTENT);
2931 #endif
2932 #if PHP_HTTP_CURL_VERSION(7,52,0)
2933 REGISTER_NS_LONG_CONSTANT("http\\Client\\Curl\\Features", "HTTPS_PROXY", CURL_VERSION_HTTPS_PROXY, CONST_CS|CONST_PERSISTENT);
2934 #endif
2935 #if PHP_HTTP_CURL_VERSION(7,56,0)
2936 REGISTER_NS_LONG_CONSTANT("http\\Client\\Curl\\Features", "MULTI_SSL", CURL_VERSION_MULTI_SSL, CONST_CS|CONST_PERSISTENT);
2937 #endif
2938 #if PHP_HTTP_CURL_VERSION(7,57,0)
2939 REGISTER_NS_LONG_CONSTANT("http\\Client\\Curl\\Features", "BROTLI", CURL_VERSION_BROTLI, CONST_CS|CONST_PERSISTENT);
2940 #endif
2941 #if PHP_HTTP_CURL_VERSION(7,64,1)
2942 REGISTER_NS_LONG_CONSTANT("http\\Client\\Curl\\Features", "ALTSVC", CURL_VERSION_ALTSVC, CONST_CS|CONST_PERSISTENT);
2943 #endif
2944 #if PHP_HTTP_CURL_VERSION(7,66,0)
2945 REGISTER_NS_LONG_CONSTANT("http\\Client\\Curl\\Features", "HTTP3", CURL_VERSION_HTTP3, CONST_CS|CONST_PERSISTENT);
2946 #endif
2947 #if PHP_HTTP_CURL_VERSION(7,72,0)
2948 REGISTER_NS_LONG_CONSTANT("http\\Client\\Curl\\Features", "ZSTD", CURL_VERSION_ZSTD, CONST_CS|CONST_PERSISTENT);
2949 REGISTER_NS_LONG_CONSTANT("http\\Client\\Curl\\Features", "UNICODE", CURL_VERSION_UNICODE, CONST_CS|CONST_PERSISTENT);
2950 #endif
2951 #if PHP_HTTP_CURL_VERSION(7,74,0)
2952 REGISTER_NS_LONG_CONSTANT("http\\Client\\Curl\\Features", "HSTS", CURL_VERSION_HSTS, CONST_CS|CONST_PERSISTENT);
2953 #endif
2954
2955
2956 /*
2957 * Version constants
2958 */
2959 REGISTER_NS_STRING_CONSTANT("http\\Client\\Curl", "VERSIONS", curl_version(), CONST_CS|CONST_PERSISTENT);
2960 REGISTER_NS_STRING_CONSTANT("http\\Client\\Curl\\Versions", "CURL", info->version, CONST_CS|CONST_PERSISTENT);
2961 REGISTER_NS_STRING_OR_NULL_CONSTANT("http\\Client\\Curl\\Versions", "SSL", info->ssl_version, CONST_CS|CONST_PERSISTENT);
2962 REGISTER_NS_STRING_OR_NULL_CONSTANT("http\\Client\\Curl\\Versions", "LIBZ", info->libz_version, CONST_CS|CONST_PERSISTENT);
2963 REGISTER_NS_STRING_OR_NULL_CONSTANT("http\\Client\\Curl\\Versions", "ARES", info->ares, CONST_CS|CONST_PERSISTENT);
2964 REGISTER_NS_STRING_OR_NULL_CONSTANT("http\\Client\\Curl\\Versions", "IDN", info->libidn, CONST_CS|CONST_PERSISTENT);
2965 tmp_ver_init();
2966 if (info->iconv_ver_num) {
2967 tmp_ptr = zend_print_ulong_to_buf(tmp_end, info->iconv_ver_num & 0xf);
2968 tmp_end = tmp_ptr - 1;
2969 tmp_ptr = zend_print_ulong_to_buf(tmp_end, info->iconv_ver_num >> 8);
2970 *tmp_end = '.';
2971 }
2972 REGISTER_NS_STRING_OR_NULL_CONSTANT("http\\Client\\Curl\\Versions", "ICONV", *tmp_ptr ? tmp_ptr : NULL, CONST_CS|CONST_PERSISTENT);
2973 #if PHP_HTTP_CURL_VERSION(7,57,0)
2974 REGISTER_NS_STRING_OR_NULL_CONSTANT("http\\Client\\Curl\\Versions", "BROTLI", info->brotli_version, CONST_CS|CONST_PERSISTENT);
2975 #endif
2976 #if PHP_HTTP_CURL_VERSION(7,66,0)
2977 REGISTER_NS_STRING_OR_NULL_CONSTANT("http\\Client\\Curl\\Versions", "NGHTTP2", info->nghttp2_version, CONST_CS|CONST_PERSISTENT);
2978 REGISTER_NS_STRING_OR_NULL_CONSTANT("http\\Client\\Curl\\Versions", "QUIC", info->quic_version, CONST_CS|CONST_PERSISTENT);
2979 #endif
2980 #if PHP_HTTP_CURL_VERSION(7,70,0)
2981 REGISTER_NS_STRING_OR_NULL_CONSTANT("http\\Client\\Curl\\Versions", "CAINFO", info->cainfo, CONST_CS|CONST_PERSISTENT);
2982 REGISTER_NS_STRING_OR_NULL_CONSTANT("http\\Client\\Curl\\Versions", "CAPATH", info->capath, CONST_CS|CONST_PERSISTENT);
2983 #endif
2984 #if PHP_HTTP_CURL_VERSION(7,72,0)
2985 REGISTER_NS_STRING_OR_NULL_CONSTANT("http\\Client\\Curl\\Versions", "ZSTD", info->zstd_version, CONST_CS|CONST_PERSISTENT);
2986 #endif
2987 #if PHP_HTTP_CURL_VERSION(7,75,0)
2988 REGISTER_NS_STRING_OR_NULL_CONSTANT("http\\Client\\Curl\\Versions", "HYPER", info->hyper_version, CONST_CS|CONST_PERSISTENT);
2989 #endif
2990
2991 }
2992
2993 /*
2994 * HTTP Protocol Version Constants
2995 */
2996 REGISTER_NS_LONG_CONSTANT("http\\Client\\Curl", "HTTP_VERSION_1_0", CURL_HTTP_VERSION_1_0, CONST_CS|CONST_PERSISTENT);
2997 REGISTER_NS_LONG_CONSTANT("http\\Client\\Curl", "HTTP_VERSION_1_1", CURL_HTTP_VERSION_1_1, CONST_CS|CONST_PERSISTENT);
2998 #if PHP_HTTP_CURL_VERSION(7,33,0)
2999 REGISTER_NS_LONG_CONSTANT("http\\Client\\Curl", "HTTP_VERSION_2_0", CURL_HTTP_VERSION_2_0, CONST_CS|CONST_PERSISTENT);
3000 #endif
3001 #if PHP_HTTP_CURL_VERSION(7,47,0)
3002 REGISTER_NS_LONG_CONSTANT("http\\Client\\Curl", "HTTP_VERSION_2TLS", CURL_HTTP_VERSION_2TLS, CONST_CS|CONST_PERSISTENT);
3003 #endif
3004 #if PHP_HTTP_CURL_VERSION(7,49,0)
3005 REGISTER_NS_LONG_CONSTANT("http\\Client\\Curl", "HTTP_VERSION_2_PRIOR_KNOWLEDGE", CURL_HTTP_VERSION_2_PRIOR_KNOWLEDGE, CONST_CS|CONST_PERSISTENT);
3006 #endif
3007 #if PHP_HTTP_CURL_VERSION(7,66,0)
3008 REGISTER_NS_LONG_CONSTANT("http\\Client\\Curl", "HTTP_VERSION_3", CURL_HTTP_VERSION_3, CONST_CS|CONST_PERSISTENT);
3009 #endif
3010 REGISTER_NS_LONG_CONSTANT("http\\Client\\Curl", "HTTP_VERSION_ANY", CURL_HTTP_VERSION_NONE, CONST_CS|CONST_PERSISTENT);
3011
3012 /*
3013 * SSL Version Constants
3014 */
3015 REGISTER_NS_LONG_CONSTANT("http\\Client\\Curl", "SSL_VERSION_TLSv1", CURL_SSLVERSION_TLSv1, CONST_CS|CONST_PERSISTENT);
3016 #if PHP_HTTP_CURL_VERSION(7,34,0)
3017 REGISTER_NS_LONG_CONSTANT("http\\Client\\Curl", "SSL_VERSION_TLSv1_0", CURL_SSLVERSION_TLSv1_0, CONST_CS|CONST_PERSISTENT);
3018 REGISTER_NS_LONG_CONSTANT("http\\Client\\Curl", "SSL_VERSION_TLSv1_1", CURL_SSLVERSION_TLSv1_1, CONST_CS|CONST_PERSISTENT);
3019 REGISTER_NS_LONG_CONSTANT("http\\Client\\Curl", "SSL_VERSION_TLSv1_2", CURL_SSLVERSION_TLSv1_2, CONST_CS|CONST_PERSISTENT);
3020 #endif
3021 #if PHP_HTTP_CURL_VERSION(7,52,0)
3022 REGISTER_NS_LONG_CONSTANT("http\\Client\\Curl", "SSL_VERSION_TLSv1_3", CURL_SSLVERSION_TLSv1_3, CONST_CS|CONST_PERSISTENT);
3023 #endif
3024 REGISTER_NS_LONG_CONSTANT("http\\Client\\Curl", "SSL_VERSION_SSLv2", CURL_SSLVERSION_SSLv2, CONST_CS|CONST_PERSISTENT);
3025 REGISTER_NS_LONG_CONSTANT("http\\Client\\Curl", "SSL_VERSION_SSLv3", CURL_SSLVERSION_SSLv3, CONST_CS|CONST_PERSISTENT);
3026 REGISTER_NS_LONG_CONSTANT("http\\Client\\Curl", "SSL_VERSION_ANY", CURL_SSLVERSION_DEFAULT, CONST_CS|CONST_PERSISTENT);
3027 #if PHP_HTTP_CURL_VERSION(7,21,4) && PHP_HTTP_HAVE_LIBCURL_TLSAUTH_TYPE
3028 REGISTER_NS_LONG_CONSTANT("http\\Client\\Curl", "TLSAUTH_SRP", CURL_TLSAUTH_SRP, CONST_CS|CONST_PERSISTENT);
3029 #endif
3030
3031 #if PHP_HTTP_CURL_VERSION(7,54,0)
3032 REGISTER_NS_LONG_CONSTANT("http\\Client\\Curl", "SSL_VERSION_MAX_DEFAULT", CURL_SSLVERSION_MAX_DEFAULT, CONST_CS|CONST_PERSISTENT);
3033 REGISTER_NS_LONG_CONSTANT("http\\Client\\Curl", "SSL_VERSION_MAX_TLSv1_0", CURL_SSLVERSION_MAX_TLSv1_0, CONST_CS|CONST_PERSISTENT);
3034 REGISTER_NS_LONG_CONSTANT("http\\Client\\Curl", "SSL_VERSION_MAX_TLSv1_1", CURL_SSLVERSION_MAX_TLSv1_1, CONST_CS|CONST_PERSISTENT);
3035 REGISTER_NS_LONG_CONSTANT("http\\Client\\Curl", "SSL_VERSION_MAX_TLSv1_2", CURL_SSLVERSION_MAX_TLSv1_2, CONST_CS|CONST_PERSISTENT);
3036 REGISTER_NS_LONG_CONSTANT("http\\Client\\Curl", "SSL_VERSION_MAX_TLSv1_3", CURL_SSLVERSION_MAX_TLSv1_3, CONST_CS|CONST_PERSISTENT);
3037 #endif
3038
3039 /*
3040 * DNS IPvX resolving
3041 */
3042 REGISTER_NS_LONG_CONSTANT("http\\Client\\Curl", "IPRESOLVE_V4", CURL_IPRESOLVE_V4, CONST_CS|CONST_PERSISTENT);
3043 REGISTER_NS_LONG_CONSTANT("http\\Client\\Curl", "IPRESOLVE_V6", CURL_IPRESOLVE_V6, CONST_CS|CONST_PERSISTENT);
3044 REGISTER_NS_LONG_CONSTANT("http\\Client\\Curl", "IPRESOLVE_ANY", CURL_IPRESOLVE_WHATEVER, CONST_CS|CONST_PERSISTENT);
3045
3046 /*
3047 * Auth Constants
3048 */
3049 REGISTER_NS_LONG_CONSTANT("http\\Client\\Curl", "AUTH_NONE", CURLAUTH_NONE, CONST_CS|CONST_PERSISTENT);
3050 REGISTER_NS_LONG_CONSTANT("http\\Client\\Curl", "AUTH_BASIC", CURLAUTH_BASIC, CONST_CS|CONST_PERSISTENT);
3051 REGISTER_NS_LONG_CONSTANT("http\\Client\\Curl", "AUTH_DIGEST", CURLAUTH_DIGEST, CONST_CS|CONST_PERSISTENT);
3052 #if PHP_HTTP_CURL_VERSION(7,19,3)
3053 REGISTER_NS_LONG_CONSTANT("http\\Client\\Curl", "AUTH_DIGEST_IE", CURLAUTH_DIGEST_IE, CONST_CS|CONST_PERSISTENT);
3054 #endif
3055 REGISTER_NS_LONG_CONSTANT("http\\Client\\Curl", "AUTH_NTLM", CURLAUTH_NTLM, CONST_CS|CONST_PERSISTENT);
3056 REGISTER_NS_LONG_CONSTANT("http\\Client\\Curl", "AUTH_GSSNEG", CURLAUTH_GSSNEGOTIATE, CONST_CS|CONST_PERSISTENT);
3057 #if PHP_HTTP_CURL_VERSION(7,38,0)
3058 REGISTER_NS_LONG_CONSTANT("http\\Client\\Curl", "AUTH_SPNEGO", CURLAUTH_NEGOTIATE, CONST_CS|CONST_PERSISTENT);
3059 #endif
3060 #if PHP_HTTP_CURL_VERSION(7,61,0)
3061 REGISTER_NS_LONG_CONSTANT("http\\Client\\Curl", "AUTH_BEARER", CURLAUTH_BEARER, CONST_CS|CONST_PERSISTENT);
3062 #endif
3063 #if PHP_HTTP_CURL_VERSION(7,75,0)
3064 REGISTER_NS_LONG_CONSTANT("http\\Client\\Curl", "AWS_SIGV4", CURLAUTH_AWS_SIGV4, CONST_CS|CONST_PERSISTENT);
3065 #endif
3066 REGISTER_NS_LONG_CONSTANT("http\\Client\\Curl", "AUTH_ANY", CURLAUTH_ANY, CONST_CS|CONST_PERSISTENT);
3067
3068 /*
3069 * Proxy Type Constants
3070 */
3071 REGISTER_NS_LONG_CONSTANT("http\\Client\\Curl", "PROXY_SOCKS4", CURLPROXY_SOCKS4, CONST_CS|CONST_PERSISTENT);
3072 REGISTER_NS_LONG_CONSTANT("http\\Client\\Curl", "PROXY_SOCKS4A", CURLPROXY_SOCKS4A, CONST_CS|CONST_PERSISTENT);
3073 REGISTER_NS_LONG_CONSTANT("http\\Client\\Curl", "PROXY_SOCKS5_HOSTNAME", CURLPROXY_SOCKS5_HOSTNAME, CONST_CS|CONST_PERSISTENT);
3074 REGISTER_NS_LONG_CONSTANT("http\\Client\\Curl", "PROXY_SOCKS5", CURLPROXY_SOCKS5, CONST_CS|CONST_PERSISTENT);
3075 REGISTER_NS_LONG_CONSTANT("http\\Client\\Curl", "PROXY_HTTP", CURLPROXY_HTTP, CONST_CS|CONST_PERSISTENT);
3076 #if PHP_HTTP_CURL_VERSION(7,19,4)
3077 REGISTER_NS_LONG_CONSTANT("http\\Client\\Curl", "PROXY_HTTP_1_0", CURLPROXY_HTTP_1_0, CONST_CS|CONST_PERSISTENT);
3078 #endif
3079
3080 /*
3081 * Post Redirection Constants
3082 */
3083 #if PHP_HTTP_CURL_VERSION(7,19,1)
3084 REGISTER_NS_LONG_CONSTANT("http\\Client\\Curl", "POSTREDIR_301", CURL_REDIR_POST_301, CONST_CS|CONST_PERSISTENT);
3085 REGISTER_NS_LONG_CONSTANT("http\\Client\\Curl", "POSTREDIR_302", CURL_REDIR_POST_302, CONST_CS|CONST_PERSISTENT);
3086 #if PHP_HTTP_CURL_VERSION(7,26,0)
3087 REGISTER_NS_LONG_CONSTANT("http\\Client\\Curl", "POSTREDIR_303", CURL_REDIR_POST_303, CONST_CS|CONST_PERSISTENT);
3088 #endif
3089 REGISTER_NS_LONG_CONSTANT("http\\Client\\Curl", "POSTREDIR_ALL", CURL_REDIR_POST_ALL, CONST_CS|CONST_PERSISTENT);
3090 #endif
3091
3092 #if PHP_HTTP_CURL_VERSION(7,64,1)
3093 REGISTER_NS_LONG_CONSTANT("http\\Client\\Curl", "ALTSVC_READONLYFILE", CURLALTSVC_READONLYFILE, CONST_CS|CONST_PERSISTENT);
3094 REGISTER_NS_LONG_CONSTANT("http\\Client\\Curl", "ALTSVC_H1", CURLALTSVC_H1, CONST_CS|CONST_PERSISTENT);
3095 REGISTER_NS_LONG_CONSTANT("http\\Client\\Curl", "ALTSVC_H2", CURLALTSVC_H2, CONST_CS|CONST_PERSISTENT);
3096 REGISTER_NS_LONG_CONSTANT("http\\Client\\Curl", "ALTSVC_H3", CURLALTSVC_H3, CONST_CS|CONST_PERSISTENT);
3097 #endif
3098 #if PHP_HTTP_CURL_VERSION(7,74,0)
3099 REGISTER_NS_LONG_CONSTANT("http\\Client\\Curl", "HSTS_ENABLE", CURLHSTS_ENABLE, CONST_CS|CONST_PERSISTENT);
3100 REGISTER_NS_LONG_CONSTANT("http\\Client\\Curl", "HSTS_READONLYFILE", CURLHSTS_READONLYFILE, CONST_CS|CONST_PERSISTENT);
3101 #endif
3102 return SUCCESS;
3103 }
3104
3105 PHP_MSHUTDOWN_FUNCTION(http_client_curl)
3106 {
3107 php_persistent_handle_cleanup(PHP_HTTP_G->client.curl.driver.client_name, NULL);
3108 php_persistent_handle_cleanup(PHP_HTTP_G->client.curl.driver.request_name, NULL);
3109 zend_string_release(PHP_HTTP_G->client.curl.driver.client_name);
3110 zend_string_release(PHP_HTTP_G->client.curl.driver.request_name);
3111 zend_string_release(PHP_HTTP_G->client.curl.driver.driver_name);
3112
3113 php_http_options_dtor(&php_http_curle_options);
3114 php_http_options_dtor(&php_http_curlm_options);
3115
3116 return SUCCESS;
3117 }
3118
3119 #endif /* PHP_HTTP_HAVE_LIBCURL */
3120
3121 /*
3122 * Local variables:
3123 * tab-width: 4
3124 * c-basic-offset: 4
3125 * End:
3126 * vim600: noet sw=4 ts=4 fdm=marker
3127 * vim<600: noet sw=4 ts=4
3128 */