Skip to content

Commit

Permalink
add headers_sent
Browse files Browse the repository at this point in the history
  • Loading branch information
kojidev committed Dec 12, 2024
1 parent 2398687 commit 54cb7bc
Show file tree
Hide file tree
Showing 5 changed files with 32 additions and 4 deletions.
1 change: 1 addition & 0 deletions builtin-functions/kphp-full/_functions.txt
Original file line number Diff line number Diff line change
Expand Up @@ -141,6 +141,7 @@ function ob_get_flush () ::: string | false;
function ob_get_length () ::: int | false;
function ob_get_level () ::: int;

function headers_sent () ::: bool;
function header ($str ::: string, $replace ::: bool = true, $http_response_code ::: int = 0) ::: void;
function headers_list () ::: string[];
function send_http_103_early_hints($headers ::: string[]) ::: void;
Expand Down
16 changes: 12 additions & 4 deletions runtime/interface.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -369,6 +369,10 @@ array<string> f$headers_list() {
return result;
}

bool f$headers_sent() {
return headers_sent;
}

void f$send_http_103_early_hints(const array<string> & headers) {
string header("HTTP/1.1 103 Early Hints\r\n");
for (const auto & h : headers) {
Expand Down Expand Up @@ -568,9 +572,11 @@ static int ob_merge_buffers() {
void f$flush() {
php_assert(ob_cur_buffer >= 0 && php_worker.has_value());
// Run custom headers handler before body processing
if (headers_custom_handler_function && !headers_sent && query_type == QUERY_TYPE_HTTP) {
if (!headers_sent && query_type == QUERY_TYPE_HTTP) {
headers_sent = true;
headers_custom_handler_function();
if (headers_custom_handler_function) {
headers_custom_handler_function();
}
}
string_buffer const * http_body = compress_http_query_body(&oub[ob_system_level]);
string_buffer const * http_headers = nullptr;
Expand All @@ -586,9 +592,11 @@ void f$flush() {

void f$fastcgi_finish_request(int64_t exit_code) {
// Run custom headers handler before body processing
if (headers_custom_handler_function && !headers_sent && query_type == QUERY_TYPE_HTTP) {
if (!headers_sent && query_type == QUERY_TYPE_HTTP) {
headers_sent = true;
headers_custom_handler_function();
if (headers_custom_handler_function) {
headers_custom_handler_function();
}
}
int ob_total_buffer = ob_merge_buffers();
if (php_worker.has_value() && php_worker->flushed_http_connection) {
Expand Down
2 changes: 2 additions & 0 deletions runtime/interface.h
Original file line number Diff line number Diff line change
Expand Up @@ -47,6 +47,8 @@ int64_t f$ob_get_level();

void f$flush();

bool f$headers_sent();

void f$header(const string &str, bool replace = true, int64_t http_response_code = 0);

array<string> f$headers_list();
Expand Down
4 changes: 4 additions & 0 deletions tests/python/tests/http_server/php/index.php
Original file line number Diff line number Diff line change
Expand Up @@ -178,6 +178,10 @@ public function work(string $output) {

break;
}
} else if ($_SERVER["PHP_SELF"] === "/test_headers_sent") {
echo (int)headers_sent();
flush();
echo (int)headers_sent();
} else if ($_SERVER["PHP_SELF"] === "/test_ignore_user_abort") {
register_shutdown_function('shutdown_function');
/** @var I */
Expand Down
13 changes: 13 additions & 0 deletions tests/python/tests/http_server/test_headers_sent.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,13 @@
import socket
from python.lib.testcase import KphpServerAutoTestCase
from python.lib.http_client import RawResponse

class TestHeadersSent(KphpServerAutoTestCase):
def test(self):
request = b"GET /test_headers_sent HTTP/1.1\r\nHost:localhost\r\n\r\n"
with socket.socket(socket.AF_INET, socket.SOCK_STREAM) as s:
s.connect(('127.0.0.1', self.kphp_server.http_port))
s.send(request)

body = RawResponse(s.recv(4096)).content + s.recv(20)
self.assertEqual(b"01", body)

0 comments on commit 54cb7bc

Please sign in to comment.