EntityContextTest.php 14 KB

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