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

Breadcrumb

  1. Drupal Core 11.1.x

TestSuite.php

Same filename in this branch
  1. 11.1.x vendor/phpunit/phpunit/src/TextUI/Configuration/Value/TestSuite.php
  2. 11.1.x vendor/phpunit/phpunit/src/Event/Value/TestSuite/TestSuite.php

Namespace

PHPUnit\Framework

File

vendor/phpunit/phpunit/src/Framework/TestSuite.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 PHP_EOL;
use function array_keys;
use function array_map;
use function array_pop;
use function array_reverse;
use function assert;
use function call_user_func;
use function class_exists;
use function count;
use function implode;
use function is_callable;
use function is_file;
use function is_subclass_of;
use function sprintf;
use function str_ends_with;
use function str_starts_with;
use function trim;
use Iterator;
use IteratorAggregate;
use PHPUnit\Event;
use PHPUnit\Event\Code\TestMethod;
use PHPUnit\Event\NoPreviousThrowableException;
use PHPUnit\Metadata\Api\Dependencies;
use PHPUnit\Metadata\Api\Groups;
use PHPUnit\Metadata\Api\HookMethods;
use PHPUnit\Metadata\Api\Requirements;
use PHPUnit\Metadata\MetadataCollection;
use PHPUnit\Runner\Exception as RunnerException;
use PHPUnit\Runner\Filter\Factory;
use PHPUnit\Runner\PhptTestCase;
use PHPUnit\Runner\TestSuiteLoader;
use PHPUnit\TestRunner\TestResult\Facade as TestResultFacade;
use PHPUnit\Util\Filter;
use PHPUnit\Util\Reflection;
use PHPUnit\Util\Test as TestUtil;
use ReflectionClass;
use ReflectionMethod;
use SebastianBergmann\CodeCoverage\InvalidArgumentException;
use SebastianBergmann\CodeCoverage\UnintentionallyCoveredCodeException;
use Throwable;

/**
 * @template-implements IteratorAggregate<int, Test>
 *
 * @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
 */
class TestSuite implements IteratorAggregate, Reorderable, SelfDescribing, Test {
    
    /**
     * @psalm-var non-empty-string
     */
    private string $name;
    
    /**
     * @psalm-var array<string,list<Test>>
     */
    private array $groups = [];
    
    /**
     * @psalm-var ?list<ExecutionOrderDependency>
     */
    private ?array $requiredTests = null;
    
    /**
     * @psalm-var list<Test>
     */
    private array $tests = [];
    
    /**
     * @psalm-var ?list<ExecutionOrderDependency>
     */
    private ?array $providedTests = null;
    private ?Factory $iteratorFilter = null;
    private bool $wasRun = false;
    
    /**
     * @psalm-param non-empty-string $name
     */
    public static function empty(string $name) : static {
        return new static($name);
    }
    
    /**
     * @psalm-param class-string $className
     */
    public static function fromClassName(string $className) : static {
        assert(class_exists($className));
        $class = new ReflectionClass($className);
        return static::fromClassReflector($class);
    }
    public static function fromClassReflector(ReflectionClass $class) : static {
        $testSuite = new static($class->getName());
        $constructor = $class->getConstructor();
        if ($constructor !== null && !$constructor->isPublic()) {
            Event\Facade::emitter()->testRunnerTriggeredWarning(sprintf('Class "%s" has no public constructor.', $class->getName()));
            return $testSuite;
        }
        foreach (Reflection::publicMethodsInTestClass($class) as $method) {
            if ($method->getDeclaringClass()
                ->getName() === Assert::class) {
                continue;
            }
            if ($method->getDeclaringClass()
                ->getName() === TestCase::class) {
                continue;
            }
            if (!TestUtil::isTestMethod($method)) {
                continue;
            }
            $testSuite->addTestMethod($class, $method);
        }
        if ($testSuite->isEmpty()) {
            Event\Facade::emitter()->testRunnerTriggeredWarning(sprintf('No tests found in class "%s".', $class->getName()));
        }
        return $testSuite;
    }
    
    /**
     * @psalm-param non-empty-string $name
     */
    private final function __construct(string $name) {
        $this->name = $name;
    }
    
    /**
     * Returns a string representation of the test suite.
     */
    public function toString() : string {
        return $this->name();
    }
    
