Skip to main content
Drupal API
User account menu
  • Log in

Breadcrumb

  1. Drupal Core 11.1.x

TestCase.php

Same filename in this branch
  1. 11.1.x vendor/slevomat/coding-standard/SlevomatCodingStandard/Sniffs/TestCase.php

Namespace

PHPUnit\Framework

File

vendor/phpunit/phpunit/src/Framework/TestCase.php

View source
<?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\Framework;

use const LC_ALL;
use const LC_COLLATE;
use const LC_CTYPE;
use const LC_MONETARY;
use const LC_NUMERIC;
use const LC_TIME;
use const PATHINFO_FILENAME;
use const PHP_EOL;
use const PHP_URL_PATH;
use function array_is_list;
use function array_keys;
use function array_map;
use function array_merge;
use function array_values;
use function assert;
use function basename;
use function chdir;
use function class_exists;
use function clearstatcache;
use function count;
use function defined;
use function explode;
use function getcwd;
use function implode;
use function in_array;
use function ini_set;
use function is_array;
use function is_callable;
use function is_int;
use function is_object;
use function is_string;
use function libxml_clear_errors;
use function method_exists;
use function ob_end_clean;
use function ob_get_clean;
use function ob_get_contents;
use function ob_get_level;
use function ob_start;
use function parse_url;
use function pathinfo;
use function preg_replace;
use function setlocale;
use function sprintf;
use function str_contains;
use function trim;
use AssertionError;
use DeepCopy\DeepCopy;
use PHPUnit\Event;
use PHPUnit\Event\NoPreviousThrowableException;
use PHPUnit\Event\RuntimeException;
use PHPUnit\Event\TestData\MoreThanOneDataSetFromDataProviderException;
use PHPUnit\Framework\Constraint\Exception as ExceptionConstraint;
use PHPUnit\Framework\Constraint\ExceptionCode;
use PHPUnit\Framework\Constraint\ExceptionMessageIsOrContains;
use PHPUnit\Framework\Constraint\ExceptionMessageMatchesRegularExpression;
use PHPUnit\Framework\MockObject\Exception as MockObjectException;
use PHPUnit\Framework\MockObject\Generator\Generator as MockGenerator;
use PHPUnit\Framework\MockObject\MockBuilder;
use PHPUnit\Framework\MockObject\MockObject;
use PHPUnit\Framework\MockObject\MockObjectInternal;
use PHPUnit\Framework\MockObject\Rule\AnyInvokedCount as AnyInvokedCountMatcher;
use PHPUnit\Framework\MockObject\Rule\InvokedAtLeastCount as InvokedAtLeastCountMatcher;
use PHPUnit\Framework\MockObject\Rule\InvokedAtLeastOnce as InvokedAtLeastOnceMatcher;
use PHPUnit\Framework\MockObject\Rule\InvokedAtMostCount as InvokedAtMostCountMatcher;
use PHPUnit\Framework\MockObject\Rule\InvokedCount as InvokedCountMatcher;
use PHPUnit\Framework\MockObject\Stub;
use PHPUnit\Framework\MockObject\Stub\ConsecutiveCalls as ConsecutiveCallsStub;
use PHPUnit\Framework\MockObject\Stub\Exception as ExceptionStub;
use PHPUnit\Framework\MockObject\Stub\ReturnArgument as ReturnArgumentStub;
use PHPUnit\Framework\MockObject\Stub\ReturnCallback as ReturnCallbackStub;
use PHPUnit\Framework\MockObject\Stub\ReturnSelf as ReturnSelfStub;
use PHPUnit\Framework\MockObject\Stub\ReturnStub;
use PHPUnit\Framework\MockObject\Stub\ReturnValueMap as ReturnValueMapStub;
use PHPUnit\Framework\TestSize\TestSize;
use PHPUnit\Framework\TestStatus\TestStatus;
use PHPUnit\Metadata\Api\Groups;
use PHPUnit\Metadata\Api\HookMethods;
use PHPUnit\Metadata\Api\Requirements;
use PHPUnit\Metadata\Parser\Registry as MetadataRegistry;
use PHPUnit\TestRunner\TestResult\PassedTests;
use PHPUnit\TextUI\Configuration\Registry as ConfigurationRegistry;
use PHPUnit\Util\Cloner;
use PHPUnit\Util\Test as TestUtil;
use ReflectionClass;
use ReflectionException;
use ReflectionMethod;
use ReflectionObject;
use ReflectionParameter;
use SebastianBergmann\CodeCoverage\StaticAnalysisCacheNotConfiguredException;
use SebastianBergmann\CodeCoverage\UnintentionallyCoveredCodeException;
use SebastianBergmann\Comparator\Comparator;
use SebastianBergmann\Comparator\Factory as ComparatorFactory;
use SebastianBergmann\Diff\Differ;
use SebastianBergmann\Diff\Output\UnifiedDiffOutputBuilder;
use SebastianBergmann\Exporter\Exporter;
use SebastianBergmann\GlobalState\ExcludeList as GlobalStateExcludeList;
use SebastianBergmann\GlobalState\Restorer;
use SebastianBergmann\GlobalState\Snapshot;
use SebastianBergmann\Invoker\TimeoutException;
use SebastianBergmann\ObjectEnumerator\Enumerator;
use SebastianBergmann\RecursionContext\Context;
use Throwable;

/**
 * @no-named-arguments Parameter names are not covered by the backward compatibility promise for PHPUnit
 */
abstract class TestCase extends Assert implements Reorderable, SelfDescribing, Test {
    private const LOCALE_CATEGORIES = [
        LC_ALL,
        LC_COLLATE,
        LC_CTYPE,
        LC_MONETARY,
        LC_NUMERIC,
        LC_TIME,
    ];
    private ?bool $backupGlobals = null;
    
    /**
     * @psalm-var list<string>
     */
    private array $backupGlobalsExcludeList = [];
    private ?bool $backupStaticProperties = null;
    
    /**
     * @psalm-var array<string,list<class-string>>
     */
    private array $backupStaticPropertiesExcludeList = [];
    private ?Snapshot $snapshot = null;
    private ?bool $runClassInSeparateProcess = null;
    private ?bool $runTestInSeparateProcess = null;
    private bool $preserveGlobalState = false;
    private bool $inIsolation = false;
    private ?string $expectedException = null;
    private ?string $expectedExceptionMessage = null;
    private ?string $expectedExceptionMessageRegExp = null;
    private null|int|string $expectedExceptionCode = null;
    
    /**
     * @psalm-var list<ExecutionOrderDependency>
     */
    private array $providedTests = [];
    private array $data = [];
    private int|string $dataName = '';
    
    /**
     * @psalm-var non-empty-string
     */
    private string $name;
    
    /**
     * @psalm-var list<string>
     */
    private array $groups = [];
    
    /**
     * @psalm-var list<ExecutionOrderDependency>
     */
    private array $dependencies = [];
    private array $dependencyInput = [];
    
    /**
     * @psalm-var array<string,string>
     */
    private array $iniSettings = [];
    private array $locale = [];
    
    /**
     * @psalm-var list<MockObjectInternal>
     */
    private array $mockObjects = [];
    private bool $registerMockObjectsFromTestArgumentsRecursively = false;
    private TestStatus $status;
    private int $numberOfAssertionsPerformed = 0;
    private mixed $testResult = null;
    private string $output = '';
    private ?string $outputExpectedRegex = null;
    private ?string $outputExpectedString = null;
    private bool $outputBufferingActive = false;
    private int $outputBufferingLevel;
    private bool $outputRetrievedForAssertion = false;
    private bool $doesNotPerformAssertions = false;
    
    /**
     * @psalm-var list<Comparator>
     */
    private array $customComparators = [];
    private ?Event\Code\TestMethod $testValueObjectForEvents = null;
    private bool $wasPrepared = false;
    
    /**
     * @psalm-var array<class-string, true>
     */
    private array $failureTypes = [];
    
    /**
     * Returns a matcher that matches when the method is executed
     * zero or more times.
     */
    public static final function any() : AnyInvokedCountMatcher {
        return new AnyInvokedCountMatcher();
    }
    
