ControllerTest.php 40 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351
  1. <?php
  2. /**
  3. * CakePHP(tm) : Rapid Development Framework (https://cakephp.org)
  4. * Copyright (c) Cake Software Foundation, Inc. (https://cakefoundation.org)
  5. *
  6. * Licensed under The MIT License
  7. * For full copyright and license information, please see the LICENSE.txt
  8. * Redistributions of files must retain the above copyright notice.
  9. *
  10. * @copyright Copyright (c) Cake Software Foundation, Inc. (https://cakefoundation.org)
  11. * @link https://cakephp.org CakePHP Project
  12. * @since 1.2.0
  13. * @license https://opensource.org/licenses/mit-license.php MIT License
  14. */
  15. namespace Cake\Test\TestCase\Controller;
  16. use Cake\Controller\Component;
  17. use Cake\Controller\Controller;
  18. use Cake\Core\Configure;
  19. use Cake\Core\Plugin;
  20. use Cake\Event\Event;
  21. use Cake\Http\Response;
  22. use Cake\Http\ServerRequest;
  23. use Cake\Routing\Router;
  24. use Cake\TestSuite\TestCase;
  25. use TestApp\Controller\Admin\PostsController;
  26. use TestPlugin\Controller\TestPluginController;
  27. /**
  28. * AppController class
  29. */
  30. class ControllerTestAppController extends Controller
  31. {
  32. /**
  33. * helpers property
  34. *
  35. * @var array
  36. */
  37. public $helpers = ['Html'];
  38. /**
  39. * modelClass property
  40. *
  41. * @var string
  42. */
  43. public $modelClass = 'Posts';
  44. /**
  45. * components property
  46. *
  47. * @var array
  48. */
  49. public $components = ['Cookie'];
  50. }
  51. /**
  52. * TestController class
  53. */
  54. class TestController extends ControllerTestAppController
  55. {
  56. /**
  57. * Theme property
  58. *
  59. * @var string
  60. */
  61. public $theme = 'Foo';
  62. /**
  63. * helpers property
  64. *
  65. * @var array
  66. */
  67. public $helpers = ['Html'];
  68. /**
  69. * components property
  70. *
  71. * @var array
  72. */
  73. public $components = ['Security'];
  74. /**
  75. * modelClass property
  76. *
  77. * @var string
  78. */
  79. public $modelClass = 'Comments';
  80. /**
  81. * beforeFilter handler
  82. *
  83. * @param \Cake\Event\Event $event
  84. * @return void
  85. */
  86. public function beforeFilter(Event $event)
  87. {
  88. }
  89. /**
  90. * index method
  91. *
  92. * @param mixed $testId
  93. * @param mixed $testTwoId
  94. * @return void
  95. */
  96. public function index($testId, $testTwoId)
  97. {
  98. $this->request = $this->request->withParsedBody([
  99. 'testId' => $testId,
  100. 'test2Id' => $testTwoId
  101. ]);
  102. }
  103. /**
  104. * view method
  105. *
  106. * @param mixed $testId
  107. * @param mixed $testTwoId
  108. * @return void
  109. */
  110. public function view($testId, $testTwoId)
  111. {
  112. $this->request = $this->request->withParsedBody([
  113. 'testId' => $testId,
  114. 'test2Id' => $testTwoId
  115. ]);
  116. }
  117. public function returner()
  118. {
  119. return 'I am from the controller.';
  120. }
  121. //@codingStandardsIgnoreStart
  122. protected function protected_m()
  123. {
  124. }
  125. private function private_m()
  126. {
  127. }
  128. public function _hidden()
  129. {
  130. }
  131. //@codingStandardsIgnoreEnd
  132. public function admin_add()
  133. {
  134. }
  135. }
  136. /**
  137. * TestComponent class
  138. */
  139. class TestComponent extends Component
  140. {
  141. /**
  142. * beforeRedirect method
  143. *
  144. * @return void
  145. */
  146. public function beforeRedirect()
  147. {
  148. }
  149. /**
  150. * initialize method
  151. *
  152. * @param array $config
  153. * @return void
  154. */
  155. public function initialize(array $config)
  156. {
  157. }
  158. /**
  159. * startup method
  160. *
  161. * @param Event $event
  162. * @return void
  163. */
  164. public function startup(Event $event)
  165. {
  166. }
  167. /**
  168. * shutdown method
  169. *
  170. * @param Event $event
  171. * @return void
  172. */
  173. public function shutdown(Event $event)
  174. {
  175. }
  176. /**
  177. * beforeRender callback
  178. *
  179. * @param Event $event
  180. * @return void
  181. */
  182. public function beforeRender(Event $event)
  183. {
  184. $controller = $event->getSubject();
  185. if ($this->viewclass) {
  186. $controller->viewClass = $this->viewclass;
  187. }
  188. }
  189. }
  190. /**
  191. * AnotherTestController class
  192. */
  193. class AnotherTestController extends ControllerTestAppController
  194. {
  195. }
  196. /**
  197. * ControllerTest class
  198. */
  199. class ControllerTest extends TestCase
  200. {
  201. /**
  202. * fixtures property
  203. *
  204. * @var array
  205. */
  206. public $fixtures = [
  207. 'core.comments',
  208. 'core.posts'
  209. ];
  210. /**
  211. * reset environment.
  212. *
  213. * @return void
  214. */
  215. public function setUp()
  216. {
  217. parent::setUp();
  218. static::setAppNamespace();
  219. Router::reload();
  220. }
  221. /**
  222. * tearDown
  223. *
  224. * @return void
  225. */
  226. public function tearDown()
  227. {
  228. parent::tearDown();
  229. Plugin::unload();
  230. }
  231. /**
  232. * test autoload modelClass
  233. *
  234. * @return void
  235. */
  236. public function testTableAutoload()
  237. {
  238. $request = new ServerRequest('controller_posts/index');
  239. $response = $this->getMockBuilder('Cake\Http\Response')->getMock();
  240. $Controller = new Controller($request, $response);
  241. $Controller->modelClass = 'SiteArticles';
  242. $this->assertFalse($Controller->Articles);
  243. $this->assertInstanceOf(
  244. 'Cake\ORM\Table',
  245. $Controller->SiteArticles
  246. );
  247. unset($Controller->SiteArticles);
  248. $Controller->modelClass = 'Articles';
  249. $this->assertFalse($Controller->SiteArticles);
  250. $this->assertInstanceOf(
  251. 'TestApp\Model\Table\ArticlesTable',
  252. $Controller->Articles
  253. );
  254. }
  255. /**
  256. * testLoadModel method
  257. *
  258. * @return void
  259. */
  260. public function testLoadModel()
  261. {
  262. $request = new ServerRequest('controller_posts/index');
  263. $response = $this->getMockBuilder('Cake\Http\Response')->getMock();
  264. $Controller = new Controller($request, $response);
  265. $this->assertFalse(isset($Controller->Articles));
  266. $result = $Controller->loadModel('Articles');
  267. $this->assertInstanceOf(
  268. 'TestApp\Model\Table\ArticlesTable',
  269. $result
  270. );
  271. $this->assertInstanceOf(
  272. 'TestApp\Model\Table\ArticlesTable',
  273. $Controller->Articles
  274. );
  275. }
  276. /**
  277. * testLoadModel method from a plugin controller
  278. *
  279. * @return void
  280. */
  281. public function testLoadModelInPlugins()
  282. {
  283. Plugin::load('TestPlugin');
  284. $Controller = new TestPluginController();
  285. $Controller->setPlugin('TestPlugin');
  286. $this->assertFalse(isset($Controller->TestPluginComments));
  287. $result = $Controller->loadModel('TestPlugin.TestPluginComments');
  288. $this->assertInstanceOf(
  289. 'TestPlugin\Model\Table\TestPluginCommentsTable',
  290. $result
  291. );
  292. $this->assertInstanceOf(
  293. 'TestPlugin\Model\Table\TestPluginCommentsTable',
  294. $Controller->TestPluginComments
  295. );
  296. }
  297. /**
  298. * Test that the constructor sets modelClass properly.
  299. *
  300. * @return void
  301. */
  302. public function testConstructSetModelClass()
  303. {
  304. Plugin::load('TestPlugin');
  305. $request = new ServerRequest();
  306. $response = new Response();
  307. $controller = new \TestApp\Controller\PostsController($request, $response);
  308. $this->assertEquals('Posts', $controller->modelClass);
  309. $this->assertInstanceOf('Cake\ORM\Table', $controller->Posts);
  310. $controller = new \TestApp\Controller\Admin\PostsController($request, $response);
  311. $this->assertEquals('Posts', $controller->modelClass);
  312. $this->assertInstanceOf('Cake\ORM\Table', $controller->Posts);
  313. $request = $request->withParam('plugin', 'TestPlugin');
  314. $controller = new \TestPlugin\Controller\Admin\CommentsController($request, $response);
  315. $this->assertEquals('TestPlugin.Comments', $controller->modelClass);
  316. $this->assertInstanceOf('TestPlugin\Model\Table\CommentsTable', $controller->Comments);
  317. }
  318. /**
  319. * testConstructClassesWithComponents method
  320. *
  321. * @return void
  322. */
  323. public function testConstructClassesWithComponents()
  324. {
  325. Plugin::load('TestPlugin');
  326. $Controller = new TestPluginController(new ServerRequest(), new Response());
  327. $Controller->loadComponent('TestPlugin.Other');
  328. $this->assertInstanceOf('TestPlugin\Controller\Component\OtherComponent', $Controller->Other);
  329. }
  330. /**
  331. * testRender method
  332. *
  333. * @return void
  334. */
  335. public function testRender()
  336. {
  337. Plugin::load('TestPlugin');
  338. $request = new ServerRequest([
  339. 'url' => 'controller_posts/index',
  340. 'params' => [
  341. 'action' => 'header'
  342. ]
  343. ]);
  344. $Controller = new Controller($request, new Response());
  345. $Controller->viewBuilder()->setTemplatePath('Posts');
  346. $result = $Controller->render('index');
  347. $this->assertRegExp('/posts index/', (string)$result);
  348. $Controller->viewBuilder()->setTemplate('index');
  349. $result = $Controller->render();
  350. $this->assertRegExp('/posts index/', (string)$result);
  351. $result = $Controller->render('/Element/test_element');
  352. $this->assertRegExp('/this is the test element/', (string)$result);
  353. }
  354. /**
  355. * test view rendering changing response
  356. *
  357. * @return void
  358. */
  359. public function testRenderViewChangesResponse()
  360. {
  361. $request = new ServerRequest([
  362. 'url' => 'controller_posts/index',
  363. 'params' => [
  364. 'action' => 'header'
  365. ]
  366. ]);
  367. $controller = new Controller($request, new Response());
  368. $controller->viewBuilder()->setTemplatePath('Posts');
  369. $result = $controller->render('header');
  370. $this->assertContains('header template', (string)$result);
  371. $this->assertTrue($controller->response->hasHeader('X-view-template'));
  372. $this->assertSame('yes', $controller->response->getHeaderLine('X-view-template'));
  373. }
  374. /**
  375. * test that a component beforeRender can change the controller view class.
  376. *
  377. * @return void
  378. */
  379. public function testBeforeRenderCallbackChangingViewClass()
  380. {
  381. $Controller = new Controller(new ServerRequest, new Response());
  382. $Controller->getEventManager()->on('Controller.beforeRender', function (Event $event) {
  383. $controller = $event->getSubject();
  384. $controller->viewClass = 'Json';
  385. });
  386. $Controller->set([
  387. 'test' => 'value',
  388. '_serialize' => ['test']
  389. ]);
  390. $debug = Configure::read('debug');
  391. Configure::write('debug', false);
  392. $result = $Controller->render('index');
  393. $this->assertEquals('{"test":"value"}', (string)$result->getBody());
  394. Configure::write('debug', $debug);
  395. }
  396. /**
  397. * test that a component beforeRender can change the controller view class.
  398. *
  399. * @return void
  400. */
  401. public function testBeforeRenderEventCancelsRender()
  402. {
  403. $Controller = new Controller(new ServerRequest, new Response());
  404. $Controller->getEventManager()->on('Controller.beforeRender', function (Event $event) {
  405. return false;
  406. });
  407. $result = $Controller->render('index');
  408. $this->assertInstanceOf('Cake\Http\Response', $result);
  409. }
  410. /**
  411. * Generates status codes for redirect test.
  412. *
  413. * @return array
  414. */
  415. public static function statusCodeProvider()
  416. {
  417. return [
  418. [300, 'Multiple Choices'],
  419. [301, 'Moved Permanently'],
  420. [302, 'Found'],
  421. [303, 'See Other'],
  422. [304, 'Not Modified'],
  423. [305, 'Use Proxy'],
  424. [307, 'Temporary Redirect'],
  425. [403, 'Forbidden'],
  426. ];
  427. }
  428. /**
  429. * testRedirect method
  430. *
  431. * @dataProvider statusCodeProvider
  432. * @return void
  433. */
  434. public function testRedirectByCode($code, $msg)
  435. {
  436. $Controller = new Controller(null, new Response());
  437. $response = $Controller->redirect('http://cakephp.org', (int)$code);
  438. $this->assertSame($response, $Controller->response);
  439. $this->assertEquals($code, $response->getStatusCode());
  440. $this->assertEquals('http://cakephp.org', $response->getHeaderLine('Location'));
  441. $this->assertFalse($Controller->isAutoRenderEnabled());
  442. }
  443. /**
  444. * test that beforeRedirect callbacks can set the URL that is being redirected to.
  445. *
  446. * @return void
  447. */
  448. public function testRedirectBeforeRedirectModifyingUrl()
  449. {
  450. $Controller = new Controller(null, new Response());
  451. $Controller->getEventManager()->on('Controller.beforeRedirect', function (Event $event, $url, Response $response) {
  452. $controller = $event->getSubject();
  453. $controller->response = $response->withLocation('https://book.cakephp.org');
  454. });
  455. $response = $Controller->redirect('http://cakephp.org', 301);
  456. $this->assertEquals('https://book.cakephp.org', $response->getHeaderLine('Location'));
  457. $this->assertEquals(301, $response->getStatusCode());
  458. }
  459. /**
  460. * test that beforeRedirect callback returning null doesn't affect things.
  461. *
  462. * @return void
  463. */
  464. public function testRedirectBeforeRedirectModifyingStatusCode()
  465. {
  466. $response = new Response();
  467. $Controller = new Controller(null, $response);
  468. $Controller->getEventManager()->on('Controller.beforeRedirect', function (Event $event, $url, Response $response) {
  469. $controller = $event->getSubject();
  470. $controller->response = $response->withStatus(302);
  471. });
  472. $response = $Controller->redirect('http://cakephp.org', 301);
  473. $this->assertEquals('http://cakephp.org', $response->getHeaderLine('Location'));
  474. $this->assertEquals(302, $response->getStatusCode());
  475. }
  476. public function testRedirectBeforeRedirectListenerReturnResponse()
  477. {
  478. $Response = $this->getMockBuilder('Cake\Http\Response')
  479. ->setMethods(['stop', 'header', 'statusCode'])
  480. ->getMock();
  481. $Controller = new Controller(null, $Response);
  482. $newResponse = new Response;
  483. $Controller->getEventManager()->on('Controller.beforeRedirect', function (Event $event, $url, Response $response) use ($newResponse) {
  484. return $newResponse;
  485. });
  486. $result = $Controller->redirect('http://cakephp.org');
  487. $this->assertSame($newResponse, $result);
  488. $this->assertSame($newResponse, $Controller->response);
  489. }
  490. /**
  491. * testMergeVars method
  492. *
  493. * @return void
  494. */
  495. public function testMergeVars()
  496. {
  497. $request = new ServerRequest();
  498. $TestController = new TestController($request);
  499. $expected = [
  500. 'Html' => null,
  501. ];
  502. $this->assertEquals($expected, $TestController->helpers);
  503. $expected = [
  504. 'Security' => null,
  505. 'Cookie' => null,
  506. ];
  507. $this->assertEquals($expected, $TestController->components);
  508. $TestController = new AnotherTestController($request);
  509. $this->assertEquals(
  510. 'Posts',
  511. $TestController->modelClass,
  512. 'modelClass should not be overwritten when defined.'
  513. );
  514. }
  515. /**
  516. * testReferer method
  517. *
  518. * @return void
  519. */
  520. public function testReferer()
  521. {
  522. $request = $this->getMockBuilder('Cake\Http\ServerRequest')
  523. ->setMethods(['referer'])
  524. ->getMock();
  525. $request->expects($this->any())->method('referer')
  526. ->with(true)
  527. ->will($this->returnValue('/posts/index'));
  528. $Controller = new Controller($request);
  529. $result = $Controller->referer(null, true);
  530. $this->assertEquals('/posts/index', $result);
  531. $request = $this->getMockBuilder('Cake\Http\ServerRequest')
  532. ->setMethods(['referer'])
  533. ->getMock();
  534. $request->expects($this->any())->method('referer')
  535. ->with(true)
  536. ->will($this->returnValue('/posts/index'));
  537. $Controller = new Controller($request);
  538. $result = $Controller->referer(['controller' => 'posts', 'action' => 'index'], true);
  539. $this->assertEquals('/posts/index', $result);
  540. $request = $this->getMockBuilder('Cake\Http\ServerRequest')
  541. ->setMethods(['referer'])
  542. ->getMock();
  543. $request->expects($this->any())->method('referer')
  544. ->with(false)
  545. ->will($this->returnValue('http://localhost/posts/index'));
  546. $Controller = new Controller($request);
  547. $result = $Controller->referer();
  548. $this->assertEquals('http://localhost/posts/index', $result);
  549. $Controller = new Controller(null);
  550. $result = $Controller->referer();
  551. $this->assertEquals('/', $result);
  552. }
  553. /**
  554. * Test that the referer is not absolute if it is '/'.
  555. *
  556. * This avoids the base path being applied twice on string urls.
  557. *
  558. * @return void
  559. */
  560. public function testRefererSlash()
  561. {
  562. $request = $this->getMockBuilder('Cake\Http\ServerRequest')
  563. ->setMethods(['referer'])
  564. ->getMock();
  565. $request = $request->withAttribute('base', '/base');
  566. Router::pushRequest($request);
  567. $request->expects($this->any())->method('referer')
  568. ->will($this->returnValue('/'));
  569. $controller = new Controller($request);
  570. $result = $controller->referer('/', true);
  571. $this->assertEquals('/', $result);
  572. $controller = new Controller($request);
  573. $result = $controller->referer('/some/path', true);
  574. $this->assertEquals('/some/path', $result);
  575. }
  576. /**
  577. * testSetAction method
  578. *
  579. * @return void
  580. */
  581. public function testSetAction()
  582. {
  583. $request = new ServerRequest('controller_posts/index');
  584. $TestController = new TestController($request);
  585. $TestController->setAction('view', 1, 2);
  586. $expected = ['testId' => 1, 'test2Id' => 2];
  587. $this->assertSame($expected, $TestController->request->getData());
  588. $this->assertSame('view', $TestController->request->getParam('action'));
  589. }
  590. /**
  591. * Tests that the startup process calls the correct functions
  592. *
  593. * @return void
  594. */
  595. public function testStartupProcess()
  596. {
  597. $eventManager = $this->getMockBuilder('Cake\Event\EventManager')->getMock();
  598. $controller = new Controller(null, null, null, $eventManager);
  599. $eventManager->expects($this->at(0))->method('dispatch')
  600. ->with(
  601. $this->logicalAnd(
  602. $this->isInstanceOf('Cake\Event\Event'),
  603. $this->attributeEqualTo('_name', 'Controller.initialize'),
  604. $this->attributeEqualTo('_subject', $controller)
  605. )
  606. )
  607. ->will($this->returnValue($this->getMockBuilder('Cake\Event\Event')->disableOriginalConstructor()->getMock()));
  608. $eventManager->expects($this->at(1))->method('dispatch')
  609. ->with(
  610. $this->logicalAnd(
  611. $this->isInstanceOf('Cake\Event\Event'),
  612. $this->attributeEqualTo('_name', 'Controller.startup'),
  613. $this->attributeEqualTo('_subject', $controller)
  614. )
  615. )
  616. ->will($this->returnValue($this->getMockBuilder('Cake\Event\Event')->disableOriginalConstructor()->getMock()));
  617. $controller->startupProcess();
  618. }
  619. /**
  620. * Tests that the shutdown process calls the correct functions
  621. *
  622. * @return void
  623. */
  624. public function testShutdownProcess()
  625. {
  626. $eventManager = $this->getMockBuilder('Cake\Event\EventManager')->getMock();
  627. $controller = new Controller(null, null, null, $eventManager);
  628. $eventManager->expects($this->once())->method('dispatch')
  629. ->with(
  630. $this->logicalAnd(
  631. $this->isInstanceOf('Cake\Event\Event'),
  632. $this->attributeEqualTo('_name', 'Controller.shutdown'),
  633. $this->attributeEqualTo('_subject', $controller)
  634. )
  635. )
  636. ->will($this->returnValue($this->getMockBuilder('Cake\Event\Event')->disableOriginalConstructor()->getMock()));
  637. $controller->shutdownProcess();
  638. }
  639. /**
  640. * test using Controller::paginate()
  641. *
  642. * @return void
  643. */
  644. public function testPaginate()
  645. {
  646. $request = new ServerRequest(['url' => 'controller_posts/index']);
  647. $response = $this->getMockBuilder('Cake\Http\Response')
  648. ->setMethods(['httpCodes'])
  649. ->getMock();
  650. $Controller = new Controller($request, $response);
  651. $Controller->request = $Controller->request->withQueryParams([
  652. 'posts' => [
  653. 'page' => 2,
  654. 'limit' => 2,
  655. ]
  656. ]);
  657. $this->assertEquals([], $Controller->paginate);
  658. $this->assertNotContains('Paginator', $Controller->helpers);
  659. $this->assertArrayNotHasKey('Paginator', $Controller->helpers);
  660. $results = $Controller->paginate('Posts');
  661. $this->assertInstanceOf('Cake\Datasource\ResultSetInterface', $results);
  662. $this->assertCount(3, $results);
  663. $results = $Controller->paginate($this->getTableLocator()->get('Posts'));
  664. $this->assertInstanceOf('Cake\Datasource\ResultSetInterface', $results);
  665. $this->assertCount(3, $results);
  666. $paging = $Controller->request->getParam('paging');
  667. $this->assertSame($paging['Posts']['page'], 1);
  668. $this->assertSame($paging['Posts']['pageCount'], 1);
  669. $this->assertFalse($paging['Posts']['prevPage']);
  670. $this->assertFalse($paging['Posts']['nextPage']);
  671. $this->assertNull($paging['Posts']['scope']);
  672. $results = $Controller->paginate($this->getTableLocator()->get('Posts'), ['scope' => 'posts']);
  673. $this->assertInstanceOf('Cake\Datasource\ResultSetInterface', $results);
  674. $this->assertCount(1, $results);
  675. $paging = $Controller->request->getParam('paging');
  676. $this->assertSame($paging['Posts']['page'], 2);
  677. $this->assertSame($paging['Posts']['pageCount'], 2);
  678. $this->assertTrue($paging['Posts']['prevPage']);
  679. $this->assertFalse($paging['Posts']['nextPage']);
  680. $this->assertSame($paging['Posts']['scope'], 'posts');
  681. }
  682. /**
  683. * test that paginate uses modelClass property.
  684. *
  685. * @return void
  686. */
  687. public function testPaginateUsesModelClass()
  688. {
  689. $request = new ServerRequest([
  690. 'url' => 'controller_posts/index',
  691. ]);
  692. $response = $this->getMockBuilder('Cake\Http\Response')
  693. ->setMethods(['httpCodes'])
  694. ->getMock();
  695. $Controller = new Controller($request, $response);
  696. $Controller->modelClass = 'Posts';
  697. $results = $Controller->paginate();
  698. $this->assertInstanceOf('Cake\Datasource\ResultSetInterface', $results);
  699. }
  700. /**
  701. * testMissingAction method
  702. *
  703. * @return void
  704. */
  705. public function testInvokeActionMissingAction()
  706. {
  707. $this->expectException(\Cake\Controller\Exception\MissingActionException::class);
  708. $this->expectExceptionMessage('Action TestController::missing() could not be found, or is not accessible.');
  709. $url = new ServerRequest([
  710. 'url' => 'test/missing',
  711. 'params' => ['controller' => 'Test', 'action' => 'missing']
  712. ]);
  713. $response = $this->getMockBuilder('Cake\Http\Response')->getMock();
  714. $Controller = new TestController($url, $response);
  715. $Controller->invokeAction();
  716. }
  717. /**
  718. * test invoking private methods.
  719. *
  720. * @return void
  721. */
  722. public function testInvokeActionPrivate()
  723. {
  724. $this->expectException(\Cake\Controller\Exception\MissingActionException::class);
  725. $this->expectExceptionMessage('Action TestController::private_m() could not be found, or is not accessible.');
  726. $url = new ServerRequest([
  727. 'url' => 'test/private_m/',
  728. 'params' => ['controller' => 'Test', 'action' => 'private_m']
  729. ]);
  730. $response = $this->getMockBuilder('Cake\Http\Response')->getMock();
  731. $Controller = new TestController($url, $response);
  732. $Controller->invokeAction();
  733. }
  734. /**
  735. * test invoking protected methods.
  736. *
  737. * @return void
  738. */
  739. public function testInvokeActionProtected()
  740. {
  741. $this->expectException(\Cake\Controller\Exception\MissingActionException::class);
  742. $this->expectExceptionMessage('Action TestController::protected_m() could not be found, or is not accessible.');
  743. $url = new ServerRequest([
  744. 'url' => 'test/protected_m/',
  745. 'params' => ['controller' => 'Test', 'action' => 'protected_m']
  746. ]);
  747. $response = $this->getMockBuilder('Cake\Http\Response')->getMock();
  748. $Controller = new TestController($url, $response);
  749. $Controller->invokeAction();
  750. }
  751. /**
  752. * test invoking controller methods.
  753. *
  754. * @return void
  755. */
  756. public function testInvokeActionBaseMethods()
  757. {
  758. $this->expectException(\Cake\Controller\Exception\MissingActionException::class);
  759. $this->expectExceptionMessage('Action TestController::redirect() could not be found, or is not accessible.');
  760. $url = new ServerRequest([
  761. 'url' => 'test/redirect/',
  762. 'params' => ['controller' => 'Test', 'action' => 'redirect']
  763. ]);
  764. $response = $this->getMockBuilder('Cake\Http\Response')->getMock();
  765. $Controller = new TestController($url, $response);
  766. $Controller->invokeAction();
  767. }
  768. /**
  769. * test invoking controller methods.
  770. *
  771. * @return void
  772. */
  773. public function testInvokeActionReturnValue()
  774. {
  775. $url = new ServerRequest([
  776. 'url' => 'test/returner/',
  777. 'params' => [
  778. 'controller' => 'Test',
  779. 'action' => 'returner',
  780. 'pass' => []
  781. ]
  782. ]);
  783. $response = $this->getMockBuilder('Cake\Http\Response')->getMock();
  784. $Controller = new TestController($url, $response);
  785. $result = $Controller->invokeAction();
  786. $this->assertEquals('I am from the controller.', $result);
  787. }
  788. /**
  789. * test invoking controller methods with passed params
  790. *
  791. * @return void
  792. */
  793. public function testInvokeActionWithPassedParams()
  794. {
  795. $url = new ServerRequest([
  796. 'url' => 'test/index/1/2',
  797. 'params' => [
  798. 'controller' => 'Test',
  799. 'action' => 'index',
  800. 'pass' => ['param1' => '1', 'param2' => '2']
  801. ]
  802. ]);
  803. $response = $this->getMockBuilder('Cake\Http\Response')->getMock();
  804. $Controller = new TestController($url, $response);
  805. $result = $Controller->invokeAction();
  806. $this->assertEquals(
  807. ['testId' => '1', 'test2Id' => '2'],
  808. $Controller->request->getData()
  809. );
  810. }
  811. /**
  812. * test that a classes namespace is used in the viewPath.
  813. *
  814. * @return void
  815. */
  816. public function testViewPathConventions()
  817. {
  818. $request = new ServerRequest([
  819. 'url' => 'admin/posts',
  820. 'params' => ['prefix' => 'admin']
  821. ]);
  822. $response = $this->getMockBuilder('Cake\Http\Response')->getMock();
  823. $Controller = new \TestApp\Controller\Admin\PostsController($request, $response);
  824. $Controller->getEventManager()->on('Controller.beforeRender', function (Event $e) {
  825. return $e->getSubject()->response;
  826. });
  827. $Controller->render();
  828. $this->assertEquals('Admin' . DS . 'Posts', $Controller->viewBuilder()->getTemplatePath());
  829. $request = $request->withParam('prefix', 'admin/super');
  830. $response = $this->getMockBuilder('Cake\Http\Response')->getMock();
  831. $Controller = new \TestApp\Controller\Admin\PostsController($request, $response);
  832. $Controller->getEventManager()->on('Controller.beforeRender', function (Event $e) {
  833. return $e->getSubject()->response;
  834. });
  835. $Controller->render();
  836. $this->assertEquals('Admin' . DS . 'Super' . DS . 'Posts', $Controller->viewBuilder()->getTemplatePath());
  837. $request = new ServerRequest([
  838. 'url' => 'pages/home',
  839. 'params' => [
  840. 'prefix' => false
  841. ]
  842. ]);
  843. $Controller = new \TestApp\Controller\PagesController($request, $response);
  844. $Controller->getEventManager()->on('Controller.beforeRender', function (Event $e) {
  845. return $e->getSubject()->response;
  846. });
  847. $Controller->render();
  848. $this->assertEquals('Pages', $Controller->viewBuilder()->getTemplatePath());
  849. }
  850. /**
  851. * Test the components() method.
  852. *
  853. * @return void
  854. */
  855. public function testComponents()
  856. {
  857. $request = new ServerRequest('/');
  858. $response = $this->getMockBuilder('Cake\Http\Response')->getMock();
  859. $controller = new TestController($request, $response);
  860. $this->assertInstanceOf('Cake\Controller\ComponentRegistry', $controller->components());
  861. $result = $controller->components();
  862. $this->assertSame($result, $controller->components());
  863. }
  864. /**
  865. * Test the components() method with the custom ObjectRegistry.
  866. *
  867. * @return void
  868. */
  869. public function testComponentsWithCustomRegistry()
  870. {
  871. $request = new ServerRequest('/');
  872. $response = $this->getMockBuilder('Cake\Http\Response')->getMock();
  873. $componentRegistry = $this->getMockBuilder('Cake\Controller\ComponentRegistry')
  874. ->setMethods(['offsetGet'])
  875. ->getMock();
  876. $controller = new TestController($request, $response, null, null, $componentRegistry);
  877. $this->assertInstanceOf(get_class($componentRegistry), $controller->components());
  878. $result = $controller->components();
  879. $this->assertSame($result, $controller->components());
  880. }
  881. /**
  882. * Test adding a component
  883. *
  884. * @return void
  885. */
  886. public function testLoadComponent()
  887. {
  888. $request = new ServerRequest('/');
  889. $response = $this->getMockBuilder('Cake\Http\Response')->getMock();
  890. $controller = new TestController($request, $response);
  891. $result = $controller->loadComponent('Paginator');
  892. $this->assertInstanceOf('Cake\Controller\Component\PaginatorComponent', $result);
  893. $this->assertSame($result, $controller->Paginator);
  894. $registry = $controller->components();
  895. $this->assertTrue(isset($registry->Paginator));
  896. }
  897. /**
  898. * Test adding a component that is a duplicate.
  899. *
  900. * @return void
  901. */
  902. public function testLoadComponentDuplicate()
  903. {
  904. $request = new ServerRequest('/');
  905. $response = $this->getMockBuilder('Cake\Http\Response')->getMock();
  906. $controller = new TestController($request, $response);
  907. $this->assertNotEmpty($controller->loadComponent('Paginator'));
  908. $this->assertNotEmpty($controller->loadComponent('Paginator'));
  909. try {
  910. $controller->loadComponent('Paginator', ['bad' => 'settings']);
  911. $this->fail('No exception');
  912. } catch (\RuntimeException $e) {
  913. $this->assertContains('The "Paginator" alias has already been loaded', $e->getMessage());
  914. }
  915. }
  916. /**
  917. * Test the isAction method.
  918. *
  919. * @return void
  920. */
  921. public function testIsAction()
  922. {
  923. $request = new ServerRequest('/');
  924. $response = $this->getMockBuilder('Cake\Http\Response')->getMock();
  925. $controller = new TestController($request, $response);
  926. $this->assertFalse($controller->isAction('redirect'));
  927. $this->assertFalse($controller->isAction('beforeFilter'));
  928. $this->assertTrue($controller->isAction('index'));
  929. }
  930. /**
  931. * Test declared deprecated properties like $theme are properly passed to view.
  932. *
  933. * @return void
  934. */
  935. public function testDeclaredDeprecatedProperty()
  936. {
  937. $controller = new TestController(new ServerRequest(), new Response());
  938. $theme = $controller->theme;
  939. // @codingStandardsIgnoreStart
  940. $this->assertEquals($theme, @$controller->createView()->theme);
  941. // @codingStandardsIgnoreEnd
  942. }
  943. /**
  944. * Test that view variables are being set after the beforeRender event gets dispatched
  945. *
  946. * @return void
  947. */
  948. public function testBeforeRenderViewVariables()
  949. {
  950. $controller = new PostsController();
  951. $controller->getEventManager()->on('Controller.beforeRender', function (Event $event) {
  952. /* @var Controller $controller */
  953. $controller = $event->getSubject();
  954. $controller->set('testVariable', 'test');
  955. });
  956. $controller->render('index');
  957. $this->assertArrayHasKey('testVariable', $controller->View->viewVars);
  958. }
  959. /**
  960. * Test name getter and setter.
  961. *
  962. * @return void
  963. */
  964. public function testName()
  965. {
  966. $controller = new PostsController();
  967. $this->assertEquals('Posts', $controller->getName());
  968. $this->assertSame($controller, $controller->setName('Articles'));
  969. $this->assertEquals('Articles', $controller->getName());
  970. }
  971. /**
  972. * Test plugin getter and setter.
  973. *
  974. * @return void
  975. */
  976. public function testPlugin()
  977. {
  978. $controller = new PostsController();
  979. $this->assertEquals('', $controller->getPlugin());
  980. $this->assertSame($controller, $controller->setPlugin('Articles'));
  981. $this->assertEquals('Articles', $controller->getPlugin());
  982. }
  983. /**
  984. * Test request getter and setter.
  985. *
  986. * @return void
  987. */
  988. public function testRequest()
  989. {
  990. $controller = new PostsController();
  991. $this->assertInstanceOf(ServerRequest::class, $controller->getRequest());
  992. $request = new ServerRequest([
  993. 'params' => [
  994. 'plugin' => 'Posts',
  995. 'pass' => [
  996. 'foo',
  997. 'bar'
  998. ]
  999. ]
  1000. ]);
  1001. $this->assertSame($controller, $controller->setRequest($request));
  1002. $this->assertSame($request, $controller->getRequest());
  1003. $this->assertEquals('Posts', $controller->getRequest()->getParam('plugin'));
  1004. $this->assertEquals(['foo', 'bar'], $controller->passedArgs);
  1005. }
  1006. /**
  1007. * Test response getter and setter.
  1008. *
  1009. * @return void
  1010. */
  1011. public function testResponse()
  1012. {
  1013. $controller = new PostsController();
  1014. $this->assertInstanceOf(Response::class, $controller->getResponse());
  1015. $response = new Response;
  1016. $this->assertSame($controller, $controller->setResponse($response));
  1017. $this->assertSame($response, $controller->getResponse());
  1018. }
  1019. /**
  1020. * Test autoRender getter and setter.
  1021. *
  1022. * @return void
  1023. */
  1024. public function testAutoRender()
  1025. {
  1026. $controller = new PostsController();
  1027. $this->assertTrue($controller->isAutoRenderEnabled());
  1028. $this->assertSame($controller, $controller->disableAutoRender());
  1029. $this->assertFalse($controller->isAutoRenderEnabled());
  1030. $this->assertSame($controller, $controller->enableAutoRender());
  1031. $this->assertTrue($controller->isAutoRenderEnabled());
  1032. }
  1033. /**
  1034. * Tests deprecated controller properties work
  1035. *
  1036. * @group deprecated
  1037. * @param $property Deprecated property name
  1038. * @param $getter Getter name
  1039. * @param $setter Setter name
  1040. * @param mixed $value Value to be set
  1041. * @return void
  1042. * @dataProvider deprecatedControllerPropertyProvider
  1043. */
  1044. public function testDeprecatedControllerProperty($property, $getter, $setter, $value)
  1045. {
  1046. $controller = new AnotherTestController();
  1047. $this->deprecated(function () use ($controller, $property, $value) {
  1048. $controller->$property = $value;
  1049. $this->assertSame($value, $controller->$property);
  1050. });
  1051. $this->assertSame($value, $controller->{$getter}());
  1052. }
  1053. /**
  1054. * Tests deprecated controller properties message
  1055. *
  1056. * @group deprecated
  1057. * @param $property Deprecated property name
  1058. * @param $getter Getter name
  1059. * @param $setter Setter name
  1060. * @param mixed $value Value to be set
  1061. * @return void
  1062. * @expectedException PHPUnit\Framework\Error\Deprecated
  1063. * @expectedExceptionMessageRegExp /^Controller::\$\w+ is deprecated(.*)/
  1064. * @dataProvider deprecatedControllerPropertyProvider
  1065. */
  1066. public function testDeprecatedControllerPropertySetterMessage($property, $getter, $setter, $value)
  1067. {
  1068. $controller = new AnotherTestController();
  1069. $this->withErrorReporting(E_ALL, function () use ($controller, $property, $value) {
  1070. $controller->$property = $value;
  1071. });
  1072. }
  1073. /**
  1074. * Tests deprecated controller properties message
  1075. *
  1076. * @group deprecated
  1077. * @param $property Deprecated property name
  1078. * @param $getter Getter name
  1079. * @param $setter Setter name
  1080. * @param mixed $value Value to be set
  1081. * @return void
  1082. * @expectedException PHPUnit\Framework\Error\Deprecated
  1083. * @expectedExceptionMessageRegExp /^Controller::\$\w+ is deprecated(.*)/
  1084. * @dataProvider deprecatedControllerPropertyProvider
  1085. */
  1086. public function testDeprecatedControllerPropertyGetterMessage($property, $getter, $setter, $value)
  1087. {
  1088. $controller = new AnotherTestController();
  1089. $controller->{$setter}($value);
  1090. $this->withErrorReporting(E_ALL, function () use ($controller, $property) {
  1091. $controller->$property;
  1092. });
  1093. }
  1094. /**
  1095. * Data provider for testing deprecated view properties
  1096. *
  1097. * @return array
  1098. */
  1099. public function deprecatedControllerPropertyProvider()
  1100. {
  1101. return [
  1102. ['name', 'getName', 'setName', 'Foo'],
  1103. ['plugin', 'getPlugin', 'setPlugin', 'Foo'],
  1104. ['autoRender', 'isAutoRenderEnabled', 'disableAutoRender', false],
  1105. ];
  1106. }
  1107. /**
  1108. * Tests deprecated view properties work
  1109. *
  1110. * @group deprecated
  1111. * @param $property Deprecated property name
  1112. * @param $getter Getter name
  1113. * @param $setter Setter name
  1114. * @param mixed $value Value to be set
  1115. * @return void
  1116. * @dataProvider deprecatedViewPropertyProvider
  1117. */
  1118. public function testDeprecatedViewProperty($property, $getter, $setter, $value)
  1119. {
  1120. $controller = new AnotherTestController();
  1121. $this->deprecated(function () use ($controller, $property, $value) {
  1122. $controller->$property = $value;
  1123. $this->assertSame($value, $controller->$property);
  1124. });
  1125. $this->assertSame($value, $controller->viewBuilder()->{$getter}());
  1126. }
  1127. /**
  1128. * Tests deprecated view properties message
  1129. *
  1130. * @group deprecated
  1131. * @param $property Deprecated property name
  1132. * @param $getter Getter name
  1133. * @param $setter Setter name
  1134. * @param mixed $value Value to be set
  1135. * @return void
  1136. * @expectedException PHPUnit\Framework\Error\Deprecated
  1137. * @expectedExceptionMessageRegExp /^Controller::\$\w+ is deprecated(.*)/
  1138. * @dataProvider deprecatedViewPropertyProvider
  1139. */
  1140. public function testDeprecatedViewPropertySetterMessage($property, $getter, $setter, $value)
  1141. {
  1142. $controller = new AnotherTestController();
  1143. $this->withErrorReporting(E_ALL, function () use ($controller, $property, $value) {
  1144. $controller->$property = $value;
  1145. });
  1146. }
  1147. /**
  1148. * Tests deprecated view properties message
  1149. *
  1150. * @group deprecated
  1151. * @param $property Deprecated property name
  1152. * @param $getter Getter name
  1153. * @param $setter Setter name
  1154. * @param mixed $value Value to be set
  1155. * @return void
  1156. * @expectedException PHPUnit\Framework\Error\Deprecated
  1157. * @expectedExceptionMessageRegExp /^Controller::\$\w+ is deprecated(.*)/
  1158. * @dataProvider deprecatedViewPropertyProvider
  1159. */
  1160. public function testDeprecatedViewPropertyGetterMessage($property, $getter, $setter, $value)
  1161. {
  1162. $controller = new AnotherTestController();
  1163. $controller->viewBuilder()->{$setter}($value);
  1164. $this->withErrorReporting(E_ALL, function () use ($controller, $property) {
  1165. $result = $controller->$property;
  1166. });
  1167. }
  1168. /**
  1169. * Data provider for testing deprecated view properties
  1170. *
  1171. * @return array
  1172. */
  1173. public function deprecatedViewPropertyProvider()
  1174. {
  1175. return [
  1176. ['layout', 'getLayout', 'setLayout', 'custom'],
  1177. ['view', 'getTemplate', 'setTemplate', 'view'],
  1178. ['theme', 'getTheme', 'setTheme', 'Modern'],
  1179. ['autoLayout', 'isAutoLayoutEnabled', 'enableAutoLayout', false],
  1180. ['viewPath', 'getTemplatePath', 'setTemplatePath', 'Templates'],
  1181. ['layoutPath', 'getLayoutPath', 'setLayoutPath', 'Layouts'],
  1182. ];
  1183. }
  1184. }