    /**
     * Adds a test to the suite.
     */
    public function addTest(Test $test, array $groups = []) : void {
        $class = new ReflectionClass($test);
        if ($class->isAbstract()) {
            return;
        }
        $this->tests[] = $test;
        $this->clearCaches();
        if ($test instanceof self && empty($groups)) {
            $groups = $test->groups();
        }
        if ($this->containsOnlyVirtualGroups($groups)) {
            $groups[] = 'default';
        }
        foreach ($groups as $group) {
            if (!isset($this->groups[$group])) {
                $this->groups[$group] = [
                    $test,
                ];
            }
            else {
                $this->groups[$group][] = $test;
            }
        }
        if ($test instanceof TestCase) {
            $test->setGroups($groups);
        }
    }
    
    /**
     * Adds the tests from the given class to the suite.
     *
     * @throws Exception
     */
    public function addTestSuite(ReflectionClass $testClass) : void {
        if ($testClass->isAbstract()) {
            throw new Exception(sprintf('Class %s is abstract', $testClass->getName()));
        }
        if (!$testClass->isSubclassOf(TestCase::class)) {
            throw new Exception(sprintf('Class %s is not a subclass of %s', $testClass->getName(), TestCase::class));
        }
        $this->addTest(self::fromClassReflector($testClass));
    }
    
    /**
     * Wraps both <code>addTest()</code> and <code>addTestSuite</code>
     * as well as the separate import statements for the user's convenience.
     *
     * If the named file cannot be read or there are no new tests that can be
     * added, a <code>PHPUnit\Framework\WarningTestCase</code> will be created instead,
     * leaving the current test run untouched.
     *
     * @throws Exception
     */
    public function addTestFile(string $filename) : void {
        try {
            if (str_ends_with($filename, '.phpt') && is_file($filename)) {
                $this->addTest(new PhptTestCase($filename));
            }
            else {
                $this->addTestSuite((new TestSuiteLoader())->load($filename));
            }
        } catch (RunnerException $e) {
            Event\Facade::emitter()->testRunnerTriggeredWarning($e->getMessage());
        }
    }
    
    /**
     * Wrapper for addTestFile() that adds multiple test files.
     *
     * @throws Exception
     */
    public function addTestFiles(iterable $fileNames) : void {
        foreach ($fileNames as $filename) {
            $this->addTestFile((string) $filename);
        }
    }
    
    /**
     * Counts the number of test cases that will be run by this test.
     */
    public function count() : int {
        $numTests = 0;
        foreach ($this as $test) {
            $numTests += count($test);
        }
        return $numTests;
    }
    public function isEmpty() : bool {
        foreach ($this as $test) {
            if (count($test) !== 0) {
                return false;
            }
        }
        return true;
    }
    
    /**
     * @psalm-return non-empty-string
     */
    public function name() : string {
        return $this->name;
    }
    
    /**
     * Returns the test groups of the suite.
     *
     * @psalm-return list<string>
     */
    public function groups() : array {
        return array_map('strval', array_keys($this->groups));
    }
    public function groupDetails() : array {
        return $this->groups;
    }
    
    /**
     * @throws CodeCoverageException
     * @throws Event\RuntimeException
     * @throws Exception
     * @throws InvalidArgumentException
     * @throws NoPreviousThrowableException
     * @throws UnintentionallyCoveredCodeException
     */
    public function run() : void {
        if ($this->wasRun) {
            // @codeCoverageIgnoreStart
            throw new Exception('The tests aggregated by this TestSuite were already run');
            // @codeCoverageIgnoreEnd
        }
        $this->wasRun = true;
        if ($this->isEmpty()) {
            return;
        }
        $emitter = Event\Facade::emitter();
        $testSuiteValueObjectForEvents = Event\TestSuite\TestSuiteBuilder::from($this);
        $emitter->testSuiteStarted($testSuiteValueObjectForEvents);
        if (!$this->invokeMethodsBeforeFirstTest($emitter, $testSuiteValueObjectForEvents)) {
            return;
        }
        
        /** @psalm-var list<Test> $tests */
        $tests = [];
        foreach ($this as $test) {
            $tests[] = $test;
        }
        $tests = array_reverse($tests);
        $this->tests = [];
        $this->groups = [];
        while (($test = array_pop($tests)) !== null) {
            if (TestResultFacade::shouldStop()) {
                $emitter->testRunnerExecutionAborted();
                break;
            }
            $test->run();
        }
        $this->invokeMethodsAfterLastTest($emitter);
        $emitter->testSuiteFinished($testSuiteValueObjectForEvents);
    }
    