    /**
     * Returns a matcher that matches when the method is never executed.
     */
    public static final function never() : InvokedCountMatcher {
        return new InvokedCountMatcher(0);
    }
    
    /**
     * Returns a matcher that matches when the method is executed
     * at least N times.
     */
    public static final function atLeast(int $requiredInvocations) : InvokedAtLeastCountMatcher {
        return new InvokedAtLeastCountMatcher($requiredInvocations);
    }
    
    /**
     * Returns a matcher that matches when the method is executed at least once.
     */
    public static final function atLeastOnce() : InvokedAtLeastOnceMatcher {
        return new InvokedAtLeastOnceMatcher();
    }
    
    /**
     * Returns a matcher that matches when the method is executed exactly once.
     */
    public static final function once() : InvokedCountMatcher {
        return new InvokedCountMatcher(1);
    }
    
    /**
     * Returns a matcher that matches when the method is executed
     * exactly $count times.
     */
    public static final function exactly(int $count) : InvokedCountMatcher {
        return new InvokedCountMatcher($count);
    }
    
    /**
     * Returns a matcher that matches when the method is executed
     * at most N times.
     */
    public static final function atMost(int $allowedInvocations) : InvokedAtMostCountMatcher {
        return new InvokedAtMostCountMatcher($allowedInvocations);
    }
    
    /**
     * @deprecated Use <code>$double->willReturn()</code> instead of <code>$double->will($this->returnValue())</code>
     * @see https://github.com/sebastianbergmann/phpunit/issues/5423
     *
     * @codeCoverageIgnore
     */
    public static final function returnValue(mixed $value) : ReturnStub {
        return new ReturnStub($value);
    }
    
    /**
     * @deprecated Use <code>$double->willReturnMap()</code> instead of <code>$double->will($this->returnValueMap())</code>
     * @see https://github.com/sebastianbergmann/phpunit/issues/5423
     *
     * @codeCoverageIgnore
     */
    public static final function returnValueMap(array $valueMap) : ReturnValueMapStub {
        return new ReturnValueMapStub($valueMap);
    }
    
    /**
     * @deprecated Use <code>$double->willReturnArgument()</code> instead of <code>$double->will($this->returnArgument())</code>
     * @see https://github.com/sebastianbergmann/phpunit/issues/5423
     *
     * @codeCoverageIgnore
     */
    public static final function returnArgument(int $argumentIndex) : ReturnArgumentStub {
        return new ReturnArgumentStub($argumentIndex);
    }
    
    /**
     * @deprecated Use <code>$double->willReturnCallback()</code> instead of <code>$double->will($this->returnCallback())</code>
     * @see https://github.com/sebastianbergmann/phpunit/issues/5423
     *
     * @codeCoverageIgnore
     */
    public static final function returnCallback(callable $callback) : ReturnCallbackStub {
        return new ReturnCallbackStub($callback);
    }
    
    /**
     * @deprecated Use <code>$double->willReturnSelf()</code> instead of <code>$double->will($this->returnSelf())</code>
     * @see https://github.com/sebastianbergmann/phpunit/issues/5423
     *
     * @codeCoverageIgnore
     */
    public static final function returnSelf() : ReturnSelfStub {
        return new ReturnSelfStub();
    }
    public static final function throwException(Throwable $exception) : ExceptionStub {
        return new ExceptionStub($exception);
    }
    
    /**
     * @deprecated Use <code>$double->willReturn()</code> instead of <code>$double->will($this->onConsecutiveCalls())</code>
     * @see https://github.com/sebastianbergmann/phpunit/issues/5423
     * @see https://github.com/sebastianbergmann/phpunit/issues/5425
     *
     * @codeCoverageIgnore
     */
    public static final function onConsecutiveCalls(mixed ...$arguments) : ConsecutiveCallsStub {
        return new ConsecutiveCallsStub($arguments);
    }
    
    /**
     * @psalm-param non-empty-string $name
     *
     * @internal This method is not covered by the backward compatibility promise for PHPUnit
     */
    public function __construct(string $name) {
        $this->setName($name);
        $this->status = TestStatus::unknown();
    }
    
    /**
     * This method is called before the first test of this test class is run.
     *
     * @codeCoverageIgnore
     */
    public static function setUpBeforeClass() : void {
    }
    
    /**
     * This method is called after the last test of this test class is run.
     *
     * @codeCoverageIgnore
     */
    public static function tearDownAfterClass() : void {
    }
    
    /**
     * This method is called before each test.
     *
     * @codeCoverageIgnore
     */
    protected function setUp() : void {
    }
    
    /**
     * Performs assertions shared by all tests of a test case.
     *
     * This method is called between setUp() and test.
     *
     * @codeCoverageIgnore
     */
    protected function assertPreConditions() : void {
    }
    
    /**
     * Performs assertions shared by all tests of a test case.
     *
     * This method is called between test and tearDown().
     *
     * @codeCoverageIgnore
     */
    protected function assertPostConditions() : void {
    }
    
    /**
     * This method is called after each test.
     *
     * @codeCoverageIgnore
     */
    protected function tearDown() : void {
    }
    
    /**
     * Returns a string representation of the test case.
     *
     * @throws Exception
     *
     * @internal This method is not covered by the backward compatibility promise for PHPUnit
     */
    public function toString() : string {
        $buffer = sprintf('%s::%s', (new ReflectionClass($this))->getName(), $this->name);
        return $buffer . $this->dataSetAsStringWithData();
    }
    
    /**
     * @internal This method is not covered by the backward compatibility promise for PHPUnit
     */
    public final function count() : int {
        return 1;
    }
    public final function getActualOutputForAssertion() : string {
        $this->outputRetrievedForAssertion = true;
        return $this->output();
    }
    public final function expectOutputRegex(string $expectedRegex) : void {
        $this->outputExpectedRegex = $expectedRegex;
    }
    public final function expectOutputString(string $expectedString) : void {
        $this->outputExpectedString = $expectedString;
    }
    
    /**
     * @psalm-param class-string<Throwable> $exception
     */
    public final function expectException(string $exception) : void {
        $this->expectedException = $exception;
    }
    public final function expectExceptionCode(int|string $code) : void {
        $this->expectedExceptionCode = $code;
    }
    public final function expectExceptionMessage(string $message) : void {
        $this->expectedExceptionMessage = $message;
    }
    public final function expectExceptionMessageMatches(string $regularExpression) : void {
        $this->expectedExceptionMessageRegExp = $regularExpression;
    }
    
    /**
     * Sets up an expectation for an exception to be raised by the code under test.
     * Information for expected exception class, expected exception message, and
     * expected exception code are retrieved from a given Exception object.
     */
    public final function expectExceptionObject(\Exception $exception) : void {
        $this->expectException($exception::class);
        $this->expectExceptionMessage($exception->getMessage());
        $this->expectExceptionCode($exception->getCode());
    }
    public final function expectNotToPerformAssertions() : void {
        $this->doesNotPerformAssertions = true;
    }
    
    /**
     * @internal This method is not covered by the backward compatibility promise for PHPUnit
     */
    public final function status() : TestStatus {
        return $this->status;
    }
    
    /**
     * @throws \PHPUnit\Runner\Exception
     * @throws \PHPUnit\Util\Exception
     * @throws \SebastianBergmann\CodeCoverage\InvalidArgumentException
     * @throws \SebastianBergmann\Template\InvalidArgumentException
     * @throws CodeCoverageException
     * @throws Exception
     * @throws MoreThanOneDataSetFromDataProviderException
     * @throws NoPreviousThrowableException
     * @throws ProcessIsolationException
     * @throws StaticAnalysisCacheNotConfiguredException
     * @throws UnintentionallyCoveredCodeException
     *
     * @internal This method is not covered by the backward compatibility promise for PHPUnit
     */
    public final function run() : void {
        if (!$this->handleDependencies()) {
            return;
        }
        if (!$this->shouldRunInSeparateProcess() || $this->requirementsNotSatisfied()) {
            (new TestRunner())->run($this);
        }
        else {
            (new TestRunner())->runInSeparateProcess($this, $this->runClassInSeparateProcess && !$this->runTestInSeparateProcess, $this->preserveGlobalState);
        }
    }
    
