. * All rights reserved. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions * are met: * * * Redistributions of source code must retain the above copyright * notice, this list of conditions and the following disclaimer. * * * Redistributions in binary form must reproduce the above copyright * notice, this list of conditions and the following disclaimer in * the documentation and/or other materials provided with the * distribution. * * * Neither the name of Sebastian Bergmann nor the names of his * contributors may be used to endorse or promote products derived * from this software without specific prior written permission. * * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE * COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, * BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER * CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN * ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE * POSSIBILITY OF SUCH DAMAGE. * * @category Testing * @package PHPUnit * @author Sebastian Bergmann * @copyright 2002-2008 Sebastian Bergmann * @license http://www.opensource.org/licenses/bsd-license.php BSD License * @version SVN: $Id: TestSuite.php 2157 2008-01-17 13:06:52Z sb $ * @link http://www.phpunit.de/ * @since File available since Release 2.0.0 */ require_once 'PHPUnit/Framework.php'; require_once 'PHPUnit/Extensions/PhptTestCase.php'; require_once 'PHPUnit/Runner/BaseTestRunner.php'; require_once 'PHPUnit/Util/Class.php'; require_once 'PHPUnit/Util/Fileloader.php'; require_once 'PHPUnit/Util/Filter.php'; require_once 'PHPUnit/Util/Test.php'; require_once 'PHPUnit/Util/TestSuiteIterator.php'; PHPUnit_Util_Filter::addFileToFilter(__FILE__, 'PHPUNIT'); if (! class_exists('PHPUnit_Framework_TestSuite', FALSE)) { /** * A TestSuite is a composite of Tests. It runs a collection of test cases. * * Here is an example using the dynamic test definition. * * * addTest(new MathTest('testPass')); * ?> * * * Alternatively, a TestSuite can extract the tests to be run automatically. * To do so you pass a ReflectionClass instance for your * PHPUnit_Framework_TestCase class to the PHPUnit_Framework_TestSuite * constructor. * * * * * * This constructor creates a suite with all the methods starting with * "test" that take no arguments. * * @category Testing * @package PHPUnit * @author Sebastian Bergmann * @copyright 2002-2008 Sebastian Bergmann * @license http://www.opensource.org/licenses/bsd-license.php BSD License * @version Release: 3.2.9 * @link http://www.phpunit.de/ * @since Class available since Release 2.0.0 */ class PHPUnit_Framework_TestSuite implements PHPUnit_Framework_Test, PHPUnit_Framework_SelfDescribing, IteratorAggregate { /** * Fixture that is shared between the tests of this test suite. * * @var mixed * @access protected */ protected $sharedFixture; /** * The name of the test suite. * * @var string * @access protected */ protected $name = ''; /** * The test groups of the test suite. * * @var array * @access protected */ protected $groups = array(); /** * The tests in the test suite. * * @var array * @access protected */ protected $tests = array(); /** * The number of tests in the test suite. * * @var integer * @access protected */ protected $numTests = - 1; /** * 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 InvalidArgumentException * @access public */ public function __construct($theClass = '', $name = '') { $argumentsValid = FALSE; if (is_object($theClass) && $theClass instanceof ReflectionClass) { $argumentsValid = TRUE; } else if (is_string($theClass) && $theClass !== '' && class_exists($theClass, FALSE)) { $argumentsValid = TRUE; if ($name == '') { $name = $theClass; } $theClass = new ReflectionClass($theClass); } else if (is_string($theClass)) { $this->setName($theClass); return; } if (! $argumentsValid) { throw new InvalidArgumentException(); } PHPUnit_Util_Filter::addFileToFilter(realpath($theClass->getFilename()), 'TESTS'); 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; } $className = $theClass->getName(); $names = array(); $classGroups = PHPUnit_Util_Test::getGroups($theClass); foreach ($theClass->getMethods() as $method) { if (strpos($method->getDeclaringClass()->getName(), 'PHPUnit_') !== 0) { $this->addTestMethod($method, PHPUnit_Util_Test::getGroups($method, $classGroups), $names, $theClass); } } if (empty($this->tests)) { $this->addTest(self::warning(sprintf('No tests found in class "%s".', $theClass->getName()))); } } /** * Returns a string representation of the test suite. * * @return string * @access public */ public function toString() { return $this->getName(); } /** * Adds a test to the suite. * * @param PHPUnit_Framework_Test $test * @param array $groups * @access public */ public function addTest(PHPUnit_Framework_Test $test, $groups = array()) { $this->tests[] = $test; $this->numTests = - 1; if ($test instanceof PHPUnit_Framework_TestSuite && empty($groups)) { $groups = $test->getGroups(); } if (empty($groups)) { $groups = array('__nogroup__'); } foreach ($groups as $group) { if (! isset($this->groups[$group])) { $this->groups[$group] = array($test); } else { $this->groups[$group][] = $test; } } } /** * Adds the tests from the given class to the suite. * * @param mixed $testClass * @throws InvalidArgumentException * @access public */ public function addTestSuite($testClass) { if (is_string($testClass) && class_exists($testClass)) { $testClass = new ReflectionClass($testClass); } if (! is_object($testClass)) { throw new InvalidArgumentException(); } if ($testClass instanceof PHPUnit_Framework_TestSuite) { $this->addTest($testClass); } else if ($testClass instanceof ReflectionClass) { $suiteMethod = FALSE; if (! $testClass->isAbstract()) { if ($testClass->hasMethod(PHPUnit_Runner_BaseTestRunner::SUITE_METHODNAME)) { $method = $testClass->getMethod(PHPUnit_Runner_BaseTestRunner::SUITE_METHODNAME); if ($method->isStatic()) { $this->addTest($method->invoke(NULL, $testClass->getName())); $suiteMethod = TRUE; } } } if (! $suiteMethod) { $this->addTest(new PHPUnit_Framework_TestSuite($testClass)); } } else { throw new InvalidArgumentException(); } } /** * Wraps both addTest() and addTestSuite * 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 PHPUnit_Framework_Warning will be created instead, * leaving the current test run untouched. * * @param string $filename * @param boolean $syntaxCheck * @param array $phptOptions Array with ini settings for the php instance * run, key being the name if the setting, * value the ini value. * @throws InvalidArgumentException * @access public * @since Method available since Release 2.3.0 * @author Stefano F. Rausch */ public function addTestFile($filename, $syntaxCheck = TRUE, $phptOptions = array()) { if (! is_string($filename)) { throw new InvalidArgumentException(); } if (file_exists($filename) && substr($filename, - 5) == '.phpt') { $this->addTest(new PHPUnit_Extensions_PhptTestCase($filename, $phptOptions)); return; } if (! file_exists($filename)) { $includePaths = explode(PATH_SEPARATOR, get_include_path()); foreach ($includePaths as $includePath) { $file = $includePath . DIRECTORY_SEPARATOR . $filename; if (file_exists($file)) { $filename = $file; break; } } } PHPUnit_Util_Class::collectStart(); PHPUnit_Util_Fileloader::checkAndLoad($filename, $syntaxCheck); $newClasses = PHPUnit_Util_Class::collectEnd(); $testsFound = FALSE; foreach ($newClasses as $className) { $class = new ReflectionClass($className); if (! $class->isAbstract()) { if ($class->hasMethod(PHPUnit_Runner_BaseTestRunner::SUITE_METHODNAME)) { $method = $class->getMethod(PHPUnit_Runner_BaseTestRunner::SUITE_METHODNAME); if ($method->isStatic()) { $this->addTest($method->invoke(NULL, $className)); $testsFound = TRUE; } } else if ($class->implementsInterface('PHPUnit_Framework_Test')) { $this->addTestSuite($class); $testsFound = TRUE; } } } if (! $testsFound) { $this->addTest(new PHPUnit_Framework_Warning('No tests found in file "' . $filename . '".')); } $this->numTests = - 1; } /** * Wrapper for addTestFile() that adds multiple test files. * * @param array|Iterator $filenames * @throws InvalidArgumentException * @access public * @since Method available since Release 2.3.0 */ public function addTestFiles($filenames) { if (! (is_array($filenames) || (is_object($filenames) && $filenames instanceof Iterator))) { throw new InvalidArgumentException(); } foreach ($filenames as $filename) { $this->addTestFile((string)$filename); } } /** * Counts the number of test cases that will be run by this test. * * @return integer * @access public */ public function count() { if ($this->numTests > - 1) { return $this->numTests; } $this->numTests = 0; foreach ($this->tests as $test) { $this->numTests += count($test); } return $this->numTests; } /** * @param ReflectionClass $theClass * @param string $name * @return PHPUnit_Framework_Test * @access public * @static */ public static function createTest(ReflectionClass $theClass, $name) { $className = $theClass->getName(); $method = new ReflectionMethod($className, $name); $docComment = $method->getDocComment(); if (! $theClass->isInstantiable()) { return self::warning(sprintf('Cannot instantiate class "%s".', $className)); } // @expectedException ExceptionClass on TestCase::testMethod() // @expectedException ExceptionClass message on TestCase::testMethod() // @expectedException ExceptionClass message code on TestCase::testMethod() if (preg_match('(@expectedException\s+([:.\w]+)(?:[\t ]+(\S*))?(?:[\t ]+(\S*))?[\t ]*$)m', $docComment, $matches)) { $expectedException = $matches[1]; if (isset($matches[2])) { $expectedExceptionMessage = trim($matches[2]); } else { $expectedExceptionMessage = ''; } if (isset($matches[3])) { $expectedExceptionCode = (int)$matches[3]; } else { $expectedExceptionCode = 0; } } $constructor = $theClass->getConstructor(); if ($constructor !== NULL) { $parameters = $constructor->getParameters(); // TestCase() or TestCase($name) if (count($parameters) < 2) { $test = new $className(); } // TestCase($name, $data) else { $data = PHPUnit_Util_Test::getProvidedData($className, $name); if (is_array($data) || $data instanceof Iterator) { $test = new PHPUnit_Framework_TestSuite($className . '::' . $name); foreach ($data as $_dataName => $_data) { $_test = new $className($name, $_data, $_dataName); if ($_test instanceof PHPUnit_Framework_TestCase && isset($expectedException)) { $_test->setExpectedException($expectedException, $expectedExceptionMessage, $expectedExceptionCode); } $test->addTest($_test); } } else { $test = new $className(); } } } if ($test instanceof PHPUnit_Framework_TestCase) { $test->setName($name); if (isset($expectedException)) { $test->setExpectedException($expectedException, $expectedExceptionMessage, $expectedExceptionCode); } } return $test; } /** * Creates a default TestResult object. * * @return PHPUnit_Framework_TestResult * @access protected */ protected function createResult() { return new PHPUnit_Framework_TestResult(); } /** * Returns the name of the suite. * * @return string * @access public */ public function getName() { return $this->name; } /** * Returns the test groups of the suite. * * @return array * @access public * @since Method available since Release 3.2.0 */ public function getGroups() { return array_keys($this->groups); } /** * Runs the tests and collects their result in a TestResult. * * @param PHPUnit_Framework_TestResult $result * @param mixed $filter * @param array $groups * @param array $excludeGroups * @return PHPUnit_Framework_TestResult * @throws InvalidArgumentException * @access public */ public function run(PHPUnit_Framework_TestResult $result = NULL, $filter = FALSE, array $groups = array(), array $excludeGroups = array()) { if ($result === NULL) { $result = $this->createResult(); } try { $this->setUp(); } catch (PHPUnit_Framework_SkippedTestSuiteError $e) { $numTests = count($this); for ($i = 0; $i < $numTests; $i ++) { $result->addFailure($this, $e, 0); } return $result; } $result->startTestSuite($this); $tests = array(); if (empty($excludeGroups)) { if (empty($groups)) { $tests = $this->tests; } else { foreach ($groups as $group) { if (isset($this->groups[$group])) { $tests = array_merge($tests, $this->groups[$group]); } } } } else { foreach ($this->groups as $_group => $_tests) { if (! in_array($_group, $excludeGroups)) { $tests = array_merge($tests, $_tests); } } } foreach ($tests as $test) { if ($result->shouldStop()) { break; } if ($test instanceof PHPUnit_Framework_TestSuite) { $test->setSharedFixture($this->sharedFixture); $test->run($result, $filter, $groups, $excludeGroups); } else { $runTest = TRUE; if ($filter !== FALSE) { $name = $test->getName(); if ($name !== NULL && preg_match($filter, $name) == 0) { $runTest = FALSE; } } if ($runTest) { if ($test instanceof PHPUnit_Framework_TestCase) { $test->setSharedFixture($this->sharedFixture); } $this->runTest($test, $result); } } } $result->endTestSuite($this); $this->tearDown(); return $result; } /** * Runs a test. * * @param PHPUnit_Framework_Test $test * @param PHPUnit_Framework_TestResult $testResult * @access public */ public function runTest(PHPUnit_Framework_Test $test, PHPUnit_Framework_TestResult $result) { $test->run($result); } /** * Sets the name of the suite. * * @param string * @access public */ public function setName($name) { $this->name = $name; } /** * Returns the test at the given index. * * @param integer * @return PHPUnit_Framework_Test * @access public */ public function testAt($index) { if (isset($this->tests[$index])) { return $this->tests[$index]; } else { return FALSE; } } /** * Returns the number of tests in this suite. * * @return integer * @access public */ public function testCount() { return count($this->tests); } /** * Returns the tests as an enumeration. * * @return array * @access public */ public function tests() { return $this->tests; } /** * Mark the test suite as skipped. * * @param string $message * @throws PHPUnit_Framework_SkippedTestSuiteError * @access public * @since Method available since Release 3.0.0 */ public function markTestSuiteSkipped($message = '') { throw new PHPUnit_Framework_SkippedTestSuiteError($message); } /** * @param ReflectionMethod $method * @param string $groups * @param array $names * @param ReflectionClass $theClass * @access protected */ protected function addTestMethod(ReflectionMethod $method, $groups, Array &$names, ReflectionClass $theClass) { $name = $method->getName(); if (in_array($name, $names)) { return; } if ($this->isPublicTestMethod($method)) { $names[] = $name; $this->addTest(self::createTest($theClass, $name), $groups); } else if ($this->isTestMethod($method)) { $this->addTest(self::warning(sprintf('Test method "%s" is not public.', $name))); } } /** * @param ReflectionMethod $method * @return boolean * @access public * @static */ public static function isPublicTestMethod(ReflectionMethod $method) { return (self::isTestMethod($method) && $method->isPublic()); } /** * @param ReflectionMethod $method * @return boolean * @access public * @static */ public static function isTestMethod(ReflectionMethod $method) { if (strpos($method->name, 'test') === 0) { return TRUE; } // @story on TestCase::testMethod() // @test on TestCase::testMethod() return strpos($method->getDocComment(), '@test') !== FALSE || strpos($method->getDocComment(), '@story') !== FALSE; } /** * @param string $message * @return PHPUnit_Framework_Warning * @access protected */ protected static function warning($message) { return new PHPUnit_Framework_Warning($message); } /** * Sets the shared fixture for the tests of this test suite. * * @param mixed $sharedFixture * @access public * @since Method available since Release 3.1.0 */ public function setSharedFixture($sharedFixture) { $this->sharedFixture = $sharedFixture; } /** * Returns an iterator for this test suite. * * @return RecursiveIteratorIterator * @access public * @since Method available since Release 3.1.0 */ public function getIterator() { return new RecursiveIteratorIterator(new PHPUnit_Util_TestSuiteIterator($this)); } /** * Template Method that is called before the tests * of this test suite are run. * * @access protected * @since Method available since Release 3.1.0 */ protected function setUp() {} /** * Template Method that is called after the tests * of this test suite have finished running. * * @access protected * @since Method available since Release 3.1.0 */ protected function tearDown() {} } } ?>