EntityContextTest.php 23 KB

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