    /**
     * Returns a builder object to create mock objects using a fluent interface.
     *
     * @psalm-template RealInstanceType of object
     *
     * @psalm-param class-string<RealInstanceType> $className
     *
     * @psalm-return MockBuilder<RealInstanceType>
     */
    public final function getMockBuilder(string $className) : MockBuilder {
        return new MockBuilder($this, $className);
    }
    
    /**
     * @internal This method is not covered by the backward compatibility promise for PHPUnit
     */
    public final function groups() : array {
        return $this->groups;
    }
    
    /**
     * @internal This method is not covered by the backward compatibility promise for PHPUnit
     */
    public final function setGroups(array $groups) : void {
        $this->groups = $groups;
    }
    
    /**
     * @internal This method is not covered by the backward compatibility promise for PHPUnit
     */
    public final function nameWithDataSet() : string {
        return $this->name . $this->dataSetAsString();
    }
    
    /**
     * @psalm-return non-empty-string
     *
     * @internal This method is not covered by the backward compatibility promise for PHPUnit
     */
    public final function name() : string {
        return $this->name;
    }
    
    /**
     * @internal This method is not covered by the backward compatibility promise for PHPUnit
     */
    public final function size() : TestSize {
        return (new Groups())->size(static::class, $this->name);
    }
    
    /**
     * @internal This method is not covered by the backward compatibility promise for PHPUnit
     */
    public final function hasUnexpectedOutput() : bool {
        if ($this->output === '') {
            return false;
        }
        if ($this->expectsOutput()) {
            return false;
        }
        return true;
    }
    
    /**
     * @internal This method is not covered by the backward compatibility promise for PHPUnit
     */
    public final function output() : string {
        if (!$this->outputBufferingActive) {
            return $this->output;
        }
        return (string) ob_get_contents();
    }
    
    /**
     * @internal This method is not covered by the backward compatibility promise for PHPUnit
     */
    public final function doesNotPerformAssertions() : bool {
        return $this->doesNotPerformAssertions;
    }
    
    /**
     * @internal This method is not covered by the backward compatibility promise for PHPUnit
     */
    public final function expectsOutput() : bool {
        return $this->hasExpectationOnOutput() || $this->outputRetrievedForAssertion;
    }
    
    /**
     * @internal This method is not covered by the backward compatibility promise for PHPUnit
     *
     * @deprecated
     *
     * @codeCoverageIgnore
     */
    public final function registerMockObjectsFromTestArgumentsRecursively() : void {
        $this->registerMockObjectsFromTestArgumentsRecursively = true;
    }
    
    /**
     * @throws Throwable
     *
     * @internal This method is not covered by the backward compatibility promise for PHPUnit
     */
    public final function runBare() : void {
        $emitter = Event\Facade::emitter();
        $emitter->testPreparationStarted($this->valueObjectForEvents());
        $this->snapshotGlobalState();
        $this->startOutputBuffering();
        clearstatcache();
        $hookMethods = (new HookMethods())->hookMethods(static::class);
        $hasMetRequirements = false;
        $this->numberOfAssertionsPerformed = 0;
        $currentWorkingDirectory = getcwd();
        try {
            $this->checkRequirements();
            $hasMetRequirements = true;
            if ($this->inIsolation) {
                // @codeCoverageIgnoreStart
                $this->invokeBeforeClassHookMethods($hookMethods, $emitter);
                // @codeCoverageIgnoreEnd
            }
            if (method_exists(static::class, $this->name) && MetadataRegistry::parser()->forClassAndMethod(static::class, $this->name)
                ->isDoesNotPerformAssertions()
                ->isNotEmpty()) {
                $this->doesNotPerformAssertions = true;
            }
            $this->invokeBeforeTestHookMethods($hookMethods, $emitter);
            $this->invokePreConditionHookMethods($hookMethods, $emitter);
            $emitter->testPrepared($this->valueObjectForEvents());
            $this->wasPrepared = true;
            $this->testResult = $this->runTest();
            $this->verifyMockObjects();
            $this->invokePostConditionHookMethods($hookMethods, $emitter);
            $this->status = TestStatus::success();
        } catch (IncompleteTest $e) {
            $this->status = TestStatus::incomplete($e->getMessage());
            $emitter->testMarkedAsIncomplete($this->valueObjectForEvents(), Event\Code\ThrowableBuilder::from($e));
        } catch (SkippedTest $e) {
            $this->status = TestStatus::skipped($e->getMessage());
            $emitter->testSkipped($this->valueObjectForEvents(), $e->getMessage());
        } catch (AssertionError|AssertionFailedError $e) {
            if (!$this->wasPrepared) {
                $this->wasPrepared = true;
                $emitter->testPreparationFailed($this->valueObjectForEvents());
            }
            $this->status = TestStatus::failure($e->getMessage());
            $emitter->testFailed($this->valueObjectForEvents(), Event\Code\ThrowableBuilder::from($e), Event\Code\ComparisonFailureBuilder::from($e));
        } catch (TimeoutException $e) {
            $this->status = TestStatus::risky($e->getMessage());
        } catch (Throwable $_e) {
            if ($this->isRegisteredFailure($_e)) {
                $this->status = TestStatus::failure($_e->getMessage());
                $emitter->testFailed($this->valueObjectForEvents(), Event\Code\ThrowableBuilder::from($_e), null);
            }
            else {
                $e = $this->transformException($_e);
                $this->status = TestStatus::error($e->getMessage());
                $emitter->testErrored($this->valueObjectForEvents(), Event\Code\ThrowableBuilder::from($e));
            }
        }
        $outputBufferingStopped = false;
        if (!isset($e) && $this->hasExpectationOnOutput() && $this->stopOutputBuffering()) {
            $outputBufferingStopped = true;
            $this->performAssertionsOnOutput();
        }
        if ($this->status
            ->isSuccess()) {
            $emitter->testPassed($this->valueObjectForEvents());
            if (!$this->usesDataProvider()) {
                PassedTests::instance()->testMethodPassed($this->valueObjectForEvents(), $this->testResult);
            }
        }
        try {
            $this->mockObjects = [];
        } catch (Throwable $t) {
            Event\Facade::emitter()->testErrored($this->valueObjectForEvents(), Event\Code\ThrowableBuilder::from($t));
        }
        // Tear down the fixture. An exception raised in tearDown() will be
        // caught and passed on when no exception was raised before.
        try {
            if ($hasMetRequirements) {
                $this->invokeAfterTestHookMethods($hookMethods, $emitter);
                if ($this->inIsolation) {
                    // @codeCoverageIgnoreStart
                    $this->invokeAfterClassHookMethods($hookMethods, $emitter);
                    // @codeCoverageIgnoreEnd
                }
            }
        } catch (AssertionError|AssertionFailedError $e) {
            $this->status = TestStatus::failure($e->getMessage());
            $emitter->testFailed($this->valueObjectForEvents(), Event\Code\ThrowableBuilder::from($e), Event\Code\ComparisonFailureBuilder::from($e));
        } catch (Throwable $exceptionRaisedDuringTearDown) {
            if (!isset($e)) {
                $this->status = TestStatus::error($exceptionRaisedDuringTearDown->getMessage());
                $e = $exceptionRaisedDuringTearDown;
                $emitter->testErrored($this->valueObjectForEvents(), Event\Code\ThrowableBuilder::from($exceptionRaisedDuringTearDown));
            }
        }
        if (!$outputBufferingStopped) {
            $this->stopOutputBuffering();
        }
        clearstatcache();
        if ($currentWorkingDirectory !== getcwd()) {
            chdir($currentWorkingDirectory);
        }
        $this->restoreGlobalState();
        $this->unregisterCustomComparators();
        $this->cleanupIniSettings();
        $this->cleanupLocaleSettings();
        libxml_clear_errors();
        $this->testValueObjectForEvents = null;
        if (isset($e)) {
            $this->onNotSuccessfulTest($e);
        }
    }
    
