EntityContextTest.php 21 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859
  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 getting primary key data.
  52. *
  53. * @return void
  54. */
  55. public function testPrimaryKey() {
  56. $row = new Article();
  57. $context = new EntityContext($this->request, [
  58. 'entity' => $row,
  59. ]);
  60. $this->assertEquals(['id'], $context->primaryKey());
  61. }
  62. /**
  63. * Test isPrimaryKey
  64. *
  65. * @return void
  66. */
  67. public function testIsPrimaryKey() {
  68. $row = new Article();
  69. $context = new EntityContext($this->request, [
  70. 'entity' => $row,
  71. ]);
  72. $this->assertTrue($context->isPrimaryKey('id'));
  73. $this->assertFalse($context->isPrimaryKey('title'));
  74. $this->assertTrue($context->isPrimaryKey('1.id'));
  75. $this->assertTrue($context->isPrimaryKey('Articles.1.id'));
  76. $this->assertTrue($context->isPrimaryKey('comments.0.id'));
  77. $this->assertTrue($context->isPrimaryKey('1.comments.0.id'));
  78. $this->assertFalse($context->isPrimaryKey('1.comments.0.comment'));
  79. $this->assertFalse($context->isPrimaryKey('Articles.1.comments.0.comment'));
  80. }
  81. /**
  82. * Test isCreate on a single entity.
  83. *
  84. * @return void
  85. */
  86. public function testIsCreateSingle() {
  87. $row = new Entity();
  88. $context = new EntityContext($this->request, [
  89. 'entity' => $row,
  90. ]);
  91. $this->assertTrue($context->isCreate());
  92. $row->isNew(false);
  93. $this->assertFalse($context->isCreate());
  94. $row->isNew(true);
  95. $this->assertTrue($context->isCreate());
  96. }
  97. /**
  98. * Test isCreate on a collection.
  99. *
  100. * @dataProvider collectionProvider
  101. * @return void
  102. */
  103. public function testIsCreateCollection($collection) {
  104. $context = new EntityContext($this->request, [
  105. 'entity' => $collection,
  106. ]);
  107. $this->assertTrue($context->isCreate());
  108. }
  109. /**
  110. * Test an invalid table scope throws an error.
  111. *
  112. * @expectedException \RuntimeException
  113. * @expectedExceptionMessage Unable to find table class for current entity
  114. */
  115. public function testInvalidTable() {
  116. $row = new \StdClass();
  117. $context = new EntityContext($this->request, [
  118. 'entity' => $row,
  119. ]);
  120. }
  121. /**
  122. * Test operations with no entity.
  123. *
  124. * @return void
  125. */
  126. public function testOperationsNoEntity() {
  127. $context = new EntityContext($this->request, [
  128. 'table' => 'Articles'
  129. ]);
  130. $this->assertNull($context->val('title'));
  131. $this->assertFalse($context->isRequired('title'));
  132. $this->assertFalse($context->hasError('title'));
  133. $this->assertEquals('string', $context->type('title'));
  134. $this->assertEquals([], $context->error('title'));
  135. $this->assertEquals(
  136. ['length' => null, 'precision' => null],
  137. $context->attributes('title')
  138. );
  139. }
  140. /**
  141. * Test operations that lack a table argument.
  142. *
  143. * @return void
  144. */
  145. public function testOperationsNoTableArg() {
  146. $row = new Article([
  147. 'title' => 'Test entity',
  148. 'body' => 'Something new'
  149. ]);
  150. $row->errors('title', ['Title is required.']);
  151. $context = new EntityContext($this->request, [
  152. 'entity' => $row,
  153. ]);
  154. $result = $context->val('title');
  155. $this->assertEquals($row->title, $result);
  156. $result = $context->error('title');
  157. $this->assertEquals($row->errors('title'), $result);
  158. $this->assertTrue($context->hasError('title'));
  159. }
  160. /**
  161. * Test collection operations that lack a table argument.
  162. *
  163. * @dataProvider collectionProvider
  164. * @return void
  165. */
  166. public function testCollectionOperationsNoTableArg($collection) {
  167. $context = new EntityContext($this->request, [
  168. 'entity' => $collection,
  169. ]);
  170. $result = $context->val('0.title');
  171. $this->assertEquals('First post', $result);
  172. $result = $context->error('1.body');
  173. $this->assertEquals(['Not long enough'], $result);
  174. }
  175. /**
  176. * Data provider for testing collections.
  177. *
  178. * @return array
  179. */
  180. public static function collectionProvider() {
  181. $one = new Entity([
  182. 'title' => 'First post',
  183. 'body' => 'Stuff',
  184. 'user' => new Entity(['username' => 'mark'])
  185. ]);
  186. $one->errors('title', 'Required field');
  187. $two = new Entity([
  188. 'title' => 'Second post',
  189. 'body' => 'Some text',
  190. 'user' => new Entity(['username' => 'jose'])
  191. ]);
  192. $two->errors('body', 'Not long enough');
  193. return [
  194. 'array' => [[$one, $two]],
  195. 'basic iterator' => [new ArrayObject([$one, $two])],
  196. 'array iterator' => [new ArrayIterator([$one, $two])],
  197. 'collection' => [new Collection([$one, $two])],
  198. ];
  199. }
  200. /**
  201. * Test operations on a collection of entities.
  202. *
  203. * @dataProvider collectionProvider
  204. * @return void
  205. */
  206. public function testValOnCollections($collection) {
  207. $context = new EntityContext($this->request, [
  208. 'entity' => $collection,
  209. 'table' => 'Articles',
  210. ]);
  211. $result = $context->val('0.title');
  212. $this->assertEquals('First post', $result);
  213. $result = $context->val('0.user.username');
  214. $this->assertEquals('mark', $result);
  215. $result = $context->val('1.title');
  216. $this->assertEquals('Second post', $result);
  217. $result = $context->val('1.user.username');
  218. $this->assertEquals('jose', $result);
  219. $this->assertNull($context->val('nope'));
  220. $this->assertNull($context->val('99.title'));
  221. }
  222. /**
  223. * Test operations on a collection of entities when prefixing with the
  224. * table name
  225. *
  226. * @dataProvider collectionProvider
  227. * @return void
  228. */
  229. public function testValOnCollectionsWithRootName($collection) {
  230. $context = new EntityContext($this->request, [
  231. 'entity' => $collection,
  232. 'table' => 'Articles',
  233. ]);
  234. $result = $context->val('Articles.0.title');
  235. $this->assertEquals('First post', $result);
  236. $result = $context->val('Articles.0.user.username');
  237. $this->assertEquals('mark', $result);
  238. $result = $context->val('Articles.1.title');
  239. $this->assertEquals('Second post', $result);
  240. $result = $context->val('Articles.1.user.username');
  241. $this->assertEquals('jose', $result);
  242. $this->assertNull($context->val('Articles.99.title'));
  243. }
  244. /**
  245. * Test error operations on a collection of entities.
  246. *
  247. * @dataProvider collectionProvider
  248. * @return void
  249. */
  250. public function testErrorsOnCollections($collection) {
  251. $context = new EntityContext($this->request, [
  252. 'entity' => $collection,
  253. 'table' => 'Articles',
  254. ]);
  255. $this->assertTrue($context->hasError('0.title'));
  256. $this->assertEquals(['Required field'], $context->error('0.title'));
  257. $this->assertFalse($context->hasError('0.body'));
  258. $this->assertFalse($context->hasError('1.title'));
  259. $this->assertEquals(['Not long enough'], $context->error('1.body'));
  260. $this->assertTrue($context->hasError('1.body'));
  261. $this->assertFalse($context->hasError('nope'));
  262. $this->assertFalse($context->hasError('99.title'));
  263. }
  264. /**
  265. * Test schema operations on a collection of entities.
  266. *
  267. * @dataProvider collectionProvider
  268. * @return void
  269. */
  270. public function testSchemaOnCollections($collection) {
  271. $this->_setupTables();
  272. $context = new EntityContext($this->request, [
  273. 'entity' => $collection,
  274. 'table' => 'Articles',
  275. ]);
  276. $this->assertEquals('string', $context->type('0.title'));
  277. $this->assertEquals('text', $context->type('1.body'));
  278. $this->assertEquals('string', $context->type('0.user.username'));
  279. $this->assertEquals('string', $context->type('1.user.username'));
  280. $this->assertEquals('string', $context->type('99.title'));
  281. $this->assertNull($context->type('0.nope'));
  282. $expected = ['length' => 255, 'precision' => null];
  283. $this->assertEquals($expected, $context->attributes('0.user.username'));
  284. }
  285. /**
  286. * Test validation operations on a collection of entities.
  287. *
  288. * @dataProvider collectionProvider
  289. * @return void
  290. */
  291. public function testValidatorsOnCollections($collection) {
  292. $this->_setupTables();
  293. $context = new EntityContext($this->request, [
  294. 'entity' => $collection,
  295. 'table' => 'Articles',
  296. 'validator' => [
  297. 'Articles' => 'create',
  298. 'Users' => 'custom',
  299. ]
  300. ]);
  301. $this->assertFalse($context->isRequired('nope'));
  302. $this->assertTrue($context->isRequired('0.title'));
  303. $this->assertTrue($context->isRequired('0.user.username'));
  304. $this->assertFalse($context->isRequired('1.body'));
  305. $this->assertTrue($context->isRequired('99.title'));
  306. $this->assertFalse($context->isRequired('99.nope'));
  307. }
  308. /**
  309. * Test reading data.
  310. *
  311. * @return void
  312. */
  313. public function testValBasic() {
  314. $row = new Entity([
  315. 'title' => 'Test entity',
  316. 'body' => 'Something new'
  317. ]);
  318. $context = new EntityContext($this->request, [
  319. 'entity' => $row,
  320. 'table' => 'Articles',
  321. ]);
  322. $result = $context->val('title');
  323. $this->assertEquals($row->title, $result);
  324. $result = $context->val('body');
  325. $this->assertEquals($row->body, $result);
  326. $result = $context->val('nope');
  327. $this->assertNull($result);
  328. }
  329. /**
  330. * Test that val() reads from the request.
  331. *
  332. * @return void
  333. */
  334. public function testValReadsRequest() {
  335. $this->request->data = [
  336. 'title' => 'New title',
  337. 'notInEntity' => 'yes',
  338. ];
  339. $row = new Entity([
  340. 'title' => 'Test entity',
  341. 'body' => 'Something new'
  342. ]);
  343. $context = new EntityContext($this->request, [
  344. 'entity' => $row,
  345. 'table' => 'Articles',
  346. ]);
  347. $this->assertEquals('New title', $context->val('title'));
  348. $this->assertEquals('yes', $context->val('notInEntity'));
  349. $this->assertEquals($row->body, $context->val('body'));
  350. }
  351. /**
  352. * Test reading values from associated entities.
  353. *
  354. * @return void
  355. */
  356. public function testValAssociated() {
  357. $row = new Entity([
  358. 'title' => 'Test entity',
  359. 'user' => new Entity([
  360. 'username' => 'mark',
  361. 'fname' => 'Mark'
  362. ]),
  363. 'comments' => [
  364. new Entity(['comment' => 'Test comment']),
  365. new Entity(['comment' => 'Second comment']),
  366. ]
  367. ]);
  368. $context = new EntityContext($this->request, [
  369. 'entity' => $row,
  370. 'table' => 'Articles',
  371. ]);
  372. $result = $context->val('user.fname');
  373. $this->assertEquals($row->user->fname, $result);
  374. $result = $context->val('comments.0.comment');
  375. $this->assertEquals($row->comments[0]->comment, $result);
  376. $result = $context->val('comments.1.comment');
  377. $this->assertEquals($row->comments[1]->comment, $result);
  378. $result = $context->val('comments.0.nope');
  379. $this->assertNull($result);
  380. $result = $context->val('comments.0.nope.no_way');
  381. $this->assertNull($result);
  382. }
  383. /**
  384. * Test reading values from associated entities.
  385. *
  386. * @return void
  387. */
  388. public function testValAssociatedHasMany() {
  389. $row = new Entity([
  390. 'title' => 'First post',
  391. 'user' => new Entity([
  392. 'username' => 'mark',
  393. 'fname' => 'Mark',
  394. 'articles' => [
  395. new Entity(['title' => 'First post']),
  396. new Entity(['title' => 'Second post']),
  397. ]
  398. ]),
  399. ]);
  400. $context = new EntityContext($this->request, [
  401. 'entity' => $row,
  402. 'table' => 'Articles',
  403. ]);
  404. $result = $context->val('user.articles.0.title');
  405. $this->assertEquals('First post', $result);
  406. $result = $context->val('user.articles.1.title');
  407. $this->assertEquals('Second post', $result);
  408. }
  409. /**
  410. * Test reading values for magic _ids input
  411. *
  412. * @return void
  413. */
  414. public function testValAssociatedDefaultIds() {
  415. $row = new Entity([
  416. 'title' => 'First post',
  417. 'user' => new Entity([
  418. 'username' => 'mark',
  419. 'fname' => 'Mark',
  420. 'groups' => [
  421. new Entity(['title' => 'PHP', 'id' => 1]),
  422. new Entity(['title' => 'Javascript', 'id' => 2]),
  423. ]
  424. ]),
  425. ]);
  426. $context = new EntityContext($this->request, [
  427. 'entity' => $row,
  428. 'table' => 'Articles',
  429. ]);
  430. $result = $context->val('user.groups._ids');
  431. $this->assertEquals([1, 2], $result);
  432. }
  433. /**
  434. * Test reading values for magic _ids input
  435. *
  436. * @return void
  437. */
  438. public function testValAssociatedCustomIds() {
  439. $row = new Entity([
  440. 'title' => 'First post',
  441. 'user' => new Entity([
  442. 'username' => 'mark',
  443. 'fname' => 'Mark',
  444. 'groups' => [
  445. new Entity(['title' => 'PHP', 'thing' => 1]),
  446. new Entity(['title' => 'Javascript', 'thing' => 4]),
  447. ]
  448. ]),
  449. ]);
  450. $context = new EntityContext($this->request, [
  451. 'entity' => $row,
  452. 'table' => 'Articles',
  453. ]);
  454. TableRegistry::get('Users')->belongsToMany('Groups');
  455. TableRegistry::get('Groups')->primaryKey('thing');
  456. $result = $context->val('user.groups._ids');
  457. $this->assertEquals([1, 4], $result);
  458. }
  459. /**
  460. * Test validator as a string.
  461. *
  462. * @return void
  463. */
  464. public function testIsRequiredStringValidator() {
  465. $this->_setupTables();
  466. $context = new EntityContext($this->request, [
  467. 'entity' => new Entity(),
  468. 'table' => 'Articles',
  469. 'validator' => 'create',
  470. ]);
  471. $this->assertTrue($context->isRequired('title'));
  472. $this->assertFalse($context->isRequired('body'));
  473. $this->assertFalse($context->isRequired('Herp.derp.derp'));
  474. $this->assertFalse($context->isRequired('nope'));
  475. $this->assertFalse($context->isRequired(''));
  476. }
  477. /**
  478. * Test isRequired on associated entities.
  479. *
  480. * @return void
  481. */
  482. public function testIsRequiredAssociatedHasMany() {
  483. $this->_setupTables();
  484. $comments = TableRegistry::get('Comments');
  485. $validator = $comments->validator();
  486. $validator->add('user_id', 'number', [
  487. 'rule' => 'numeric',
  488. ]);
  489. $row = new Entity([
  490. 'title' => 'My title',
  491. 'comments' => [
  492. new Entity(['comment' => 'First comment']),
  493. new Entity(['comment' => 'Second comment']),
  494. ]
  495. ]);
  496. $context = new EntityContext($this->request, [
  497. 'entity' => $row,
  498. 'table' => 'Articles',
  499. 'validator' => 'default',
  500. ]);
  501. $this->assertTrue($context->isRequired('comments.0.user_id'));
  502. $this->assertFalse($context->isRequired('comments.0.other'));
  503. $this->assertFalse($context->isRequired('user.0.other'));
  504. $this->assertFalse($context->isRequired(''));
  505. }
  506. /**
  507. * Test isRequired on associated entities with custom validators.
  508. *
  509. * @return void
  510. */
  511. public function testIsRequiredAssociatedValidator() {
  512. $this->_setupTables();
  513. $row = new Entity([
  514. 'title' => 'My title',
  515. 'comments' => [
  516. new Entity(['comment' => 'First comment']),
  517. new Entity(['comment' => 'Second comment']),
  518. ]
  519. ]);
  520. $context = new EntityContext($this->request, [
  521. 'entity' => $row,
  522. 'table' => 'Articles',
  523. 'validator' => [
  524. 'Articles' => 'create',
  525. 'Comments' => 'custom'
  526. ]
  527. ]);
  528. $this->assertTrue($context->isRequired('title'));
  529. $this->assertFalse($context->isRequired('body'));
  530. $this->assertTrue($context->isRequired('comments.0.comment'));
  531. $this->assertTrue($context->isRequired('comments.1.comment'));
  532. }
  533. /**
  534. * Test isRequired on associated entities.
  535. *
  536. * @return void
  537. */
  538. public function testIsRequiredAssociatedBelongsTo() {
  539. $this->_setupTables();
  540. $row = new Entity([
  541. 'title' => 'My title',
  542. 'user' => new Entity(['username' => 'Mark']),
  543. ]);
  544. $context = new EntityContext($this->request, [
  545. 'entity' => $row,
  546. 'table' => 'Articles',
  547. 'validator' => [
  548. 'Articles' => 'create',
  549. 'Users' => 'custom'
  550. ]
  551. ]);
  552. $this->assertTrue($context->isRequired('user.username'));
  553. $this->assertFalse($context->isRequired('user.first_name'));
  554. }
  555. /**
  556. * Test type() basic
  557. *
  558. * @return void
  559. */
  560. public function testType() {
  561. $this->_setupTables();
  562. $row = new Entity([
  563. 'title' => 'My title',
  564. 'body' => 'Some content',
  565. ]);
  566. $context = new EntityContext($this->request, [
  567. 'entity' => $row,
  568. 'table' => 'Articles',
  569. ]);
  570. $this->assertEquals('string', $context->type('title'));
  571. $this->assertEquals('text', $context->type('body'));
  572. $this->assertEquals('integer', $context->type('user_id'));
  573. $this->assertNull($context->type('nope'));
  574. }
  575. /**
  576. * Test getting types for associated records.
  577. *
  578. * @return void
  579. */
  580. public function testTypeAssociated() {
  581. $this->_setupTables();
  582. $row = new Entity([
  583. 'title' => 'My title',
  584. 'user' => new Entity(['username' => 'Mark']),
  585. ]);
  586. $context = new EntityContext($this->request, [
  587. 'entity' => $row,
  588. 'table' => 'Articles',
  589. ]);
  590. $this->assertEquals('string', $context->type('user.username'));
  591. $this->assertEquals('text', $context->type('user.bio'));
  592. $this->assertNull($context->type('user.nope'));
  593. }
  594. /**
  595. * Test attributes for fields.
  596. *
  597. * @return void
  598. */
  599. public function testAttributes() {
  600. $this->_setupTables();
  601. $row = new Entity([
  602. 'title' => 'My title',
  603. 'user' => new Entity(['username' => 'Mark']),
  604. ]);
  605. $context = new EntityContext($this->request, [
  606. 'entity' => $row,
  607. 'table' => 'Articles',
  608. ]);
  609. $expected = [
  610. 'length' => 255, 'precision' => null
  611. ];
  612. $this->assertEquals($expected, $context->attributes('title'));
  613. $expected = [
  614. 'length' => null, 'precision' => null
  615. ];
  616. $this->assertEquals($expected, $context->attributes('body'));
  617. $expected = [
  618. 'length' => 10, 'precision' => 3
  619. ];
  620. $this->assertEquals($expected, $context->attributes('user.rating'));
  621. }
  622. /**
  623. * Test hasError
  624. *
  625. * @return void
  626. */
  627. public function testHasError() {
  628. $this->_setupTables();
  629. $row = new Entity([
  630. 'title' => 'My title',
  631. 'user' => new Entity(['username' => 'Mark']),
  632. ]);
  633. $row->errors('title', []);
  634. $row->errors('body', 'Gotta have one');
  635. $row->errors('user_id', ['Required field']);
  636. $context = new EntityContext($this->request, [
  637. 'entity' => $row,
  638. 'table' => 'Articles',
  639. ]);
  640. $this->assertFalse($context->hasError('title'));
  641. $this->assertFalse($context->hasError('nope'));
  642. $this->assertTrue($context->hasError('body'));
  643. $this->assertTrue($context->hasError('user_id'));
  644. }
  645. /**
  646. * Test hasError on associated records
  647. *
  648. * @return void
  649. */
  650. public function testHasErrorAssociated() {
  651. $this->_setupTables();
  652. $row = new Entity([
  653. 'title' => 'My title',
  654. 'user' => new Entity(['username' => 'Mark']),
  655. ]);
  656. $row->errors('title', []);
  657. $row->errors('body', 'Gotta have one');
  658. $row->user->errors('username', ['Required']);
  659. $context = new EntityContext($this->request, [
  660. 'entity' => $row,
  661. 'table' => 'Articles',
  662. ]);
  663. $this->assertTrue($context->hasError('user.username'));
  664. $this->assertFalse($context->hasError('user.nope'));
  665. $this->assertFalse($context->hasError('no.nope'));
  666. }
  667. /**
  668. * Test error
  669. *
  670. * @return void
  671. */
  672. public function testError() {
  673. $this->_setupTables();
  674. $row = new Entity([
  675. 'title' => 'My title',
  676. 'user' => new Entity(['username' => 'Mark']),
  677. ]);
  678. $row->errors('title', []);
  679. $row->errors('body', 'Gotta have one');
  680. $row->errors('user_id', ['Required field']);
  681. $row->user->errors('username', ['Required']);
  682. $context = new EntityContext($this->request, [
  683. 'entity' => $row,
  684. 'table' => 'Articles',
  685. ]);
  686. $this->assertEquals([], $context->error('title'));
  687. $expected = ['Gotta have one'];
  688. $this->assertEquals($expected, $context->error('body'));
  689. $expected = ['Required'];
  690. $this->assertEquals($expected, $context->error('user.username'));
  691. }
  692. /**
  693. * Setup tables for tests.
  694. *
  695. * @return void
  696. */
  697. protected function _setupTables() {
  698. $articles = TableRegistry::get('Articles');
  699. $articles->belongsTo('Users');
  700. $articles->hasMany('Comments');
  701. $comments = TableRegistry::get('Comments');
  702. $users = TableRegistry::get('Users');
  703. $users->hasMany('Articles');
  704. $articles->schema([
  705. 'id' => ['type' => 'integer', 'length' => 11, 'null' => false],
  706. 'title' => ['type' => 'string', 'length' => 255],
  707. 'user_id' => ['type' => 'integer', 'length' => 11, 'null' => false],
  708. 'body' => ['type' => 'text']
  709. ]);
  710. $users->schema([
  711. 'id' => ['type' => 'integer', 'length' => 11],
  712. 'username' => ['type' => 'string', 'length' => 255],
  713. 'bio' => ['type' => 'text'],
  714. 'rating' => ['type' => 'decimal', 'length' => 10, 'precision' => 3],
  715. ]);
  716. $validator = new Validator();
  717. $validator->add('title', 'minlength', [
  718. 'rule' => ['minlength', 10]
  719. ])
  720. ->add('body', 'maxlength', [
  721. 'rule' => ['maxlength', 1000]
  722. ])->allowEmpty('body');
  723. $articles->validator('create', $validator);
  724. $validator = new Validator();
  725. $validator->add('username', 'length', [
  726. 'rule' => ['minlength', 10]
  727. ]);
  728. $users->validator('custom', $validator);
  729. $validator = new Validator();
  730. $validator->add('comment', 'length', [
  731. 'rule' => ['minlength', 10]
  732. ]);
  733. $comments->validator('custom', $validator);
  734. }
  735. /**
  736. * Test the fieldnames method.
  737. *
  738. * @return void
  739. */
  740. public function testFieldNames() {
  741. $context = new EntityContext($this->request, [
  742. 'entity' => new Entity(),
  743. 'table' => 'Articles',
  744. ]);
  745. $articles = TableRegistry::get('Articles');
  746. $this->assertEquals($articles->schema()->columns(), $context->fieldNames());
  747. }
  748. }