mirror of
https://github.com/php/frankenphp.git
synced 2026-03-24 00:52:11 +01:00
* refactor: better memory management * wip * tmp * introduce a go-like api * upgraded to PHP 8.2 * Fix thread safety issues * fix tests * wip * refactor worker * worker prototype * fix populate env * session * improve tests * fix Caddy tests * refactor
529 lines
13 KiB
C
529 lines
13 KiB
C
#include <errno.h>
|
|
#include <stdint.h>
|
|
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
#include "_cgo_export.h"
|
|
#include "php.h"
|
|
#include "SAPI.h"
|
|
#include "ext/standard/head.h"
|
|
#include "ext/session/php_session.h"
|
|
#include "php_main.h"
|
|
#include "php_variables.h"
|
|
#include "php_output.h"
|
|
#include "Zend/zend_alloc.h"
|
|
|
|
|
|
// Helper functions copied from the PHP source code
|
|
|
|
#include "php.h"
|
|
#include "SAPI.h"
|
|
|
|
// main/php_variables.c
|
|
|
|
static zend_always_inline void php_register_variable_quick(const char *name, size_t name_len, zval *val, HashTable *ht)
|
|
{
|
|
zend_string *key = zend_string_init_interned(name, name_len, 0);
|
|
|
|
zend_hash_update_ind(ht, key, val);
|
|
zend_string_release_ex(key, 0);
|
|
}
|
|
|
|
static inline void php_register_server_variables(void)
|
|
{
|
|
zval tmp;
|
|
zval *arr = &PG(http_globals)[TRACK_VARS_SERVER];
|
|
HashTable *ht;
|
|
|
|
zval_ptr_dtor_nogc(arr);
|
|
array_init(arr);
|
|
|
|
/* Server variables */
|
|
if (sapi_module.register_server_variables) {
|
|
sapi_module.register_server_variables(arr);
|
|
}
|
|
ht = Z_ARRVAL_P(arr);
|
|
|
|
/* PHP Authentication support */
|
|
if (SG(request_info).auth_user) {
|
|
ZVAL_STRING(&tmp, SG(request_info).auth_user);
|
|
php_register_variable_quick("PHP_AUTH_USER", sizeof("PHP_AUTH_USER")-1, &tmp, ht);
|
|
}
|
|
if (SG(request_info).auth_password) {
|
|
ZVAL_STRING(&tmp, SG(request_info).auth_password);
|
|
php_register_variable_quick("PHP_AUTH_PW", sizeof("PHP_AUTH_PW")-1, &tmp, ht);
|
|
}
|
|
if (SG(request_info).auth_digest) {
|
|
ZVAL_STRING(&tmp, SG(request_info).auth_digest);
|
|
php_register_variable_quick("PHP_AUTH_DIGEST", sizeof("PHP_AUTH_DIGEST")-1, &tmp, ht);
|
|
}
|
|
|
|
/* store request init time */
|
|
ZVAL_DOUBLE(&tmp, sapi_get_request_time());
|
|
php_register_variable_quick("REQUEST_TIME_FLOAT", sizeof("REQUEST_TIME_FLOAT")-1, &tmp, ht);
|
|
ZVAL_LONG(&tmp, zend_dval_to_lval(Z_DVAL(tmp)));
|
|
php_register_variable_quick("REQUEST_TIME", sizeof("REQUEST_TIME")-1, &tmp, ht);
|
|
}
|
|
|
|
|
|
// ext/session/php_session.c
|
|
|
|
/* Initialized in MINIT, readonly otherwise. */
|
|
static int my_module_number = 0;
|
|
|
|
/* Dispatched by RINIT and by php_session_destroy */
|
|
static inline void php_rinit_session_globals(void) /* {{{ */
|
|
{
|
|
/* Do NOT init PS(mod_user_names) here! */
|
|
/* TODO: These could be moved to MINIT and removed. These should be initialized by php_rshutdown_session_globals() always when execution is finished. */
|
|
PS(id) = NULL;
|
|
PS(session_status) = php_session_none;
|
|
PS(in_save_handler) = 0;
|
|
PS(set_handler) = 0;
|
|
PS(mod_data) = NULL;
|
|
PS(mod_user_is_open) = 0;
|
|
PS(define_sid) = 1;
|
|
PS(session_vars) = NULL;
|
|
PS(module_number) = my_module_number;
|
|
ZVAL_UNDEF(&PS(http_session_vars));
|
|
}
|
|
/* }}} */
|
|
|
|
/* Dispatched by RSHUTDOWN and by php_session_destroy */
|
|
static inline void php_rshutdown_session_globals(void) /* {{{ */
|
|
{
|
|
/* Do NOT destroy PS(mod_user_names) here! */
|
|
if (!Z_ISUNDEF(PS(http_session_vars))) {
|
|
zval_ptr_dtor(&PS(http_session_vars));
|
|
ZVAL_UNDEF(&PS(http_session_vars));
|
|
}
|
|
if (PS(mod_data) || PS(mod_user_implemented)) {
|
|
zend_try {
|
|
PS(mod)->s_close(&PS(mod_data));
|
|
} zend_end_try();
|
|
}
|
|
if (PS(id)) {
|
|
zend_string_release_ex(PS(id), 0);
|
|
PS(id) = NULL;
|
|
}
|
|
|
|
if (PS(session_vars)) {
|
|
zend_string_release_ex(PS(session_vars), 0);
|
|
PS(session_vars) = NULL;
|
|
}
|
|
|
|
/* User save handlers may end up directly here by misuse, bugs in user script, etc. */
|
|
/* Set session status to prevent error while restoring save handler INI value. */
|
|
PS(session_status) = php_session_none;
|
|
}
|
|
/* }}} */
|
|
|
|
// End of copied functions
|
|
|
|
typedef struct frankenphp_server_context {
|
|
uintptr_t request;
|
|
uintptr_t requests_chan;
|
|
char *worker_filename;
|
|
char *cookie_data;
|
|
} frankenphp_server_context;
|
|
|
|
ZEND_BEGIN_ARG_INFO_EX(arginfo_frankenphp_handle_request, 0, 0, 1)
|
|
ZEND_ARG_CALLABLE_INFO(false, handler, false)
|
|
ZEND_END_ARG_INFO()
|
|
|
|
PHP_FUNCTION(frankenphp_handle_request) {
|
|
zend_fcall_info fci;
|
|
zend_fcall_info_cache fcc;
|
|
|
|
if (zend_parse_parameters(ZEND_NUM_ARGS(), "f", &fci, &fcc) == FAILURE) {
|
|
RETURN_THROWS();
|
|
}
|
|
|
|
frankenphp_server_context *ctx = SG(server_context);
|
|
|
|
uintptr_t request = go_frankenphp_worker_handle_request_start(ctx->requests_chan);
|
|
if (!request) {
|
|
RETURN_FALSE;
|
|
}
|
|
|
|
// Call the PHP func
|
|
zval retval = {0};
|
|
fci.size = sizeof fci;
|
|
fci.retval = &retval;
|
|
|
|
zend_call_function(&fci, &fcc);
|
|
|
|
php_session_flush(1);
|
|
|
|
go_frankenphp_worker_handle_request_end(request);
|
|
|
|
// Adapted from php_request_shutdown
|
|
|
|
zend_try {
|
|
php_output_end_all();
|
|
} zend_end_try();
|
|
|
|
|
|
zend_try {
|
|
php_output_deactivate();
|
|
} zend_end_try();
|
|
|
|
php_rshutdown_session_globals();
|
|
php_rinit_session_globals();
|
|
|
|
RETURN_TRUE;
|
|
}
|
|
|
|
// Adapted from php_request_startup()
|
|
int frankenphp_worker_reset_server_context() {
|
|
int retval = SUCCESS;
|
|
|
|
zend_try {
|
|
//PG(in_error_log) = 0;
|
|
//PG(during_request_startup) = 1;
|
|
|
|
php_output_activate();
|
|
|
|
/* initialize global variables */
|
|
//PG(modules_activated) = 0;
|
|
PG(header_is_being_sent) = 0;
|
|
PG(connection_status) = PHP_CONNECTION_NORMAL;
|
|
//PG(in_user_include) = 0;
|
|
|
|
// Keep the current execution context
|
|
//zend_activate();
|
|
sapi_activate();
|
|
|
|
#ifdef ZEND_SIGNALS
|
|
//zend_signal_activate();
|
|
#endif
|
|
|
|
if (PG(max_input_time) == -1) {
|
|
zend_set_timeout(EG(timeout_seconds), 1);
|
|
} else {
|
|
zend_set_timeout(PG(max_input_time), 1);
|
|
}
|
|
|
|
/* Disable realpath cache if an open_basedir is set */
|
|
//if (PG(open_basedir) && *PG(open_basedir)) {
|
|
// CWDG(realpath_cache_size_limit) = 0;
|
|
//}
|
|
|
|
if (PG(expose_php)) {
|
|
sapi_add_header(SAPI_PHP_VERSION_HEADER, sizeof(SAPI_PHP_VERSION_HEADER)-1, 1);
|
|
}
|
|
|
|
if (PG(output_handler) && PG(output_handler)[0]) {
|
|
zval oh;
|
|
|
|
ZVAL_STRING(&oh, PG(output_handler));
|
|
php_output_start_user(&oh, 0, PHP_OUTPUT_HANDLER_STDFLAGS);
|
|
zval_ptr_dtor(&oh);
|
|
} else if (PG(output_buffering)) {
|
|
php_output_start_user(NULL, PG(output_buffering) > 1 ? PG(output_buffering) : 0, PHP_OUTPUT_HANDLER_STDFLAGS);
|
|
} else if (PG(implicit_flush)) {
|
|
php_output_set_implicit_flush(1);
|
|
}
|
|
|
|
/* We turn this off in php_execute_script() */
|
|
/* PG(during_request_startup) = 0; */
|
|
|
|
php_register_server_variables();
|
|
|
|
zend_hash_update(&EG(symbol_table), ZSTR_KNOWN(ZEND_STR_AUTOGLOBAL_SERVER), &PG(http_globals)[TRACK_VARS_SERVER]);
|
|
Z_ADDREF(PG(http_globals)[TRACK_VARS_SERVER]);
|
|
|
|
php_hash_environment();
|
|
} zend_catch {
|
|
retval = FAILURE;
|
|
} zend_end_try();
|
|
|
|
return retval;
|
|
}
|
|
|
|
static const zend_function_entry frankenphp_ext_functions[] = {
|
|
PHP_FE(frankenphp_handle_request, arginfo_frankenphp_handle_request)
|
|
PHP_FE_END
|
|
};
|
|
|
|
static zend_module_entry frankenphp_module = {
|
|
STANDARD_MODULE_HEADER,
|
|
"frankenphp",
|
|
frankenphp_ext_functions, /* function table */
|
|
NULL, /* initialization */
|
|
NULL, /* shutdown */
|
|
NULL, /* request initialization */
|
|
NULL, /* request shutdown */
|
|
NULL, /* information */
|
|
"dev",
|
|
STANDARD_MODULE_PROPERTIES
|
|
};
|
|
|
|
uintptr_t frankenphp_clean_server_context() {
|
|
frankenphp_server_context *ctx = SG(server_context);
|
|
if (ctx == NULL) return 0;
|
|
|
|
free(SG(request_info.auth_password));
|
|
SG(request_info.auth_password) = NULL;
|
|
|
|
free(SG(request_info.auth_user));
|
|
SG(request_info.auth_user) = NULL;
|
|
|
|
free((char *) SG(request_info.request_method));
|
|
SG(request_info.request_method) = NULL;
|
|
|
|
free(SG(request_info.query_string));
|
|
SG(request_info.query_string) = NULL;
|
|
|
|
free((char *) SG(request_info.content_type));
|
|
SG(request_info.content_type) = NULL;
|
|
|
|
free(SG(request_info.path_translated));
|
|
SG(request_info.path_translated) = NULL;
|
|
|
|
free(SG(request_info.request_uri));
|
|
SG(request_info.request_uri) = NULL;
|
|
|
|
return ctx->request;
|
|
}
|
|
|
|
uintptr_t frankenphp_request_shutdown()
|
|
{
|
|
php_request_shutdown((void *) 0);
|
|
|
|
frankenphp_server_context *ctx = SG(server_context);
|
|
|
|
free(ctx->cookie_data);
|
|
((frankenphp_server_context*) SG(server_context))->cookie_data = NULL;
|
|
|
|
uintptr_t rh = frankenphp_clean_server_context();
|
|
|
|
free(ctx);
|
|
SG(server_context) = NULL;
|
|
|
|
return rh;
|
|
}
|
|
|
|
// set worker to 0 if not in worker mode
|
|
int frankenphp_create_server_context(uintptr_t requests_chan, char* worker_filename)
|
|
{
|
|
frankenphp_server_context *ctx;
|
|
|
|
(void) ts_resource(0);
|
|
|
|
// todo: use a pool
|
|
ctx = malloc(sizeof(frankenphp_server_context));
|
|
if (ctx == NULL) return FAILURE;
|
|
|
|
ctx->request = 0;
|
|
ctx->requests_chan = requests_chan;
|
|
ctx->worker_filename = worker_filename;
|
|
ctx->cookie_data = NULL;
|
|
|
|
SG(server_context) = ctx;
|
|
|
|
return SUCCESS;
|
|
}
|
|
|
|
void frankenphp_update_server_context(
|
|
uintptr_t request,
|
|
|
|
const char *request_method,
|
|
char *query_string,
|
|
zend_long content_length,
|
|
char *path_translated,
|
|
char *request_uri,
|
|
const char *content_type,
|
|
char *auth_user,
|
|
char *auth_password,
|
|
int proto_num
|
|
) {
|
|
frankenphp_server_context *ctx = SG(server_context);
|
|
|
|
ctx->request = request;
|
|
|
|
SG(request_info).auth_password = auth_password;
|
|
SG(request_info).auth_user = auth_user;
|
|
SG(request_info).request_method = request_method;
|
|
SG(request_info).query_string = query_string;
|
|
SG(request_info).content_type = content_type;
|
|
SG(request_info).content_length = content_length;
|
|
SG(request_info).path_translated = path_translated;
|
|
SG(request_info).request_uri = request_uri;
|
|
SG(request_info).proto_num = proto_num;
|
|
}
|
|
|
|
static int frankenphp_startup(sapi_module_struct *sapi_module)
|
|
{
|
|
return php_module_startup(sapi_module, &frankenphp_module, 1);
|
|
}
|
|
|
|
static int frankenphp_deactivate(void)
|
|
{
|
|
// TODO: flush everything
|
|
return SUCCESS;
|
|
}
|
|
|
|
static size_t frankenphp_ub_write(const char *str, size_t str_length)
|
|
{
|
|
frankenphp_server_context* ctx = SG(server_context);
|
|
|
|
if (ctx->request == 0) return 0; // TODO: write on stdout?
|
|
|
|
return go_ub_write(ctx->request, (char *) str, str_length);
|
|
}
|
|
|
|
static int frankenphp_send_headers(sapi_headers_struct *sapi_headers)
|
|
{
|
|
if (SG(request_info).no_headers == 1) {
|
|
return SAPI_HEADER_SENT_SUCCESSFULLY;
|
|
}
|
|
|
|
sapi_header_struct *h;
|
|
zend_llist_position pos;
|
|
int status;
|
|
frankenphp_server_context* ctx = SG(server_context);
|
|
|
|
if (ctx->request == 0) return SAPI_HEADER_SEND_FAILED;
|
|
|
|
h = zend_llist_get_first_ex(&sapi_headers->headers, &pos);
|
|
while (h) {
|
|
go_add_header(ctx->request, h->header, h->header_len);
|
|
h = zend_llist_get_next_ex(&sapi_headers->headers, &pos);
|
|
}
|
|
|
|
if (!SG(sapi_headers).http_status_line) {
|
|
status = SG(sapi_headers).http_response_code;
|
|
if (!status) status = 200;
|
|
} else {
|
|
status = atoi((SG(sapi_headers).http_status_line) + 9);
|
|
}
|
|
|
|
go_write_header(ctx->request, status);
|
|
|
|
return SAPI_HEADER_SENT_SUCCESSFULLY;
|
|
}
|
|
|
|
static size_t frankenphp_read_post(char *buffer, size_t count_bytes)
|
|
{
|
|
frankenphp_server_context* ctx = SG(server_context);
|
|
|
|
if (ctx->request == 0) return 0;
|
|
|
|
return go_read_post(ctx->request, buffer, count_bytes);
|
|
}
|
|
|
|
static char* frankenphp_read_cookies(void)
|
|
{
|
|
frankenphp_server_context* ctx = SG(server_context);
|
|
|
|
if (ctx->request == 0) return "";
|
|
|
|
ctx->cookie_data = go_read_cookies(ctx->request);
|
|
|
|
return ctx->cookie_data;
|
|
}
|
|
|
|
static void frankenphp_register_variables(zval *track_vars_array)
|
|
{
|
|
// https://www.php.net/manual/en/reserved.variables.server.php
|
|
frankenphp_server_context* ctx = SG(server_context);
|
|
|
|
if (ctx->request == 0 && ctx->worker_filename != NULL) {
|
|
// todo: also register PHP_SELF etc
|
|
php_register_variable_safe("SCRIPT_FILENAME", ctx->worker_filename, strlen(ctx->worker_filename), track_vars_array);
|
|
}
|
|
|
|
// todo: import or not environment variables set in the parent process?
|
|
//php_import_environment_variables(track_vars_array);
|
|
|
|
go_register_variables(ctx->request, track_vars_array);
|
|
}
|
|
|
|
static void frankenphp_log_message(const char *message, int syslog_type_int)
|
|
{
|
|
// TODO: call Go logger
|
|
}
|
|
|
|
sapi_module_struct frankenphp_sapi_module = {
|
|
"frankenphp", /* name */
|
|
"FrankenPHP", /* pretty name */
|
|
|
|
frankenphp_startup, /* startup */
|
|
php_module_shutdown_wrapper, /* shutdown */
|
|
|
|
NULL, /* activate */
|
|
frankenphp_deactivate, /* deactivate */
|
|
|
|
frankenphp_ub_write, /* unbuffered write */
|
|
NULL, /* flush */
|
|
NULL, /* get uid */
|
|
NULL, /* getenv */
|
|
|
|
php_error, /* error handler */
|
|
|
|
NULL, /* header handler */
|
|
frankenphp_send_headers, /* send headers handler */
|
|
NULL, /* send header handler */
|
|
|
|
frankenphp_read_post, /* read POST data */
|
|
frankenphp_read_cookies, /* read Cookies */
|
|
|
|
frankenphp_register_variables, /* register server variables */
|
|
frankenphp_log_message, /* Log message */
|
|
NULL, /* Get request time */
|
|
NULL, /* Child terminate */
|
|
|
|
STANDARD_SAPI_MODULE_PROPERTIES
|
|
};
|
|
|
|
int frankenphp_init() {
|
|
#ifndef ZTS
|
|
return FAILURE;
|
|
#endif
|
|
|
|
php_tsrm_startup();
|
|
zend_signal_startup();
|
|
sapi_startup(&frankenphp_sapi_module);
|
|
|
|
return frankenphp_sapi_module.startup(&frankenphp_sapi_module);
|
|
}
|
|
|
|
void frankenphp_shutdown()
|
|
{
|
|
php_module_shutdown();
|
|
sapi_shutdown();
|
|
tsrm_shutdown();
|
|
}
|
|
|
|
int frankenphp_request_startup()
|
|
{
|
|
if (php_request_startup() == SUCCESS) {
|
|
return SUCCESS;
|
|
}
|
|
|
|
fprintf(stderr, "problem in php_request_startup\n");
|
|
|
|
php_request_shutdown((void *) 0);
|
|
frankenphp_server_context *ctx = SG(server_context);
|
|
SG(server_context) = NULL;
|
|
free(ctx);
|
|
|
|
return FAILURE;
|
|
}
|
|
|
|
int frankenphp_execute_script(const char* file_name)
|
|
{
|
|
int status = FAILURE;
|
|
|
|
zend_file_handle file_handle;
|
|
zend_stream_init_filename(&file_handle, file_name);
|
|
|
|
zend_first_try {
|
|
status = php_execute_script(&file_handle);
|
|
} zend_catch {
|
|
/* int exit_status = EG(exit_status); */
|
|
} zend_end_try();
|
|
|
|
return status;
|
|
}
|