    /**
     * @psalm-param non-empty-string $name
     *
     * @internal This method is not covered by the backward compatibility promise for PHPUnit
     */
    public final function setName(string $name) : void {
        $this->name = $name;
        if (is_callable($this->sortId(), true)) {
            $this->providedTests = [
                new ExecutionOrderDependency($this->sortId()),
            ];
        }
    }
    
    /**
     * @psalm-param list<ExecutionOrderDependency> $dependencies
     *
     * @internal This method is not covered by the backward compatibility promise for PHPUnit
     */
    public final function setDependencies(array $dependencies) : void {
        $this->dependencies = $dependencies;
    }
    
    /**
     * @internal This method is not covered by the backward compatibility promise for PHPUnit
     *
     * @codeCoverageIgnore
     */
    public final function setDependencyInput(array $dependencyInput) : void {
        $this->dependencyInput = $dependencyInput;
    }
    
    /**
     * @internal This method is not covered by the backward compatibility promise for PHPUnit
     */
    public final function dependencyInput() : array {
        return $this->dependencyInput;
    }
    
    /**
     * @internal This method is not covered by the backward compatibility promise for PHPUnit
     */
    public final function hasDependencyInput() : bool {
        return !empty($this->dependencyInput);
    }
    
    /**
     * @internal This method is not covered by the backward compatibility promise for PHPUnit
     */
    public final function setBackupGlobals(bool $backupGlobals) : void {
        $this->backupGlobals = $backupGlobals;
    }
    
    /**
     * @internal This method is not covered by the backward compatibility promise for PHPUnit
     */
    public final function setBackupGlobalsExcludeList(array $backupGlobalsExcludeList) : void {
        $this->backupGlobalsExcludeList = $backupGlobalsExcludeList;
    }
    
    /**
     * @internal This method is not covered by the backward compatibility promise for PHPUnit
     */
    public final function setBackupStaticProperties(bool $backupStaticProperties) : void {
        $this->backupStaticProperties = $backupStaticProperties;
    }
    
    /**
     * @internal This method is not covered by the backward compatibility promise for PHPUnit
     */
    public final function setBackupStaticPropertiesExcludeList(array $backupStaticPropertiesExcludeList) : void {
        $this->backupStaticPropertiesExcludeList = $backupStaticPropertiesExcludeList;
    }
    
    /**
     * @internal This method is not covered by the backward compatibility promise for PHPUnit
     */
    public final function setRunTestInSeparateProcess(bool $runTestInSeparateProcess) : void {
        if ($this->runTestInSeparateProcess === null) {
            $this->runTestInSeparateProcess = $runTestInSeparateProcess;
        }
    }
    
    /**
     * @internal This method is not covered by the backward compatibility promise for PHPUnit
     */
    public final function setRunClassInSeparateProcess(bool $runClassInSeparateProcess) : void {
        $this->runClassInSeparateProcess = $runClassInSeparateProcess;
    }
    
    /**
     * @internal This method is not covered by the backward compatibility promise for PHPUnit
     */
    public final function setPreserveGlobalState(bool $preserveGlobalState) : void {
        $this->preserveGlobalState = $preserveGlobalState;
    }
    
    /**
     * @internal This method is not covered by the backward compatibility promise for PHPUnit
     *
     * @codeCoverageIgnore
     */
    public final function setInIsolation(bool $inIsolation) : void {
        $this->inIsolation = $inIsolation;
    }
    
    /**
     * @internal This method is not covered by the backward compatibility promise for PHPUnit
     *
     * @codeCoverageIgnore
     */
    public final function result() : mixed {
        return $this->testResult;
    }
    
    /**
     * @internal This method is not covered by the backward compatibility promise for PHPUnit
     */
    public final function setResult(mixed $result) : void {
        $this->testResult = $result;
    }
    
    /**
     * @internal This method is not covered by the backward compatibility promise for PHPUnit
     */
    public final function registerMockObject(MockObject $mockObject) : void {
        assert($mockObject instanceof MockObjectInternal);
        $this->mockObjects[] = $mockObject;
    }
    
    /**
     * @internal This method is not covered by the backward compatibility promise for PHPUnit
     */
    public final function addToAssertionCount(int $count) : void {
        $this->numberOfAssertionsPerformed += $count;
    }
    
    /**
     * @internal This method is not covered by the backward compatibility promise for PHPUnit
     */
    public final function numberOfAssertionsPerformed() : int {
        return $this->numberOfAssertionsPerformed;
    }
    
    /**
     * @internal This method is not covered by the backward compatibility promise for PHPUnit
     */
    public final function usesDataProvider() : bool {
        return !empty($this->data);
    }
    
    /**
     * @internal This method is not covered by the backward compatibility promise for PHPUnit
     */
    public final function dataName() : int|string {
        return $this->dataName;
    }
    
    /**
     * @internal This method is not covered by the backward compatibility promise for PHPUnit
     */
    public final function dataSetAsString() : string {
        $buffer = '';
        if (!empty($this->data)) {
            if (is_int($this->dataName)) {
                $buffer .= sprintf(' with data set #%d', $this->dataName);
            }
            else {
                $buffer .= sprintf(' with data set "%s"', $this->dataName);
            }
        }
        return $buffer;
    }
    
    /**
     * @internal This method is not covered by the backward compatibility promise for PHPUnit
     */
    public final function dataSetAsStringWithData() : string {
        if (empty($this->data)) {
            return '';
        }
        return $this->dataSetAsString() . sprintf(' (%s)', (new Exporter())->shortenedRecursiveExport($this->data));
    }
    
    /**
     * @internal This method is not covered by the backward compatibility promise for PHPUnit
     */
    public final function providedData() : array {
        return $this->data;
    }
    
    /**
     * @internal This method is not covered by the backward compatibility promise for PHPUnit
     */
    public final function sortId() : string {
        $id = $this->name;
        if (!str_contains($id, '::')) {
            $id = static::class . '::' . $id;
        }
        if ($this->usesDataProvider()) {
            $id .= $this->dataSetAsString();
        }
        return $id;
    }
    
    /**
     * @psalm-return list<ExecutionOrderDependency>
     *
     * @internal This method is not covered by the backward compatibility promise for PHPUnit
     */
    public final function provides() : array {
        return $this->providedTests;
    }
    
    /**
     * @psalm-return list<ExecutionOrderDependency>
     *
     * @internal This method is not covered by the backward compatibility promise for PHPUnit
     */
    public final function requires() : array {
        return $this->dependencies;
    }
    
    /**
     * @throws RuntimeException
     *
     * @internal This method is not covered by the backward compatibility promise for PHPUnit
     */
    public final function setData(int|string $dataName, array $data) : void {
        $this->dataName = $dataName;
        $this->data = $data;
        if (array_is_list($data)) {
            return;
        }
        try {
            $reflector = new ReflectionMethod($this, $this->name);
            $parameters = array_map(static fn(ReflectionParameter $parameter) => $parameter->name, $reflector->getParameters());
            foreach (array_keys($data) as $parameter) {
                if (is_string($parameter) && !in_array($parameter, $parameters, true)) {
                    Event\Facade::emitter()->testTriggeredPhpunitDeprecation($this->valueObjectForEvents(), sprintf('Providing invalid named argument $%s for method %s::%s() is deprecated and will not be supported in PHPUnit 11.0.', $parameter, $this::class, $this->name));
                }
            }
            // @codeCoverageIgnoreStart
        } catch (ReflectionException $e) {
            throw new RuntimeException($e->getMessage(), $e->getCode(), $e);
        }
        // @codeCoverageIgnoreEnd
    }
    
    /**
     * @internal This method is not covered by the backward compatibility promise for PHPUnit
     *
     * @throws MoreThanOneDataSetFromDataProviderException
     */
    public final function valueObjectForEvents() : Event\Code\TestMethod {
        if ($this->testValueObjectForEvents !== null) {
            return $this->testValueObjectForEvents;
        }
        $this->testValueObjectForEvents = Event\Code\TestMethodBuilder::fromTestCase($this);
        return $this->testValueObjectForEvents;
    }
    
