ControllerTest.php 41 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344
  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([
  703. 'url' => 'test/missing',
  704. 'params' => ['controller' => 'Test', 'action' => 'missing']
  705. ]);
  706. $response = $this->getMockBuilder('Cake\Http\Response')->getMock();
  707. $Controller = new TestController($url, $response);
  708. $Controller->invokeAction();
  709. }
  710. /**
  711. * test invoking private methods.
  712. *
  713. * @return void
  714. */
  715. public function testInvokeActionPrivate()
  716. {
  717. $this->expectException(\Cake\Controller\Exception\MissingActionException::class);
  718. $this->expectExceptionMessage('Action TestController::private_m() could not be found, or is not accessible.');
  719. $url = new ServerRequest([
  720. 'url' => 'test/private_m/',
  721. 'params' => ['controller' => 'Test', 'action' => 'private_m']
  722. ]);
  723. $response = $this->getMockBuilder('Cake\Http\Response')->getMock();
  724. $Controller = new TestController($url, $response);
  725. $Controller->invokeAction();
  726. }
  727. /**
  728. * test invoking protected methods.
  729. *
  730. * @return void
  731. */
  732. public function testInvokeActionProtected()
  733. {
  734. $this->expectException(\Cake\Controller\Exception\MissingActionException::class);
  735. $this->expectExceptionMessage('Action TestController::protected_m() could not be found, or is not accessible.');
  736. $url = new ServerRequest([
  737. 'url' => 'test/protected_m/',
  738. 'params' => ['controller' => 'Test', 'action' => 'protected_m']
  739. ]);
  740. $response = $this->getMockBuilder('Cake\Http\Response')->getMock();
  741. $Controller = new TestController($url, $response);
  742. $Controller->invokeAction();
  743. }
  744. /**
  745. * test invoking controller methods.
  746. *
  747. * @return void
  748. */
  749. public function testInvokeActionBaseMethods()
  750. {
  751. $this->expectException(\Cake\Controller\Exception\MissingActionException::class);
  752. $this->expectExceptionMessage('Action TestController::redirect() could not be found, or is not accessible.');
  753. $url = new ServerRequest([
  754. 'url' => 'test/redirect/',
  755. 'params' => ['controller' => 'Test', 'action' => 'redirect']
  756. ]);
  757. $response = $this->getMockBuilder('Cake\Http\Response')->getMock();
  758. $Controller = new TestController($url, $response);
  759. $Controller->invokeAction();
  760. }
  761. /**
  762. * test invoking controller methods.
  763. *
  764. * @return void
  765. */
  766. public function testInvokeActionReturnValue()
  767. {
  768. $url = new ServerRequest([
  769. 'url' => 'test/returner/',
  770. 'params' => [
  771. 'controller' => 'Test',
  772. 'action' => 'returner',
  773. 'pass' => []
  774. ]
  775. ]);
  776. $response = $this->getMockBuilder('Cake\Http\Response')->getMock();
  777. $Controller = new TestController($url, $response);
  778. $result = $Controller->invokeAction();
  779. $this->assertEquals('I am from the controller.', $result);
  780. }
  781. /**
  782. * test invoking controller methods with passed params
  783. *
  784. * @return void
  785. */
  786. public function testInvokeActionWithPassedParams()
  787. {
  788. $url = new ServerRequest([
  789. 'url' => 'test/index/1/2',
  790. 'params' => [
  791. 'controller' => 'Test',
  792. 'action' => 'index',
  793. 'pass' => ['param1' => '1', 'param2' => '2']
  794. ]
  795. ]);
  796. $response = $this->getMockBuilder('Cake\Http\Response')->getMock();
  797. $Controller = new TestController($url, $response);
  798. $result = $Controller->invokeAction();
  799. $this->assertEquals(
  800. ['testId' => '1', 'test2Id' => '2'],
  801. $Controller->request->data
  802. );
  803. }
  804. /**
  805. * test that a classes namespace is used in the viewPath.
  806. *
  807. * @return void
  808. */
  809. public function testViewPathConventions()
  810. {
  811. $request = new ServerRequest([
  812. 'url' => 'admin/posts',
  813. 'params' => ['prefix' => 'admin']
  814. ]);
  815. $response = $this->getMockBuilder('Cake\Http\Response')->getMock();
  816. $Controller = new \TestApp\Controller\Admin\PostsController($request, $response);
  817. $Controller->getEventManager()->on('Controller.beforeRender', function (Event $e) {
  818. return $e->getSubject()->response;
  819. });
  820. $Controller->render();
  821. $this->assertEquals('Admin' . DS . 'Posts', $Controller->viewBuilder()->templatePath());
  822. $request = $request->withParam('prefix', 'admin/super');
  823. $response = $this->getMockBuilder('Cake\Http\Response')->getMock();
  824. $Controller = new \TestApp\Controller\Admin\PostsController($request, $response);
  825. $Controller->getEventManager()->on('Controller.beforeRender', function (Event $e) {
  826. return $e->getSubject()->response;
  827. });
  828. $Controller->render();
  829. $this->assertEquals('Admin' . DS . 'Super' . DS . 'Posts', $Controller->viewBuilder()->templatePath());
  830. $request = new ServerRequest([
  831. 'url' => 'pages/home',
  832. 'params' => [
  833. 'prefix' => false
  834. ]
  835. ]);
  836. $Controller = new \TestApp\Controller\PagesController($request, $response);
  837. $Controller->getEventManager()->on('Controller.beforeRender', function (Event $e) {
  838. return $e->getSubject()->response;
  839. });
  840. $Controller->render();
  841. $this->assertEquals('Pages', $Controller->viewBuilder()->templatePath());
  842. }
  843. /**
  844. * Test the components() method.
  845. *
  846. * @return void
  847. */
  848. public function testComponents()
  849. {
  850. $request = new ServerRequest('/');
  851. $response = $this->getMockBuilder('Cake\Http\Response')->getMock();
  852. $controller = new TestController($request, $response);
  853. $this->assertInstanceOf('Cake\Controller\ComponentRegistry', $controller->components());
  854. $result = $controller->components();
  855. $this->assertSame($result, $controller->components());
  856. }
  857. /**
  858. * Test the components() method with the custom ObjectRegistry.
  859. *
  860. * @return void
  861. */
  862. public function testComponentsWithCustomRegistry()
  863. {
  864. $request = new ServerRequest('/');
  865. $response = $this->getMockBuilder('Cake\Http\Response')->getMock();
  866. $componentRegistry = $this->getMockBuilder('Cake\Controller\ComponentRegistry')
  867. ->setMethods(['offsetGet'])
  868. ->getMock();
  869. $controller = new TestController($request, $response, null, null, $componentRegistry);
  870. $this->assertInstanceOf(get_class($componentRegistry), $controller->components());
  871. $result = $controller->components();
  872. $this->assertSame($result, $controller->components());
  873. }
  874. /**
  875. * Test adding a component
  876. *
  877. * @return void
  878. */
  879. public function testLoadComponent()
  880. {
  881. $request = new ServerRequest('/');
  882. $response = $this->getMockBuilder('Cake\Http\Response')->getMock();
  883. $controller = new TestController($request, $response);
  884. $result = $controller->loadComponent('Paginator');
  885. $this->assertInstanceOf('Cake\Controller\Component\PaginatorComponent', $result);
  886. $this->assertSame($result, $controller->Paginator);
  887. $registry = $controller->components();
  888. $this->assertTrue(isset($registry->Paginator));
  889. }
  890. /**
  891. * Test adding a component that is a duplicate.
  892. *
  893. * @return void
  894. */
  895. public function testLoadComponentDuplicate()
  896. {
  897. $request = new ServerRequest('/');
  898. $response = $this->getMockBuilder('Cake\Http\Response')->getMock();
  899. $controller = new TestController($request, $response);
  900. $this->assertNotEmpty($controller->loadComponent('Paginator'));
  901. $this->assertNotEmpty($controller->loadComponent('Paginator'));
  902. try {
  903. $controller->loadComponent('Paginator', ['bad' => 'settings']);
  904. $this->fail('No exception');
  905. } catch (\RuntimeException $e) {
  906. $this->assertContains('The "Paginator" alias has already been loaded', $e->getMessage());
  907. }
  908. }
  909. /**
  910. * Test the isAction method.
  911. *
  912. * @return void
  913. */
  914. public function testIsAction()
  915. {
  916. $request = new ServerRequest('/');
  917. $response = $this->getMockBuilder('Cake\Http\Response')->getMock();
  918. $controller = new TestController($request, $response);
  919. $this->assertFalse($controller->isAction('redirect'));
  920. $this->assertFalse($controller->isAction('beforeFilter'));
  921. $this->assertTrue($controller->isAction('index'));
  922. }
  923. /**
  924. * Test declared deprecated properties like $theme are properly passed to view.
  925. *
  926. * @return void
  927. */
  928. public function testDeclaredDeprecatedProperty()
  929. {
  930. $controller = new TestController(new ServerRequest(), new Response());
  931. $theme = $controller->theme;
  932. // @codingStandardsIgnoreStart
  933. $this->assertEquals($theme, @$controller->createView()->theme);
  934. // @codingStandardsIgnoreEnd
  935. }
  936. /**
  937. * Test that view variables are being set after the beforeRender event gets dispatched
  938. *
  939. * @return void
  940. */
  941. public function testBeforeRenderViewVariables()
  942. {
  943. $controller = new PostsController();
  944. $controller->getEventManager()->on('Controller.beforeRender', function (Event $event) {
  945. /* @var Controller $controller */
  946. $controller = $event->getSubject();
  947. $controller->set('testVariable', 'test');
  948. });
  949. $controller->render('index');
  950. $this->assertArrayHasKey('testVariable', $controller->View->viewVars);
  951. }
  952. /**
  953. * Test name getter and setter.
  954. *
  955. * @return void
  956. */
  957. public function testName()
  958. {
  959. $controller = new PostsController();
  960. $this->assertEquals('Posts', $controller->getName());
  961. $this->assertSame($controller, $controller->setName('Articles'));
  962. $this->assertEquals('Articles', $controller->getName());
  963. }
  964. /**
  965. * Test plugin getter and setter.
  966. *
  967. * @return void
  968. */
  969. public function testPlugin()
  970. {
  971. $controller = new PostsController();
  972. $this->assertEquals('', $controller->getPlugin());
  973. $this->assertSame($controller, $controller->setPlugin('Articles'));
  974. $this->assertEquals('Articles', $controller->getPlugin());
  975. }
  976. /**
  977. * Test request getter and setter.
  978. *
  979. * @return void
  980. */
  981. public function testRequest()
  982. {
  983. $controller = new PostsController();
  984. $this->assertInstanceOf(ServerRequest::class, $controller->getRequest());
  985. $request = new ServerRequest([
  986. 'params' => [
  987. 'plugin' => 'Posts',
  988. 'pass' => [
  989. 'foo',
  990. 'bar'
  991. ]
  992. ]
  993. ]);
  994. $this->assertSame($controller, $controller->setRequest($request));
  995. $this->assertSame($request, $controller->getRequest());
  996. $this->assertEquals('Posts', $controller->getRequest()->getParam('plugin'));
  997. $this->assertEquals(['foo', 'bar'], $controller->passedArgs);
  998. }
  999. /**
  1000. * Test response getter and setter.
  1001. *
  1002. * @return void
  1003. */
  1004. public function testResponse()
  1005. {
  1006. $controller = new PostsController();
  1007. $this->assertInstanceOf(Response::class, $controller->getResponse());
  1008. $response = new Response;
  1009. $this->assertSame($controller, $controller->setResponse($response));
  1010. $this->assertSame($response, $controller->getResponse());
  1011. }
  1012. /**
  1013. * Test autoRender getter and setter.
  1014. *
  1015. * @return void
  1016. */
  1017. public function testAutoRender()
  1018. {
  1019. $controller = new PostsController();
  1020. $this->assertTrue($controller->isAutoRenderEnabled());
  1021. $this->assertSame($controller, $controller->disableAutoRender());
  1022. $this->assertFalse($controller->isAutoRenderEnabled());
  1023. $this->assertSame($controller, $controller->enableAutoRender());
  1024. $this->assertTrue($controller->isAutoRenderEnabled());
  1025. }
  1026. /**
  1027. * Tests deprecated controller properties work
  1028. *
  1029. * @group deprecated
  1030. * @param $property Deprecated property name
  1031. * @param $getter Getter name
  1032. * @param $setter Setter name
  1033. * @param mixed $value Value to be set
  1034. * @return void
  1035. * @dataProvider deprecatedControllerPropertyProvider
  1036. */
  1037. public function testDeprecatedControllerProperty($property, $getter, $setter, $value)
  1038. {
  1039. $controller = new AnotherTestController();
  1040. $this->deprecated(function () use ($controller, $property, $value) {
  1041. $controller->$property = $value;
  1042. $this->assertSame($value, $controller->$property);
  1043. });
  1044. $this->assertSame($value, $controller->{$getter}());
  1045. }
  1046. /**
  1047. * Tests deprecated controller properties message
  1048. *
  1049. * @group deprecated
  1050. * @param $property Deprecated property name
  1051. * @param $getter Getter name
  1052. * @param $setter Setter name
  1053. * @param mixed $value Value to be set
  1054. * @return void
  1055. * @expectedException PHPUnit\Framework\Error\Deprecated
  1056. * @expectedExceptionMessageRegExp /^Controller::\$\w+ is deprecated(.*)/
  1057. * @dataProvider deprecatedControllerPropertyProvider
  1058. */
  1059. public function testDeprecatedControllerPropertySetterMessage($property, $getter, $setter, $value)
  1060. {
  1061. $controller = new AnotherTestController();
  1062. $this->withErrorReporting(E_ALL, function () use ($controller, $property, $value) {
  1063. $controller->$property = $value;
  1064. });
  1065. }
  1066. /**
  1067. * Tests deprecated controller properties message
  1068. *
  1069. * @group deprecated
  1070. * @param $property Deprecated property name
  1071. * @param $getter Getter name
  1072. * @param $setter Setter name
  1073. * @param mixed $value Value to be set
  1074. * @return void
  1075. * @expectedException PHPUnit\Framework\Error\Deprecated
  1076. * @expectedExceptionMessageRegExp /^Controller::\$\w+ is deprecated(.*)/
  1077. * @dataProvider deprecatedControllerPropertyProvider
  1078. */
  1079. public function testDeprecatedControllerPropertyGetterMessage($property, $getter, $setter, $value)
  1080. {
  1081. $controller = new AnotherTestController();
  1082. $controller->{$setter}($value);
  1083. $this->withErrorReporting(E_ALL, function () use ($controller, $property) {
  1084. $controller->$property;
  1085. });
  1086. }
  1087. /**
  1088. * Data provider for testing deprecated view properties
  1089. *
  1090. * @return array
  1091. */
  1092. public function deprecatedControllerPropertyProvider()
  1093. {
  1094. return [
  1095. ['name', 'getName', 'setName', 'Foo'],
  1096. ['plugin', 'getPlugin', 'setPlugin', 'Foo'],
  1097. ['autoRender', 'isAutoRenderEnabled', 'disableAutoRender', false],
  1098. ];
  1099. }
  1100. /**
  1101. * Tests deprecated view properties work
  1102. *
  1103. * @group deprecated
  1104. * @param $property Deprecated property name
  1105. * @param $getter Getter name
  1106. * @param $setter Setter name
  1107. * @param mixed $value Value to be set
  1108. * @return void
  1109. * @dataProvider deprecatedViewPropertyProvider
  1110. */
  1111. public function testDeprecatedViewProperty($property, $getter, $setter, $value)
  1112. {
  1113. $controller = new AnotherTestController();
  1114. $this->deprecated(function () use ($controller, $property, $value) {
  1115. $controller->$property = $value;
  1116. $this->assertSame($value, $controller->$property);
  1117. });
  1118. $this->assertSame($value, $controller->viewBuilder()->{$getter}());
  1119. }
  1120. /**
  1121. * Tests deprecated view properties message
  1122. *
  1123. * @group deprecated
  1124. * @param $property Deprecated property name
  1125. * @param $getter Getter name
  1126. * @param $setter Setter name
  1127. * @param mixed $value Value to be set
  1128. * @return void
  1129. * @expectedException PHPUnit\Framework\Error\Deprecated
  1130. * @expectedExceptionMessageRegExp /^Controller::\$\w+ is deprecated(.*)/
  1131. * @dataProvider deprecatedViewPropertyProvider
  1132. */
  1133. public function testDeprecatedViewPropertySetterMessage($property, $getter, $setter, $value)
  1134. {
  1135. $controller = new AnotherTestController();
  1136. $this->withErrorReporting(E_ALL, function () use ($controller, $property, $value) {
  1137. $controller->$property = $value;
  1138. });
  1139. }
  1140. /**
  1141. * Tests deprecated view properties message
  1142. *
  1143. * @group deprecated
  1144. * @param $property Deprecated property name
  1145. * @param $getter Getter name
  1146. * @param $setter Setter name
  1147. * @param mixed $value Value to be set
  1148. * @return void
  1149. * @expectedException PHPUnit\Framework\Error\Deprecated
  1150. * @expectedExceptionMessageRegExp /^Controller::\$\w+ is deprecated(.*)/
  1151. * @dataProvider deprecatedViewPropertyProvider
  1152. */
  1153. public function testDeprecatedViewPropertyGetterMessage($property, $getter, $setter, $value)
  1154. {
  1155. $controller = new AnotherTestController();
  1156. $controller->viewBuilder()->{$setter}($value);
  1157. $this->withErrorReporting(E_ALL, function () use ($controller, $property) {
  1158. $result = $controller->$property;
  1159. });
  1160. }
  1161. /**
  1162. * Data provider for testing deprecated view properties
  1163. *
  1164. * @return array
  1165. */
  1166. public function deprecatedViewPropertyProvider()
  1167. {
  1168. return [
  1169. ['layout', 'getLayout', 'setLayout', 'custom'],
  1170. ['view', 'getTemplate', 'setTemplate', 'view'],
  1171. ['theme', 'getTheme', 'setTheme', 'Modern'],
  1172. ['autoLayout', 'isAutoLayoutEnabled', 'enableAutoLayout', false],
  1173. ['viewPath', 'getTemplatePath', 'setTemplatePath', 'Templates'],
  1174. ['layoutPath', 'getLayoutPath', 'setLayoutPath', 'Layouts'],
  1175. ];
  1176. }
  1177. }