File: /home/marketing.cfbon.ru/public_html/vendor/phpunit/phpunit/src/Runner/ErrorHandler.php
<?php declare(strict_types=1);
/*
* This file is part of PHPUnit.
*
* (c) Sebastian Bergmann <sebastian@phpunit.de>
*
* For the full copyright and license information, please view the LICENSE
* file that was distributed with this source code.
*/
namespace PHPUnit\Runner;
use const DEBUG_BACKTRACE_IGNORE_ARGS;
use const E_COMPILE_ERROR;
use const E_COMPILE_WARNING;
use const E_CORE_ERROR;
use const E_CORE_WARNING;
use const E_DEPRECATED;
use const E_ERROR;
use const E_NOTICE;
use const E_PARSE;
use const E_RECOVERABLE_ERROR;
use const E_USER_DEPRECATED;
use const E_USER_ERROR;
use const E_USER_NOTICE;
use const E_USER_WARNING;
use const E_WARNING;
use function array_keys;
use function array_values;
use function debug_backtrace;
use function defined;
use function error_reporting;
use function restore_error_handler;
use function set_error_handler;
use function sprintf;
use PHPUnit\Event;
use PHPUnit\Event\Code\IssueTrigger\IssueTrigger;
use PHPUnit\Event\Code\NoTestCaseObjectOnCallStackException;
use PHPUnit\Event\Code\TestMethod;
use PHPUnit\Runner\Baseline\Baseline;
use PHPUnit\Runner\Baseline\Issue;
use PHPUnit\TextUI\Configuration\Registry;
use PHPUnit\TextUI\Configuration\Source;
use PHPUnit\TextUI\Configuration\SourceFilter;
use PHPUnit\Util\ExcludeList;
/**
* @no-named-arguments Parameter names are not covered by the backward compatibility promise for PHPUnit
*
* @internal This class is not covered by the backward compatibility promise for PHPUnit
*/
final class ErrorHandler
{
private const UNHANDLEABLE_LEVELS = E_ERROR | E_PARSE | E_CORE_ERROR | E_CORE_WARNING | E_COMPILE_ERROR | E_COMPILE_WARNING;
private const INSUPPRESSIBLE_LEVELS = E_ERROR | E_PARSE | E_CORE_ERROR | E_COMPILE_ERROR | E_USER_ERROR | E_RECOVERABLE_ERROR;
private static ?self $instance = null;
private ?Baseline $baseline = null;
private bool $enabled = false;
private ?int $originalErrorReportingLevel = null;
private readonly Source $source;
/**
* @var ?array{functions: list<non-empty-string>, methods: list<array{className: class-string, methodName: non-empty-string}>}
*/
private ?array $deprecationTriggers = null;
public static function instance(): self
{
return self::$instance ?? self::$instance = new self(Registry::get()->source());
}
private function __construct(Source $source)
{
$this->source = $source;
}
/**
* @throws NoTestCaseObjectOnCallStackException
*/
public function __invoke(int $errorNumber, string $errorString, string $errorFile, int $errorLine): bool
{
$suppressed = (error_reporting() & ~self::INSUPPRESSIBLE_LEVELS) === 0;
if ($suppressed && (new ExcludeList)->isExcluded($errorFile)) {
return false;
}
/**
* E_STRICT is deprecated since PHP 8.4.
*
* @see https://github.com/sebastianbergmann/phpunit/issues/5956
*/
if (defined('E_STRICT') && $errorNumber === 2048) {
$errorNumber = E_NOTICE;
}
$test = Event\Code\TestMethodBuilder::fromCallStack();
$ignoredByBaseline = $this->ignoredByBaseline($errorFile, $errorLine, $errorString);
$ignoredByTest = $test->metadata()->isIgnoreDeprecations()->isNotEmpty();
switch ($errorNumber) {
case E_NOTICE:
Event\Facade::emitter()->testTriggeredPhpNotice(
$test,
$errorString,
$errorFile,
$errorLine,
$suppressed,
$ignoredByBaseline,
);
break;
case E_USER_NOTICE:
Event\Facade::emitter()->testTriggeredNotice(
$test,
$errorString,
$errorFile,
$errorLine,
$suppressed,
$ignoredByBaseline,
);
break;
case E_WARNING:
Event\Facade::emitter()->testTriggeredPhpWarning(
$test,
$errorString,
$errorFile,
$errorLine,
$suppressed,
$ignoredByBaseline,
);
break;
case E_USER_WARNING:
Event\Facade::emitter()->testTriggeredWarning(
$test,
$errorString,
$errorFile,
$errorLine,
$suppressed,
$ignoredByBaseline,
);
break;
case E_DEPRECATED:
Event\Facade::emitter()->testTriggeredPhpDeprecation(
$test,
$errorString,
$errorFile,
$errorLine,
$suppressed,
$ignoredByBaseline,
$ignoredByTest,
$this->trigger($test, false),
);
break;
case E_USER_DEPRECATED:
$deprecationFrame = $this->guessDeprecationFrame();
Event\Facade::emitter()->testTriggeredDeprecation(
$test,
$errorString,
$deprecationFrame['file'] ?? $errorFile,
$deprecationFrame['line'] ?? $errorLine,
$suppressed,
$ignoredByBaseline,
$ignoredByTest,
$this->trigger($test, true),
$this->stackTrace(),
);
break;
case E_USER_ERROR:
Event\Facade::emitter()->testTriggeredError(
$test,
$errorString,
$errorFile,
$errorLine,
$suppressed,
);
throw new ErrorException('E_USER_ERROR was triggered');
default:
return false;
}
return false;
}
public function enable(): void
{
if ($this->enabled) {
return;
}
$oldErrorHandler = set_error_handler($this);
if ($oldErrorHandler !== null) {
restore_error_handler();
return;
}
$this->enabled = true;
$this->originalErrorReportingLevel = error_reporting();
error_reporting($this->originalErrorReportingLevel & self::UNHANDLEABLE_LEVELS);
}
public function disable(): void
{
if (!$this->enabled) {
return;
}
restore_error_handler();
error_reporting(error_reporting() | $this->originalErrorReportingLevel);
$this->enabled = false;
$this->originalErrorReportingLevel = null;
}
public function useBaseline(Baseline $baseline): void
{
$this->baseline = $baseline;
}
/**
* @param array{functions: list<non-empty-string>, methods: list<array{className: class-string, methodName: non-empty-string}>} $deprecationTriggers
*/
public function useDeprecationTriggers(array $deprecationTriggers): void
{
$this->deprecationTriggers = $deprecationTriggers;
}
/**
* @param non-empty-string $file
* @param positive-int $line
* @param non-empty-string $description
*/
private function ignoredByBaseline(string $file, int $line, string $description): bool
{
if ($this->baseline === null) {
return false;
}
return $this->baseline->has(Issue::from($file, $line, null, $description));
}
private function trigger(TestMethod $test, bool $filterTrigger): IssueTrigger
{
if (!$this->source->notEmpty()) {
return IssueTrigger::unknown();
}
$trace = $this->filteredStackTrace($filterTrigger);
$triggeredInFirstPartyCode = false;
$triggerCalledFromFirstPartyCode = false;
if (isset($trace[0]['file'])) {
if ($trace[0]['file'] === $test->file()) {
return IssueTrigger::test();
}
if (SourceFilter::instance()->includes($trace[0]['file'])) {
$triggeredInFirstPartyCode = true;
}
}
if (isset($trace[1]['file']) &&
($trace[1]['file'] === $test->file() ||
SourceFilter::instance()->includes($trace[1]['file']))) {
$triggerCalledFromFirstPartyCode = true;
}
if ($triggerCalledFromFirstPartyCode) {
if ($triggeredInFirstPartyCode) {
return IssueTrigger::self();
}
return IssueTrigger::direct();
}
return IssueTrigger::indirect();
}
/**
* @return list<array{file: string, line: int, class?: string, function?: string, type: string}>
*/
private function filteredStackTrace(bool $filterDeprecationTriggers): array
{
$trace = $this->errorStackTrace();
if ($this->deprecationTriggers === null || !$filterDeprecationTriggers) {
return array_values($trace);
}
foreach (array_keys($trace) as $frame) {
foreach ($this->deprecationTriggers['functions'] as $function) {
if ($this->frameIsFunction($trace[$frame], $function)) {
unset($trace[$frame]);
continue 2;
}
}
foreach ($this->deprecationTriggers['methods'] as $method) {
if ($this->frameIsMethod($trace[$frame], $method)) {
unset($trace[$frame]);
continue 2;
}
}
}
return array_values($trace);
}
/**
* @return ?array{file: non-empty-string, line: positive-int}
*/
private function guessDeprecationFrame(): ?array
{
if ($this->deprecationTriggers === null) {
return null;
}
$trace = $this->errorStackTrace();
foreach ($trace as $frame) {
foreach ($this->deprecationTriggers['functions'] as $function) {
if ($this->frameIsFunction($frame, $function)) {
return $frame;
}
}
foreach ($this->deprecationTriggers['methods'] as $method) {
if ($this->frameIsMethod($frame, $method)) {
return $frame;
}
}
}
return null;
}
/**
* @return list<array{file: string, line: ?int, class?: class-string, function?: string, type: string}>
*/
private function errorStackTrace(): array
{
$trace = debug_backtrace(DEBUG_BACKTRACE_IGNORE_ARGS);
$i = 0;
do {
unset($trace[$i]);
} while (self::class === ($trace[++$i]['class'] ?? null));
return array_values($trace);
}
/**
* @param array{class? : class-string, function?: non-empty-string} $frame
* @param non-empty-string $function
*/
private function frameIsFunction(array $frame, string $function): bool
{
return !isset($frame['class']) && isset($frame['function']) && $frame['function'] === $function;
}
/**
* @param array{class? : class-string, function?: non-empty-string} $frame
* @param array{className: class-string, methodName: non-empty-string} $method
*/
private function frameIsMethod(array $frame, array $method): bool
{
return isset($frame['class']) &&
$frame['class'] === $method['className'] &&
isset($frame['function']) &&
$frame['function'] === $method['methodName'];
}
/**
* @return non-empty-string
*/
private function stackTrace(): string
{
$buffer = '';
$excludeList = new ExcludeList(true);
foreach ($this->errorStackTrace() as $frame) {
/**
* @see https://github.com/sebastianbergmann/phpunit/issues/6043
*/
if (!isset($frame['file'])) {
continue;
}
if ($excludeList->isExcluded($frame['file'])) {
continue;
}
$buffer .= sprintf(
"%s:%s\n",
$frame['file'],
$frame['line'] ?? '?',
);
}
return $buffer;
}
}