    /**
     * @internal This method is not covered by the backward compatibility promise for PHPUnit
     */
    public final function wasPrepared() : bool {
        return $this->wasPrepared;
    }
    protected final function registerComparator(Comparator $comparator) : void {
        ComparatorFactory::getInstance()->register($comparator);
        Event\Facade::emitter()->testRegisteredComparator($comparator::class);
        $this->customComparators[] = $comparator;
    }
    
    /**
     * @psalm-param class-string $classOrInterface
     */
    protected final function registerFailureType(string $classOrInterface) : void {
        $this->failureTypes[$classOrInterface] = true;
    }
    
    /**
     * @throws AssertionFailedError
     * @throws Exception
     * @throws ExpectationFailedException
     * @throws Throwable
     *
     * @internal This method is not covered by the backward compatibility promise for PHPUnit
     */
    protected function runTest() : mixed {
        $testArguments = array_merge($this->data, $this->dependencyInput);
        $this->registerMockObjectsFromTestArguments($testArguments);
        try {
            $testResult = $this->{$this->name}(...array_values($testArguments));
        } catch (Throwable $exception) {
            if (!$this->shouldExceptionExpectationsBeVerified($exception)) {
                throw $exception;
            }
            $this->verifyExceptionExpectations($exception);
            return null;
        }
        $this->expectedExceptionWasNotRaised();
        return $testResult;
    }
    
    /**
     * This method is a wrapper for the ini_set() function that automatically
     * resets the modified php.ini setting to its original value after the
     * test is run.
     *
     * @throws Exception
     *
     * @deprecated https://github.com/sebastianbergmann/phpunit/issues/5214
     *
     * @codeCoverageIgnore
     */
    protected function iniSet(string $varName, string $newValue) : void {
        $currentValue = ini_set($varName, $newValue);
        if ($currentValue !== false) {
            $this->iniSettings[$varName] = $currentValue;
        }
        else {
            throw new Exception(sprintf('INI setting "%s" could not be set to "%s".', $varName, $newValue));
        }
    }
    
    /**
     * This method is a wrapper for the setlocale() function that automatically
     * resets the locale to its original value after the test is run.
     *
     * @throws Exception
     *
     * @deprecated https://github.com/sebastianbergmann/phpunit/issues/5216
     *
     * @codeCoverageIgnore
     */
    protected function setLocale(mixed ...$arguments) : void {
        if (count($arguments) < 2) {
            throw new Exception();
        }
        [
            $category,
            $locale,
        ] = $arguments;
        if (!in_array($category, self::LOCALE_CATEGORIES, true)) {
            throw new Exception();
        }
        if (!is_array($locale) && !is_string($locale)) {
            throw new Exception();
        }
        $this->locale[$category] = setlocale($category, '0');
        $result = setlocale(...$arguments);
        if ($result === false) {
            throw new Exception('The locale functionality is not implemented on your platform, ' . 'the specified locale does not exist or the category name is ' . 'invalid.');
        }
    }
    
    /**
     * Creates a mock object for the specified interface or class.
     *
     * @psalm-template RealInstanceType of object
     *
     * @psalm-param class-string<RealInstanceType> $originalClassName
     *
     * @psalm-return MockObject&RealInstanceType
     *
     * @throws InvalidArgumentException
     * @throws MockObjectException
     * @throws NoPreviousThrowableException
     */
    protected function createMock(string $originalClassName) : MockObject {
        $mock = (new MockGenerator())->testDouble($originalClassName, true, callOriginalConstructor: false, callOriginalClone: false, cloneArguments: false, allowMockingUnknownTypes: false);
        assert($mock instanceof $originalClassName);
        assert($mock instanceof MockObject);
        $this->registerMockObject($mock);
        Event\Facade::emitter()->testCreatedMockObject($originalClassName);
        return $mock;
    }
    
    /**
     * @psalm-param list<class-string> $interfaces
     *
     * @throws MockObjectException
     */
    protected function createMockForIntersectionOfInterfaces(array $interfaces) : MockObject {
        $mock = (new MockGenerator())->testDoubleForInterfaceIntersection($interfaces, true);
        assert($mock instanceof MockObject);
        $this->registerMockObject($mock);
        Event\Facade::emitter()->testCreatedMockObjectForIntersectionOfInterfaces($interfaces);
        return $mock;
    }
    
    /**
     * Creates (and configures) a mock object for the specified interface or class.
     *
     * @psalm-template RealInstanceType of object
     *
     * @psalm-param class-string<RealInstanceType> $originalClassName
     *
     * @psalm-return MockObject&RealInstanceType
     *
     * @throws InvalidArgumentException
     * @throws MockObjectException
     * @throws NoPreviousThrowableException
     */
    protected function createConfiguredMock(string $originalClassName, array $configuration) : MockObject {
        $o = $this->createMock($originalClassName);
        foreach ($configuration as $method => $return) {
            $o->method($method)
                ->willReturn($return);
        }
        return $o;
    }
    
    /**
     * Creates a partial mock object for the specified interface or class.
     *
     * @psalm-param list<non-empty-string> $methods
     *
     * @psalm-template RealInstanceType of object
     *
     * @psalm-param class-string<RealInstanceType> $originalClassName
     *
     * @psalm-return MockObject&RealInstanceType
     *
     * @throws InvalidArgumentException
     * @throws MockObjectException
     */
    protected function createPartialMock(string $originalClassName, array $methods) : MockObject {
        $partialMock = $this->getMockBuilder($originalClassName)
            ->disableOriginalConstructor()
            ->disableOriginalClone()
            ->disableArgumentCloning()
            ->disallowMockingUnknownTypes()
            ->onlyMethods($methods)
            ->getMock();
        Event\Facade::emitter()->testCreatedPartialMockObject($originalClassName, ...$methods);
        return $partialMock;
    }
    
    /**
     * Creates a test proxy for the specified class.
     *
     * @psalm-template RealInstanceType of object
     *
     * @psalm-param class-string<RealInstanceType> $originalClassName
     *
     * @psalm-return MockObject&RealInstanceType
     *
     * @throws InvalidArgumentException
     * @throws MockObjectException
     *
     * @deprecated https://github.com/sebastianbergmann/phpunit/issues/5240
     */
    protected function createTestProxy(string $originalClassName, array $constructorArguments = []) : MockObject {
        $testProxy = $this->getMockBuilder($originalClassName)
            ->setConstructorArgs($constructorArguments)
            ->enableProxyingToOriginalMethods()
            ->getMock();
        Event\Facade::emitter()->testCreatedTestProxy($originalClassName, $constructorArguments);
        return $testProxy;
    }
    
    /**
     * Creates a mock object for the specified abstract class with all abstract
     * methods of the class mocked. Concrete methods are not mocked by default.
     * To mock concrete methods, use the 7th parameter ($mockedMethods).
     *
     * @psalm-template RealInstanceType of object
     *
     * @psalm-param class-string<RealInstanceType> $originalClassName
     *
     * @psalm-return MockObject&RealInstanceType
     *
     * @throws InvalidArgumentException
     * @throws MockObjectException
     *
     * @deprecated https://github.com/sebastianbergmann/phpunit/issues/5241
     */
    protected function getMockForAbstractClass(string $originalClassName, array $arguments = [], string $mockClassName = '', bool $callOriginalConstructor = true, bool $callOriginalClone = true, bool $callAutoload = true, array $mockedMethods = [], bool $cloneArguments = false) : MockObject {
        $mockObject = (new MockGenerator())->mockObjectForAbstractClass($originalClassName, $arguments, $mockClassName, $callOriginalConstructor, $callOriginalClone, $callAutoload, $mockedMethods, $cloneArguments);
        $this->registerMockObject($mockObject);
        Event\Facade::emitter()->testCreatedMockObjectForAbstractClass($originalClassName);
        assert($mockObject instanceof $originalClassName);
        assert($mockObject instanceof MockObject);
        return $mockObject;
    }
    
