php-src/ext/openssl/tests/ServerClientTestCase.inc
2024-12-31 15:05:18 +01:00

229 lines
6.1 KiB
PHP

<?php
const WORKER_ARGV_VALUE = 'RUN_WORKER';
const WORKER_DEFAULT_NAME = 'server';
function phpt_notify(string $worker = WORKER_DEFAULT_NAME, string $message = ""): void
{
ServerClientTestCase::getInstance()->notify($worker, $message);
}
function phpt_wait($worker = WORKER_DEFAULT_NAME, $timeout = null): ?string
{
return ServerClientTestCase::getInstance()->wait($worker, $timeout);
}
function phpt_notify_server_start($server): void
{
ServerClientTestCase::getInstance()->notify_server_start($server);
}
function phpt_has_sslv3() {
static $result = null;
if (!is_null($result)) {
return $result;
}
$server = @stream_socket_server('sslv3://127.0.0.1:10013');
if ($result = !!$server) {
fclose($server);
}
return $result;
}
function phpt_extract_tls_records($rawData) {
$records = [];
$offset = 0;
$dataLength = strlen($rawData);
while ($offset < $dataLength) {
// Ensure there's enough data left for the header.
if ($offset + 5 > $dataLength) {
break;
}
// Extract the length of the current record.
$length = unpack("n", substr($rawData, $offset + 3, 2))[1];
// Check if the total length is within the bounds of the rawData.
if ($offset + 5 + $length > $dataLength) {
break;
}
// Extract the record and add it to the records array.
$records[] = substr($rawData, $offset, 5 + $length);
// Move the offset past the current record.
$offset += 5 + $length;
}
return $records;
}
/**
* This is a singleton to let the wait/notify functions work
* I know it's horrible, but it's a means to an end
*/
class ServerClientTestCase
{
private $isWorker = false;
private $workerHandle = [];
private $workerStdIn = [];
private $workerStdOut = [];
private static $instance;
public static function getInstance($isWorker = false)
{
if (!isset(self::$instance)) {
self::$instance = new self($isWorker);
}
return self::$instance;
}
public function __construct($isWorker = false)
{
if (!isset(self::$instance)) {
self::$instance = $this;
}
$this->isWorker = $isWorker;
}
private function spawnWorkerProcess($worker, $code)
{
if (defined("PHP_WINDOWS_VERSION_MAJOR")) {
$ini = php_ini_loaded_file();
$cmd = sprintf(
'%s %s "%s" %s',
PHP_BINARY, $ini ? "-n -c $ini" : "",
__FILE__,
WORKER_ARGV_VALUE
);
} else {
$cmd = sprintf(
'%s %s "%s" %s %s',
PHP_BINARY,
getenv('TEST_PHP_EXTRA_ARGS'),
__FILE__,
WORKER_ARGV_VALUE,
$worker
);
}
$this->workerHandle[$worker] = proc_open(
$cmd,
[['pipe', 'r'], ['pipe', 'w'], STDERR],
$pipes
);
$this->workerStdIn[$worker] = $pipes[0];
$this->workerStdOut[$worker] = $pipes[1];
fwrite($this->workerStdIn[$worker], $code . "\n---\n");
}
private function cleanupWorkerProcess($worker)
{
fclose($this->workerStdIn[$worker]);
fclose($this->workerStdOut[$worker]);
proc_close($this->workerHandle[$worker]);
}
private function stripPhpTagsFromCode($code)
{
return preg_replace('/^\s*<\?(?:php)?|\?>\s*$/i', '', $code);
}
public function runWorker()
{
$code = '';
while (1) {
$line = fgets(STDIN);
if (trim($line) === "---") {
break;
}
$code .= $line;
}
eval($code);
}
/**
* Run client and all workers
*
* @param string $clientCode The client PHP code
* @param string|array $workerCode
* @param bool $ephemeral Select whether automatic port selection and automatic awaiting is used
* @return void
* @throws Exception
*/
public function run(string $clientCode, string|array $workerCode, bool $ephemeral = true): void
{
if (!is_array($workerCode)) {
$workerCode = [WORKER_DEFAULT_NAME => $workerCode];
}
reset($workerCode);
$code = current($workerCode);
$worker = key($workerCode);
while ($worker != null) {
$this->spawnWorkerProcess($worker, $this->stripPhpTagsFromCode($code));
$code = next($workerCode);
if ($ephemeral) {
$addr = trim($this->wait($worker));
if (empty($addr)) {
throw new \Exception("Failed server start");
}
if ($code === false) {
$clientCode = preg_replace('/{{\s*ADDR\s*}}/', $addr, $clientCode);
} else {
$code = preg_replace('/{{\s*ADDR\s*}}/', $addr, $code);
}
}
$worker = key($workerCode);
}
eval($this->stripPhpTagsFromCode($clientCode));
foreach ($workerCode as $worker => $code) {
$this->cleanupWorkerProcess($worker);
}
}
public function wait($worker, $timeout = null): ?string
{
$handle = $this->isWorker ? STDIN : $this->workerStdOut[$worker];
if ($timeout === null) {
return fgets($handle);
}
stream_set_blocking($handle, false);
$read = [$handle];
$result = stream_select($read, $write, $except, $timeout);
if (!$result) {
return null;
}
$result = fgets($handle);
stream_set_blocking($handle, true);
return $result;
}
public function notify(string $worker, string $message = ""): void
{
fwrite($this->isWorker ? STDOUT : $this->workerStdIn[$worker], "$message\n");
}
public function notify_server_start($server): void
{
echo stream_socket_get_name($server, false) . "\n";
}
}
if (isset($argv[1]) && $argv[1] === WORKER_ARGV_VALUE) {
ServerClientTestCase::getInstance(true)->runWorker();
}