    /**
     * Returns the tests as an enumeration.
     *
     * @psalm-return list<Test>
     */
    public function tests() : array {
        return $this->tests;
    }
    
    /**
     * Set tests of the test suite.
     *
     * @psalm-param list<Test> $tests
     */
    public function setTests(array $tests) : void {
        $this->tests = $tests;
    }
    
    /**
     * Mark the test suite as skipped.
     *
     * @throws SkippedTestSuiteError
     */
    public function markTestSuiteSkipped(string $message = '') : never {
        throw new SkippedTestSuiteError($message);
    }
    
    /**
     * Returns an iterator for this test suite.
     */
    public function getIterator() : Iterator {
        $iterator = new TestSuiteIterator($this);
        if ($this->iteratorFilter !== null) {
            $iterator = $this->iteratorFilter
                ->factory($iterator, $this);
        }
        return $iterator;
    }
    public function injectFilter(Factory $filter) : void {
        $this->iteratorFilter = $filter;
        foreach ($this as $test) {
            if ($test instanceof self) {
                $test->injectFilter($filter);
            }
        }
    }
    
    /**
     * @psalm-return list<ExecutionOrderDependency>
     */
    public function provides() : array {
        if ($this->providedTests === null) {
            $this->providedTests = [];
            if (is_callable($this->sortId(), true)) {
                $this->providedTests[] = new ExecutionOrderDependency($this->sortId());
            }
            foreach ($this->tests as $test) {
                if (!$test instanceof Reorderable) {
                    continue;
                }
                $this->providedTests = ExecutionOrderDependency::mergeUnique($this->providedTests, $test->provides());
            }
        }
        return $this->providedTests;
    }
    
    /**
     * @psalm-return list<ExecutionOrderDependency>
     */
    public function requires() : array {
        if ($this->requiredTests === null) {
            $this->requiredTests = [];
            foreach ($this->tests as $test) {
                if (!$test instanceof Reorderable) {
                    continue;
                }
                $this->requiredTests = ExecutionOrderDependency::mergeUnique(ExecutionOrderDependency::filterInvalid($this->requiredTests), $test->requires());
            }
            $this->requiredTests = ExecutionOrderDependency::diff($this->requiredTests, $this->provides());
        }
        return $this->requiredTests;
    }
    public function sortId() : string {
        return $this->name() . '::class';
    }
    
    /**
     * @psalm-assert-if-true class-string $this->name
     */
    public function isForTestClass() : bool {
        return class_exists($this->name, false) && is_subclass_of($this->name, TestCase::class);
    }
    
    /**
     * @throws Event\TestData\MoreThanOneDataSetFromDataProviderException
     * @throws Exception
     */
    protected function addTestMethod(ReflectionClass $class, ReflectionMethod $method) : void {
        $className = $class->getName();
        $methodName = $method->getName();
        assert(!empty($methodName));
        try {
            $test = (new TestBuilder())->build($class, $methodName);
        } catch (InvalidDataProviderException $e) {
            Event\Facade::emitter()->testTriggeredPhpunitError(new TestMethod($className, $methodName, $class->getFileName(), $method->getStartLine(), Event\Code\TestDoxBuilder::fromClassNameAndMethodName($className, $methodName), MetadataCollection::fromArray([]), Event\TestData\TestDataCollection::fromArray([])), sprintf("The data provider specified for %s::%s is invalid\n%s", $className, $methodName, $this->throwableToString($e)));
            return;
        }
        if ($test instanceof TestCase || $test instanceof DataProviderTestSuite) {
            $test->setDependencies(Dependencies::dependencies($class->getName(), $methodName));
        }
        $this->addTest($test, (new Groups())->groups($class->getName(), $methodName));
    }
    private function clearCaches() : void {
        $this->providedTests = null;
        $this->requiredTests = null;
    }
    private function containsOnlyVirtualGroups(array $groups) : bool {
        foreach ($groups as $group) {
            if (!str_starts_with($group, '__phpunit_')) {
                return false;
            }
        }
        return true;
    }
    private function methodDoesNotExistOrIsDeclaredInTestCase(string $methodName) : bool {
        $reflector = new ReflectionClass($this->name);
        return !$reflector->hasMethod($methodName) || $reflector->getMethod($methodName)
            ->getDeclaringClass()
            ->getName() === TestCase::class;
    }
    