    /**
     * Creates a mock object based on the given WSDL file.
     *
     * @throws MockObjectException
     *
     * @deprecated https://github.com/sebastianbergmann/phpunit/issues/5242
     */
    protected function getMockFromWsdl(string $wsdlFile, string $originalClassName = '', string $mockClassName = '', array $methods = [], bool $callOriginalConstructor = true, array $options = []) : MockObject {
        if ($originalClassName === '') {
            $fileName = pathinfo(basename(parse_url($wsdlFile, PHP_URL_PATH)), PATHINFO_FILENAME);
            $originalClassName = preg_replace('/\\W/', '', $fileName);
        }
        if (!class_exists($originalClassName)) {
            eval((new MockGenerator())->generateClassFromWsdl($wsdlFile, $originalClassName, $methods, $options));
        }
        $mockObject = (new MockGenerator())->testDouble($originalClassName, true, $methods, [
            '',
            $options,
        ], $mockClassName, $callOriginalConstructor, false, false);
        Event\Facade::emitter()->testCreatedMockObjectFromWsdl($wsdlFile, $originalClassName, $mockClassName, $methods, $callOriginalConstructor, $options);
        assert($mockObject instanceof MockObject);
        $this->registerMockObject($mockObject);
        return $mockObject;
    }
    
    /**
     * Creates a mock object for the specified trait with all abstract methods
     * of the trait mocked. Concrete methods to mock can be specified with the
     * `$mockedMethods` parameter.
     *
     * @psalm-param trait-string $traitName
     *
     * @throws InvalidArgumentException
     * @throws MockObjectException
     *
     * @deprecated https://github.com/sebastianbergmann/phpunit/issues/5243
     */
    protected function getMockForTrait(string $traitName, array $arguments = [], string $mockClassName = '', bool $callOriginalConstructor = true, bool $callOriginalClone = true, bool $callAutoload = true, array $mockedMethods = [], bool $cloneArguments = false) : MockObject {
        $mockObject = (new MockGenerator())->mockObjectForTrait($traitName, $arguments, $mockClassName, $callOriginalConstructor, $callOriginalClone, $callAutoload, $mockedMethods, $cloneArguments);
        $this->registerMockObject($mockObject);
        Event\Facade::emitter()->testCreatedMockObjectForTrait($traitName);
        return $mockObject;
    }
    
    /**
     * Creates an object that uses the specified trait.
     *
     * @psalm-param trait-string $traitName
     *
     * @throws MockObjectException
     *
     * @deprecated https://github.com/sebastianbergmann/phpunit/issues/5244
     */
    protected function getObjectForTrait(string $traitName, array $arguments = [], string $traitClassName = '', bool $callOriginalConstructor = true, bool $callOriginalClone = true, bool $callAutoload = true) : object {
        return (new MockGenerator())->objectForTrait($traitName, $traitClassName, $callAutoload, $callOriginalConstructor, $arguments);
    }
    protected function transformException(Throwable $t) : Throwable {
        return $t;
    }
    
    /**
     * This method is called when a test method did not execute successfully.
     *
     * @throws Throwable
     */
    protected function onNotSuccessfulTest(Throwable $t) : never {
        throw $t;
    }
    
    /**
     * @throws Throwable
     */
    private function verifyMockObjects() : void {
        foreach ($this->mockObjects as $mockObject) {
            if ($mockObject->__phpunit_hasMatchers()) {
                $this->numberOfAssertionsPerformed++;
            }
            $mockObject->__phpunit_verify($this->shouldInvocationMockerBeReset($mockObject));
        }
    }
    
    /**
     * @throws SkippedTest
     */
    private function checkRequirements() : void {
        if (!$this->name || !method_exists($this, $this->name)) {
            return;
        }
        $missingRequirements = (new Requirements())->requirementsNotSatisfiedFor(static::class, $this->name);
        if (!empty($missingRequirements)) {
            $this->markTestSkipped(implode(PHP_EOL, $missingRequirements));
        }
    }
    private function handleDependencies() : bool {
        if ([] === $this->dependencies || $this->inIsolation) {
            return true;
        }
        $passedTests = PassedTests::instance();
        foreach ($this->dependencies as $dependency) {
            if (!$dependency->isValid()) {
                $this->markErrorForInvalidDependency();
                return false;
            }
            if ($dependency->targetIsClass()) {
                $dependencyClassName = $dependency->getTargetClassName();
                if (!class_exists($dependencyClassName)) {
                    $this->markErrorForInvalidDependency($dependency);
                    return false;
                }
                if (!$passedTests->hasTestClassPassed($dependencyClassName)) {
                    $this->markSkippedForMissingDependency($dependency);
                    return false;
                }
                continue;
            }
            $dependencyTarget = $dependency->getTarget();
            if (!$passedTests->hasTestMethodPassed($dependencyTarget)) {
                if (!$this->isCallableTestMethod($dependencyTarget)) {
                    $this->markErrorForInvalidDependency($dependency);
                }
                else {
                    $this->markSkippedForMissingDependency($dependency);
                }
                return false;
            }
            if ($passedTests->isGreaterThan($dependencyTarget, $this->size())) {
                Event\Facade::emitter()->testConsideredRisky($this->valueObjectForEvents(), 'This test depends on a test that is larger than itself');
                return false;
            }
            $returnValue = $passedTests->returnValue($dependencyTarget);
            if ($dependency->deepClone()) {
                $deepCopy = new DeepCopy();
                $deepCopy->skipUncloneable(false);
                $this->dependencyInput[$dependencyTarget] = $deepCopy->copy($returnValue);
            }
            elseif ($dependency->shallowClone()) {
                $this->dependencyInput[$dependencyTarget] = clone $returnValue;
            }
            else {
                $this->dependencyInput[$dependencyTarget] = $returnValue;
            }
        }
        $this->testValueObjectForEvents = null;
        return true;
    }
    
    /**
     * @throws Exception
     * @throws MoreThanOneDataSetFromDataProviderException
     * @throws NoPreviousThrowableException
     */
    private function markErrorForInvalidDependency(?ExecutionOrderDependency $dependency = null) : void {
        $message = 'This test has an invalid dependency';
        if ($dependency !== null) {
            $message = sprintf('This test depends on "%s" which does not exist', $dependency->targetIsClass() ? $dependency->getTargetClassName() : $dependency->getTarget());
        }
        $exception = new InvalidDependencyException($message);
        Event\Facade::emitter()->testErrored($this->valueObjectForEvents(), Event\Code\ThrowableBuilder::from($exception));
        $this->status = TestStatus::error($message);
    }
    
    /**
     * @throws MoreThanOneDataSetFromDataProviderException
     */
    private function markSkippedForMissingDependency(ExecutionOrderDependency $dependency) : void {
        $message = sprintf('This test depends on "%s" to pass', $dependency->getTarget());
        Event\Facade::emitter()->testSkipped($this->valueObjectForEvents(), $message);
        $this->status = TestStatus::skipped($message);
    }
    private function startOutputBuffering() : void {
        ob_start();
        $this->outputBufferingActive = true;
        $this->outputBufferingLevel = ob_get_level();
    }
    
    /**
     * @throws MoreThanOneDataSetFromDataProviderException
     */
    private function stopOutputBuffering() : bool {
        $bufferingLevel = ob_get_level();
        if ($bufferingLevel !== $this->outputBufferingLevel) {
            if ($bufferingLevel > $this->outputBufferingLevel) {
                $message = 'Test code or tested code did not close its own output buffers';
            }
            else {
                $message = 'Test code or tested code closed output buffers other than its own';
            }
            while (ob_get_level() >= $this->outputBufferingLevel) {
                ob_end_clean();
            }
            Event\Facade::emitter()->testConsideredRisky($this->valueObjectForEvents(), $message);
            $this->status = TestStatus::risky($message);
            return false;
        }
        $this->output = ob_get_clean();
        $this->outputBufferingActive = false;
        $this->outputBufferingLevel = ob_get_level();
        return true;
    }
    private function snapshotGlobalState() : void {
        if ($this->runTestInSeparateProcess || $this->inIsolation || !$this->backupGlobals && !$this->backupStaticProperties) {
            return;
        }
        $snapshot = $this->createGlobalStateSnapshot($this->backupGlobals === true);
        $this->snapshot = $snapshot;
    }
    
