xTest.php 7.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291
  1. <?php
  2. namespace mindplay\test\lib;
  3. use mindplay\test\lib\ResultPrinter\ResultPrinter;
  4. /**
  5. * A base class to support simple unit tests.
  6. *
  7. * To define a test, declare a method with no arguments, prefixing it's name with "test",
  8. * for example: function testCanReadXmlFeed().
  9. *
  10. * If you declare an init() method, this will be run once before proceeding with the tests.
  11. *
  12. * If you declare a setup() and/or teardown() method, these will be run before/after each test.
  13. *
  14. * @todo document missing parameters and return-types
  15. */
  16. abstract class xTest
  17. {
  18. private $result;
  19. /**
  20. * Test runner.
  21. *
  22. * @var xTestRunner
  23. */
  24. private $testRunner;
  25. /**
  26. * Result printer.
  27. *
  28. * @var ResultPrinter
  29. */
  30. private $resultPrinter;
  31. /**
  32. * The name of the expected Exception.
  33. *
  34. * @var mixed
  35. */
  36. private $expectedException = null;
  37. /**
  38. * The message of the expected Exception.
  39. *
  40. * @var string
  41. */
  42. private $expectedExceptionMessage = '';
  43. /**
  44. * The code of the expected Exception.
  45. *
  46. * @var integer
  47. */
  48. private $expectedExceptionCode;
  49. /**
  50. * Sets result printer.
  51. *
  52. * @param ResultPrinter $resultPrinter Result printer.
  53. * @return void
  54. */
  55. public function setResultPrinter(ResultPrinter $resultPrinter)
  56. {
  57. $this->resultPrinter = $resultPrinter;
  58. }
  59. /**
  60. * Run this test.
  61. *
  62. * @param xTestRunner $testRunner Test runner.
  63. * @return boolean
  64. */
  65. public function run(xTestRunner $testRunner)
  66. {
  67. $this->testRunner = $testRunner;
  68. $this->resultPrinter->testHeader($this);
  69. $reflection = new \ReflectionClass(get_class($this));
  70. $methods = $reflection->getMethods();
  71. $passed = $count = 0;
  72. if (method_exists($this, 'init')) {
  73. try {
  74. $this->init();
  75. } catch (\Exception $exception) {
  76. echo '<tr style="color:white; background:red;"><td>init() failed</td><td><pre>' . $exception . '</pre></td></tr></table>';
  77. return false;
  78. }
  79. }
  80. foreach ($methods as $method) {
  81. if (substr($method->name, 0, 4) == 'test') {
  82. $this->result = null;
  83. $test = $method->name;
  84. $name = substr($test, 4);
  85. if (count($_GET) && isset($_GET[$name]) && $_GET[$name] !== '') {
  86. continue;
  87. }
  88. $this->testRunner->startCoverageCollector($test);
  89. if (method_exists($this, 'setup')) {
  90. $this->setup();
  91. }
  92. $exception = null;
  93. try {
  94. $this->$test();
  95. } catch (\Exception $exception) {
  96. }
  97. try {
  98. $this->assertException($exception);
  99. } catch (xTestException $subException) {
  100. }
  101. $count++;
  102. if ($this->result === true) {
  103. $passed++;
  104. }
  105. if (method_exists($this, 'teardown')) {
  106. $this->teardown();
  107. }
  108. $this->setExpectedException(null, '', null);
  109. $this->testRunner->stopCoverageCollector();
  110. $this->resultPrinter->testCaseResult($method, $this->getResultColor(), $this->getResultMessage());
  111. }
  112. }
  113. $this->resultPrinter->testFooter($this, $count, $passed);
  114. return $passed == $count;
  115. }
  116. /**
  117. * Checks that given exception matches expected one.
  118. *
  119. * @param \Exception $e Exception.
  120. * @return void
  121. */
  122. private function assertException(\Exception $e = null)
  123. {
  124. if (!is_string($this->expectedException)) {
  125. if ($e && !(($e instanceof xTestException) && $e->getCode() == xTestException::FAIL)) {
  126. $this->result = (string)$e;
  127. }
  128. return;
  129. }
  130. $this->check(
  131. $e instanceof \Exception,
  132. 'Exception of "' . $this->expectedException . '" class was not thrown'
  133. );
  134. $this->check(
  135. get_class($e) == $this->expectedException,
  136. 'Exception with "' . get_class($e) . '" class thrown instead of "' . $this->expectedException . '"'
  137. );
  138. if (is_string($this->expectedExceptionMessage) && !empty($this->expectedExceptionMessage)) {
  139. $this->check(
  140. $e->getMessage() == $this->expectedExceptionMessage,
  141. 'Exception with "' . $e->getMessage() . '" message thrown instead of "' . $this->expectedExceptionMessage . '"'
  142. );
  143. }
  144. if ($this->expectedExceptionCode !== null) {
  145. $this->check(
  146. $e->getCode() == $this->expectedExceptionCode,
  147. 'Exception with "' . $e->getCode() . '" code thrown instead of "' . $this->expectedExceptionCode . '"'
  148. );
  149. }
  150. $this->pass();
  151. }
  152. /**
  153. * Returns test result color.
  154. *
  155. * @return string
  156. */
  157. private function getResultColor()
  158. {
  159. if ($this->result !== true) {
  160. $color = 'red';
  161. } elseif ($this->result === null) {
  162. $color = 'blue';
  163. } else {
  164. $color = 'green';
  165. }
  166. return $color;
  167. }
  168. /**
  169. * Returns test result message.
  170. *
  171. * @return string
  172. */
  173. private function getResultMessage()
  174. {
  175. if ($this->result === true) {
  176. $result = 'PASS';
  177. } elseif ($this->result === null) {
  178. $result = 'FAIL: Incomplete Test';
  179. } else {
  180. $result = 'FAIL' . (is_string($this->result) ? ': ' . $this->result : '');
  181. }
  182. return $result;
  183. }
  184. /**
  185. * Calling this method during a test flags a test as passed or failed.
  186. *
  187. * @param bool $pass bool If this expression evaluates as true, the test is passed
  188. * @param bool|string $result string Optional - if supplied, should contain a brief description of why the test failed
  189. */
  190. protected function check($pass, $result = false)
  191. {
  192. if ($pass) {
  193. $this->pass();
  194. } else {
  195. $this->fail($result);
  196. }
  197. }
  198. /**
  199. * Calling this method during a test manually flags a test as passed
  200. */
  201. protected function pass()
  202. {
  203. if ($this->result === null) {
  204. $this->result = true;
  205. }
  206. }
  207. /**
  208. * Calling this method during a test manually flags a test as failed
  209. *
  210. * @param bool|string $result string Optional - if supplied, should contain a brief description of why the test failed
  211. *
  212. * @throws xTestException
  213. */
  214. protected function fail($result = false)
  215. {
  216. $this->result = $result;
  217. throw new xTestException();
  218. }
  219. /**
  220. * Calling this method during a test flags a test as passed if two values are exactly (===) the same.
  221. *
  222. * @param mixed $a mixed Any value
  223. * @param mixed $b mixed Any value - if exactly the same as $a, the test is passed
  224. * @param bool|string $fail string Optional - if supplied, should contain a brief description of why the test failed
  225. */
  226. protected function eq($a, $b, $fail = false)
  227. {
  228. if ($a === $b) {
  229. $this->pass();
  230. } else {
  231. $this->fail($fail === false ? var_export($a, true) . ' !== ' . var_export($b, true) : $fail);
  232. }
  233. }
  234. /**
  235. * Sets expected exception.
  236. *
  237. * @param mixed $exceptionName Exception class name.
  238. * @param string $exceptionMessage Exception message.
  239. * @param integer $exceptionCode Exception code.
  240. */
  241. public function setExpectedException($exceptionName, $exceptionMessage = '', $exceptionCode = null)
  242. {
  243. $this->expectedException = $exceptionName;
  244. $this->expectedExceptionMessage = $exceptionMessage;
  245. $this->expectedExceptionCode = $exceptionCode;
  246. }
  247. }