EntityContextTest.php 15 KB

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