    /**
     * @throws Exception
     */
    private function throwableToString(Throwable $t) : string {
        $message = $t->getMessage();
        if (empty(trim($message))) {
            $message = '<no message>';
        }
        if ($t instanceof InvalidDataProviderException) {
            return sprintf("%s\n%s", $message, Filter::getFilteredStacktrace($t));
        }
        return sprintf("%s: %s\n%s", $t::class, $message, Filter::getFilteredStacktrace($t));
    }
    
    /**
     * @throws Exception
     * @throws NoPreviousThrowableException
     */
    private function invokeMethodsBeforeFirstTest(Event\Emitter $emitter, Event\TestSuite\TestSuite $testSuiteValueObjectForEvents) : bool {
        if (!$this->isForTestClass()) {
            return true;
        }
        $methodsCalledBeforeFirstTest = [];
        $beforeClassMethods = (new HookMethods())->hookMethods($this->name)['beforeClass'];
        try {
            foreach ($beforeClassMethods as $beforeClassMethod) {
                if ($this->methodDoesNotExistOrIsDeclaredInTestCase($beforeClassMethod)) {
                    continue;
                }
                if ($missingRequirements = (new Requirements())->requirementsNotSatisfiedFor($this->name, $beforeClassMethod)) {
                    $this->markTestSuiteSkipped(implode(PHP_EOL, $missingRequirements));
                }
                $methodCalledBeforeFirstTest = new Event\Code\ClassMethod($this->name, $beforeClassMethod);
                $emitter->testBeforeFirstTestMethodCalled($this->name, $methodCalledBeforeFirstTest);
                $methodsCalledBeforeFirstTest[] = $methodCalledBeforeFirstTest;
                call_user_func([
                    $this->name,
                    $beforeClassMethod,
                ]);
            }
        } catch (SkippedTest|SkippedTestSuiteError $e) {
            $emitter->testSuiteSkipped($testSuiteValueObjectForEvents, $e->getMessage());
            return false;
        } catch (Throwable $t) {
            assert(isset($methodCalledBeforeFirstTest));
            $emitter->testBeforeFirstTestMethodErrored($this->name, $methodCalledBeforeFirstTest, Event\Code\ThrowableBuilder::from($t));
            if (!empty($methodsCalledBeforeFirstTest)) {
                $emitter->testBeforeFirstTestMethodFinished($this->name, ...$methodsCalledBeforeFirstTest);
            }
            return false;
        }
        if (!empty($methodsCalledBeforeFirstTest)) {
            $emitter->testBeforeFirstTestMethodFinished($this->name, ...$methodsCalledBeforeFirstTest);
        }
        return true;
    }
    private function invokeMethodsAfterLastTest(Event\Emitter $emitter) : void {
        if (!$this->isForTestClass()) {
            return;
        }
        $methodsCalledAfterLastTest = [];
        $afterClassMethods = (new HookMethods())->hookMethods($this->name)['afterClass'];
        foreach ($afterClassMethods as $afterClassMethod) {
            if ($this->methodDoesNotExistOrIsDeclaredInTestCase($afterClassMethod)) {
                continue;
            }
            try {
                call_user_func([
                    $this->name,
                    $afterClassMethod,
                ]);
                $methodCalledAfterLastTest = new Event\Code\ClassMethod($this->name, $afterClassMethod);
                $emitter->testAfterLastTestMethodCalled($this->name, $methodCalledAfterLastTest);
                $methodsCalledAfterLastTest[] = $methodCalledAfterLastTest;
            } catch (Throwable) {
                // @todo
            }
        }
        if (!empty($methodsCalledAfterLastTest)) {
            $emitter->testAfterLastTestMethodFinished($this->name, ...$methodsCalledAfterLastTest);
        }
    }

}

Classes

Title Deprecated Summary
TestSuite @template-implements IteratorAggregate<int, Test>
RSS feed
Powered by Drupal