EntityContextTest.php 14 KB

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