    /**
     * @throws MoreThanOneDataSetFromDataProviderException
     */
    private function restoreGlobalState() : void {
        if (!$this->snapshot instanceof Snapshot) {
            return;
        }
        if (ConfigurationRegistry::get()->beStrictAboutChangesToGlobalState()) {
            $this->compareGlobalStateSnapshots($this->snapshot, $this->createGlobalStateSnapshot($this->backupGlobals === true));
        }
        $restorer = new Restorer();
        if ($this->backupGlobals) {
            $restorer->restoreGlobalVariables($this->snapshot);
        }
        if ($this->backupStaticProperties) {
            $restorer->restoreStaticProperties($this->snapshot);
        }
        $this->snapshot = null;
    }
    private function createGlobalStateSnapshot(bool $backupGlobals) : Snapshot {
        $excludeList = new GlobalStateExcludeList();
        foreach ($this->backupGlobalsExcludeList as $globalVariable) {
            $excludeList->addGlobalVariable($globalVariable);
        }
        if (!defined('PHPUNIT_TESTSUITE')) {
            $excludeList->addClassNamePrefix('PHPUnit');
            $excludeList->addClassNamePrefix('SebastianBergmann\\CodeCoverage');
            $excludeList->addClassNamePrefix('SebastianBergmann\\FileIterator');
            $excludeList->addClassNamePrefix('SebastianBergmann\\Invoker');
            $excludeList->addClassNamePrefix('SebastianBergmann\\Template');
            $excludeList->addClassNamePrefix('SebastianBergmann\\Timer');
            $excludeList->addStaticProperty(ComparatorFactory::class, 'instance');
            foreach ($this->backupStaticPropertiesExcludeList as $class => $properties) {
                foreach ($properties as $property) {
                    $excludeList->addStaticProperty($class, $property);
                }
            }
        }
        return new Snapshot($excludeList, $backupGlobals, (bool) $this->backupStaticProperties, false, false, false, false, false, false, false);
    }
    
    /**
     * @throws MoreThanOneDataSetFromDataProviderException
     */
    private function compareGlobalStateSnapshots(Snapshot $before, Snapshot $after) : void {
        $backupGlobals = $this->backupGlobals === null || $this->backupGlobals;
        if ($backupGlobals) {
            $this->compareGlobalStateSnapshotPart($before->globalVariables(), $after->globalVariables(), "--- Global variables before the test\n+++ Global variables after the test\n");
            $this->compareGlobalStateSnapshotPart($before->superGlobalVariables(), $after->superGlobalVariables(), "--- Super-global variables before the test\n+++ Super-global variables after the test\n");
        }
        if ($this->backupStaticProperties) {
            $this->compareGlobalStateSnapshotPart($before->staticProperties(), $after->staticProperties(), "--- Static properties before the test\n+++ Static properties after the test\n");
        }
    }
    
    /**
     * @throws MoreThanOneDataSetFromDataProviderException
     */
    private function compareGlobalStateSnapshotPart(array $before, array $after, string $header) : void {
        if ($before != $after) {
            $differ = new Differ(new UnifiedDiffOutputBuilder($header));
            $exporter = new Exporter();
            Event\Facade::emitter()->testConsideredRisky($this->valueObjectForEvents(), 'This test modified global state but was not expected to do so' . PHP_EOL . trim($differ->diff($exporter->export($before), $exporter->export($after))));
        }
    }
    private function shouldInvocationMockerBeReset(MockObject $mock) : bool {
        $enumerator = new Enumerator();
        if (in_array($mock, $enumerator->enumerate($this->dependencyInput), true)) {
            return false;
        }
        if (!is_array($this->testResult) && !is_object($this->testResult)) {
            return true;
        }
        return !in_array($mock, $enumerator->enumerate($this->testResult), true);
    }
    
    /**
     * @deprecated
     */
    private function registerMockObjectsFromTestArguments(array $testArguments, Context $context = new Context()) : void {
        if ($this->registerMockObjectsFromTestArgumentsRecursively) {
            foreach ((new Enumerator())->enumerate($testArguments) as $object) {
                if ($object instanceof MockObject) {
                    $this->registerMockObject($object);
                }
            }
        }
        else {
            foreach ($testArguments as &$testArgument) {
                if ($testArgument instanceof MockObject) {
                    $testArgument = Cloner::clone($testArgument);
                    $this->registerMockObject($testArgument);
                }
                elseif (is_array($testArgument) && !$context->contains($testArgument)) {
                    $testArgumentCopy = $testArgument;
                    $context->add($testArgument);
                    $this->registerMockObjectsFromTestArguments($testArgumentCopy, $context);
                }
            }
        }
    }
    private function unregisterCustomComparators() : void {
        $factory = ComparatorFactory::getInstance();
        foreach ($this->customComparators as $comparator) {
            $factory->unregister($comparator);
        }
        $this->customComparators = [];
    }
    private function cleanupIniSettings() : void {
        foreach ($this->iniSettings as $varName => $oldValue) {
            ini_set($varName, $oldValue);
        }
        $this->iniSettings = [];
    }
    private function cleanupLocaleSettings() : void {
        foreach ($this->locale as $category => $locale) {
            setlocale($category, $locale);
        }
        $this->locale = [];
    }
    
    /**
     * @throws Exception
     */
    private function shouldExceptionExpectationsBeVerified(Throwable $throwable) : bool {
        $result = false;
        if ($this->expectedException !== null || $this->expectedExceptionCode !== null || $this->expectedExceptionMessage !== null || $this->expectedExceptionMessageRegExp !== null) {
            $result = true;
        }
        if ($throwable instanceof Exception) {
            $result = false;
        }
        if (is_string($this->expectedException)) {
            try {
                $reflector = new ReflectionClass($this->expectedException);
                // @codeCoverageIgnoreStart
            } catch (ReflectionException $e) {
                throw new Exception($e->getMessage(), $e->getCode(), $e);
            }
            // @codeCoverageIgnoreEnd
            if ($this->expectedException === 'PHPUnit\\Framework\\Exception' || $this->expectedException === '\\PHPUnit\\Framework\\Exception' || $reflector->isSubclassOf(Exception::class)) {
                $result = true;
            }
        }
        return $result;
    }
    private function shouldRunInSeparateProcess() : bool {
        if ($this->inIsolation) {
            return false;
        }
        if ($this->runTestInSeparateProcess) {
            return true;
        }
        if ($this->runClassInSeparateProcess) {
            return true;
        }
        return ConfigurationRegistry::get()->processIsolation();
    }
    private function isCallableTestMethod(string $dependency) : bool {
        [
            $className,
            $methodName,
        ] = explode('::', $dependency);
        if (!class_exists($className)) {
            return false;
        }
        $class = new ReflectionClass($className);
        if (!$class->isSubclassOf(__CLASS__)) {
            return false;
        }
        if (!$class->hasMethod($methodName)) {
            return false;
        }
        return TestUtil::isTestMethod($class->getMethod($methodName));
    }
    
    /**
     * @throws Exception
     * @throws ExpectationFailedException
     * @throws MoreThanOneDataSetFromDataProviderException
     * @throws NoPreviousThrowableException
     */
    private function performAssertionsOnOutput() : void {
        try {
            if ($this->outputExpectedRegex !== null) {
                $this->assertMatchesRegularExpression($this->outputExpectedRegex, $this->output);
            }
            elseif ($this->outputExpectedString !== null) {
                $this->assertSame($this->outputExpectedString, $this->output);
            }
        } catch (ExpectationFailedException $e) {
            $this->status = TestStatus::failure($e->getMessage());
            Event\Facade::emitter()->testFailed($this->valueObjectForEvents(), Event\Code\ThrowableBuilder::from($e), Event\Code\ComparisonFailureBuilder::from($e));
            throw $e;
        }
    }
    
    /**
     * @throws Throwable
     *
     * @codeCoverageIgnore
     */
    private function invokeBeforeClassHookMethods(array $hookMethods, Event\Emitter $emitter) : void {
        $this->invokeHookMethods($hookMethods['beforeClass'], $emitter, 'testBeforeFirstTestMethodCalled', 'testBeforeFirstTestMethodFinished');
    }
    
