xTestRunner.php 3.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148
  1. <?php
  2. namespace mindplay\test\lib;
  3. use mindplay\test\lib\ResultPrinter\CliResultPrinter;
  4. use mindplay\test\lib\ResultPrinter\ResultPrinter;
  5. use mindplay\test\lib\ResultPrinter\WebResultPrinter;
  6. /**
  7. * This class implements a very simple test suite runner and code
  8. * coverage benchmarking (where supported by the xdebug extension).
  9. */
  10. class xTestRunner
  11. {
  12. protected $rootPath;
  13. /**
  14. * Code coverage information tracker.
  15. *
  16. * @var \PHP_CodeCoverage
  17. */
  18. protected $coverage;
  19. /**
  20. * Result printer.
  21. *
  22. * @var ResultPrinter
  23. */
  24. protected $resultPrinter;
  25. /**
  26. * Creates result printer based on environment.
  27. *
  28. * @return ResultPrinter
  29. */
  30. public static function createResultPrinter()
  31. {
  32. if (PHP_SAPI == 'cli') {
  33. return new CliResultPrinter(new Colors());
  34. }
  35. return new WebResultPrinter();
  36. }
  37. /**
  38. * Creates test runner instance.
  39. *
  40. * @param string $rootPath The absolute path to the root folder of the test suite.
  41. * @param ResultPrinter $resultPrinter Result printer.
  42. * @throws \Exception
  43. */
  44. public function __construct($rootPath, ResultPrinter $resultPrinter)
  45. {
  46. if (!is_dir($rootPath)) {
  47. throw new \Exception("{$rootPath} is not a directory");
  48. }
  49. $this->rootPath = $rootPath;
  50. $this->resultPrinter = $resultPrinter;
  51. try {
  52. $this->coverage = new \PHP_CodeCoverage();
  53. $this->coverage->filter()->addDirectoryToWhitelist($rootPath);
  54. } catch (\PHP_CodeCoverage_Exception $e) {
  55. // can't collect coverage
  56. }
  57. }
  58. /**
  59. * Returns library root path.
  60. *
  61. * @return string
  62. */
  63. public function getRootPath()
  64. {
  65. return $this->rootPath;
  66. }
  67. /**
  68. * Starts coverage information collection for a test.
  69. *
  70. * @param string $testName Test name.
  71. * @return void
  72. */
  73. public function startCoverageCollector($testName)
  74. {
  75. if (isset($this->coverage)) {
  76. $this->coverage->start($testName);
  77. }
  78. }
  79. /**
  80. * Stops coverage information collection.
  81. *
  82. * @return void
  83. */
  84. public function stopCoverageCollector()
  85. {
  86. if (isset($this->coverage)) {
  87. $this->coverage->stop();
  88. }
  89. }
  90. /**
  91. * Runs a suite of unit tests
  92. *
  93. * @param string $directory Directory with tests.
  94. * @param string $suffix Test file suffix.
  95. * @throws \Exception When invalid test found.
  96. * @return boolean
  97. */
  98. public function run($directory, $suffix)
  99. {
  100. $this->resultPrinter->suiteHeader($this, $directory . '/*' . $suffix);
  101. $passed = true;
  102. $facade = new \File_Iterator_Facade;
  103. $old_handler = set_error_handler(array($this, 'handlePHPErrors'));
  104. foreach ($facade->getFilesAsArray($directory, $suffix) as $path) {
  105. $test = require($path);
  106. if (!$test instanceof xTest) {
  107. throw new \Exception("'{$path}' is not a valid unit test");
  108. }
  109. $test->setResultPrinter($this->resultPrinter);
  110. $passed = $passed && $test->run($this);
  111. }
  112. if ($old_handler) {
  113. set_error_handler($old_handler);
  114. } else {
  115. restore_error_handler();
  116. }
  117. $this->resultPrinter->createCodeCoverageReport($this->coverage);
  118. $this->resultPrinter->suiteFooter($this);
  119. return $passed;
  120. }
  121. public function handlePHPErrors($errno, $errstr)
  122. {
  123. throw new xTestException($errstr, xTestException::PHP_ERROR);
  124. }
  125. }