b374k
m1n1 1.01
Apache/2.4.41 (Ubuntu)
Linux vmi616275.contaboserver.net 5.4.0-84-generic #94-Ubuntu SMP Thu Aug 26 20:27:37 UTC 2021 x86_64
uid=33(www-data) gid=33(www-data) groups=33(www-data)
server ip : 62.171.164.128 | your ip : 127.0.0.1
safemode OFF
 >  / home / dev2.destoffenstraat.com / vendor / phpunit / phpunit / src / Framework /
Filename/home/dev2.destoffenstraat.com/vendor/phpunit/phpunit/src/Framework/TestSuite.php
Size27.37 kb
Permissionrw-r--r--
Ownerroot : root
Create time17-Aug-2025 10:26
Last modified01-Feb-2019 06:22
Last accessed23-Aug-2025 03:56
Actionsedit | rename | delete | download (gzip)
Viewtext | code | image
<?php
/*
* 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 Iterator;
use IteratorAggregate;
use PHPUnit\Runner\BaseTestRunner;
use PHPUnit\Runner\Filter\Factory;
use PHPUnit\Runner\PhptTestCase;
use PHPUnit\Util\Fileloader;
use PHPUnit\Util\InvalidArgumentHelper;
use ReflectionClass;
use ReflectionMethod;
use Throwable;

/**
* A TestSuite is a composite of Tests. It runs a collection of test cases.
*/
class TestSuite implements Test, SelfDescribing, IteratorAggregate
{
/**
* Last count of tests in this suite.
*
* @var int|null
*/
private $cachedNumTests;

/**
* Enable or disable the backup and restoration of the $GLOBALS array.
*
* @var bool
*/
protected $backupGlobals;

/**
* Enable or disable the backup and restoration of static attributes.
*
* @var bool
*/
protected $backupStaticAttributes;

/**
* @var bool
*/
private $beStrictAboutChangesToGlobalState;

/**
* @var bool
*/
protected $runTestInSeparateProcess = false;

/**
* The name of the test suite.
*
* @var string
*/
protected $name = '';

/**
* The test groups of the test suite.
*
* @var array
*/
protected $groups = [];

/**
* The tests in the test suite.
*
* @var TestCase[]
*/
protected $tests = [];

/**
* The number of tests in the test suite.
*
* @var int
*/
protected $numTests = -1;

/**
* @var bool
*/
protected $testCase = false;

/**
* @var array
*/
protected $foundClasses = [];

/**
* @var Factory
*/
private $iteratorFilter;

/**
* @var string[]
*/
private $declaredClasses;

/**
* Constructs a new TestSuite:
*
* - PHPUnit\Framework\TestSuite() constructs an empty TestSuite.
*
* - PHPUnit\Framework\TestSuite(ReflectionClass) constructs a
* TestSuite from the given class.
*
* - PHPUnit\Framework\TestSuite(ReflectionClass, String)
* constructs a TestSuite from the given class with the given
* name.
*
* - PHPUnit\Framework\TestSuite(String) either constructs a
* TestSuite from the given class (if the passed string is the
* name of an existing class) or constructs an empty TestSuite
* with the given name.
*
* @param mixed $theClass
* @param string $name
*
* @throws Exception
*/
public function __construct($theClass = '', $name = '')
{
$this->declaredClasses = \get_declared_classes();

$argumentsValid = false;

if (\is_object($theClass) &&
$theClass instanceof ReflectionClass) {
$argumentsValid = true;
} elseif (\is_string($theClass) &&
$theClass !== '' &&
\class_exists($theClass, false)) {
$argumentsValid = true;

if ($name == '') {
$name = $theClass;
}

$theClass = new ReflectionClass($theClass);
} elseif (\is_string($theClass)) {
$this->setName($theClass);

return;
}

if (!$argumentsValid) {
throw new Exception;
}

if (!$theClass->isSubclassOf(TestCase::class)) {
throw new Exception(
'Class "' . $theClass->name . '" does not extend PHPUnit\Framework\TestCase.'
);
}

if ($name != '') {
$this->setName($name);
} else {
$this->setName($theClass->getName());
}

$constructor = $theClass->getConstructor();

if ($constructor !== null &&
!$constructor->isPublic()) {
$this->addTest(
self::warning(
\sprintf(
'Class "%s" has no public constructor.',
$theClass->getName()
)
)
);

return;
}

foreach ($theClass->getMethods() as $method) {
$this->addTestMethod($theClass, $method);
}

if (empty($this->tests)) {
$this->addTest(
self::warning(
\sprintf(
'No tests found in class "%s".',
$theClass->getName()
)
)
);
}

$this->testCase = true;
}

/**
* Returns a string representation of the test suite.
*
* @return string
*/
public function toString()
{
return $this->getName();
}

/**
* Adds a test to the suite.
*
* @param Test $test
* @param array $groups
*/
public function addTest(Test $test, $groups = [])
{
$class = new ReflectionClass($test);

if (!$class->isAbstract()) {
$this->tests[] = $test;
$this->numTests = -1;

if ($test instanceof self && empty($groups)) {
$groups = $test->getGroups();
}

if (empty($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.
*
* @param mixed $testClass
*
* @throws Exception
*/
public function addTestSuite($testClass)
{
if (\is_string($testClass) && \class_exists($testClass)) {
$testClass = new ReflectionClass($testClass);
}

if (!\is_object($testClass)) {
throw InvalidArgumentHelper::factory(
1,
'class name or object'
);
}

if ($testClass instanceof self) {
$this->addTest($testClass);
} elseif ($testClass instanceof ReflectionClass) {
$suiteMethod = false;

if (!$testClass->isAbstract() && $testClass->hasMethod(BaseTestRunner::SUITE_METHODNAME)) {
$method = $testClass->getMethod(
BaseTestRunner::SUITE_METHODNAME
);

if ($method->isStatic()) {
$this->addTest(
$method->invoke(null, $testClass->getName())
);

$suiteMethod = true;
}
}

if (!$suiteMethod && !$testClass->isAbstract() && $testClass->isSubclassOf(TestCase::class)) {
$this->addTest(new self($testClass));
}
} else {
throw new Exception;
}
}

/**
* 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.
*
* @param string $filename
*
* @throws Exception
*/
public function addTestFile($filename)
{
if (!\is_string($filename)) {
throw InvalidArgumentHelper::factory(1, 'string');
}

if (\file_exists($filename) && \substr($filename, -5) == '.phpt') {
$this->addTest(
new PhptTestCase($filename)
);

return;
}

// The given file may contain further stub classes in addition to the
// test class itself. Figure out the actual test class.
$filename = Fileloader::checkAndLoad($filename);
$newClasses = \array_diff(\get_declared_classes(), $this->declaredClasses);

// The diff is empty in case a parent class (with test methods) is added
// AFTER a child class that inherited from it. To account for that case,
// accumulate all discovered classes, so the parent class may be found in
// a later invocation.
if (!empty($newClasses)) {
// On the assumption that test classes are defined first in files,
// process discovered classes in approximate LIFO order, so as to
// avoid unnecessary reflection.
$this->foundClasses = \array_merge($newClasses, $this->foundClasses);
$this->declaredClasses = \get_declared_classes();
}

// The test class's name must match the filename, either in full, or as
// a PEAR/PSR-0 prefixed shortname ('NameSpace_ShortName'), or as a
// PSR-1 local shortname ('NameSpace\ShortName'). The comparison must be
// anchored to prevent false-positive matches (e.g., 'OtherShortName').
$shortname = \basename($filename, '.php');
$shortnameRegEx = '/(?:^|_|\\\\)' . \preg_quote($shortname, '/') . '$/';

foreach ($this->foundClasses as $i => $className) {
if (\preg_match($shortnameRegEx, $className)) {
$class = new ReflectionClass($className);

if ($class->getFileName() == $filename) {
$newClasses = [$className];
unset($this->foundClasses[$i]);

break;
}
}
}

foreach ($newClasses as $className) {
$class = new ReflectionClass($className);

if (\dirname($class->getFileName()) === __DIR__) {
continue;
}

if (!$class->isAbstract()) {
if ($class->hasMethod(BaseTestRunner::SUITE_METHODNAME)) {
$method = $class->getMethod(
BaseTestRunner::SUITE_METHODNAME
);

if ($method->isStatic()) {
$this->addTest($method->invoke(null, $className));
}
} elseif ($class->implementsInterface(Test::class)) {
$this->addTestSuite($class);
}
}
}

$this->numTests = -1;
}

/**
* Wrapper for addTestFile() that adds multiple test files.
*
* @param array|Iterator $filenames
*
* @throws Exception
*/
public function addTestFiles($filenames)
{
if (!(\is_array($filenames) ||
(\is_object($filenames) && $filenames instanceof Iterator))) {
throw InvalidArgumentHelper::factory(
1,
'array or iterator'
);
}

foreach ($filenames as $filename) {
$this->addTestFile((string) $filename);
}
}

/**
* Counts the number of test cases that will be run by this test.
*
* @param bool $preferCache Indicates if cache is preferred.
*
* @return int
*/
public function count($preferCache = false)
{
if ($preferCache && $this->cachedNumTests !== null) {
return $this->cachedNumTests;
}

$numTests = 0;

foreach ($this as $test) {
$numTests += \count($test);
}

$this->cachedNumTests = $numTests;

return $numTests;
}

/**
* @param ReflectionClass $theClass
* @param string $name
*
* @return Test
*
* @throws Exception
*/
public static function createTest(ReflectionClass $theClass, $name)
{
$className = $theClass->getName();

if (!$theClass->isInstantiable()) {
return self::warning(
\sprintf('Cannot instantiate class "%s".', $className)
);
}

$backupSettings = \PHPUnit\Util\Test::getBackupSettings(
$className,
$name
);

$preserveGlobalState = \PHPUnit\Util\Test::getPreserveGlobalStateSettings(
$className,
$name
);

$runTestInSeparateProcess = \PHPUnit\Util\Test::getProcessIsolationSettings(
$className,
$name
);

$runClassInSeparateProcess = \PHPUnit\Util\Test::getClassProcessIsolationSettings(
$className,
$name
);

$constructor = $theClass->getConstructor();

if ($constructor !== null) {
$parameters = $constructor->getParameters();

// TestCase() or TestCase($name)
if (\count($parameters) < 2) {
$test = new $className;
} // TestCase($name, $data)
else {
try {
$data = \PHPUnit\Util\Test::getProvidedData(
$className,
$name
);
} catch (IncompleteTestError $e) {
$message = \sprintf(
'Test for %s::%s marked incomplete by data provider',
$className,
$name
);

$_message = $e->getMessage();

if (!empty($_message)) {
$message .= "\n" . $_message;
}

$data = self::incompleteTest($className, $name, $message);
} catch (SkippedTestError $e) {
$message = \sprintf(
'Test for %s::%s skipped by data provider',
$className,
$name
);

$_message = $e->getMessage();

if (!empty($_message)) {
$message .= "\n" . $_message;
}

$data = self::skipTest($className, $name, $message);
} catch (Throwable $_t) {
$t = $_t;
} catch (Exception $_t) {
$t = $_t;
}

if (isset($t)) {
$message = \sprintf(
'The data provider specified for %s::%s is invalid.',
$className,
$name
);

$_message = $t->getMessage();

if (!empty($_message)) {
$message .= "\n" . $_message;
}

$data = self::warning($message);
}

// Test method with @dataProvider.
if (isset($data)) {
$test = new DataProviderTestSuite(
$className . '::' . $name
);

if (empty($data)) {
$data = self::warning(
\sprintf(
'No tests found in suite "%s".',
$test->getName()
)
);
}

$groups = \PHPUnit\Util\Test::getGroups($className, $name);

if ($data instanceof WarningTestCase ||
$data instanceof SkippedTestCase ||
$data instanceof IncompleteTestCase) {
$test->addTest($data, $groups);
} else {
foreach ($data as $_dataName => $_data) {
$_test = new $className($name, $_data, $_dataName);

if ($runTestInSeparateProcess) {
$_test->setRunTestInSeparateProcess(true);

if ($preserveGlobalState !== null) {
$_test->setPreserveGlobalState($preserveGlobalState);
}
}

if ($runClassInSeparateProcess) {
$_test->setRunClassInSeparateProcess(true);

if ($preserveGlobalState !== null) {
$_test->setPreserveGlobalState($preserveGlobalState);
}
}

if ($backupSettings['backupGlobals'] !== null) {
$_test->setBackupGlobals(
$backupSettings['backupGlobals']
);
}

if ($backupSettings['backupStaticAttributes'] !== null) {
$_test->setBackupStaticAttributes(
$backupSettings['backupStaticAttributes']
);
}

$test->addTest($_test, $groups);
}
}
} else {
$test = new $className;
}
}
}

if (!isset($test)) {
throw new Exception('No valid test provided.');
}

if ($test instanceof TestCase) {
$test->setName($name);

if ($runTestInSeparateProcess) {
$test->setRunTestInSeparateProcess(true);

if ($preserveGlobalState !== null) {
$test->setPreserveGlobalState($preserveGlobalState);
}
}

if ($backupSettings['backupGlobals'] !== null) {
$test->setBackupGlobals($backupSettings['backupGlobals']);
}

if ($backupSettings['backupStaticAttributes'] !== null) {
$test->setBackupStaticAttributes(
$backupSettings['backupStaticAttributes']
);
}
}

return $test;
}

/**
* Creates a default TestResult object.
*
* @return TestResult
*/
protected function createResult()
{
return new TestResult;
}

/**
* Returns the name of the suite.
*
* @return string
*/
public function getName()
{
return $this->name;
}

/**
* Returns the test groups of the suite.
*
* @return array
*/
public function getGroups()
{
return \array_keys($this->groups);
}

public function getGroupDetails()
{
return $this->groups;
}

/**
* Set tests groups of the test case
*
* @param array $groups
*/
public function setGroupDetails(array $groups)
{
$this->groups = $groups;
}

/**
* Runs the tests and collects their result in a TestResult.
*
* @param TestResult $result
*
* @return TestResult
*/
public function run(TestResult $result = null)
{
if ($result === null) {
$result = $this->createResult();
}

if (\count($this) == 0) {
return $result;
}

$hookMethods = \PHPUnit\Util\Test::getHookMethods($this->name);

$result->startTestSuite($this);

try {
$this->setUp();

foreach ($hookMethods['beforeClass'] as $beforeClassMethod) {
if ($this->testCase === true &&
\class_exists($this->name, false) &&
\method_exists($this->name, $beforeClassMethod)) {
if ($missingRequirements = \PHPUnit\Util\Test::getMissingRequirements($this->name, $beforeClassMethod)) {
$this->markTestSuiteSkipped(\implode(PHP_EOL, $missingRequirements));
}

\call_user_func([$this->name, $beforeClassMethod]);
}
}
} catch (SkippedTestSuiteError $e) {
$numTests = \count($this);

for ($i = 0; $i < $numTests; $i++) {
$result->startTest($this);
$result->addFailure($this, $e, 0);
$result->endTest($this, 0);
}

$this->tearDown();
$result->endTestSuite($this);

return $result;
} catch (Throwable $_t) {
$t = $_t;
} catch (Exception $_t) {
$t = $_t;
}

if (isset($t)) {
$numTests = \count($this);

for ($i = 0; $i < $numTests; $i++) {
if ($result->shouldStop()) {
break;
}

$result->startTest($this);
$result->addError($this, $t, 0);
$result->endTest($this, 0);
}

$this->tearDown();
$result->endTestSuite($this);

return $result;
}

foreach ($this as $test) {
if ($result->shouldStop()) {
break;
}

if ($test instanceof TestCase || $test instanceof self) {
$test->setBeStrictAboutChangesToGlobalState($this->beStrictAboutChangesToGlobalState);
$test->setBackupGlobals($this->backupGlobals);
$test->setBackupStaticAttributes($this->backupStaticAttributes);
$test->setRunTestInSeparateProcess($this->runTestInSeparateProcess);
}

$test->run($result);
}

foreach ($hookMethods['afterClass'] as $afterClassMethod) {
if ($this->testCase === true && \class_exists($this->name, false) && \method_exists($this->name, $afterClassMethod)) {
\call_user_func([$this->name, $afterClassMethod]);
}
}

$this->tearDown();

$result->endTestSuite($this);

return $result;
}

/**
* @param bool $runTestInSeparateProcess
*
* @throws Exception
*/
public function setRunTestInSeparateProcess($runTestInSeparateProcess)
{
if (\is_bool($runTestInSeparateProcess)) {
$this->runTestInSeparateProcess = $runTestInSeparateProcess;
} else {
throw InvalidArgumentHelper::factory(1, 'boolean');
}
}

/**
* Runs a test.
*
* @deprecated
*
* @param Test $test
* @param TestResult $result
*/
public function runTest(Test $test, TestResult $result)
{
$test->run($result);
}

/**
* Sets the name of the suite.
*
* @param string
*/
public function setName($name)
{
$this->name = $name;
}

/**
* Returns the test at the given index.
*
* @param int|false
*
* @return Test|false
*/
public function testAt($index)
{
if (isset($this->tests[$index])) {
return $this->tests[$index];
}

return false;
}

/**
* Returns the tests as an enumeration.
*
* @return array
*/
public function tests()
{
return $this->tests;
}

/**
* Set tests of the test suite
*
* @param array $tests
*/
public function setTests(array $tests)
{
$this->tests = $tests;
}

/**
* Mark the test suite as skipped.
*
* @param string $message
*
* @throws SkippedTestSuiteError
*/
public function markTestSuiteSkipped($message = '')
{
throw new SkippedTestSuiteError($message);
}

/**
* @param ReflectionClass $class
* @param ReflectionMethod $method
*/
protected function addTestMethod(ReflectionClass $class, ReflectionMethod $method)
{
if (!$this->isTestMethod($method)) {
return;
}

$name = $method->getName();

if (!$method->isPublic()) {
$this->addTest(
self::warning(
\sprintf(
'Test method "%s" in test class "%s" is not public.',
$name,
$class->getName()
)
)
);

return;
}

$test = self::createTest($class, $name);

if ($test instanceof TestCase || $test instanceof DataProviderTestSuite) {
$test->setDependencies(
\PHPUnit\Util\Test::getDependencies($class->getName(), $name)
);
}

$this->addTest(
$test,
\PHPUnit\Util\Test::getGroups($class->getName(), $name)
);
}

/**
* @param ReflectionMethod $method
*
* @return bool
*/
public static function isTestMethod(ReflectionMethod $method)
{
if (\strpos($method->name, 'test') === 0) {
return true;
}

// @scenario on TestCase::testMethod()
// @test on TestCase::testMethod()
$docComment = $method->getDocComment();

return \strpos($docComment, '@test') !== false ||
\strpos($docComment, '@scenario') !== false;
}

/**
* @param string $message
*
* @return WarningTestCase
*/
protected static function warning($message)
{
return new WarningTestCase($message);
}

/**
* @param string $class
* @param string $methodName
* @param string $message
*
* @return SkippedTestCase
*/
protected static function skipTest($class, $methodName, $message)
{
return new SkippedTestCase($class, $methodName, $message);
}

/**
* @param string $class
* @param string $methodName
* @param string $message
*
* @return IncompleteTestCase
*/
protected static function incompleteTest($class, $methodName, $message)
{
return new IncompleteTestCase($class, $methodName, $message);
}

/**
* @param bool $beStrictAboutChangesToGlobalState
*/
public function setBeStrictAboutChangesToGlobalState($beStrictAboutChangesToGlobalState)
{
if (null === $this->beStrictAboutChangesToGlobalState && \is_bool($beStrictAboutChangesToGlobalState)) {
$this->beStrictAboutChangesToGlobalState = $beStrictAboutChangesToGlobalState;
}
}

/**
* @param bool $backupGlobals
*/
public function setBackupGlobals($backupGlobals)
{
if (null === $this->backupGlobals && \is_bool($backupGlobals)) {
$this->backupGlobals = $backupGlobals;
}
}

/**
* @param bool $backupStaticAttributes
*/
public function setBackupStaticAttributes($backupStaticAttributes)
{
if (null === $this->backupStaticAttributes && \is_bool($backupStaticAttributes)) {
$this->backupStaticAttributes = $backupStaticAttributes;
}
}

/**
* Returns an iterator for this test suite.
*
* @return TestSuiteIterator
*/
public function getIterator()
{
$iterator = new TestSuiteIterator($this);

if ($this->iteratorFilter !== null) {
$iterator = $this->iteratorFilter->factory($iterator, $this);
}

return $iterator;
}

public function injectFilter(Factory $filter)
{
$this->iteratorFilter = $filter;
foreach ($this as $test) {
if ($test instanceof self) {
$test->injectFilter($filter);
}
}
}

/**
* Template Method that is called before the tests
* of this test suite are run.
*/
protected function setUp()
{
}

/**
* Template Method that is called after the tests
* of this test suite have finished running.
*/
protected function tearDown()
{
}
}