EntityContextTest.php 17 KB

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