ControllerFactoryTest.php 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377
  1. <?php
  2. declare(strict_types=1);
  3. /**
  4. * CakePHP(tm) : Rapid Development Framework (https://cakephp.org)
  5. * Copyright (c) Cake Software Foundation, Inc. (https://cakefoundation.org)
  6. *
  7. * Licensed under The MIT License
  8. * For full copyright and license information, please see the LICENSE.txt
  9. * Redistributions of files must retain the above copyright notice.
  10. *
  11. * @copyright Copyright (c) Cake Software Foundation, Inc. (https://cakefoundation.org)
  12. * @link https://cakephp.org CakePHP(tm) Project
  13. * @since 3.3.0
  14. * @license https://opensource.org/licenses/mit-license.php MIT License
  15. */
  16. namespace Cake\Test\TestCase\Controller;
  17. use Cake\Controller\ControllerFactory;
  18. use Cake\Http\Exception\MissingControllerException;
  19. use Cake\Http\Response;
  20. use Cake\Http\ServerRequest;
  21. use Cake\TestSuite\TestCase;
  22. /**
  23. * Test case for ControllerFactory.
  24. */
  25. class ControllerFactoryTest extends TestCase
  26. {
  27. /**
  28. * @var \Cake\Controller\ControllerFactory
  29. */
  30. protected $factory;
  31. /**
  32. * Setup
  33. *
  34. * @return void
  35. */
  36. public function setUp(): void
  37. {
  38. parent::setUp();
  39. static::setAppNamespace();
  40. $this->factory = new ControllerFactory();
  41. }
  42. /**
  43. * Test building an application controller
  44. *
  45. * @return void
  46. */
  47. public function testApplicationController()
  48. {
  49. $request = new ServerRequest([
  50. 'url' => 'cakes/index',
  51. 'params' => [
  52. 'controller' => 'Cakes',
  53. 'action' => 'index',
  54. ],
  55. ]);
  56. $result = $this->factory->create($request);
  57. $this->assertInstanceOf('TestApp\Controller\CakesController', $result);
  58. $this->assertSame($request, $result->getRequest());
  59. }
  60. /**
  61. * Test building a prefixed app controller.
  62. *
  63. * @return void
  64. */
  65. public function testPrefixedAppControllerDeprecated()
  66. {
  67. $this->deprecated(function () {
  68. $request = new ServerRequest([
  69. 'url' => 'admin/posts/index',
  70. 'params' => [
  71. 'prefix' => 'admin',
  72. 'controller' => 'Posts',
  73. 'action' => 'index',
  74. ],
  75. ]);
  76. $result = $this->factory->create($request);
  77. $this->assertInstanceOf(
  78. 'TestApp\Controller\Admin\PostsController',
  79. $result
  80. );
  81. $this->assertSame($request, $result->getRequest());
  82. });
  83. }
  84. /**
  85. * Test building a nested prefix app controller
  86. *
  87. * @return void
  88. */
  89. public function testNestedPrefixedAppController()
  90. {
  91. $request = new ServerRequest([
  92. 'url' => 'admin/sub/posts/index',
  93. 'params' => [
  94. 'prefix' => 'Admin/Sub',
  95. 'controller' => 'Posts',
  96. 'action' => 'index',
  97. ],
  98. ]);
  99. $result = $this->factory->create($request);
  100. $this->assertInstanceOf(
  101. 'TestApp\Controller\Admin\Sub\PostsController',
  102. $result
  103. );
  104. $this->assertSame($request, $result->getRequest());
  105. }
  106. /**
  107. * Test building a plugin controller
  108. *
  109. * @return void
  110. */
  111. public function testPluginController()
  112. {
  113. $request = new ServerRequest([
  114. 'url' => 'test_plugin/test_plugin/index',
  115. 'params' => [
  116. 'plugin' => 'TestPlugin',
  117. 'controller' => 'TestPlugin',
  118. 'action' => 'index',
  119. ],
  120. ]);
  121. $result = $this->factory->create($request);
  122. $this->assertInstanceOf(
  123. 'TestPlugin\Controller\TestPluginController',
  124. $result
  125. );
  126. $this->assertSame($request, $result->getRequest());
  127. }
  128. /**
  129. * Test building a vendored plugin controller.
  130. *
  131. * @return void
  132. */
  133. public function testVendorPluginController()
  134. {
  135. $request = new ServerRequest([
  136. 'url' => 'test_plugin_three/ovens/index',
  137. 'params' => [
  138. 'plugin' => 'Company/TestPluginThree',
  139. 'controller' => 'Ovens',
  140. 'action' => 'index',
  141. ],
  142. ]);
  143. $result = $this->factory->create($request);
  144. $this->assertInstanceOf(
  145. 'Company\TestPluginThree\Controller\OvensController',
  146. $result
  147. );
  148. $this->assertSame($request, $result->getRequest());
  149. }
  150. /**
  151. * Test building a prefixed plugin controller
  152. *
  153. * @return void
  154. */
  155. public function testPrefixedPluginController()
  156. {
  157. $request = new ServerRequest([
  158. 'url' => 'test_plugin/admin/comments',
  159. 'params' => [
  160. 'prefix' => 'Admin',
  161. 'plugin' => 'TestPlugin',
  162. 'controller' => 'Comments',
  163. 'action' => 'index',
  164. ],
  165. ]);
  166. $result = $this->factory->create($request);
  167. $this->assertInstanceOf(
  168. 'TestPlugin\Controller\Admin\CommentsController',
  169. $result
  170. );
  171. $this->assertSame($request, $result->getRequest());
  172. }
  173. /**
  174. * @return void
  175. */
  176. public function testAbstractClassFailure()
  177. {
  178. $this->expectException(MissingControllerException::class);
  179. $this->expectExceptionMessage('Controller class Abstract could not be found.');
  180. $request = new ServerRequest([
  181. 'url' => 'abstract/index',
  182. 'params' => [
  183. 'controller' => 'Abstract',
  184. 'action' => 'index',
  185. ],
  186. ]);
  187. $this->factory->create($request);
  188. }
  189. /**
  190. * @return void
  191. */
  192. public function testInterfaceFailure()
  193. {
  194. $this->expectException(MissingControllerException::class);
  195. $this->expectExceptionMessage('Controller class Interface could not be found.');
  196. $request = new ServerRequest([
  197. 'url' => 'interface/index',
  198. 'params' => [
  199. 'controller' => 'Interface',
  200. 'action' => 'index',
  201. ],
  202. ]);
  203. $this->factory->create($request);
  204. }
  205. /**
  206. * @return void
  207. */
  208. public function testMissingClassFailure()
  209. {
  210. $this->expectException(MissingControllerException::class);
  211. $this->expectExceptionMessage('Controller class Invisible could not be found.');
  212. $request = new ServerRequest([
  213. 'url' => 'interface/index',
  214. 'params' => [
  215. 'controller' => 'Invisible',
  216. 'action' => 'index',
  217. ],
  218. ]);
  219. $this->factory->create($request);
  220. }
  221. /**
  222. * @return void
  223. */
  224. public function testSlashedControllerFailure()
  225. {
  226. $this->expectException(MissingControllerException::class);
  227. $this->expectExceptionMessage('Controller class Admin/Posts could not be found.');
  228. $request = new ServerRequest([
  229. 'url' => 'admin/posts/index',
  230. 'params' => [
  231. 'controller' => 'Admin/Posts',
  232. 'action' => 'index',
  233. ],
  234. ]);
  235. $this->factory->create($request);
  236. }
  237. /**
  238. * @return void
  239. */
  240. public function testAbsoluteReferenceFailure()
  241. {
  242. $this->expectException(MissingControllerException::class);
  243. $this->expectExceptionMessage('Controller class TestApp\Controller\CakesController could not be found.');
  244. $request = new ServerRequest([
  245. 'url' => 'interface/index',
  246. 'params' => [
  247. 'controller' => 'TestApp\Controller\CakesController',
  248. 'action' => 'index',
  249. ],
  250. ]);
  251. $this->factory->create($request);
  252. }
  253. /**
  254. * Test building controller name when passing no controller name
  255. *
  256. * @return void
  257. */
  258. public function testGetControllerClassNoControllerName()
  259. {
  260. $request = new ServerRequest([
  261. 'url' => 'test_plugin_three/ovens/index',
  262. 'params' => [
  263. 'plugin' => 'Company/TestPluginThree',
  264. 'controller' => 'Ovens',
  265. 'action' => 'index',
  266. ],
  267. ]);
  268. $result = $this->factory->getControllerClass($request);
  269. $this->assertSame('Company\TestPluginThree\Controller\OvensController', $result);
  270. }
  271. /**
  272. * Test invoke with autorender
  273. *
  274. * @return void
  275. */
  276. public function testInvokeAutoRender()
  277. {
  278. $request = new ServerRequest([
  279. 'url' => 'posts',
  280. 'params' => [
  281. 'controller' => 'Posts',
  282. 'action' => 'index',
  283. 'pass' => [],
  284. ],
  285. ]);
  286. $controller = $this->factory->create($request);
  287. $result = $this->factory->invoke($controller);
  288. $this->assertInstanceOf(Response::class, $result);
  289. $this->assertStringContainsString('posts index', (string)$result->getBody());
  290. }
  291. /**
  292. * Test dispatch with autorender=false
  293. *
  294. * @return void
  295. */
  296. public function testInvokeAutoRenderFalse()
  297. {
  298. $request = new ServerRequest([
  299. 'url' => 'posts',
  300. 'params' => [
  301. 'controller' => 'Cakes',
  302. 'action' => 'noRender',
  303. 'pass' => [],
  304. ],
  305. ]);
  306. $controller = $this->factory->create($request);
  307. $result = $this->factory->invoke($controller);
  308. $this->assertInstanceOf(Response::class, $result);
  309. $this->assertStringContainsString('autoRender false body', (string)$result->getBody());
  310. }
  311. /**
  312. * Ensure that a controller's startup event can stop the request.
  313. *
  314. * @return void
  315. */
  316. public function testStartupProcessAbort()
  317. {
  318. $request = new ServerRequest([
  319. 'url' => 'cakes/index',
  320. 'params' => [
  321. 'plugin' => null,
  322. 'controller' => 'Cakes',
  323. 'action' => 'index',
  324. 'stop' => 'startup',
  325. 'pass' => [],
  326. ],
  327. ]);
  328. $controller = $this->factory->create($request);
  329. $result = $this->factory->invoke($controller);
  330. $this->assertSame('startup stop', (string)$result->getBody());
  331. }
  332. /**
  333. * Ensure that a controllers startup process can emit a response
  334. *
  335. * @return void
  336. */
  337. public function testShutdownProcessResponse()
  338. {
  339. $request = new ServerRequest([
  340. 'url' => 'cakes/index',
  341. 'params' => [
  342. 'plugin' => null,
  343. 'controller' => 'Cakes',
  344. 'action' => 'index',
  345. 'stop' => 'shutdown',
  346. 'pass' => [],
  347. ],
  348. ]);
  349. $controller = $this->factory->create($request);
  350. $result = $this->factory->invoke($controller);
  351. $this->assertSame('shutdown stop', (string)$result->getBody());
  352. }
  353. }