EntityContextTest.php 16 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663
  1. <?php
  2. /**
  3. * CakePHP(tm) : Rapid Development Framework (http://cakephp.org)
  4. * Copyright (c) Cake Software Foundation, Inc. (http://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. (http://cakefoundation.org)
  11. * @link http://cakephp.org CakePHP(tm) Project
  12. * @since CakePHP(tm) v 3.0
  13. * @license http://www.opensource.org/licenses/mit-license.php MIT License
  14. */
  15. namespace Cake\Test\TestCase\View\Form;
  16. use ArrayIterator;
  17. use ArrayObject;
  18. use Cake\Collection\Collection;
  19. use Cake\Network\Request;
  20. use Cake\ORM\Entity;
  21. use Cake\ORM\Table;
  22. use Cake\ORM\TableRegistry;
  23. use Cake\TestSuite\TestCase;
  24. use Cake\Validation\Validator;
  25. use Cake\View\Form\EntityContext;
  26. /**
  27. * Test stub.
  28. */
  29. class Article extends Entity {
  30. }
  31. /**
  32. * Entity context test case.
  33. */
  34. class EntityContextTest extends TestCase {
  35. /**
  36. * Fixtures to use.
  37. *
  38. * @var array
  39. */
  40. public $fixtures = ['core.article', 'core.comment'];
  41. /**
  42. * setup method.
  43. *
  44. * @return void
  45. */
  46. public function setUp() {
  47. parent::setUp();
  48. $this->request = new Request();
  49. }
  50. /**
  51. * Test an invalid table scope throws an error.
  52. *
  53. * @expectedException \RuntimeException
  54. * @expectedExceptionMessage Unable to find table class for current entity
  55. */
  56. public function testInvalidTable() {
  57. $row = new \StdClass();
  58. $context = new EntityContext($this->request, [
  59. 'entity' => $row,
  60. ]);
  61. }
  62. /**
  63. * Test operations with no entity.
  64. *
  65. * @return void
  66. */
  67. public function testOperationsNoEntity() {
  68. $context = new EntityContext($this->request, [
  69. 'table' => 'Articles'
  70. ]);
  71. $this->assertNull($context->val('title'));
  72. $this->assertFalse($context->isRequired('title'));
  73. $this->assertFalse($context->hasError('title'));
  74. $this->assertEquals('string', $context->type('title'));
  75. $this->assertEquals([], $context->error('title'));
  76. $this->assertEquals(
  77. ['length' => null, 'precision' => null],
  78. $context->attributes('title')
  79. );
  80. }
  81. /**
  82. * Test operations that lack a table argument.
  83. *
  84. * @return void
  85. */
  86. public function testOperationsNoTableArg() {
  87. $row = new Article([
  88. 'title' => 'Test entity',
  89. 'body' => 'Something new'
  90. ]);
  91. $row->errors('title', ['Title is required.']);
  92. $context = new EntityContext($this->request, [
  93. 'entity' => $row,
  94. ]);
  95. $result = $context->val('title');
  96. $this->assertEquals($row->title, $result);
  97. $result = $context->error('title');
  98. $this->assertEquals($row->errors('title'), $result);
  99. $this->assertTrue($context->hasError('title'));
  100. }
  101. /**
  102. * Test collection operations that lack a table argument.
  103. *
  104. * @dataProvider collectionProvider
  105. * @return void
  106. */
  107. public function testCollectionOperationsNoTableArg($collection) {
  108. $context = new EntityContext($this->request, [
  109. 'entity' => $collection,
  110. ]);
  111. $result = $context->val('0.title');
  112. $this->assertEquals('First post', $result);
  113. $result = $context->error('1.body');
  114. $this->assertEquals(['Not long enough'], $result);
  115. }
  116. /**
  117. * Data provider for testing collections.
  118. *
  119. * @return array
  120. */
  121. public static function collectionProvider() {
  122. $one = new Entity([
  123. 'title' => 'First post',
  124. 'body' => 'Stuff',
  125. 'user' => new Entity(['username' => 'mark'])
  126. ]);
  127. $one->errors('title', 'Required field');
  128. $two = new Entity([
  129. 'title' => 'Second post',
  130. 'body' => 'Some text',
  131. 'user' => new Entity(['username' => 'jose'])
  132. ]);
  133. $two->errors('body', 'Not long enough');
  134. return [
  135. 'array' => [[$one, $two]],
  136. 'basic iterator' => [new ArrayObject([$one, $two])],
  137. 'array iterator' => [new ArrayIterator([$one, $two])],
  138. 'collection' => [new Collection([$one, $two])],
  139. ];
  140. }
  141. /**
  142. * Test operations on a collection of entities.
  143. *
  144. * @dataProvider collectionProvider
  145. * @return void
  146. */
  147. public function testValOnCollections($collection) {
  148. $context = new EntityContext($this->request, [
  149. 'entity' => $collection,
  150. 'table' => 'Articles',
  151. ]);
  152. $result = $context->val('0.title');
  153. $this->assertEquals('First post', $result);
  154. $result = $context->val('0.user.username');
  155. $this->assertEquals('mark', $result);
  156. $result = $context->val('1.title');
  157. $this->assertEquals('Second post', $result);
  158. $result = $context->val('1.user.username');
  159. $this->assertEquals('jose', $result);
  160. }
  161. /**
  162. * Test error operations on a collection of entities.
  163. *
  164. * @dataProvider collectionProvider
  165. * @return void
  166. */
  167. public function testErrorsOnCollections($collection) {
  168. $context = new EntityContext($this->request, [
  169. 'entity' => $collection,
  170. 'table' => 'Articles',
  171. ]);
  172. $this->assertTrue($context->hasError('0.title'));
  173. $this->assertEquals(['Required field'], $context->error('0.title'));
  174. $this->assertFalse($context->hasError('0.body'));
  175. $this->assertFalse($context->hasError('1.title'));
  176. $this->assertEquals(['Not long enough'], $context->error('1.body'));
  177. $this->assertTrue($context->hasError('1.body'));
  178. }
  179. /**
  180. * Test schema operations on a collection of entities.
  181. *
  182. * @dataProvider collectionProvider
  183. * @return void
  184. */
  185. public function testSchemaOnCollections($collection) {
  186. $this->_setupTables();
  187. $context = new EntityContext($this->request, [
  188. 'entity' => $collection,
  189. 'table' => 'Articles',
  190. ]);
  191. $this->assertEquals('string', $context->type('0.title'));
  192. $this->assertEquals('text', $context->type('1.body'));
  193. $this->assertEquals('string', $context->type('0.user.username'));
  194. $this->assertEquals('string', $context->type('1.user.username'));
  195. $this->assertNull($context->type('0.nope'));
  196. $expected = ['length' => 255, 'precision' => null];
  197. $this->assertEquals($expected, $context->attributes('0.user.username'));
  198. }
  199. /**
  200. * Test validation operations on a collection of entities.
  201. *
  202. * @dataProvider collectionProvider
  203. * @return void
  204. */
  205. public function testValidatorsOnCollections($collection) {
  206. $this->_setupTables();
  207. $context = new EntityContext($this->request, [
  208. 'entity' => $collection,
  209. 'table' => 'Articles',
  210. 'validator' => [
  211. 'Articles' => 'create',
  212. 'Users' => 'custom',
  213. ]
  214. ]);
  215. $this->assertTrue($context->isRequired('0.title'));
  216. $this->assertFalse($context->isRequired('1.body'));
  217. $this->assertTrue($context->isRequired('0.user.username'));
  218. }
  219. /**
  220. * Test reading data.
  221. *
  222. * @return void
  223. */
  224. public function testValBasic() {
  225. $row = new Entity([
  226. 'title' => 'Test entity',
  227. 'body' => 'Something new'
  228. ]);
  229. $context = new EntityContext($this->request, [
  230. 'entity' => $row,
  231. 'table' => 'Articles',
  232. ]);
  233. $result = $context->val('title');
  234. $this->assertEquals($row->title, $result);
  235. $result = $context->val('body');
  236. $this->assertEquals($row->body, $result);
  237. $result = $context->val('nope');
  238. $this->assertNull($result);
  239. }
  240. /**
  241. * Test reading values from associated entities.
  242. *
  243. * @return void
  244. */
  245. public function testValAssociated() {
  246. $row = new Entity([
  247. 'title' => 'Test entity',
  248. 'user' => new Entity([
  249. 'username' => 'mark',
  250. 'fname' => 'Mark'
  251. ]),
  252. 'comments' => [
  253. new Entity(['comment' => 'Test comment']),
  254. new Entity(['comment' => 'Second comment']),
  255. ]
  256. ]);
  257. $context = new EntityContext($this->request, [
  258. 'entity' => $row,
  259. 'table' => 'Articles',
  260. ]);
  261. $result = $context->val('user.fname');
  262. $this->assertEquals($row->user->fname, $result);
  263. $result = $context->val('comments.0.comment');
  264. $this->assertEquals($row->comments[0]->comment, $result);
  265. $result = $context->val('comments.1.comment');
  266. $this->assertEquals($row->comments[1]->comment, $result);
  267. $result = $context->val('comments.0.nope');
  268. $this->assertNull($result);
  269. $result = $context->val('comments.0.nope.no_way');
  270. $this->assertNull($result);
  271. }
  272. /**
  273. * Test reading values from associated entities.
  274. *
  275. * @return void
  276. */
  277. public function testValAssociatedHasMany() {
  278. $row = new Entity([
  279. 'title' => 'First post',
  280. 'user' => new Entity([
  281. 'username' => 'mark',
  282. 'fname' => 'Mark',
  283. 'articles' => [
  284. new Entity(['title' => 'First post']),
  285. new Entity(['title' => 'Second post']),
  286. ]
  287. ]),
  288. ]);
  289. $context = new EntityContext($this->request, [
  290. 'entity' => $row,
  291. 'table' => 'Articles',
  292. ]);
  293. $result = $context->val('user.articles.0.title');
  294. $this->assertEquals('First post', $result);
  295. $result = $context->val('user.articles.1.title');
  296. $this->assertEquals('Second post', $result);
  297. }
  298. /**
  299. * Test validator as a string.
  300. *
  301. * @return void
  302. */
  303. public function testIsRequiredStringValidator() {
  304. $this->_setupTables();
  305. $context = new EntityContext($this->request, [
  306. 'entity' => new Entity(),
  307. 'table' => 'Articles',
  308. 'validator' => 'create',
  309. ]);
  310. $this->assertTrue($context->isRequired('title'));
  311. $this->assertFalse($context->isRequired('body'));
  312. $this->assertFalse($context->isRequired('Herp.derp.derp'));
  313. $this->assertFalse($context->isRequired('nope'));
  314. $this->assertFalse($context->isRequired(''));
  315. }
  316. /**
  317. * Test isRequired on associated entities.
  318. *
  319. * @return void
  320. */
  321. public function testIsRequiredAssociatedHasMany() {
  322. $this->_setupTables();
  323. $comments = TableRegistry::get('Comments');
  324. $validator = $comments->validator();
  325. $validator->add('user_id', 'number', [
  326. 'rule' => 'numeric',
  327. ]);
  328. $row = new Entity([
  329. 'title' => 'My title',
  330. 'comments' => [
  331. new Entity(['comment' => 'First comment']),
  332. new Entity(['comment' => 'Second comment']),
  333. ]
  334. ]);
  335. $context = new EntityContext($this->request, [
  336. 'entity' => $row,
  337. 'table' => 'Articles',
  338. 'validator' => 'default',
  339. ]);
  340. $this->assertTrue($context->isRequired('comments.0.user_id'));
  341. $this->assertFalse($context->isRequired('comments.0.other'));
  342. $this->assertFalse($context->isRequired('user.0.other'));
  343. $this->assertFalse($context->isRequired(''));
  344. }
  345. /**
  346. * Test isRequired on associated entities with custom validators.
  347. *
  348. * @return void
  349. */
  350. public function testIsRequiredAssociatedValidator() {
  351. $this->_setupTables();
  352. $row = new Entity([
  353. 'title' => 'My title',
  354. 'comments' => [
  355. new Entity(['comment' => 'First comment']),
  356. new Entity(['comment' => 'Second comment']),
  357. ]
  358. ]);
  359. $context = new EntityContext($this->request, [
  360. 'entity' => $row,
  361. 'table' => 'Articles',
  362. 'validator' => [
  363. 'Articles' => 'create',
  364. 'Comments' => 'custom'
  365. ]
  366. ]);
  367. $this->assertTrue($context->isRequired('title'));
  368. $this->assertFalse($context->isRequired('body'));
  369. $this->assertTrue($context->isRequired('comments.0.comment'));
  370. $this->assertTrue($context->isRequired('comments.1.comment'));
  371. }
  372. /**
  373. * Test isRequired on associated entities.
  374. *
  375. * @return void
  376. */
  377. public function testIsRequiredAssociatedBelongsTo() {
  378. $this->_setupTables();
  379. $row = new Entity([
  380. 'title' => 'My title',
  381. 'user' => new Entity(['username' => 'Mark']),
  382. ]);
  383. $context = new EntityContext($this->request, [
  384. 'entity' => $row,
  385. 'table' => 'Articles',
  386. 'validator' => [
  387. 'Articles' => 'create',
  388. 'Users' => 'custom'
  389. ]
  390. ]);
  391. $this->assertTrue($context->isRequired('user.username'));
  392. $this->assertFalse($context->isRequired('user.first_name'));
  393. }
  394. /**
  395. * Test type() basic
  396. *
  397. * @return void
  398. */
  399. public function testType() {
  400. $this->_setupTables();
  401. $row = new Entity([
  402. 'title' => 'My title',
  403. 'body' => 'Some content',
  404. ]);
  405. $context = new EntityContext($this->request, [
  406. 'entity' => $row,
  407. 'table' => 'Articles',
  408. ]);
  409. $this->assertEquals('string', $context->type('title'));
  410. $this->assertEquals('text', $context->type('body'));
  411. $this->assertEquals('integer', $context->type('user_id'));
  412. $this->assertNull($context->type('nope'));
  413. }
  414. /**
  415. * Test getting types for associated records.
  416. *
  417. * @return void
  418. */
  419. public function testTypeAssociated() {
  420. $this->_setupTables();
  421. $row = new Entity([
  422. 'title' => 'My title',
  423. 'user' => new Entity(['username' => 'Mark']),
  424. ]);
  425. $context = new EntityContext($this->request, [
  426. 'entity' => $row,
  427. 'table' => 'Articles',
  428. ]);
  429. $this->assertEquals('string', $context->type('user.username'));
  430. $this->assertEquals('text', $context->type('user.bio'));
  431. $this->assertNull($context->type('user.nope'));
  432. }
  433. /**
  434. * Test attributes for fields.
  435. *
  436. * @return void
  437. */
  438. public function testAttributes() {
  439. $this->_setupTables();
  440. $row = new Entity([
  441. 'title' => 'My title',
  442. 'user' => new Entity(['username' => 'Mark']),
  443. ]);
  444. $context = new EntityContext($this->request, [
  445. 'entity' => $row,
  446. 'table' => 'Articles',
  447. ]);
  448. $expected = [
  449. 'length' => 255, 'precision' => null
  450. ];
  451. $this->assertEquals($expected, $context->attributes('title'));
  452. $expected = [
  453. 'length' => null, 'precision' => null
  454. ];
  455. $this->assertEquals($expected, $context->attributes('body'));
  456. $expected = [
  457. 'length' => 10, 'precision' => 3
  458. ];
  459. $this->assertEquals($expected, $context->attributes('user.rating'));
  460. }
  461. /**
  462. * Test hasError
  463. *
  464. * @return void
  465. */
  466. public function testHasError() {
  467. $this->_setupTables();
  468. $row = new Entity([
  469. 'title' => 'My title',
  470. 'user' => new Entity(['username' => 'Mark']),
  471. ]);
  472. $row->errors('title', []);
  473. $row->errors('body', 'Gotta have one');
  474. $row->errors('user_id', ['Required field']);
  475. $context = new EntityContext($this->request, [
  476. 'entity' => $row,
  477. 'table' => 'Articles',
  478. ]);
  479. $this->assertFalse($context->hasError('title'));
  480. $this->assertFalse($context->hasError('nope'));
  481. $this->assertTrue($context->hasError('body'));
  482. $this->assertTrue($context->hasError('user_id'));
  483. }
  484. /**
  485. * Test hasError on associated records
  486. *
  487. * @return void
  488. */
  489. public function testHasErrorAssociated() {
  490. $this->_setupTables();
  491. $row = new Entity([
  492. 'title' => 'My title',
  493. 'user' => new Entity(['username' => 'Mark']),
  494. ]);
  495. $row->errors('title', []);
  496. $row->errors('body', 'Gotta have one');
  497. $row->user->errors('username', ['Required']);
  498. $context = new EntityContext($this->request, [
  499. 'entity' => $row,
  500. 'table' => 'Articles',
  501. ]);
  502. $this->assertTrue($context->hasError('user.username'));
  503. $this->assertFalse($context->hasError('user.nope'));
  504. $this->assertFalse($context->hasError('no.nope'));
  505. }
  506. /**
  507. * Test error
  508. *
  509. * @return void
  510. */
  511. public function testError() {
  512. $this->_setupTables();
  513. $row = new Entity([
  514. 'title' => 'My title',
  515. 'user' => new Entity(['username' => 'Mark']),
  516. ]);
  517. $row->errors('title', []);
  518. $row->errors('body', 'Gotta have one');
  519. $row->errors('user_id', ['Required field']);
  520. $row->user->errors('username', ['Required']);
  521. $context = new EntityContext($this->request, [
  522. 'entity' => $row,
  523. 'table' => 'Articles',
  524. ]);
  525. $this->assertEquals([], $context->error('title'));
  526. $expected = ['Gotta have one'];
  527. $this->assertEquals($expected, $context->error('body'));
  528. $expected = ['Required'];
  529. $this->assertEquals($expected, $context->error('user.username'));
  530. }
  531. /**
  532. * Setup tables for tests.
  533. *
  534. * @return void
  535. */
  536. protected function _setupTables() {
  537. $articles = TableRegistry::get('Articles');
  538. $articles->belongsTo('Users');
  539. $articles->hasMany('Comments');
  540. $comments = TableRegistry::get('Comments');
  541. $users = TableRegistry::get('Users');
  542. $users->hasMany('Articles');
  543. $articles->schema([
  544. 'id' => ['type' => 'integer', 'length' => 11, 'null' => false],
  545. 'title' => ['type' => 'string', 'length' => 255],
  546. 'user_id' => ['type' => 'integer', 'length' => 11, 'null' => false],
  547. 'body' => ['type' => 'text']
  548. ]);
  549. $users->schema([
  550. 'id' => ['type' => 'integer', 'length' => 11],
  551. 'username' => ['type' => 'string', 'length' => 255],
  552. 'bio' => ['type' => 'text'],
  553. 'rating' => ['type' => 'decimal', 'length' => 10, 'precision' => 3],
  554. ]);
  555. $validator = new Validator();
  556. $validator->add('title', 'minlength', [
  557. 'rule' => ['minlength', 10]
  558. ])
  559. ->add('body', 'maxlength', [
  560. 'rule' => ['maxlength', 1000]
  561. ])->allowEmpty('body');
  562. $articles->validator('create', $validator);
  563. $validator = new Validator();
  564. $validator->add('username', 'length', [
  565. 'rule' => ['minlength', 10]
  566. ]);
  567. $users->validator('custom', $validator);
  568. $validator = new Validator();
  569. $validator->add('comment', 'length', [
  570. 'rule' => ['minlength', 10]
  571. ]);
  572. $comments->validator('custom', $validator);
  573. }
  574. }