mirror of
https://github.com/php/php-src.git
synced 2025-08-15 21:48:51 +02:00
229 lines
6.1 KiB
PHP
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();
|
|
}
|