2 +--------------------------------------------------------------------+
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 +--------------------------------------------------------------------+
13 #include "php_http_api.h"
14 #include "php_http_client.h"
16 #include <ext/spl/spl_observer.h>
19 * array of name => php_http_client_driver_t*
21 static HashTable php_http_client_drivers
;
23 static void php_http_client_driver_hash_dtor(zval
*pData
)
25 pefree(Z_PTR_P(pData
), 1);
28 ZEND_RESULT_CODE
php_http_client_driver_add(php_http_client_driver_t
*driver
)
30 return zend_hash_add_mem(&php_http_client_drivers
, driver
->driver_name
, (void *) driver
, sizeof(php_http_client_driver_t
))
34 php_http_client_driver_t
*php_http_client_driver_get(zend_string
*name
)
37 php_http_client_driver_t
*tmp
;
39 if (name
&& (tmp
= zend_hash_find_ptr(&php_http_client_drivers
, name
))) {
42 if ((ztmp
= zend_hash_get_current_data(&php_http_client_drivers
))) {
48 static int apply_driver_list(zval
*p
, void *arg
)
50 php_http_client_driver_t
*d
= Z_PTR_P(p
);
53 ZVAL_STR(&zname
, d
->driver_name
);
55 zend_hash_next_index_insert(arg
, &zname
);
56 return ZEND_HASH_APPLY_KEEP
;
59 void php_http_client_driver_list(HashTable
*ht
)
61 zend_hash_apply_with_argument(&php_http_client_drivers
, apply_driver_list
, ht
);
64 void php_http_client_options_set_subr(zval
*instance
, char *key
, size_t len
, zval
*opts
, int overwrite
)
66 if (overwrite
|| (opts
&& zend_hash_num_elements(Z_ARRVAL_P(opts
)))) {
67 zend_class_entry
*this_ce
= Z_OBJCE_P(instance
);
68 zval old_opts_tmp
, *old_opts
, new_opts
, *entry
= NULL
;
70 array_init(&new_opts
);
71 old_opts
= zend_read_property(this_ce
, instance
, ZEND_STRL("options"), 0, &old_opts_tmp
);
72 if (Z_TYPE_P(old_opts
) == IS_ARRAY
) {
73 array_copy(Z_ARRVAL_P(old_opts
), Z_ARRVAL(new_opts
));
77 if (opts
&& zend_hash_num_elements(Z_ARRVAL_P(opts
))) {
79 zend_symtable_str_update(Z_ARRVAL(new_opts
), key
, len
, opts
);
81 zend_symtable_str_del(Z_ARRVAL(new_opts
), key
, len
);
83 } else if (opts
&& zend_hash_num_elements(Z_ARRVAL_P(opts
))) {
84 if ((entry
= zend_symtable_str_find(Z_ARRVAL(new_opts
), key
, len
))) {
85 array_join(Z_ARRVAL_P(opts
), Z_ARRVAL_P(entry
), 0, 0);
88 zend_symtable_str_update(Z_ARRVAL(new_opts
), key
, len
, opts
);
92 zend_update_property(this_ce
, instance
, ZEND_STRL("options"), &new_opts
);
93 zval_ptr_dtor(&new_opts
);
97 void php_http_client_options_set(zval
*instance
, zval
*opts
)
99 php_http_arrkey_t key
;
101 zend_class_entry
*this_ce
= Z_OBJCE_P(instance
);
102 zend_bool is_client
= instanceof_function(this_ce
, php_http_client_class_entry
);
104 array_init(&new_opts
);
106 if (!opts
|| !zend_hash_num_elements(Z_ARRVAL_P(opts
))) {
107 zend_update_property(this_ce
, instance
, ZEND_STRL("options"), &new_opts
);
108 zval_ptr_dtor(&new_opts
);
110 zval old_opts_tmp
, *old_opts
, add_opts
, *opt
;
112 array_init(&add_opts
);
113 /* some options need extra attention -- thus cannot use array_merge() directly */
114 ZEND_HASH_FOREACH_KEY_VAL(Z_ARRVAL_P(opts
), key
.h
, key
.key
, opt
)
117 if (Z_TYPE_P(opt
) == IS_ARRAY
&& (zend_string_equals_literal(key
.key
, "ssl") || zend_string_equals_literal(key
.key
, "cookies"))) {
118 php_http_client_options_set_subr(instance
, key
.key
->val
, key
.key
->len
, opt
, 0);
119 } else if (is_client
&& (zend_string_equals_literal(key
.key
, "recordHistory") || zend_string_equals_literal(key
.key
, "responseMessageClass"))) {
120 zend_update_property(this_ce
, instance
, key
.key
->val
, key
.key
->len
, opt
);
121 } else if (Z_TYPE_P(opt
) == IS_NULL
) {
122 old_opts
= zend_read_property(this_ce
, instance
, ZEND_STRL("options"), 0, &old_opts_tmp
);
123 if (Z_TYPE_P(old_opts
) == IS_ARRAY
) {
124 zend_symtable_del(Z_ARRVAL_P(old_opts
), key
.key
);
128 add_assoc_zval_ex(&add_opts
, key
.key
->val
, key
.key
->len
, opt
);
132 ZEND_HASH_FOREACH_END();
134 old_opts
= zend_read_property(this_ce
, instance
, ZEND_STRL("options"), 0, &old_opts_tmp
);
135 if (Z_TYPE_P(old_opts
) == IS_ARRAY
) {
136 array_copy(Z_ARRVAL_P(old_opts
), Z_ARRVAL(new_opts
));
138 array_join(Z_ARRVAL(add_opts
), Z_ARRVAL(new_opts
), 0, 0);
139 zend_update_property(this_ce
, instance
, ZEND_STRL("options"), &new_opts
);
140 zval_ptr_dtor(&new_opts
);
141 zval_ptr_dtor(&add_opts
);
145 void php_http_client_options_get_subr(zval
*instance
, char *key
, size_t len
, zval
*return_value
)
147 zend_class_entry
*this_ce
= Z_OBJCE_P(instance
);
148 zval
*options
, opts_tmp
, *opts
= zend_read_property(this_ce
, instance
, ZEND_STRL("options"), 0, &opts_tmp
);
150 if ((Z_TYPE_P(opts
) == IS_ARRAY
) && (options
= zend_symtable_str_find(Z_ARRVAL_P(opts
), key
, len
))) {
151 RETVAL_ZVAL_FAST(options
);
155 static void queue_dtor(void *enqueued
)
157 php_http_client_enqueue_t
*e
= enqueued
;
164 php_http_client_t
*php_http_client_init(php_http_client_t
*h
, php_http_client_ops_t
*ops
, php_resource_factory_t
*rf
, void *init_arg
)
166 php_http_client_t
*free_h
= NULL
;
169 free_h
= h
= emalloc(sizeof(*h
));
171 memset(h
, 0, sizeof(*h
));
176 } else if (ops
->rsrc
) {
177 h
->rf
= php_resource_factory_init(NULL
, h
->ops
->rsrc
, h
, NULL
);
179 zend_llist_init(&h
->requests
, sizeof(php_http_client_enqueue_t
), queue_dtor
, 0);
180 zend_llist_init(&h
->responses
, sizeof(void *), NULL
, 0);
183 if (!(h
= h
->ops
->init(h
, init_arg
))) {
184 php_error_docref(NULL
, E_WARNING
, "Could not initialize client");
192 php_http_client_t
*php_http_client_copy(php_http_client_t
*from
, php_http_client_t
*to
)
194 if (from
->ops
->copy
) {
195 return from
->ops
->copy(from
, to
);
201 void php_http_client_dtor(php_http_client_t
*h
)
203 php_http_client_reset(h
);
209 php_resource_factory_free(&h
->rf
);
212 void php_http_client_free(php_http_client_t
**h
) {
214 php_http_client_dtor(*h
);
220 ZEND_RESULT_CODE
php_http_client_enqueue(php_http_client_t
*h
, php_http_client_enqueue_t
*enqueue
)
222 if (h
->ops
->enqueue
) {
223 if (php_http_client_enqueued(h
, enqueue
->request
, NULL
)) {
224 php_error_docref(NULL
, E_WARNING
, "Failed to enqueue request; request already in queue");
227 return h
->ops
->enqueue(h
, enqueue
);
233 ZEND_RESULT_CODE
php_http_client_dequeue(php_http_client_t
*h
, php_http_message_t
*request
)
235 if (h
->ops
->dequeue
) {
236 php_http_client_enqueue_t
*enqueue
= php_http_client_enqueued(h
, request
, NULL
);
239 php_error_docref(NULL
, E_WARNING
, "Failed to dequeue request; request not in queue");
242 return h
->ops
->dequeue(h
, enqueue
);
247 php_http_client_enqueue_t
*php_http_client_enqueued(php_http_client_t
*h
, void *compare_arg
, php_http_client_enqueue_cmp_func_t compare_func
)
249 zend_llist_element
*el
= NULL
;
252 for (el
= h
->requests
.head
; el
; el
= el
->next
) {
253 if (compare_func((php_http_client_enqueue_t
*) el
->data
, compare_arg
)) {
258 for (el
= h
->requests
.head
; el
; el
= el
->next
) {
259 if (((php_http_client_enqueue_t
*) el
->data
)->request
== compare_arg
) {
264 return el
? (php_http_client_enqueue_t
*) el
->data
: NULL
;
267 ZEND_RESULT_CODE
php_http_client_wait(php_http_client_t
*h
, struct timeval
*custom_timeout
)
270 return h
->ops
->wait(h
, custom_timeout
);
276 int php_http_client_once(php_http_client_t
*h
)
279 return h
->ops
->once(h
);
285 ZEND_RESULT_CODE
php_http_client_exec(php_http_client_t
*h
)
288 return h
->ops
->exec(h
);
294 void php_http_client_reset(php_http_client_t
*h
)
300 zend_llist_clean(&h
->requests
);
301 zend_llist_clean(&h
->responses
);
304 ZEND_RESULT_CODE
php_http_client_setopt(php_http_client_t
*h
, php_http_client_setopt_opt_t opt
, void *arg
)
306 if (h
->ops
->setopt
) {
307 return h
->ops
->setopt(h
, opt
, arg
);
313 ZEND_RESULT_CODE
php_http_client_getopt(php_http_client_t
*h
, php_http_client_getopt_opt_t opt
, void *arg
, void *res_ptr
)
315 if (h
->ops
->getopt
) {
316 return h
->ops
->getopt(h
, opt
, arg
, res_ptr
);
321 zend_class_entry
*php_http_client_class_entry
;
322 static zend_object_handlers php_http_client_object_handlers
;
324 void php_http_client_object_free(zend_object
*object
)
326 php_http_client_object_t
*o
= PHP_HTTP_OBJ(object
, NULL
);
328 php_http_client_free(&o
->client
);
329 zend_object_std_dtor(object
);
332 php_http_client_object_t
*php_http_client_object_new_ex(zend_class_entry
*ce
, php_http_client_t
*client
)
334 php_http_client_object_t
*o
;
336 o
= ecalloc(1, sizeof(php_http_client_object_t
) + (ce
->default_properties_count
- 1) * sizeof(zval
));
337 zend_object_std_init(&o
->zo
, ce
);
338 object_properties_init(&o
->zo
, ce
);
342 o
->zo
.handlers
= &php_http_client_object_handlers
;
347 zend_object
*php_http_client_object_new(zend_class_entry
*ce
)
349 return &php_http_client_object_new_ex(ce
, NULL
)->zo
;
352 static void handle_history(zval
*zclient
, php_http_message_t
*request
, php_http_message_t
*response
)
354 zval new_hist
, old_hist_tmp
, *old_hist
= zend_read_property(php_http_client_class_entry
, zclient
, ZEND_STRL("history"), 0, &old_hist_tmp
);
355 php_http_message_t
*req_copy
= php_http_message_copy(request
, NULL
);
356 php_http_message_t
*res_copy
= php_http_message_copy(response
, NULL
);
357 php_http_message_t
*zipped
= php_http_message_zip(res_copy
, req_copy
);
358 php_http_message_object_t
*obj
= php_http_message_object_new_ex(php_http_message_class_entry
, zipped
);
360 ZVAL_OBJ(&new_hist
, &obj
->zo
);
362 if (Z_TYPE_P(old_hist
) == IS_OBJECT
) {
363 php_http_message_object_prepend(&new_hist
, old_hist
, 1);
366 zend_update_property(php_http_client_class_entry
, zclient
, ZEND_STRL("history"), &new_hist
);
367 zval_ptr_dtor(&new_hist
);
370 static ZEND_RESULT_CODE
handle_response(void *arg
, php_http_client_t
*client
, php_http_client_enqueue_t
*e
, php_http_message_t
**request
, php_http_message_t
**response
)
372 zend_bool dequeue
= 0;
374 php_http_message_t
*msg
;
375 php_http_client_progress_state_t
*progress
;
377 ZVAL_OBJ(&zclient
, &((php_http_client_object_t
*) arg
)->zo
);
379 if ((msg
= *response
)) {
380 php_http_message_object_t
*msg_obj
;
381 zval info
, zresponse
, zrequest
, rec_hist_tmp
;
384 /* ensure the message is of type response (could be uninitialized in case of early error, like DNS) */
385 php_http_message_set_type(msg
, PHP_HTTP_RESPONSE
);
387 if (zend_is_true(zend_read_property(php_http_client_class_entry
, &zclient
, ZEND_STRL("recordHistory"), 0, &rec_hist_tmp
))) {
388 handle_history(&zclient
, *request
, *response
);
391 /* hard detach, redirects etc. are in the history */
392 php_http_message_free(&msg
->parent
);
395 msg_obj
= php_http_message_object_new_ex(php_http_client_response_class_entry
, msg
);
396 ZVAL_OBJ(&zresponse
, &msg_obj
->zo
);
397 ZVAL_OBJECT(&zrequest
, &((php_http_message_object_t
*) e
->opaque
)->zo
, 1);
399 php_http_message_object_prepend(&zresponse
, &zrequest
, 1);
402 info_ht
= HASH_OF(&info
);
403 php_http_client_getopt(client
, PHP_HTTP_CLIENT_OPT_TRANSFER_INFO
, e
->request
, &info_ht
);
404 zend_update_property(php_http_client_response_class_entry
, &zresponse
, ZEND_STRL("transferInfo"), &info
);
405 zval_ptr_dtor(&info
);
408 zend_llist_add_element(&client
->responses
, &msg_obj
);
410 if (e
->closure
.fci
.size
) {
412 zend_error_handling zeh
;
415 zend_fcall_info_argn(&e
->closure
.fci
, 1, &zresponse
);
416 zend_replace_error_handling(EH_NORMAL
, NULL
, &zeh
);
417 zend_fcall_info_call(&e
->closure
.fci
, &e
->closure
.fcc
, &retval
, NULL
);
418 zend_restore_error_handling(&zeh
);
419 zend_fcall_info_argn(&e
->closure
.fci
, 0);
421 if (Z_TYPE(retval
) == IS_TRUE
) {
424 zval_ptr_dtor(&retval
);
427 zval_ptr_dtor(&zresponse
);
428 zval_ptr_dtor(&zrequest
);
431 if (SUCCESS
== php_http_client_getopt(client
, PHP_HTTP_CLIENT_OPT_PROGRESS_INFO
, e
->request
, &progress
)) {
432 progress
->info
= "finished";
433 progress
->finished
= 1;
434 client
->callback
.progress
.func(client
->callback
.progress
.arg
, client
, e
, progress
);
438 php_http_client_dequeue(client
, e
->request
);
444 static void handle_progress(void *arg
, php_http_client_t
*client
, php_http_client_enqueue_t
*e
, php_http_client_progress_state_t
*progress
)
446 zval zrequest
, zprogress
, retval
, zclient
;
447 zend_error_handling zeh
;
450 ZVAL_OBJECT(&zclient
, &((php_http_client_object_t
*) arg
)->zo
, 1);
451 ZVAL_OBJECT(&zrequest
, &((php_http_message_object_t
*) e
->opaque
)->zo
, 1);
452 object_init(&zprogress
);
453 add_property_bool(&zprogress
, "started", progress
->started
);
454 add_property_bool(&zprogress
, "finished", progress
->finished
);
455 add_property_string(&zprogress
, "info", STR_PTR(progress
->info
));
456 add_property_double(&zprogress
, "dltotal", progress
->dl
.total
);
457 add_property_double(&zprogress
, "dlnow", progress
->dl
.now
);
458 add_property_double(&zprogress
, "ultotal", progress
->ul
.total
);
459 add_property_double(&zprogress
, "ulnow", progress
->ul
.now
);
460 zend_replace_error_handling(EH_NORMAL
, NULL
, &zeh
);
461 zend_call_method_with_2_params(&zclient
, NULL
, NULL
, "notify", &retval
, &zrequest
, &zprogress
);
462 zend_restore_error_handling(&zeh
);
463 zval_ptr_dtor(&zclient
);
464 zval_ptr_dtor(&zrequest
);
465 zval_ptr_dtor(&zprogress
);
466 zval_ptr_dtor(&retval
);
469 static void response_dtor(void *data
)
471 php_http_message_object_t
*msg_obj
= *(php_http_message_object_t
**) data
;
473 zend_objects_store_del(&msg_obj
->zo
);
476 ZEND_BEGIN_ARG_INFO_EX(ai_HttpClient_construct
, 0, 0, 0)
477 ZEND_ARG_INFO(0, driver
)
478 ZEND_ARG_INFO(0, persistent_handle_id
)
480 static PHP_METHOD(HttpClient
, __construct
)
482 zend_string
*driver_name
= NULL
, *persistent_handle_name
= NULL
;
483 php_http_client_driver_t
*driver
;
484 php_resource_factory_t
*rf
= NULL
;
485 php_http_client_object_t
*obj
;
488 php_http_expect(SUCCESS
== zend_parse_parameters(ZEND_NUM_ARGS(), "|S!S!", &driver_name
, &persistent_handle_name
), invalid_arg
, return);
490 if (!zend_hash_num_elements(&php_http_client_drivers
)) {
491 php_http_throw(unexpected_val
, "No http\\Client drivers available", NULL
);
494 if (!(driver
= php_http_client_driver_get(driver_name
))) {
495 php_http_throw(unexpected_val
, "Failed to locate \"%s\" client request handler", driver_name
? driver_name
->val
: "default");
499 object_init_ex(&os
, spl_ce_SplObjectStorage
);
500 zend_update_property(php_http_client_class_entry
, getThis(), ZEND_STRL("observers"), &os
);
503 if (persistent_handle_name
) {
504 php_persistent_handle_factory_t
*pf
;
506 if ((pf
= php_persistent_handle_concede(NULL
, driver
->client_name
, persistent_handle_name
, NULL
, NULL
))) {
507 rf
= php_resource_factory_init(NULL
, php_persistent_handle_get_resource_factory_ops(), pf
, (void (*)(void *)) php_persistent_handle_abandon
);
511 obj
= PHP_HTTP_OBJ(NULL
, getThis());
513 php_http_expect(obj
->client
= php_http_client_init(NULL
, driver
->client_ops
, rf
, NULL
), runtime
, return);
515 obj
->client
->callback
.response
.func
= handle_response
;
516 obj
->client
->callback
.response
.arg
= obj
;
517 obj
->client
->callback
.progress
.func
= handle_progress
;
518 obj
->client
->callback
.progress
.arg
= obj
;
520 obj
->client
->responses
.dtor
= response_dtor
;
523 ZEND_BEGIN_ARG_INFO_EX(ai_HttpClient_reset
, 0, 0, 0)
525 static PHP_METHOD(HttpClient
, reset
)
527 php_http_client_object_t
*obj
;
528 php_http_expect(SUCCESS
== zend_parse_parameters_none(), invalid_arg
, return);
530 obj
= PHP_HTTP_OBJ(NULL
, getThis());
533 php_http_client_reset(obj
->client
);
535 RETVAL_ZVAL_FAST(getThis());
538 static HashTable
*combined_options(zval
*client
, zval
*request
)
541 unsigned num_options
= 0;
542 zval z_roptions
, z_options_tmp
, *z_coptions
= zend_read_property(php_http_client_class_entry
, client
, ZEND_STRL("options"), 0, &z_options_tmp
);
544 if (Z_TYPE_P(z_coptions
) == IS_ARRAY
) {
545 num_options
= zend_hash_num_elements(Z_ARRVAL_P(z_coptions
));
547 ZVAL_UNDEF(&z_roptions
);
548 zend_call_method_with_0_params(request
, NULL
, NULL
, "getOptions", &z_roptions
);
549 if (Z_TYPE(z_roptions
) == IS_ARRAY
) {
550 unsigned num
= zend_hash_num_elements(Z_ARRVAL(z_roptions
));
551 if (num
> num_options
) {
555 ALLOC_HASHTABLE(options
);
556 ZEND_INIT_SYMTABLE_EX(options
, num_options
, 0);
557 if (Z_TYPE_P(z_coptions
) == IS_ARRAY
) {
558 array_copy(Z_ARRVAL_P(z_coptions
), options
);
560 if (Z_TYPE(z_roptions
) == IS_ARRAY
) {
561 array_join(Z_ARRVAL(z_roptions
), options
, 0, 0);
563 zval_ptr_dtor(&z_roptions
);
568 static void msg_queue_dtor(php_http_client_enqueue_t
*e
)
570 php_http_message_object_t
*msg_obj
= e
->opaque
;
572 zend_objects_store_del(&msg_obj
->zo
);
573 zend_hash_destroy(e
->options
);
574 FREE_HASHTABLE(e
->options
);
576 if (e
->closure
.fci
.size
) {
577 zval_ptr_dtor(&e
->closure
.fci
.function_name
);
578 if (e
->closure
.fci
.object
) {
579 zend_objects_store_del(e
->closure
.fci
.object
);
584 ZEND_BEGIN_ARG_INFO_EX(ai_HttpClient_enqueue
, 0, 0, 1)
585 ZEND_ARG_OBJ_INFO(0, request
, http
\\Client
\\Request
, 0)
586 ZEND_ARG_INFO(0, callable
)
588 static PHP_METHOD(HttpClient
, enqueue
)
591 zend_fcall_info fci
= empty_fcall_info
;
592 zend_fcall_info_cache fcc
= empty_fcall_info_cache
;
593 php_http_client_object_t
*obj
;
594 php_http_message_object_t
*msg_obj
;
595 php_http_client_enqueue_t q
;
597 php_http_expect(SUCCESS
== zend_parse_parameters(ZEND_NUM_ARGS(), "O|f", &request
, php_http_client_request_class_entry
, &fci
, &fcc
), invalid_arg
, return);
599 obj
= PHP_HTTP_OBJ(NULL
, getThis());
600 msg_obj
= PHP_HTTP_OBJ(NULL
, request
);
602 if (php_http_client_enqueued(obj
->client
, msg_obj
->message
, NULL
)) {
603 php_http_throw(bad_method_call
, "Failed to enqueue request; request already in queue", NULL
);
607 q
.request
= msg_obj
->message
;
608 q
.options
= combined_options(getThis(), request
);
609 q
.dtor
= msg_queue_dtor
;
615 Z_TRY_ADDREF(fci
.function_name
);
617 ++GC_REFCOUNT(fci
.object
);
623 php_http_expect(SUCCESS
== php_http_client_enqueue(obj
->client
, &q
), runtime
,
628 RETVAL_ZVAL_FAST(getThis());
631 ZEND_BEGIN_ARG_INFO_EX(ai_HttpClient_dequeue
, 0, 0, 1)
632 ZEND_ARG_OBJ_INFO(0, request
, http
\\Client
\\Request
, 0)
634 static PHP_METHOD(HttpClient
, dequeue
)
637 php_http_client_object_t
*obj
;
638 php_http_message_object_t
*msg_obj
;
640 php_http_expect(SUCCESS
== zend_parse_parameters(ZEND_NUM_ARGS(), "O", &request
, php_http_client_request_class_entry
), invalid_arg
, return);
642 obj
= PHP_HTTP_OBJ(NULL
, getThis());
643 msg_obj
= PHP_HTTP_OBJ(NULL
, request
);
645 if (!php_http_client_enqueued(obj
->client
, msg_obj
->message
, NULL
)) {
646 php_http_throw(bad_method_call
, "Failed to dequeue request; request not in queue", NULL
);
650 php_http_expect(SUCCESS
== php_http_client_dequeue(obj
->client
, msg_obj
->message
), runtime
, return);
652 RETVAL_ZVAL_FAST(getThis());
655 ZEND_BEGIN_ARG_INFO_EX(ai_HttpClient_requeue
, 0, 0, 1)
656 ZEND_ARG_OBJ_INFO(0, request
, http
\\Client
\\Request
, 0)
657 ZEND_ARG_INFO(0, callable
)
659 static PHP_METHOD(HttpClient
, requeue
)
662 zend_fcall_info fci
= empty_fcall_info
;
663 zend_fcall_info_cache fcc
= empty_fcall_info_cache
;
664 php_http_client_object_t
*obj
;
665 php_http_message_object_t
*msg_obj
;
666 php_http_client_enqueue_t q
;
668 php_http_expect(SUCCESS
== zend_parse_parameters(ZEND_NUM_ARGS(), "O|f", &request
, php_http_client_request_class_entry
, &fci
, &fcc
), invalid_arg
, return);
670 obj
= PHP_HTTP_OBJ(NULL
, getThis());
671 msg_obj
= PHP_HTTP_OBJ(NULL
, request
);
673 if (php_http_client_enqueued(obj
->client
, msg_obj
->message
, NULL
)) {
674 php_http_expect(SUCCESS
== php_http_client_dequeue(obj
->client
, msg_obj
->message
), runtime
, return);
677 q
.request
= msg_obj
->message
;
678 q
.options
= combined_options(getThis(), request
);
679 q
.dtor
= msg_queue_dtor
;
685 Z_TRY_ADDREF(fci
.function_name
);
687 ++GC_REFCOUNT(fci
.object
);
693 php_http_expect(SUCCESS
== php_http_client_enqueue(obj
->client
, &q
), runtime
,
698 RETVAL_ZVAL_FAST(getThis());
701 ZEND_BEGIN_ARG_INFO_EX(ai_HttpClient_count
, 0, 0, 0)
703 static PHP_METHOD(HttpClient
, count
)
705 zend_long count_mode
= -1;
707 if (SUCCESS
== zend_parse_parameters(ZEND_NUM_ARGS(), "|l", &count_mode
)) {
708 php_http_client_object_t
*obj
= PHP_HTTP_OBJ(NULL
, getThis());
710 RETVAL_LONG(zend_llist_count(&obj
->client
->requests
));
714 ZEND_BEGIN_ARG_INFO_EX(ai_HttpClient_getResponse
, 0, 0, 0)
715 ZEND_ARG_OBJ_INFO(0, request
, http
\\Client
\\Request
, 1)
717 static PHP_METHOD(HttpClient
, getResponse
)
719 zval
*zrequest
= NULL
;
720 php_http_client_object_t
*obj
;
722 php_http_expect(SUCCESS
== zend_parse_parameters(ZEND_NUM_ARGS(), "|O", &zrequest
, php_http_client_request_class_entry
), invalid_arg
, return);
724 obj
= PHP_HTTP_OBJ(NULL
, getThis());
727 /* lookup the response with the request */
728 zend_llist_element
*el
= NULL
;
729 php_http_message_object_t
*req_obj
= PHP_HTTP_OBJ(NULL
, zrequest
);
731 for (el
= obj
->client
->responses
.head
; el
; el
= el
->next
) {
732 php_http_message_object_t
*response_obj
= *(php_http_message_object_t
**) el
->data
;
734 if (response_obj
->message
->parent
== req_obj
->message
) {
735 RETURN_OBJECT(&response_obj
->zo
, 1);
739 /* not found for the request! */
740 php_http_throw(unexpected_val
, "Could not find response for the request", NULL
);
744 /* pop off the last response */
745 if (obj
->client
->responses
.tail
) {
746 php_http_message_object_t
*response_obj
= *(php_http_message_object_t
**) obj
->client
->responses
.tail
->data
;
750 RETVAL_OBJECT(&response_obj
->zo
, 1);
751 zend_llist_remove_tail(&obj
->client
->responses
);
756 ZEND_BEGIN_ARG_INFO_EX(ai_HttpClient_getHistory
, 0, 0, 0)
758 static PHP_METHOD(HttpClient
, getHistory
)
760 zval zhistory_tmp
, *zhistory
;
762 php_http_expect(SUCCESS
== zend_parse_parameters_none(), invalid_arg
, return);
764 zhistory
= zend_read_property(php_http_client_class_entry
, getThis(), ZEND_STRL("history"), 0, &zhistory_tmp
);
765 RETVAL_ZVAL_FAST(zhistory
);
768 ZEND_BEGIN_ARG_INFO_EX(ai_HttpClient_send
, 0, 0, 0)
770 static PHP_METHOD(HttpClient
, send
)
772 php_http_client_object_t
*obj
;
774 php_http_expect(SUCCESS
== zend_parse_parameters_none(), invalid_arg
, return);
776 obj
= PHP_HTTP_OBJ(NULL
, getThis());
778 php_http_expect(SUCCESS
== php_http_client_exec(obj
->client
), runtime
, return);
780 RETVAL_ZVAL_FAST(getThis());
783 ZEND_BEGIN_ARG_INFO_EX(ai_HttpClient_once
, 0, 0, 0)
785 static PHP_METHOD(HttpClient
, once
)
787 if (SUCCESS
== zend_parse_parameters_none()) {
788 php_http_client_object_t
*obj
= PHP_HTTP_OBJ(NULL
, getThis());
790 RETURN_BOOL(0 < php_http_client_once(obj
->client
));
794 ZEND_BEGIN_ARG_INFO_EX(ai_HttpClient_wait
, 0, 0, 0)
795 ZEND_ARG_INFO(0, timeout
)
797 static PHP_METHOD(HttpClient
, wait
)
801 if (SUCCESS
== zend_parse_parameters(ZEND_NUM_ARGS(), "|d", &timeout
)) {
802 struct timeval timeout_val
;
803 php_http_client_object_t
*obj
= PHP_HTTP_OBJ(NULL
, getThis());
805 timeout_val
.tv_sec
= (time_t) timeout
;
806 timeout_val
.tv_usec
= PHP_HTTP_USEC(timeout
) % PHP_HTTP_MCROSEC
;
808 RETURN_BOOL(SUCCESS
== php_http_client_wait(obj
->client
, timeout
> 0 ? &timeout_val
: NULL
));
812 ZEND_BEGIN_ARG_INFO_EX(ai_HttpClient_enablePipelining
, 0, 0, 0)
813 ZEND_ARG_INFO(0, enable
)
815 static PHP_METHOD(HttpClient
, enablePipelining
)
817 zend_bool enable
= 1;
818 php_http_client_object_t
*obj
;
820 php_http_expect(SUCCESS
== zend_parse_parameters(ZEND_NUM_ARGS(), "|b", &enable
), invalid_arg
, return);
822 obj
= PHP_HTTP_OBJ(NULL
, getThis());
824 php_http_expect(SUCCESS
== php_http_client_setopt(obj
->client
, PHP_HTTP_CLIENT_OPT_ENABLE_PIPELINING
, &enable
), unexpected_val
, return);
826 RETVAL_ZVAL_FAST(getThis());
829 ZEND_BEGIN_ARG_INFO_EX(ai_HttpClient_enableEvents
, 0, 0, 0)
830 ZEND_ARG_INFO(0, enable
)
832 static PHP_METHOD(HttpClient
, enableEvents
)
834 zend_bool enable
= 1;
835 php_http_client_object_t
*obj
;
837 php_http_expect(SUCCESS
== zend_parse_parameters(ZEND_NUM_ARGS(), "|b", &enable
), invalid_arg
, return);
839 obj
= PHP_HTTP_OBJ(NULL
, getThis());
841 php_http_expect(SUCCESS
== php_http_client_setopt(obj
->client
, PHP_HTTP_CLIENT_OPT_USE_EVENTS
, &enable
), unexpected_val
, return);
843 RETVAL_ZVAL_FAST(getThis());
846 static int notify(zend_object_iterator
*iter
, void *puser
)
848 zval
*observer
, *args
= puser
;
850 if ((observer
= iter
->funcs
->get_current_data(iter
))) {
851 int num_args
= !Z_ISUNDEF(args
[0]) + !Z_ISUNDEF(args
[1]) + !Z_ISUNDEF(args
[2]);
852 return php_http_method_call(observer
, ZEND_STRL("update"), num_args
, args
, NULL
);
857 ZEND_BEGIN_ARG_INFO_EX(ai_HttpClient_notify
, 0, 0, 0)
858 ZEND_ARG_OBJ_INFO(0, request
, http
\\Client
\\Request
, 1)
860 static PHP_METHOD(HttpClient
, notify
)
862 zval
*request
= NULL
, *zprogress
= NULL
, observers_tmp
, *observers
, args
[3];
864 php_http_expect(SUCCESS
== zend_parse_parameters(ZEND_NUM_ARGS(), "|O!o!", &request
, php_http_client_request_class_entry
, &zprogress
), invalid_arg
, return);
866 observers
= zend_read_property(php_http_client_class_entry
, getThis(), ZEND_STRL("observers"), 0, &observers_tmp
);
868 if (Z_TYPE_P(observers
) != IS_OBJECT
) {
869 php_http_throw(unexpected_val
, "Observer storage is corrupted", NULL
);
873 ZVAL_COPY(&args
[0], getThis());
875 ZVAL_COPY(&args
[1], request
);
877 ZVAL_UNDEF(&args
[1]);
880 ZVAL_COPY(&args
[2], zprogress
);
882 ZVAL_UNDEF(&args
[2]);
885 spl_iterator_apply(observers
, notify
, args
);
887 zval_ptr_dtor(getThis());
889 zval_ptr_dtor(request
);
892 zval_ptr_dtor(zprogress
);
895 RETVAL_ZVAL_FAST(getThis());
898 ZEND_BEGIN_ARG_INFO_EX(ai_HttpClient_attach
, 0, 0, 1)
899 ZEND_ARG_OBJ_INFO(0, observer
, SplObserver
, 0)
901 static PHP_METHOD(HttpClient
, attach
)
903 zval observers_tmp
, *observers
, *observer
, retval
;
905 php_http_expect(SUCCESS
== zend_parse_parameters(ZEND_NUM_ARGS(), "O", &observer
, spl_ce_SplObserver
), invalid_arg
, return);
907 observers
= zend_read_property(php_http_client_class_entry
, getThis(), ZEND_STRL("observers"), 0, &observers_tmp
);
909 if (Z_TYPE_P(observers
) != IS_OBJECT
) {
910 php_http_throw(unexpected_val
, "Observer storage is corrupted", NULL
);
915 zend_call_method_with_1_params(observers
, NULL
, NULL
, "attach", &retval
, observer
);
916 zval_ptr_dtor(&retval
);
918 RETVAL_ZVAL_FAST(getThis());
921 ZEND_BEGIN_ARG_INFO_EX(ai_HttpClient_detach
, 0, 0, 1)
922 ZEND_ARG_OBJ_INFO(0, observer
, SplObserver
, 0)
924 static PHP_METHOD(HttpClient
, detach
)
926 zval observers_tmp
, *observers
, *observer
, retval
;
928 php_http_expect(SUCCESS
== zend_parse_parameters(ZEND_NUM_ARGS(), "O", &observer
, spl_ce_SplObserver
), invalid_arg
, return);
930 observers
= zend_read_property(php_http_client_class_entry
, getThis(), ZEND_STRL("observers"), 0, &observers_tmp
);
932 if (Z_TYPE_P(observers
) != IS_OBJECT
) {
933 php_http_throw(unexpected_val
, "Observer storage is corrupted", NULL
);
938 zend_call_method_with_1_params(observers
, NULL
, NULL
, "detach", &retval
, observer
);
939 zval_ptr_dtor(&retval
);
941 RETVAL_ZVAL_FAST(getThis());
944 ZEND_BEGIN_ARG_INFO_EX(ai_HttpClient_getObservers
, 0, 0, 0)
946 static PHP_METHOD(HttpClient
, getObservers
)
948 zval observers_tmp
, *observers
;
950 php_http_expect(SUCCESS
== zend_parse_parameters_none(), invalid_arg
, return);
952 observers
= zend_read_property(php_http_client_class_entry
, getThis(), ZEND_STRL("observers"), 0, &observers_tmp
);
954 if (Z_TYPE_P(observers
) != IS_OBJECT
) {
955 php_http_throw(unexpected_val
, "Observer storage is corrupted", NULL
);
959 RETVAL_ZVAL_FAST(observers
);
962 ZEND_BEGIN_ARG_INFO_EX(ai_HttpClient_getProgressInfo
, 0, 0, 1)
963 ZEND_ARG_OBJ_INFO(0, request
, http
\\Client
\\Request
, 0)
965 static PHP_METHOD(HttpClient
, getProgressInfo
)
968 php_http_client_object_t
*obj
;
969 php_http_message_object_t
*req_obj
;
970 php_http_client_progress_state_t
*progress
;
972 php_http_expect(SUCCESS
== zend_parse_parameters(ZEND_NUM_ARGS(), "O", &request
, php_http_client_request_class_entry
), invalid_arg
, return);
974 obj
= PHP_HTTP_OBJ(NULL
, getThis());
975 req_obj
= PHP_HTTP_OBJ(NULL
, request
);
977 php_http_expect(SUCCESS
== php_http_client_getopt(obj
->client
, PHP_HTTP_CLIENT_OPT_PROGRESS_INFO
, req_obj
->message
, &progress
), unexpected_val
, return);
979 object_init(return_value
);
980 add_property_bool(return_value
, "started", progress
->started
);
981 add_property_bool(return_value
, "finished", progress
->finished
);
982 add_property_string(return_value
, "info", STR_PTR(progress
->info
));
983 add_property_double(return_value
, "dltotal", progress
->dl
.total
);
984 add_property_double(return_value
, "dlnow", progress
->dl
.now
);
985 add_property_double(return_value
, "ultotal", progress
->ul
.total
);
986 add_property_double(return_value
, "ulnow", progress
->ul
.now
);
989 ZEND_BEGIN_ARG_INFO_EX(ai_HttpClient_getTransferInfo
, 0, 0, 1)
990 ZEND_ARG_OBJ_INFO(0, request
, http
\\Client
\\Request
, 0)
992 static PHP_METHOD(HttpClient
, getTransferInfo
)
996 php_http_client_object_t
*obj
;
997 php_http_message_object_t
*req_obj
;
999 php_http_expect(SUCCESS
== zend_parse_parameters(ZEND_NUM_ARGS(), "O", &request
, php_http_client_request_class_entry
), invalid_arg
, return);
1001 obj
= PHP_HTTP_OBJ(NULL
, getThis());
1002 req_obj
= PHP_HTTP_OBJ(NULL
, request
);
1004 object_init(return_value
);
1005 info
= HASH_OF(return_value
);
1006 php_http_expect(SUCCESS
== php_http_client_getopt(obj
->client
, PHP_HTTP_CLIENT_OPT_TRANSFER_INFO
, req_obj
->message
, &info
), unexpected_val
, return);
1009 ZEND_BEGIN_ARG_INFO_EX(ai_HttpClient_setOptions
, 0, 0, 0)
1010 ZEND_ARG_ARRAY_INFO(0, options
, 1)
1011 ZEND_END_ARG_INFO();
1012 static PHP_METHOD(HttpClient
, setOptions
)
1016 php_http_expect(SUCCESS
== zend_parse_parameters(ZEND_NUM_ARGS(), "|a!/", &opts
), invalid_arg
, return);
1018 php_http_client_options_set(getThis(), opts
);
1020 RETVAL_ZVAL_FAST(getThis());
1023 ZEND_BEGIN_ARG_INFO_EX(ai_HttpClient_getOptions
, 0, 0, 0)
1024 ZEND_END_ARG_INFO();
1025 static PHP_METHOD(HttpClient
, getOptions
)
1027 if (SUCCESS
== zend_parse_parameters_none()) {
1028 zval options_tmp
, *options
= zend_read_property(php_http_client_class_entry
, getThis(), ZEND_STRL("options"), 0, &options_tmp
);
1029 RETVAL_ZVAL_FAST(options
);
1033 ZEND_BEGIN_ARG_INFO_EX(ai_HttpClient_setSslOptions
, 0, 0, 0)
1034 ZEND_ARG_ARRAY_INFO(0, ssl_option
, 1)
1035 ZEND_END_ARG_INFO();
1036 static PHP_METHOD(HttpClient
, setSslOptions
)
1040 php_http_expect(SUCCESS
== zend_parse_parameters(ZEND_NUM_ARGS(), "|a!/", &opts
), invalid_arg
, return);
1042 php_http_client_options_set_subr(getThis(), ZEND_STRL("ssl"), opts
, 1);
1044 RETVAL_ZVAL_FAST(getThis());
1047 ZEND_BEGIN_ARG_INFO_EX(ai_HttpClient_addSslOptions
, 0, 0, 0)
1048 ZEND_ARG_ARRAY_INFO(0, ssl_options
, 1)
1049 ZEND_END_ARG_INFO();
1050 static PHP_METHOD(HttpClient
, addSslOptions
)
1054 php_http_expect(SUCCESS
== zend_parse_parameters(ZEND_NUM_ARGS(), "|a!/", &opts
), invalid_arg
, return);
1056 php_http_client_options_set_subr(getThis(), ZEND_STRL("ssl"), opts
, 0);
1058 RETVAL_ZVAL_FAST(getThis());
1061 ZEND_BEGIN_ARG_INFO_EX(ai_HttpClient_getSslOptions
, 0, 0, 0)
1062 ZEND_END_ARG_INFO();
1063 static PHP_METHOD(HttpClient
, getSslOptions
)
1065 if (SUCCESS
== zend_parse_parameters_none()) {
1066 php_http_client_options_get_subr(getThis(), ZEND_STRL("ssl"), return_value
);
1070 ZEND_BEGIN_ARG_INFO_EX(ai_HttpClient_setCookies
, 0, 0, 0)
1071 ZEND_ARG_ARRAY_INFO(0, cookies
, 1)
1072 ZEND_END_ARG_INFO();
1073 static PHP_METHOD(HttpClient
, setCookies
)
1077 php_http_expect(SUCCESS
== zend_parse_parameters(ZEND_NUM_ARGS(), "|a!/", &opts
), invalid_arg
, return);
1079 php_http_client_options_set_subr(getThis(), ZEND_STRL("cookies"), opts
, 1);
1081 RETVAL_ZVAL_FAST(getThis());
1084 ZEND_BEGIN_ARG_INFO_EX(ai_HttpClient_addCookies
, 0, 0, 0)
1085 ZEND_ARG_ARRAY_INFO(0, cookies
, 1)
1086 ZEND_END_ARG_INFO();
1087 static PHP_METHOD(HttpClient
, addCookies
)
1091 php_http_expect(SUCCESS
== zend_parse_parameters(ZEND_NUM_ARGS(), "|a!/", &opts
), invalid_arg
, return);
1093 php_http_client_options_set_subr(getThis(), ZEND_STRL("cookies"), opts
, 0);
1095 RETVAL_ZVAL_FAST(getThis());
1098 ZEND_BEGIN_ARG_INFO_EX(ai_HttpClient_getCookies
, 0, 0, 0)
1099 ZEND_END_ARG_INFO();
1100 static PHP_METHOD(HttpClient
, getCookies
)
1102 if (SUCCESS
== zend_parse_parameters_none()) {
1103 php_http_client_options_get_subr(getThis(), ZEND_STRL("cookies"), return_value
);
1107 ZEND_BEGIN_ARG_INFO_EX(ai_HttpClient_getAvailableDrivers
, 0, 0, 0)
1108 ZEND_END_ARG_INFO();
1109 static PHP_METHOD(HttpClient
, getAvailableDrivers
)
1111 if (SUCCESS
== zend_parse_parameters_none()) {
1112 array_init(return_value
);
1113 php_http_client_driver_list(Z_ARRVAL_P(return_value
));
1117 static zend_function_entry php_http_client_methods
[] = {
1118 PHP_ME(HttpClient
, __construct
, ai_HttpClient_construct
, ZEND_ACC_PUBLIC
|ZEND_ACC_CTOR
)
1119 PHP_ME(HttpClient
, reset
, ai_HttpClient_reset
, ZEND_ACC_PUBLIC
)
1120 PHP_ME(HttpClient
, enqueue
, ai_HttpClient_enqueue
, ZEND_ACC_PUBLIC
)
1121 PHP_ME(HttpClient
, dequeue
, ai_HttpClient_dequeue
, ZEND_ACC_PUBLIC
)
1122 PHP_ME(HttpClient
, requeue
, ai_HttpClient_requeue
, ZEND_ACC_PUBLIC
)
1123 PHP_ME(HttpClient
, count
, ai_HttpClient_count
, ZEND_ACC_PUBLIC
)
1124 PHP_ME(HttpClient
, send
, ai_HttpClient_send
, ZEND_ACC_PUBLIC
)
1125 PHP_ME(HttpClient
, once
, ai_HttpClient_once
, ZEND_ACC_PUBLIC
)
1126 PHP_ME(HttpClient
, wait
, ai_HttpClient_wait
, ZEND_ACC_PUBLIC
)
1127 PHP_ME(HttpClient
, getResponse
, ai_HttpClient_getResponse
, ZEND_ACC_PUBLIC
)
1128 PHP_ME(HttpClient
, getHistory
, ai_HttpClient_getHistory
, ZEND_ACC_PUBLIC
)
1129 PHP_ME(HttpClient
, enablePipelining
, ai_HttpClient_enablePipelining
, ZEND_ACC_PUBLIC
)
1130 PHP_ME(HttpClient
, enableEvents
, ai_HttpClient_enableEvents
, ZEND_ACC_PUBLIC
)
1131 PHP_ME(HttpClient
, notify
, ai_HttpClient_notify
, ZEND_ACC_PUBLIC
)
1132 PHP_ME(HttpClient
, attach
, ai_HttpClient_attach
, ZEND_ACC_PUBLIC
)
1133 PHP_ME(HttpClient
, detach
, ai_HttpClient_detach
, ZEND_ACC_PUBLIC
)
1134 PHP_ME(HttpClient
, getObservers
, ai_HttpClient_getObservers
, ZEND_ACC_PUBLIC
)
1135 PHP_ME(HttpClient
, getProgressInfo
, ai_HttpClient_getProgressInfo
, ZEND_ACC_PUBLIC
)
1136 PHP_ME(HttpClient
, getTransferInfo
, ai_HttpClient_getTransferInfo
, ZEND_ACC_PUBLIC
)
1137 PHP_ME(HttpClient
, setOptions
, ai_HttpClient_setOptions
, ZEND_ACC_PUBLIC
)
1138 PHP_ME(HttpClient
, getOptions
, ai_HttpClient_getOptions
, ZEND_ACC_PUBLIC
)
1139 PHP_ME(HttpClient
, setSslOptions
, ai_HttpClient_setSslOptions
, ZEND_ACC_PUBLIC
)
1140 PHP_ME(HttpClient
, addSslOptions
, ai_HttpClient_addSslOptions
, ZEND_ACC_PUBLIC
)
1141 PHP_ME(HttpClient
, getSslOptions
, ai_HttpClient_getSslOptions
, ZEND_ACC_PUBLIC
)
1142 PHP_ME(HttpClient
, setCookies
, ai_HttpClient_setCookies
, ZEND_ACC_PUBLIC
)
1143 PHP_ME(HttpClient
, addCookies
, ai_HttpClient_addCookies
, ZEND_ACC_PUBLIC
)
1144 PHP_ME(HttpClient
, getCookies
, ai_HttpClient_getCookies
, ZEND_ACC_PUBLIC
)
1145 PHP_ME(HttpClient
, getAvailableDrivers
, ai_HttpClient_getAvailableDrivers
, ZEND_ACC_PUBLIC
|ZEND_ACC_STATIC
)
1146 EMPTY_FUNCTION_ENTRY
1149 PHP_MINIT_FUNCTION(http_client
)
1151 zend_class_entry ce
= {0};
1153 INIT_NS_CLASS_ENTRY(ce
, "http", "Client", php_http_client_methods
);
1154 php_http_client_class_entry
= zend_register_internal_class_ex(&ce
, NULL
);
1155 php_http_client_class_entry
->create_object
= php_http_client_object_new
;
1156 zend_class_implements(php_http_client_class_entry
, 2, spl_ce_SplSubject
, spl_ce_Countable
);
1157 memcpy(&php_http_client_object_handlers
, zend_get_std_object_handlers(), sizeof(zend_object_handlers
));
1158 php_http_client_object_handlers
.offset
= XtOffsetOf(php_http_client_object_t
, zo
);
1159 php_http_client_object_handlers
.free_obj
= php_http_client_object_free
;
1160 php_http_client_object_handlers
.clone_obj
= NULL
;
1161 zend_declare_property_null(php_http_client_class_entry
, ZEND_STRL("observers"), ZEND_ACC_PRIVATE
);
1162 zend_declare_property_null(php_http_client_class_entry
, ZEND_STRL("options"), ZEND_ACC_PROTECTED
);
1163 zend_declare_property_null(php_http_client_class_entry
, ZEND_STRL("history"), ZEND_ACC_PROTECTED
);
1164 zend_declare_property_bool(php_http_client_class_entry
, ZEND_STRL("recordHistory"), 0, ZEND_ACC_PUBLIC
);
1166 zend_hash_init(&php_http_client_drivers
, 2, NULL
, php_http_client_driver_hash_dtor
, 1);
1171 PHP_MSHUTDOWN_FUNCTION(http_client
)
1173 zend_hash_destroy(&php_http_client_drivers
);
1182 * vim600: noet sw=4 ts=4 fdm=marker
1183 * vim<600: noet sw=4 ts=4