ControllerTest.php 40 KB

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