    /**
     * @throws Throwable
     */
    private function invokeBeforeTestHookMethods(array $hookMethods, Event\Emitter $emitter) : void {
        $this->invokeHookMethods($hookMethods['before'], $emitter, 'testBeforeTestMethodCalled', 'testBeforeTestMethodFinished');
    }
    
    /**
     * @throws Throwable
     */
    private function invokePreConditionHookMethods(array $hookMethods, Event\Emitter $emitter) : void {
        $this->invokeHookMethods($hookMethods['preCondition'], $emitter, 'testPreConditionCalled', 'testPreConditionFinished');
    }
    
    /**
     * @throws Throwable
     */
    private function invokePostConditionHookMethods(array $hookMethods, Event\Emitter $emitter) : void {
        $this->invokeHookMethods($hookMethods['postCondition'], $emitter, 'testPostConditionCalled', 'testPostConditionFinished');
    }
    
    /**
     * @throws Throwable
     */
    private function invokeAfterTestHookMethods(array $hookMethods, Event\Emitter $emitter) : void {
        $this->invokeHookMethods($hookMethods['after'], $emitter, 'testAfterTestMethodCalled', 'testAfterTestMethodFinished');
    }
    
    /**
     * @throws Throwable
     *
     * @codeCoverageIgnore
     */
    private function invokeAfterClassHookMethods(array $hookMethods, Event\Emitter $emitter) : void {
        $this->invokeHookMethods($hookMethods['afterClass'], $emitter, 'testAfterLastTestMethodCalled', 'testAfterLastTestMethodFinished');
    }
    
    /**
     * @psalm-param list<non-empty-string> $hookMethods
     * @psalm-param 'testBeforeFirstTestMethodCalled'|'testBeforeTestMethodCalled'|'testPreConditionCalled'|'testPostConditionCalled'|'testAfterTestMethodCalled'|'testAfterLastTestMethodCalled' $calledMethod
     * @psalm-param 'testBeforeFirstTestMethodFinished'|'testBeforeTestMethodFinished'|'testPreConditionFinished'|'testPostConditionFinished'|'testAfterTestMethodFinished'|'testAfterLastTestMethodFinished' $finishedMethod
     *
     * @throws Throwable
     */
    private function invokeHookMethods(array $hookMethods, Event\Emitter $emitter, string $calledMethod, string $finishedMethod) : void {
        $methodsInvoked = [];
        foreach ($hookMethods as $methodName) {
            if ($this->methodDoesNotExistOrIsDeclaredInTestCase($methodName)) {
                continue;
            }
            try {
                $this->{$methodName}();
            } catch (Throwable $t) {
            }
            $methodInvoked = new Event\Code\ClassMethod(static::class, $methodName);
            $emitter->{$calledMethod}(static::class, $methodInvoked);
            $methodsInvoked[] = $methodInvoked;
            if (isset($t)) {
                break;
            }
        }
        if (!empty($methodsInvoked)) {
            $emitter->{$finishedMethod}(static::class, ...$methodsInvoked);
        }
        if (isset($t)) {
            throw $t;
        }
    }
    private function methodDoesNotExistOrIsDeclaredInTestCase(string $methodName) : bool {
        $reflector = new ReflectionObject($this);
        return !$reflector->hasMethod($methodName) || $reflector->getMethod($methodName)
            ->getDeclaringClass()
            ->getName() === self::class;
    }
    
    /**
     * @throws ExpectationFailedException
     */
    private function verifyExceptionExpectations(\Exception|Throwable $exception) : void {
        if ($this->expectedException !== null) {
            $this->assertThat($exception, new ExceptionConstraint($this->expectedException));
        }
        if ($this->expectedExceptionMessage !== null) {
            $this->assertThat($exception->getMessage(), new ExceptionMessageIsOrContains($this->expectedExceptionMessage));
        }
        if ($this->expectedExceptionMessageRegExp !== null) {
            $this->assertThat($exception->getMessage(), new ExceptionMessageMatchesRegularExpression($this->expectedExceptionMessageRegExp));
        }
        if ($this->expectedExceptionCode !== null) {
            $this->assertThat($exception->getCode(), new ExceptionCode($this->expectedExceptionCode));
        }
    }
    
    /**
     * @throws AssertionFailedError
     */
    private function expectedExceptionWasNotRaised() : void {
        if ($this->expectedException !== null) {
            $this->assertThat(null, new ExceptionConstraint($this->expectedException));
        }
        elseif ($this->expectedExceptionMessage !== null) {
            $this->numberOfAssertionsPerformed++;
            throw new AssertionFailedError(sprintf('Failed asserting that exception with message "%s" is thrown', $this->expectedExceptionMessage));
        }
        elseif ($this->expectedExceptionMessageRegExp !== null) {
            $this->numberOfAssertionsPerformed++;
            throw new AssertionFailedError(sprintf('Failed asserting that exception with message matching "%s" is thrown', $this->expectedExceptionMessageRegExp));
        }
        elseif ($this->expectedExceptionCode !== null) {
            $this->numberOfAssertionsPerformed++;
            throw new AssertionFailedError(sprintf('Failed asserting that exception with code "%s" is thrown', $this->expectedExceptionCode));
        }
    }
    private function isRegisteredFailure(Throwable $t) : bool {
        foreach (array_keys($this->failureTypes) as $failureType) {
            if ($t instanceof $failureType) {
                return true;
            }
        }
        return false;
    }
    
    /**
     * @internal This method is not covered by the backward compatibility promise for PHPUnit
     */
    private function hasExpectationOnOutput() : bool {
        return is_string($this->outputExpectedString) || is_string($this->outputExpectedRegex);
    }
    private function requirementsNotSatisfied() : bool {
        return (new Requirements())->requirementsNotSatisfiedFor(static::class, $this->name) !== [];
    }
    
    /**
     * Creates a test stub for the specified interface or class.
     *
     * @psalm-template RealInstanceType of object
     *
     * @psalm-param class-string<RealInstanceType> $originalClassName
     *
     * @psalm-return Stub&RealInstanceType
     *
     * @throws InvalidArgumentException
     * @throws MockObjectException
     * @throws NoPreviousThrowableException
     */
    protected static function createStub(string $originalClassName) : Stub {
        $stub = (new MockGenerator())->testDouble($originalClassName, true, callOriginalConstructor: false, callOriginalClone: false, cloneArguments: false, allowMockingUnknownTypes: false);
        Event\Facade::emitter()->testCreatedStub($originalClassName);
        assert($stub instanceof $originalClassName);
        assert($stub instanceof Stub);
        return $stub;
    }
    
    /**
     * @psalm-param list<class-string> $interfaces
     *
     * @throws MockObjectException
     */
    protected static function createStubForIntersectionOfInterfaces(array $interfaces) : Stub {
        $stub = (new MockGenerator())->testDoubleForInterfaceIntersection($interfaces, true);
        Event\Facade::emitter()->testCreatedStubForIntersectionOfInterfaces($interfaces);
        return $stub;
    }
    
    /**
     * Creates (and configures) a test stub for the specified interface or class.
     *
     * @psalm-template RealInstanceType of object
     *
     * @psalm-param class-string<RealInstanceType> $originalClassName
     *
     * @psalm-return Stub&RealInstanceType
     *
     * @throws InvalidArgumentException
     * @throws MockObjectException
     * @throws NoPreviousThrowableException
     */
    protected static final function createConfiguredStub(string $originalClassName, array $configuration) : Stub {
        $o = self::createStub($originalClassName);
        foreach ($configuration as $method => $return) {
            $o->method($method)
                ->willReturn($return);
        }
        return $o;
    }

}

Classes

Title Deprecated Summary
TestCase @no-named-arguments Parameter names are not covered by the backward compatibility promise for PHPUnit

API Navigation

  • Drupal Core 11.1.x
  • Topics
  • Classes
  • Functions
  • Constants
  • Globals
  • Files
  • Namespaces
  • Deprecated
  • Services
RSS feed
Powered by Drupal