mirror of
https://github.com/php/php-src.git
synced 2025-08-15 21:48:51 +02:00
339 lines
9.3 KiB
C
339 lines
9.3 KiB
C
/*
|
|
+----------------------------------------------------------------------+
|
|
| Copyright (c) The PHP Group |
|
|
+----------------------------------------------------------------------+
|
|
| This source file is subject to version 3.01 of the PHP license, |
|
|
| that is bundled with this package in the file LICENSE, and is |
|
|
| available through the world-wide-web at the following url: |
|
|
| https://www.php.net/license/3_01.txt |
|
|
| If you did not receive a copy of the PHP license and are unable to |
|
|
| obtain it through the world-wide-web, please send a note to |
|
|
| license@php.net so we can mail you a copy immediately. |
|
|
+----------------------------------------------------------------------+
|
|
| Author: Pierrick Charron <pierrick@php.net> |
|
|
+----------------------------------------------------------------------+
|
|
*/
|
|
|
|
#define ZEND_INCLUDE_FULL_WINDOWS_HEADERS
|
|
|
|
#ifdef HAVE_CONFIG_H
|
|
#include <config.h>
|
|
#endif
|
|
|
|
#include "php.h"
|
|
#include "Zend/zend_exceptions.h"
|
|
|
|
#include "curl_private.h"
|
|
|
|
#include <curl/curl.h>
|
|
|
|
#define SAVE_CURLSH_ERROR(__handle, __err) (__handle)->err.no = (int) __err;
|
|
|
|
/* {{{ Initialize a share curl handle */
|
|
PHP_FUNCTION(curl_share_init)
|
|
{
|
|
php_curlsh *sh;
|
|
|
|
ZEND_PARSE_PARAMETERS_NONE();
|
|
|
|
object_init_ex(return_value, curl_share_ce);
|
|
sh = Z_CURL_SHARE_P(return_value);
|
|
|
|
sh->share = curl_share_init();
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ Close a set of cURL handles */
|
|
PHP_FUNCTION(curl_share_close)
|
|
{
|
|
zval *z_sh;
|
|
|
|
ZEND_PARSE_PARAMETERS_START(1,1)
|
|
Z_PARAM_OBJECT_OF_CLASS(z_sh, curl_share_ce)
|
|
ZEND_PARSE_PARAMETERS_END();
|
|
}
|
|
/* }}} */
|
|
|
|
static bool _php_curl_share_setopt(php_curlsh *sh, zend_long option, zval *zvalue, zval *return_value) /* {{{ */
|
|
{
|
|
CURLSHcode error = CURLSHE_OK;
|
|
|
|
switch (option) {
|
|
case CURLSHOPT_SHARE:
|
|
case CURLSHOPT_UNSHARE:
|
|
error = curl_share_setopt(sh->share, option, zval_get_long(zvalue));
|
|
break;
|
|
|
|
default:
|
|
zend_argument_value_error(2, "is not a valid cURL share option");
|
|
error = CURLSHE_BAD_OPTION;
|
|
break;
|
|
}
|
|
|
|
SAVE_CURLSH_ERROR(sh, error);
|
|
|
|
return error == CURLSHE_OK;
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ Set an option for a cURL transfer */
|
|
PHP_FUNCTION(curl_share_setopt)
|
|
{
|
|
zval *z_sh, *zvalue;
|
|
zend_long options;
|
|
php_curlsh *sh;
|
|
|
|
ZEND_PARSE_PARAMETERS_START(3,3)
|
|
Z_PARAM_OBJECT_OF_CLASS(z_sh, curl_share_ce)
|
|
Z_PARAM_LONG(options)
|
|
Z_PARAM_ZVAL(zvalue)
|
|
ZEND_PARSE_PARAMETERS_END();
|
|
|
|
sh = Z_CURL_SHARE_P(z_sh);
|
|
|
|
if (_php_curl_share_setopt(sh, options, zvalue, return_value)) {
|
|
RETURN_TRUE;
|
|
} else {
|
|
RETURN_FALSE;
|
|
}
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ Return an integer containing the last share curl error number */
|
|
PHP_FUNCTION(curl_share_errno)
|
|
{
|
|
zval *z_sh;
|
|
php_curlsh *sh;
|
|
|
|
ZEND_PARSE_PARAMETERS_START(1,1)
|
|
Z_PARAM_OBJECT_OF_CLASS(z_sh, curl_share_ce)
|
|
ZEND_PARSE_PARAMETERS_END();
|
|
|
|
sh = Z_CURL_SHARE_P(z_sh);
|
|
|
|
RETURN_LONG(sh->err.no);
|
|
}
|
|
/* }}} */
|
|
|
|
|
|
/* {{{ return string describing error code */
|
|
PHP_FUNCTION(curl_share_strerror)
|
|
{
|
|
zend_long code;
|
|
const char *str;
|
|
|
|
ZEND_PARSE_PARAMETERS_START(1,1)
|
|
Z_PARAM_LONG(code)
|
|
ZEND_PARSE_PARAMETERS_END();
|
|
|
|
str = curl_share_strerror(code);
|
|
if (str) {
|
|
RETURN_STRING(str);
|
|
} else {
|
|
RETURN_NULL();
|
|
}
|
|
}
|
|
/* }}} */
|
|
|
|
/**
|
|
* Initialize a persistent curl share handle with the given share options, reusing an existing one if found.
|
|
*
|
|
* Throws an exception if the share options are invalid.
|
|
*/
|
|
PHP_FUNCTION(curl_share_init_persistent)
|
|
{
|
|
// Options specified by the user.
|
|
HashTable *share_opts = NULL;
|
|
|
|
// De-duplicated and sorted options.
|
|
zval share_opts_prop;
|
|
ZVAL_UNDEF(&share_opts_prop);
|
|
|
|
// An ID representing the unique set of options specified by the user.
|
|
zend_ulong persistent_id = 0;
|
|
|
|
php_curlsh *sh = NULL;
|
|
|
|
ZEND_PARSE_PARAMETERS_START(1, 1)
|
|
Z_PARAM_ARRAY_HT(share_opts)
|
|
ZEND_PARSE_PARAMETERS_END();
|
|
|
|
if (zend_hash_num_elements(share_opts) == 0) {
|
|
zend_argument_must_not_be_empty_error(1);
|
|
goto error;
|
|
}
|
|
|
|
ZEND_HASH_FOREACH_VAL(share_opts, zval *entry) {
|
|
ZVAL_DEREF(entry);
|
|
|
|
bool failed = false;
|
|
zend_ulong option = zval_try_get_long(entry, &failed);
|
|
|
|
if (failed) {
|
|
zend_argument_type_error(1, "must contain only int values, %s given", zend_zval_value_name(entry));
|
|
goto error;
|
|
}
|
|
|
|
switch (option) {
|
|
// Disallowed options
|
|
case CURL_LOCK_DATA_COOKIE:
|
|
zend_argument_value_error(1, "must not contain CURL_LOCK_DATA_COOKIE because sharing cookies across PHP requests is unsafe");
|
|
goto error;
|
|
|
|
// Allowed options
|
|
case CURL_LOCK_DATA_DNS:
|
|
persistent_id |= 1 << 0;
|
|
break;
|
|
case CURL_LOCK_DATA_SSL_SESSION:
|
|
persistent_id |= 1 << 1;
|
|
break;
|
|
case CURL_LOCK_DATA_CONNECT:
|
|
persistent_id |= 1 << 2;
|
|
break;
|
|
case CURL_LOCK_DATA_PSL:
|
|
persistent_id |= 1 << 3;
|
|
break;
|
|
|
|
// Unknown options
|
|
default:
|
|
zend_argument_value_error(1, "must contain only CURL_LOCK_DATA_* constants");
|
|
goto error;
|
|
}
|
|
} ZEND_HASH_FOREACH_END();
|
|
|
|
// We're now decently confident that we'll be returning a CurlSharePersistentHandle object, so we construct it here.
|
|
object_init_ex(return_value, curl_share_persistent_ce);
|
|
|
|
// Next we initialize a property field for the CurlSharePersistentHandle object with the enabled share options.
|
|
array_init(&share_opts_prop);
|
|
|
|
if (persistent_id & (1 << 0)) {
|
|
add_next_index_long(&share_opts_prop, CURL_LOCK_DATA_DNS);
|
|
}
|
|
|
|
if (persistent_id & (1 << 1)) {
|
|
add_next_index_long(&share_opts_prop, CURL_LOCK_DATA_SSL_SESSION);
|
|
}
|
|
|
|
if (persistent_id & (1 << 2)) {
|
|
add_next_index_long(&share_opts_prop, CURL_LOCK_DATA_CONNECT);
|
|
}
|
|
|
|
if (persistent_id & (1 << 3)) {
|
|
add_next_index_long(&share_opts_prop, CURL_LOCK_DATA_PSL);
|
|
}
|
|
|
|
zend_update_property(curl_share_persistent_ce, Z_OBJ_P(return_value), ZEND_STRL("options"), &share_opts_prop);
|
|
|
|
sh = Z_CURL_SHARE_P(return_value);
|
|
|
|
// If we are able to find an existing persistent share handle, we can use it and return early.
|
|
zval *persisted = zend_hash_index_find(&CURL_G(persistent_curlsh), persistent_id);
|
|
|
|
if (persisted) {
|
|
sh->share = Z_PTR_P(persisted);
|
|
|
|
goto cleanup;
|
|
}
|
|
|
|
// We could not find an existing share handle, so we'll have to create one.
|
|
sh->share = curl_share_init();
|
|
|
|
// Apply the options property to the handle. We avoid using $share_opts as zval_get_long may not necessarily return
|
|
// the same value as it did in the initial ZEND_HASH_FOREACH_VAL above.
|
|
ZEND_HASH_PACKED_FOREACH_VAL(Z_ARRVAL_P(&share_opts_prop), zval *entry) {
|
|
CURLSHcode curlsh_error = curl_share_setopt(sh->share, CURLSHOPT_SHARE, Z_LVAL_P(entry));
|
|
|
|
if (curlsh_error != CURLSHE_OK) {
|
|
zend_throw_exception_ex(NULL, 0, "Could not construct persistent cURL share handle: %s", curl_share_strerror(curlsh_error));
|
|
|
|
goto error;
|
|
}
|
|
} ZEND_HASH_FOREACH_END();
|
|
|
|
zend_hash_index_add_new_ptr(&CURL_G(persistent_curlsh), persistent_id, sh->share);
|
|
|
|
cleanup:
|
|
zval_ptr_dtor(&share_opts_prop);
|
|
|
|
return;
|
|
|
|
error:
|
|
if (sh) {
|
|
curl_share_cleanup(sh->share);
|
|
}
|
|
|
|
zval_ptr_dtor(&share_opts_prop);
|
|
|
|
RETURN_THROWS();
|
|
}
|
|
|
|
/**
|
|
* Free a persistent curl share handle from the module global HashTable.
|
|
*
|
|
* See PHP_GINIT_FUNCTION in ext/curl/interface.c.
|
|
*/
|
|
void curl_share_free_persistent_curlsh(zval *data)
|
|
{
|
|
CURLSH *handle = Z_PTR_P(data);
|
|
|
|
curl_share_cleanup(handle);
|
|
}
|
|
|
|
/* CurlShareHandle class */
|
|
|
|
static zend_object *curl_share_create_object(zend_class_entry *class_type) {
|
|
php_curlsh *intern = zend_object_alloc(sizeof(php_curlsh), class_type);
|
|
|
|
zend_object_std_init(&intern->std, class_type);
|
|
object_properties_init(&intern->std, class_type);
|
|
|
|
return &intern->std;
|
|
}
|
|
|
|
static zend_function *curl_share_get_constructor(zend_object *object) {
|
|
zend_throw_error(NULL, "Cannot directly construct CurlShareHandle, use curl_share_init() instead");
|
|
return NULL;
|
|
}
|
|
|
|
void curl_share_free_obj(zend_object *object)
|
|
{
|
|
php_curlsh *sh = curl_share_from_obj(object);
|
|
|
|
curl_share_cleanup(sh->share);
|
|
zend_object_std_dtor(&sh->std);
|
|
}
|
|
|
|
static zend_object_handlers curl_share_handlers;
|
|
|
|
void curl_share_register_handlers(void) {
|
|
curl_share_ce->create_object = curl_share_create_object;
|
|
curl_share_ce->default_object_handlers = &curl_share_handlers;
|
|
|
|
memcpy(&curl_share_handlers, &std_object_handlers, sizeof(zend_object_handlers));
|
|
curl_share_handlers.offset = XtOffsetOf(php_curlsh, std);
|
|
curl_share_handlers.free_obj = curl_share_free_obj;
|
|
curl_share_handlers.get_constructor = curl_share_get_constructor;
|
|
curl_share_handlers.clone_obj = NULL;
|
|
curl_share_handlers.compare = zend_objects_not_comparable;
|
|
}
|
|
|
|
/* CurlSharePersistentHandle class */
|
|
|
|
static zend_object_handlers curl_share_persistent_handlers;
|
|
|
|
static zend_function *curl_share_persistent_get_constructor(zend_object *object) {
|
|
zend_throw_error(NULL, "Cannot directly construct CurlSharePersistentHandle, use curl_share_init_persistent() instead");
|
|
return NULL;
|
|
}
|
|
|
|
void curl_share_persistent_register_handlers(void) {
|
|
curl_share_persistent_ce->create_object = curl_share_create_object;
|
|
curl_share_persistent_ce->default_object_handlers = &curl_share_persistent_handlers;
|
|
|
|
memcpy(&curl_share_persistent_handlers, &std_object_handlers, sizeof(zend_object_handlers));
|
|
curl_share_persistent_handlers.offset = XtOffsetOf(php_curlsh, std);
|
|
curl_share_persistent_handlers.get_constructor = curl_share_persistent_get_constructor;
|
|
curl_share_persistent_handlers.clone_obj = NULL;
|
|
curl_share_persistent_handlers.compare = zend_objects_not_comparable;
|
|
}
|