mirror of
https://github.com/php/php-src.git
synced 2025-08-16 05:58:45 +02:00
295 lines
6.7 KiB
C
295 lines
6.7 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: Sascha Schumann <sascha@schumann.cx> |
|
|
+----------------------------------------------------------------------+
|
|
*/
|
|
|
|
#include "php.h"
|
|
#include "php_session.h"
|
|
#include "mod_user.h"
|
|
|
|
const ps_module ps_mod_user = {
|
|
PS_MOD_UPDATE_TIMESTAMP(user)
|
|
};
|
|
|
|
|
|
static void ps_call_handler(zval *func, int argc, zval *argv, zval *retval)
|
|
{
|
|
int i;
|
|
if (PS(in_save_handler)) {
|
|
PS(in_save_handler) = 0;
|
|
ZVAL_UNDEF(retval);
|
|
php_error_docref(NULL, E_WARNING, "Cannot call session save handler in a recursive manner");
|
|
return;
|
|
}
|
|
PS(in_save_handler) = 1;
|
|
if (call_user_function(NULL, NULL, func, retval, argc, argv) == FAILURE) {
|
|
zval_ptr_dtor(retval);
|
|
ZVAL_UNDEF(retval);
|
|
} else if (Z_ISUNDEF_P(retval)) {
|
|
ZVAL_NULL(retval);
|
|
}
|
|
PS(in_save_handler) = 0;
|
|
for (i = 0; i < argc; i++) {
|
|
zval_ptr_dtor(&argv[i]);
|
|
}
|
|
}
|
|
|
|
#define PSF(a) PS(mod_user_names).ps_##a
|
|
|
|
static zend_result verify_bool_return_type_userland_calls(const zval* value)
|
|
{
|
|
/* Exit or exception in userland call */
|
|
if (Z_TYPE_P(value) == IS_UNDEF) {
|
|
return FAILURE;
|
|
}
|
|
if (Z_TYPE_P(value) == IS_TRUE) {
|
|
return SUCCESS;
|
|
}
|
|
if (Z_TYPE_P(value) == IS_FALSE) {
|
|
return FAILURE;
|
|
}
|
|
if ((Z_TYPE_P(value) == IS_LONG) && (Z_LVAL_P(value) == -1)) {
|
|
/* TODO Why are exception cheked? */
|
|
if (!EG(exception)) {
|
|
php_error_docref(NULL, E_DEPRECATED, "Session callback must have a return value of type bool, %s returned", zend_zval_type_name(value));
|
|
}
|
|
return FAILURE;
|
|
}
|
|
if ((Z_TYPE_P(value) == IS_LONG) && (Z_LVAL_P(value) == 0)) {
|
|
/* TODO Why are exception cheked? */
|
|
if (!EG(exception)) {
|
|
php_error_docref(NULL, E_DEPRECATED, "Session callback must have a return value of type bool, %s returned", zend_zval_type_name(value));
|
|
}
|
|
return SUCCESS;
|
|
}
|
|
if (!EG(exception)) {
|
|
zend_type_error("Session callback must have a return value of type bool, %s returned", zend_zval_type_name(value)); \
|
|
}
|
|
return FAILURE;
|
|
}
|
|
|
|
PS_OPEN_FUNC(user)
|
|
{
|
|
zval args[2];
|
|
zval retval;
|
|
zend_result ret = FAILURE;
|
|
|
|
ZEND_ASSERT(!Z_ISUNDEF(PSF(open)));
|
|
|
|
ZVAL_STRING(&args[0], (char*)save_path);
|
|
ZVAL_STRING(&args[1], (char*)session_name);
|
|
|
|
zend_try {
|
|
ps_call_handler(&PSF(open), 2, args, &retval);
|
|
} zend_catch {
|
|
PS(session_status) = php_session_none;
|
|
if (!Z_ISUNDEF(retval)) {
|
|
zval_ptr_dtor(&retval);
|
|
}
|
|
zend_bailout();
|
|
} zend_end_try();
|
|
|
|
PS(mod_user_implemented) = 1;
|
|
|
|
ret = verify_bool_return_type_userland_calls(&retval);
|
|
zval_ptr_dtor(&retval);
|
|
return ret;
|
|
}
|
|
|
|
PS_CLOSE_FUNC(user)
|
|
{
|
|
bool bailout = 0;
|
|
zval retval;
|
|
zend_result ret = FAILURE;
|
|
|
|
ZEND_ASSERT(!Z_ISUNDEF(PSF(close)));
|
|
|
|
if (!PS(mod_user_implemented)) {
|
|
/* already closed */
|
|
return SUCCESS;
|
|
}
|
|
|
|
zend_try {
|
|
ps_call_handler(&PSF(close), 0, NULL, &retval);
|
|
} zend_catch {
|
|
bailout = 1;
|
|
} zend_end_try();
|
|
|
|
PS(mod_user_implemented) = 0;
|
|
|
|
if (bailout) {
|
|
if (!Z_ISUNDEF(retval)) {
|
|
zval_ptr_dtor(&retval);
|
|
}
|
|
zend_bailout();
|
|
}
|
|
|
|
ret = verify_bool_return_type_userland_calls(&retval);
|
|
zval_ptr_dtor(&retval);
|
|
return ret;
|
|
}
|
|
|
|
PS_READ_FUNC(user)
|
|
{
|
|
zval args[1];
|
|
zval retval;
|
|
zend_result ret = FAILURE;
|
|
|
|
ZEND_ASSERT(!Z_ISUNDEF(PSF(read)));
|
|
|
|
ZVAL_STR_COPY(&args[0], key);
|
|
|
|
ps_call_handler(&PSF(read), 1, args, &retval);
|
|
|
|
if (!Z_ISUNDEF(retval)) {
|
|
if (Z_TYPE(retval) == IS_STRING) {
|
|
*val = zend_string_copy(Z_STR(retval));
|
|
ret = SUCCESS;
|
|
}
|
|
zval_ptr_dtor(&retval);
|
|
}
|
|
|
|
return ret;
|
|
}
|
|
|
|
PS_WRITE_FUNC(user)
|
|
{
|
|
zval args[2];
|
|
zval retval;
|
|
zend_result ret = FAILURE;
|
|
|
|
ZEND_ASSERT(!Z_ISUNDEF(PSF(write)));
|
|
|
|
ZVAL_STR_COPY(&args[0], key);
|
|
ZVAL_STR_COPY(&args[1], val);
|
|
|
|
ps_call_handler(&PSF(write), 2, args, &retval);
|
|
|
|
ret = verify_bool_return_type_userland_calls(&retval);
|
|
zval_ptr_dtor(&retval);
|
|
return ret;
|
|
}
|
|
|
|
PS_DESTROY_FUNC(user)
|
|
{
|
|
zval args[1];
|
|
zval retval;
|
|
zend_result ret = FAILURE;
|
|
|
|
ZEND_ASSERT(!Z_ISUNDEF(PSF(destroy)));
|
|
|
|
ZVAL_STR_COPY(&args[0], key);
|
|
|
|
ps_call_handler(&PSF(destroy), 1, args, &retval);
|
|
|
|
ret = verify_bool_return_type_userland_calls(&retval);
|
|
zval_ptr_dtor(&retval);
|
|
return ret;
|
|
}
|
|
|
|
PS_GC_FUNC(user)
|
|
{
|
|
zval args[1];
|
|
zval retval;
|
|
|
|
ZEND_ASSERT(!Z_ISUNDEF(PSF(gc)));
|
|
|
|
ZVAL_LONG(&args[0], maxlifetime);
|
|
|
|
ps_call_handler(&PSF(gc), 1, args, &retval);
|
|
|
|
if (Z_TYPE(retval) == IS_LONG) {
|
|
*nrdels = Z_LVAL(retval);
|
|
} else if (Z_TYPE(retval) == IS_TRUE) {
|
|
/* This is for older API compatibility */
|
|
*nrdels = 1;
|
|
} else {
|
|
/* Anything else is some kind of error */
|
|
*nrdels = -1; // Error
|
|
}
|
|
return *nrdels;
|
|
}
|
|
|
|
PS_CREATE_SID_FUNC(user)
|
|
{
|
|
/* maintain backwards compatibility */
|
|
if (!Z_ISUNDEF(PSF(create_sid))) {
|
|
zend_string *id = NULL;
|
|
zval retval;
|
|
|
|
ps_call_handler(&PSF(create_sid), 0, NULL, &retval);
|
|
|
|
if (!Z_ISUNDEF(retval)) {
|
|
if (Z_TYPE(retval) == IS_STRING) {
|
|
id = zend_string_copy(Z_STR(retval));
|
|
}
|
|
zval_ptr_dtor(&retval);
|
|
} else {
|
|
zend_throw_error(NULL, "No session id returned by function");
|
|
return NULL;
|
|
}
|
|
|
|
if (!id) {
|
|
zend_throw_error(NULL, "Session id must be a string");
|
|
return NULL;
|
|
}
|
|
|
|
return id;
|
|
}
|
|
|
|
/* function as defined by PS_MOD */
|
|
return php_session_create_id(mod_data);
|
|
}
|
|
|
|
PS_VALIDATE_SID_FUNC(user)
|
|
{
|
|
/* maintain backwards compatibility */
|
|
if (!Z_ISUNDEF(PSF(validate_sid))) {
|
|
zval args[1];
|
|
zval retval;
|
|
zend_result ret = FAILURE;
|
|
|
|
ZVAL_STR_COPY(&args[0], key);
|
|
|
|
ps_call_handler(&PSF(validate_sid), 1, args, &retval);
|
|
|
|
ret = verify_bool_return_type_userland_calls(&retval);
|
|
zval_ptr_dtor(&retval);
|
|
return ret;
|
|
}
|
|
|
|
/* dummy function defined by PS_MOD */
|
|
return php_session_validate_sid(mod_data, key);
|
|
}
|
|
|
|
PS_UPDATE_TIMESTAMP_FUNC(user)
|
|
{
|
|
zval args[2];
|
|
zval retval;
|
|
zend_result ret = FAILURE;
|
|
|
|
ZVAL_STR_COPY(&args[0], key);
|
|
ZVAL_STR_COPY(&args[1], val);
|
|
|
|
/* maintain backwards compatibility */
|
|
if (!Z_ISUNDEF(PSF(update_timestamp))) {
|
|
ps_call_handler(&PSF(update_timestamp), 2, args, &retval);
|
|
} else {
|
|
ps_call_handler(&PSF(write), 2, args, &retval);
|
|
}
|
|
|
|
ret = verify_bool_return_type_userland_calls(&retval);
|
|
zval_ptr_dtor(&retval);
|
|
return ret;
|
|
}
|