EntityContextTest.php 44 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485
  1. <?php
  2. declare(strict_types=1);
  3. /**
  4. * CakePHP(tm) : Rapid Development Framework (https://cakephp.org)
  5. * Copyright (c) Cake Software Foundation, Inc. (https://cakefoundation.org)
  6. *
  7. * Licensed under The MIT License
  8. * For full copyright and license information, please see the LICENSE.txt
  9. * Redistributions of files must retain the above copyright notice.
  10. *
  11. * @copyright Copyright (c) Cake Software Foundation, Inc. (https://cakefoundation.org)
  12. * @link https://cakephp.org CakePHP(tm) Project
  13. * @since 3.0.0
  14. * @license https://opensource.org/licenses/mit-license.php MIT License
  15. */
  16. namespace Cake\Test\TestCase\View\Form;
  17. use ArrayIterator;
  18. use ArrayObject;
  19. use Cake\Collection\Collection;
  20. use Cake\Http\ServerRequest;
  21. use Cake\ORM\Entity;
  22. use Cake\TestSuite\TestCase;
  23. use Cake\Validation\Validator;
  24. use Cake\View\Form\EntityContext;
  25. use TestApp\Model\Entity\Article;
  26. use TestApp\Model\Entity\ArticlesTag;
  27. use TestApp\Model\Entity\Tag;
  28. /**
  29. * Entity context test case.
  30. */
  31. class EntityContextTest extends TestCase
  32. {
  33. /**
  34. * Fixtures to use.
  35. *
  36. * @var array
  37. */
  38. public $fixtures = ['core.Articles', 'core.Comments', 'core.ArticlesTags', 'core.Tags'];
  39. /**
  40. * @var \Cake\Http\ServerRequest
  41. */
  42. protected $request;
  43. /**
  44. * setup method.
  45. *
  46. * @return void
  47. */
  48. public function setUp(): void
  49. {
  50. parent::setUp();
  51. $this->request = new ServerRequest();
  52. }
  53. /**
  54. * tests getRequiredMessage
  55. *
  56. * @return void
  57. */
  58. public function testGetRequiredMessage()
  59. {
  60. $this->_setupTables();
  61. $context = new EntityContext($this->request, [
  62. 'entity' => new Article(),
  63. 'table' => 'Articles',
  64. 'validator' => 'create',
  65. ]);
  66. $this->assertNull($context->getRequiredMessage('body'));
  67. $this->assertSame('Don\'t forget a title!', $context->getRequiredMessage('title'));
  68. }
  69. /**
  70. * Test getting entity back from context.
  71. *
  72. * @return void
  73. */
  74. public function testEntity()
  75. {
  76. $row = new Article();
  77. $context = new EntityContext($this->request, [
  78. 'entity' => $row,
  79. ]);
  80. $this->assertSame($row, $context->entity());
  81. }
  82. /**
  83. * Test getting primary key data.
  84. *
  85. * @return void
  86. */
  87. public function testPrimaryKey()
  88. {
  89. $row = new Article();
  90. $context = new EntityContext($this->request, [
  91. 'entity' => $row,
  92. ]);
  93. $this->assertEquals(['id'], $context->primaryKey());
  94. }
  95. /**
  96. * Test isPrimaryKey
  97. *
  98. * @return void
  99. */
  100. public function testIsPrimaryKey()
  101. {
  102. $this->_setupTables();
  103. $row = new Article();
  104. $context = new EntityContext($this->request, [
  105. 'entity' => $row,
  106. ]);
  107. $this->assertTrue($context->isPrimaryKey('id'));
  108. $this->assertFalse($context->isPrimaryKey('title'));
  109. $this->assertTrue($context->isPrimaryKey('1.id'));
  110. $this->assertTrue($context->isPrimaryKey('Articles.1.id'));
  111. $this->assertTrue($context->isPrimaryKey('comments.0.id'));
  112. $this->assertTrue($context->isPrimaryKey('1.comments.0.id'));
  113. $this->assertFalse($context->isPrimaryKey('1.comments.0.comment'));
  114. $this->assertFalse($context->isPrimaryKey('Articles.1.comments.0.comment'));
  115. $this->assertTrue($context->isPrimaryKey('tags.0._joinData.article_id'));
  116. $this->assertTrue($context->isPrimaryKey('tags.0._joinData.tag_id'));
  117. }
  118. /**
  119. * Test isCreate on a single entity.
  120. *
  121. * @return void
  122. */
  123. public function testIsCreateSingle()
  124. {
  125. $row = new Article();
  126. $context = new EntityContext($this->request, [
  127. 'entity' => $row,
  128. ]);
  129. $this->assertTrue($context->isCreate());
  130. $row->isNew(false);
  131. $this->assertFalse($context->isCreate());
  132. $row->isNew(true);
  133. $this->assertTrue($context->isCreate());
  134. }
  135. /**
  136. * Test isCreate on a collection.
  137. *
  138. * @dataProvider collectionProvider
  139. * @return void
  140. */
  141. public function testIsCreateCollection($collection)
  142. {
  143. $context = new EntityContext($this->request, [
  144. 'entity' => $collection,
  145. ]);
  146. $this->assertTrue($context->isCreate());
  147. }
  148. /**
  149. * Test an invalid table scope throws an error.
  150. *
  151. */
  152. public function testInvalidTable()
  153. {
  154. $this->expectException(\RuntimeException::class);
  155. $this->expectExceptionMessage('Unable to find table class for current entity');
  156. $row = new \stdClass();
  157. $context = new EntityContext($this->request, [
  158. 'entity' => $row,
  159. ]);
  160. }
  161. /**
  162. * Tests that passing a plain entity will give an error as it cannot be matched
  163. *
  164. */
  165. public function testDefaultEntityError()
  166. {
  167. $this->expectException(\RuntimeException::class);
  168. $this->expectExceptionMessage('Unable to find table class for current entity');
  169. $context = new EntityContext($this->request, [
  170. 'entity' => new Entity(),
  171. ]);
  172. }
  173. /**
  174. * Tests that the table can be derived from the entity source if it is present
  175. *
  176. * @return void
  177. */
  178. public function testTableFromEntitySource()
  179. {
  180. $entity = new Entity();
  181. $entity->setSource('Articles');
  182. $context = new EntityContext($this->request, [
  183. 'entity' => $entity,
  184. ]);
  185. $expected = ['id', 'author_id', 'title', 'body', 'published'];
  186. $this->assertEquals($expected, $context->fieldNames());
  187. }
  188. /**
  189. * Test operations with no entity.
  190. *
  191. * @return void
  192. */
  193. public function testOperationsNoEntity()
  194. {
  195. $context = new EntityContext($this->request, [
  196. 'table' => 'Articles',
  197. ]);
  198. $this->assertNull($context->val('title'));
  199. $this->assertFalse($context->isRequired('title'));
  200. $this->assertFalse($context->hasError('title'));
  201. $this->assertSame('string', $context->type('title'));
  202. $this->assertEquals([], $context->error('title'));
  203. $attrs = $context->attributes('title');
  204. $this->assertArrayHasKey('length', $attrs);
  205. $this->assertArrayHasKey('precision', $attrs);
  206. }
  207. /**
  208. * Test operations that lack a table argument.
  209. *
  210. * @return void
  211. */
  212. public function testOperationsNoTableArg()
  213. {
  214. $row = new Article([
  215. 'title' => 'Test entity',
  216. 'body' => 'Something new',
  217. ]);
  218. $row->setError('title', ['Title is required.']);
  219. $context = new EntityContext($this->request, [
  220. 'entity' => $row,
  221. ]);
  222. $result = $context->val('title');
  223. $this->assertEquals($row->title, $result);
  224. $result = $context->error('title');
  225. $this->assertEquals($row->getError('title'), $result);
  226. $this->assertTrue($context->hasError('title'));
  227. }
  228. /**
  229. * Test collection operations that lack a table argument.
  230. *
  231. * @dataProvider collectionProvider
  232. * @return void
  233. */
  234. public function testCollectionOperationsNoTableArg($collection)
  235. {
  236. $context = new EntityContext($this->request, [
  237. 'entity' => $collection,
  238. ]);
  239. $result = $context->val('0.title');
  240. $this->assertSame('First post', $result);
  241. $result = $context->error('1.body');
  242. $this->assertEquals(['Not long enough'], $result);
  243. }
  244. /**
  245. * Data provider for testing collections.
  246. *
  247. * @return array
  248. */
  249. public static function collectionProvider()
  250. {
  251. $one = new Article([
  252. 'title' => 'First post',
  253. 'body' => 'Stuff',
  254. 'user' => new Entity(['username' => 'mark']),
  255. ]);
  256. $one->setError('title', 'Required field');
  257. $two = new Article([
  258. 'title' => 'Second post',
  259. 'body' => 'Some text',
  260. 'user' => new Entity(['username' => 'jose']),
  261. ]);
  262. $two->setError('body', 'Not long enough');
  263. return [
  264. 'array' => [[$one, $two]],
  265. 'basic iterator' => [new ArrayObject([$one, $two])],
  266. 'array iterator' => [new ArrayIterator([$one, $two])],
  267. 'collection' => [new Collection([$one, $two])],
  268. ];
  269. }
  270. /**
  271. * Test operations on a collection of entities.
  272. *
  273. * @dataProvider collectionProvider
  274. * @return void
  275. */
  276. public function testValOnCollections($collection)
  277. {
  278. $context = new EntityContext($this->request, [
  279. 'entity' => $collection,
  280. 'table' => 'Articles',
  281. ]);
  282. $result = $context->val('0.title');
  283. $this->assertSame('First post', $result);
  284. $result = $context->val('0.user.username');
  285. $this->assertSame('mark', $result);
  286. $result = $context->val('1.title');
  287. $this->assertSame('Second post', $result);
  288. $result = $context->val('1.user.username');
  289. $this->assertSame('jose', $result);
  290. $this->assertNull($context->val('nope'));
  291. $this->assertNull($context->val('99.title'));
  292. }
  293. /**
  294. * Test operations on a collection of entities when prefixing with the
  295. * table name
  296. *
  297. * @dataProvider collectionProvider
  298. * @return void
  299. */
  300. public function testValOnCollectionsWithRootName($collection)
  301. {
  302. $context = new EntityContext($this->request, [
  303. 'entity' => $collection,
  304. 'table' => 'Articles',
  305. ]);
  306. $result = $context->val('Articles.0.title');
  307. $this->assertSame('First post', $result);
  308. $result = $context->val('Articles.0.user.username');
  309. $this->assertSame('mark', $result);
  310. $result = $context->val('Articles.1.title');
  311. $this->assertSame('Second post', $result);
  312. $result = $context->val('Articles.1.user.username');
  313. $this->assertSame('jose', $result);
  314. $this->assertNull($context->val('Articles.99.title'));
  315. }
  316. /**
  317. * Test error operations on a collection of entities.
  318. *
  319. * @dataProvider collectionProvider
  320. * @return void
  321. */
  322. public function testErrorsOnCollections($collection)
  323. {
  324. $context = new EntityContext($this->request, [
  325. 'entity' => $collection,
  326. 'table' => 'Articles',
  327. ]);
  328. $this->assertTrue($context->hasError('0.title'));
  329. $this->assertEquals(['Required field'], $context->error('0.title'));
  330. $this->assertFalse($context->hasError('0.body'));
  331. $this->assertFalse($context->hasError('1.title'));
  332. $this->assertEquals(['Not long enough'], $context->error('1.body'));
  333. $this->assertTrue($context->hasError('1.body'));
  334. $this->assertFalse($context->hasError('nope'));
  335. $this->assertFalse($context->hasError('99.title'));
  336. }
  337. /**
  338. * Test schema operations on a collection of entities.
  339. *
  340. * @dataProvider collectionProvider
  341. * @return void
  342. */
  343. public function testSchemaOnCollections($collection)
  344. {
  345. $this->_setupTables();
  346. $context = new EntityContext($this->request, [
  347. 'entity' => $collection,
  348. 'table' => 'Articles',
  349. ]);
  350. $this->assertSame('string', $context->type('0.title'));
  351. $this->assertSame('text', $context->type('1.body'));
  352. $this->assertSame('string', $context->type('0.user.username'));
  353. $this->assertSame('string', $context->type('1.user.username'));
  354. $this->assertSame('string', $context->type('99.title'));
  355. $this->assertNull($context->type('0.nope'));
  356. $expected = ['length' => 255, 'precision' => null];
  357. $this->assertEquals($expected, $context->attributes('0.user.username'));
  358. }
  359. /**
  360. * Test validation operations on a collection of entities.
  361. *
  362. * @dataProvider collectionProvider
  363. * @return void
  364. */
  365. public function testValidatorsOnCollections($collection)
  366. {
  367. $this->_setupTables();
  368. $context = new EntityContext($this->request, [
  369. 'entity' => $collection,
  370. 'table' => 'Articles',
  371. 'validator' => [
  372. 'Articles' => 'create',
  373. 'Users' => 'custom',
  374. ],
  375. ]);
  376. $this->assertFalse($context->isRequired('nope'));
  377. $this->assertTrue($context->isRequired('0.title'));
  378. $this->assertTrue($context->isRequired('0.user.username'));
  379. $this->assertFalse($context->isRequired('1.body'));
  380. $this->assertTrue($context->isRequired('99.title'));
  381. $this->assertFalse($context->isRequired('99.nope'));
  382. }
  383. /**
  384. * Test reading data.
  385. *
  386. * @return void
  387. */
  388. public function testValBasic()
  389. {
  390. $row = new Article([
  391. 'title' => 'Test entity',
  392. 'body' => 'Something new',
  393. ]);
  394. $context = new EntityContext($this->request, [
  395. 'entity' => $row,
  396. 'table' => 'Articles',
  397. ]);
  398. $result = $context->val('title');
  399. $this->assertEquals($row->title, $result);
  400. $result = $context->val('body');
  401. $this->assertEquals($row->body, $result);
  402. $result = $context->val('nope');
  403. $this->assertNull($result);
  404. }
  405. /**
  406. * Test default values when entity is an array.
  407. *
  408. * @return void
  409. */
  410. public function testValDefaultArray()
  411. {
  412. $context = new EntityContext($this->request, [
  413. 'entity' => new Article([
  414. 'prop' => ['title' => 'foo'],
  415. ]),
  416. 'table' => 'Articles',
  417. ]);
  418. $this->assertSame('foo', $context->val('prop.title', ['default' => 'bar']));
  419. $this->assertSame('bar', $context->val('prop.nope', ['default' => 'bar']));
  420. }
  421. /**
  422. * Test reading array values from an entity.
  423. *
  424. * @return void
  425. */
  426. public function testValGetArrayValue()
  427. {
  428. $row = new Article([
  429. 'title' => 'Test entity',
  430. 'types' => [1, 2, 3],
  431. 'tag' => [
  432. 'name' => 'Test tag',
  433. ],
  434. 'author' => new Entity([
  435. 'roles' => ['admin', 'publisher'],
  436. 'aliases' => new ArrayObject(['dave', 'david']),
  437. ]),
  438. ]);
  439. $context = new EntityContext($this->request, [
  440. 'entity' => $row,
  441. 'table' => 'Articles',
  442. ]);
  443. $result = $context->val('types');
  444. $this->assertEquals($row->types, $result);
  445. $result = $context->val('author.roles');
  446. $this->assertEquals($row->author->roles, $result);
  447. $result = $context->val('tag.name');
  448. $this->assertEquals($row->tag['name'], $result);
  449. $result = $context->val('author.aliases.0');
  450. $this->assertEquals($row->author->aliases[0], $result, 'ArrayAccess can be read');
  451. $this->assertNull($context->val('author.aliases.3'));
  452. $this->assertNull($context->val('tag.nope'));
  453. $this->assertNull($context->val('author.roles.3'));
  454. }
  455. /**
  456. * Test that val() reads from the request.
  457. *
  458. * @return void
  459. */
  460. public function testValReadsRequest()
  461. {
  462. $this->request = $this->request->withParsedBody([
  463. 'title' => 'New title',
  464. 'notInEntity' => 'yes',
  465. ]);
  466. $row = new Article([
  467. 'title' => 'Test entity',
  468. 'body' => 'Something new',
  469. ]);
  470. $context = new EntityContext($this->request, [
  471. 'entity' => $row,
  472. 'table' => 'Articles',
  473. ]);
  474. $this->assertSame('New title', $context->val('title'));
  475. $this->assertSame('yes', $context->val('notInEntity'));
  476. $this->assertEquals($row->body, $context->val('body'));
  477. }
  478. /**
  479. * Test reading values from associated entities.
  480. *
  481. * @return void
  482. */
  483. public function testValAssociated()
  484. {
  485. $row = new Article([
  486. 'title' => 'Test entity',
  487. 'user' => new Entity([
  488. 'username' => 'mark',
  489. 'fname' => 'Mark',
  490. ]),
  491. 'comments' => [
  492. new Entity(['comment' => 'Test comment']),
  493. new Entity(['comment' => 'Second comment']),
  494. ],
  495. ]);
  496. $context = new EntityContext($this->request, [
  497. 'entity' => $row,
  498. 'table' => 'Articles',
  499. ]);
  500. $result = $context->val('user.fname');
  501. $this->assertEquals($row->user->fname, $result);
  502. $result = $context->val('comments.0.comment');
  503. $this->assertEquals($row->comments[0]->comment, $result);
  504. $result = $context->val('comments.1.comment');
  505. $this->assertEquals($row->comments[1]->comment, $result);
  506. $result = $context->val('comments.0.nope');
  507. $this->assertNull($result);
  508. $result = $context->val('comments.0.nope.no_way');
  509. $this->assertNull($result);
  510. }
  511. /**
  512. * Tests that trying to get values from missing associations returns null
  513. *
  514. * @return void
  515. */
  516. public function testValMissingAssociation()
  517. {
  518. $row = new Article([
  519. 'id' => 1,
  520. ]);
  521. $context = new EntityContext($this->request, [
  522. 'entity' => $row,
  523. 'table' => 'Articles',
  524. ]);
  525. $result = $context->val('id');
  526. $this->assertEquals($row->id, $result);
  527. $this->assertNull($context->val('profile.id'));
  528. }
  529. /**
  530. * Test reading values from associated entities.
  531. *
  532. * @return void
  533. */
  534. public function testValAssociatedHasMany()
  535. {
  536. $row = new Article([
  537. 'title' => 'First post',
  538. 'user' => new Entity([
  539. 'username' => 'mark',
  540. 'fname' => 'Mark',
  541. 'articles' => [
  542. new Article(['title' => 'First post']),
  543. new Article(['title' => 'Second post']),
  544. ],
  545. ]),
  546. ]);
  547. $context = new EntityContext($this->request, [
  548. 'entity' => $row,
  549. 'table' => 'Articles',
  550. ]);
  551. $result = $context->val('user.articles.0.title');
  552. $this->assertSame('First post', $result);
  553. $result = $context->val('user.articles.1.title');
  554. $this->assertSame('Second post', $result);
  555. }
  556. /**
  557. * Test reading values for magic _ids input
  558. *
  559. * @return void
  560. */
  561. public function testValAssociatedDefaultIds()
  562. {
  563. $row = new Article([
  564. 'title' => 'First post',
  565. 'user' => new Entity([
  566. 'username' => 'mark',
  567. 'fname' => 'Mark',
  568. 'groups' => [
  569. new Entity(['title' => 'PHP', 'id' => 1]),
  570. new Entity(['title' => 'Javascript', 'id' => 2]),
  571. ],
  572. ]),
  573. ]);
  574. $context = new EntityContext($this->request, [
  575. 'entity' => $row,
  576. 'table' => 'Articles',
  577. ]);
  578. $result = $context->val('user.groups._ids');
  579. $this->assertEquals([1, 2], $result);
  580. }
  581. /**
  582. * Test reading values for magic _ids input
  583. *
  584. * @return void
  585. */
  586. public function testValAssociatedCustomIds()
  587. {
  588. $this->_setupTables();
  589. $row = new Article([
  590. 'title' => 'First post',
  591. 'user' => new Entity([
  592. 'username' => 'mark',
  593. 'fname' => 'Mark',
  594. 'groups' => [
  595. new Entity(['title' => 'PHP', 'thing' => 1]),
  596. new Entity(['title' => 'Javascript', 'thing' => 4]),
  597. ],
  598. ]),
  599. ]);
  600. $context = new EntityContext($this->request, [
  601. 'entity' => $row,
  602. 'table' => 'Articles',
  603. ]);
  604. $this->getTableLocator()->get('Users')->belongsToMany('Groups');
  605. $this->getTableLocator()->get('Groups')->setPrimaryKey('thing');
  606. $result = $context->val('user.groups._ids');
  607. $this->assertEquals([1, 4], $result);
  608. }
  609. /**
  610. * Test getting default value from table schema.
  611. *
  612. * @return void
  613. */
  614. public function testValSchemaDefault()
  615. {
  616. $table = $this->getTableLocator()->get('Articles');
  617. $column = $table->getSchema()->getColumn('title');
  618. $table->getSchema()->addColumn('title', ['default' => 'default title'] + $column);
  619. $row = $table->newEmptyEntity();
  620. $context = new EntityContext($this->request, [
  621. 'entity' => $row,
  622. 'table' => 'Articles',
  623. ]);
  624. $result = $context->val('title');
  625. $this->assertSame('default title', $result);
  626. }
  627. /**
  628. * Test getting association default value from table schema.
  629. *
  630. * @return void
  631. */
  632. public function testValAssociatedSchemaDefault()
  633. {
  634. $table = $this->getTableLocator()->get('Articles');
  635. $associatedTable = $table->hasMany('Comments')->getTarget();
  636. $column = $associatedTable->getSchema()->getColumn('comment');
  637. $associatedTable->getSchema()->addColumn('comment', ['default' => 'default comment'] + $column);
  638. $row = $table->newEmptyEntity();
  639. $context = new EntityContext($this->request, [
  640. 'entity' => $row,
  641. 'table' => 'Articles',
  642. ]);
  643. $result = $context->val('comments.0.comment');
  644. $this->assertSame('default comment', $result);
  645. }
  646. /**
  647. * Test getting association join table default value from table schema.
  648. *
  649. * @return void
  650. */
  651. public function testValAssociatedJoinTableSchemaDefault()
  652. {
  653. $table = $this->getTableLocator()->get('Articles');
  654. $joinTable = $table
  655. ->belongsToMany('Tags')
  656. ->setThrough('ArticlesTags')
  657. ->junction();
  658. $joinTable->getSchema()->addColumn('column', [
  659. 'default' => 'default join table column value',
  660. 'type' => 'text',
  661. ]);
  662. $row = $table->newEmptyEntity();
  663. $context = new EntityContext($this->request, [
  664. 'entity' => $row,
  665. 'table' => 'Articles',
  666. ]);
  667. $result = $context->val('tags.0._joinData.column');
  668. $this->assertSame('default join table column value', $result);
  669. }
  670. /**
  671. * Test validator for boolean fields.
  672. *
  673. * @return void
  674. */
  675. public function testIsRequiredBooleanField()
  676. {
  677. $this->_setupTables();
  678. $context = new EntityContext($this->request, [
  679. 'entity' => new Entity(),
  680. 'table' => 'Articles',
  681. ]);
  682. $articles = $this->getTableLocator()->get('Articles');
  683. $articles->getSchema()->addColumn('comments_on', [
  684. 'type' => 'boolean',
  685. ]);
  686. $validator = $articles->getValidator();
  687. $validator->add('comments_on', 'is_bool', [
  688. 'rule' => 'boolean',
  689. ]);
  690. $articles->setValidator('default', $validator);
  691. $this->assertFalse($context->isRequired('title'));
  692. }
  693. /**
  694. * Test validator as a string.
  695. *
  696. * @return void
  697. */
  698. public function testIsRequiredStringValidator()
  699. {
  700. $this->_setupTables();
  701. $context = new EntityContext($this->request, [
  702. 'entity' => new Entity(),
  703. 'table' => 'Articles',
  704. 'validator' => 'create',
  705. ]);
  706. $this->assertTrue($context->isRequired('title'));
  707. $this->assertFalse($context->isRequired('body'));
  708. $this->assertFalse($context->isRequired('Herp.derp.derp'));
  709. $this->assertFalse($context->isRequired('nope'));
  710. $this->assertFalse($context->isRequired(''));
  711. }
  712. /**
  713. * Test isRequired on associated entities.
  714. *
  715. * @return void
  716. */
  717. public function testIsRequiredAssociatedHasMany()
  718. {
  719. $this->_setupTables();
  720. $comments = $this->getTableLocator()->get('Comments');
  721. $validator = $comments->getValidator();
  722. $validator->add('user_id', 'number', [
  723. 'rule' => 'numeric',
  724. ]);
  725. $row = new Article([
  726. 'title' => 'My title',
  727. 'comments' => [
  728. new Entity(['comment' => 'First comment']),
  729. new Entity(['comment' => 'Second comment']),
  730. ],
  731. ]);
  732. $context = new EntityContext($this->request, [
  733. 'entity' => $row,
  734. 'table' => 'Articles',
  735. 'validator' => 'default',
  736. ]);
  737. $this->assertTrue($context->isRequired('comments.0.user_id'));
  738. $this->assertFalse($context->isRequired('comments.0.other'));
  739. $this->assertFalse($context->isRequired('user.0.other'));
  740. $this->assertFalse($context->isRequired(''));
  741. }
  742. /**
  743. * Test isRequired on associated entities with boolean fields
  744. *
  745. * @return void
  746. */
  747. public function testIsRequiredAssociatedHasManyBoolean()
  748. {
  749. $this->_setupTables();
  750. $comments = $this->getTableLocator()->get('Comments');
  751. $comments->getSchema()->addColumn('starred', 'boolean');
  752. $comments->getValidator()->add('starred', 'valid', ['rule' => 'boolean']);
  753. $row = new Article([
  754. 'title' => 'My title',
  755. 'comments' => [
  756. new Entity(['comment' => 'First comment']),
  757. ],
  758. ]);
  759. $context = new EntityContext($this->request, [
  760. 'entity' => $row,
  761. 'table' => 'Articles',
  762. 'validator' => 'default',
  763. ]);
  764. $this->assertFalse($context->isRequired('comments.0.starred'));
  765. }
  766. /**
  767. * Test isRequired on associated entities with custom validators.
  768. *
  769. * Ensures that missing associations use the correct entity class
  770. * so provider methods work correctly.
  771. *
  772. * @return void
  773. */
  774. public function testIsRequiredAssociatedCustomValidator()
  775. {
  776. $this->_setupTables();
  777. $users = $this->getTableLocator()->get('Users');
  778. $articles = $this->getTableLocator()->get('Articles');
  779. $validator = $articles->getValidator();
  780. $validator->notEmpty('title', 'nope', function ($context) {
  781. return $context['providers']['entity']->isRequired();
  782. });
  783. $articles->setValidator('default', $validator);
  784. $row = new Entity([
  785. 'username' => 'mark',
  786. ]);
  787. $context = new EntityContext($this->request, [
  788. 'entity' => $row,
  789. 'table' => 'Users',
  790. 'validator' => 'default',
  791. ]);
  792. $this->assertTrue($context->isRequired('articles.0.title'));
  793. }
  794. /**
  795. * Test isRequired on associated entities.
  796. *
  797. * @return void
  798. */
  799. public function testIsRequiredAssociatedHasManyMissingObject()
  800. {
  801. $this->_setupTables();
  802. $comments = $this->getTableLocator()->get('Comments');
  803. $validator = $comments->getValidator();
  804. $validator->allowEmpty('comment', function ($context) {
  805. return $context['providers']['entity']->isNew();
  806. });
  807. $row = new Article([
  808. 'title' => 'My title',
  809. 'comments' => [
  810. new Entity(['comment' => 'First comment'], ['markNew' => false]),
  811. ],
  812. ]);
  813. $context = new EntityContext($this->request, [
  814. 'entity' => $row,
  815. 'table' => 'Articles',
  816. 'validator' => 'default',
  817. ]);
  818. $this->assertTrue(
  819. $context->isRequired('comments.0.comment'),
  820. 'comment is required as object is not new'
  821. );
  822. $this->assertFalse(
  823. $context->isRequired('comments.1.comment'),
  824. 'comment is not required as missing object is "new"'
  825. );
  826. }
  827. /**
  828. * Test isRequired on associated entities with custom validators.
  829. *
  830. * @return void
  831. */
  832. public function testIsRequiredAssociatedValidator()
  833. {
  834. $this->_setupTables();
  835. $row = new Article([
  836. 'title' => 'My title',
  837. 'comments' => [
  838. new Entity(['comment' => 'First comment']),
  839. new Entity(['comment' => 'Second comment']),
  840. ],
  841. ]);
  842. $context = new EntityContext($this->request, [
  843. 'entity' => $row,
  844. 'table' => 'Articles',
  845. 'validator' => [
  846. 'Articles' => 'create',
  847. 'Comments' => 'custom',
  848. ],
  849. ]);
  850. $this->assertTrue($context->isRequired('title'));
  851. $this->assertFalse($context->isRequired('body'));
  852. $this->assertTrue($context->isRequired('comments.0.comment'));
  853. $this->assertTrue($context->isRequired('comments.1.comment'));
  854. }
  855. /**
  856. * Test isRequired on associated entities.
  857. *
  858. * @return void
  859. */
  860. public function testIsRequiredAssociatedBelongsTo()
  861. {
  862. $this->_setupTables();
  863. $row = new Article([
  864. 'title' => 'My title',
  865. 'user' => new Entity(['username' => 'Mark']),
  866. ]);
  867. $context = new EntityContext($this->request, [
  868. 'entity' => $row,
  869. 'table' => 'Articles',
  870. 'validator' => [
  871. 'Articles' => 'create',
  872. 'Users' => 'custom',
  873. ],
  874. ]);
  875. $this->assertTrue($context->isRequired('user.username'));
  876. $this->assertFalse($context->isRequired('user.first_name'));
  877. }
  878. /**
  879. * Test isRequired on associated join table entities.
  880. *
  881. * @return void
  882. */
  883. public function testIsRequiredAssociatedJoinTable()
  884. {
  885. $this->_setupTables();
  886. $row = new Article([
  887. 'tags' => [
  888. new Tag([
  889. '_joinData' => new ArticlesTag([
  890. 'article_id' => 1,
  891. 'tag_id' => 2,
  892. ]),
  893. ]),
  894. ],
  895. ]);
  896. $context = new EntityContext($this->request, [
  897. 'entity' => $row,
  898. 'table' => 'Articles',
  899. ]);
  900. $this->assertTrue($context->isRequired('tags.0._joinData.article_id'));
  901. $this->assertTrue($context->isRequired('tags.0._joinData.tag_id'));
  902. }
  903. /**
  904. * Test type() basic
  905. *
  906. * @return void
  907. */
  908. public function testType()
  909. {
  910. $this->_setupTables();
  911. $row = new Article([
  912. 'title' => 'My title',
  913. 'body' => 'Some content',
  914. ]);
  915. $context = new EntityContext($this->request, [
  916. 'entity' => $row,
  917. 'table' => 'Articles',
  918. ]);
  919. $this->assertSame('string', $context->type('title'));
  920. $this->assertSame('text', $context->type('body'));
  921. $this->assertSame('integer', $context->type('user_id'));
  922. $this->assertNull($context->type('nope'));
  923. }
  924. /**
  925. * Test getting types for associated records.
  926. *
  927. * @return void
  928. */
  929. public function testTypeAssociated()
  930. {
  931. $this->_setupTables();
  932. $row = new Article([
  933. 'title' => 'My title',
  934. 'user' => new Entity(['username' => 'Mark']),
  935. ]);
  936. $context = new EntityContext($this->request, [
  937. 'entity' => $row,
  938. 'table' => 'Articles',
  939. ]);
  940. $this->assertSame('string', $context->type('user.username'));
  941. $this->assertSame('text', $context->type('user.bio'));
  942. $this->assertNull($context->type('user.nope'));
  943. }
  944. /**
  945. * Test getting types for associated join data records.
  946. *
  947. * @return void
  948. */
  949. public function testTypeAssociatedJoinData()
  950. {
  951. $this->_setupTables();
  952. $row = new Article([
  953. 'tags' => [
  954. new Tag([
  955. '_joinData' => new ArticlesTag([
  956. 'article_id' => 1,
  957. 'tag_id' => 2,
  958. ]),
  959. ]),
  960. ],
  961. ]);
  962. $context = new EntityContext($this->request, [
  963. 'entity' => $row,
  964. 'table' => 'Articles',
  965. ]);
  966. $this->assertSame('integer', $context->type('tags.0._joinData.article_id'));
  967. $this->assertNull($context->type('tags.0._joinData.non_existent'));
  968. // tests the fallback behavior
  969. $this->assertSame('integer', $context->type('tags.0._joinData._joinData.article_id'));
  970. $this->assertSame('integer', $context->type('tags.0._joinData.non_existent.article_id'));
  971. $this->assertNull($context->type('tags.0._joinData._joinData.non_existent'));
  972. $this->assertNull($context->type('tags.0._joinData.non_existent'));
  973. }
  974. /**
  975. * Test attributes for fields.
  976. *
  977. * @return void
  978. */
  979. public function testAttributes()
  980. {
  981. $this->_setupTables();
  982. $row = new Article([
  983. 'title' => 'My title',
  984. 'user' => new Entity(['username' => 'Mark']),
  985. 'tags' => [
  986. new Tag([
  987. '_joinData' => new ArticlesTag([
  988. 'article_id' => 1,
  989. 'tag_id' => 2,
  990. ]),
  991. ]),
  992. ],
  993. ]);
  994. $context = new EntityContext($this->request, [
  995. 'entity' => $row,
  996. 'table' => 'Articles',
  997. ]);
  998. $expected = [
  999. 'length' => 255, 'precision' => null,
  1000. ];
  1001. $this->assertEquals($expected, $context->attributes('title'));
  1002. $expected = [
  1003. 'length' => null, 'precision' => null,
  1004. ];
  1005. $this->assertEquals($expected, $context->attributes('body'));
  1006. $expected = [
  1007. 'length' => 10, 'precision' => 3,
  1008. ];
  1009. $this->assertEquals($expected, $context->attributes('user.rating'));
  1010. $expected = [
  1011. 'length' => 11, 'precision' => null,
  1012. ];
  1013. $this->assertEquals($expected, $context->attributes('tags.0._joinData.article_id'));
  1014. }
  1015. /**
  1016. * Test hasError
  1017. *
  1018. * @return void
  1019. */
  1020. public function testHasError()
  1021. {
  1022. $this->_setupTables();
  1023. $row = new Article([
  1024. 'title' => 'My title',
  1025. 'user' => new Entity(['username' => 'Mark']),
  1026. ]);
  1027. $row->setError('title', []);
  1028. $row->setError('body', 'Gotta have one');
  1029. $row->setError('user_id', ['Required field']);
  1030. $context = new EntityContext($this->request, [
  1031. 'entity' => $row,
  1032. 'table' => 'Articles',
  1033. ]);
  1034. $this->assertFalse($context->hasError('title'));
  1035. $this->assertFalse($context->hasError('nope'));
  1036. $this->assertTrue($context->hasError('body'));
  1037. $this->assertTrue($context->hasError('user_id'));
  1038. }
  1039. /**
  1040. * Test hasError on associated records
  1041. *
  1042. * @return void
  1043. */
  1044. public function testHasErrorAssociated()
  1045. {
  1046. $this->_setupTables();
  1047. $row = new Article([
  1048. 'title' => 'My title',
  1049. 'user' => new Entity(['username' => 'Mark']),
  1050. ]);
  1051. $row->setError('title', []);
  1052. $row->setError('body', 'Gotta have one');
  1053. $row->user->setError('username', ['Required']);
  1054. $context = new EntityContext($this->request, [
  1055. 'entity' => $row,
  1056. 'table' => 'Articles',
  1057. ]);
  1058. $this->assertTrue($context->hasError('user.username'));
  1059. $this->assertFalse($context->hasError('user.nope'));
  1060. $this->assertFalse($context->hasError('no.nope'));
  1061. }
  1062. /**
  1063. * Test error
  1064. *
  1065. * @return void
  1066. */
  1067. public function testError()
  1068. {
  1069. $this->_setupTables();
  1070. $row = new Article([
  1071. 'title' => 'My title',
  1072. 'user' => new Entity(['username' => 'Mark']),
  1073. ]);
  1074. $row->setError('title', []);
  1075. $row->setError('body', 'Gotta have one');
  1076. $row->setError('user_id', ['Required field']);
  1077. $row->user->setError('username', ['Required']);
  1078. $context = new EntityContext($this->request, [
  1079. 'entity' => $row,
  1080. 'table' => 'Articles',
  1081. ]);
  1082. $this->assertEquals([], $context->error('title'));
  1083. $expected = ['Gotta have one'];
  1084. $this->assertEquals($expected, $context->error('body'));
  1085. $expected = ['Required'];
  1086. $this->assertEquals($expected, $context->error('user.username'));
  1087. }
  1088. /**
  1089. * Test error on associated entities.
  1090. *
  1091. * @return void
  1092. */
  1093. public function testErrorAssociatedHasMany()
  1094. {
  1095. $this->_setupTables();
  1096. $comments = $this->getTableLocator()->get('Comments');
  1097. $row = new Article([
  1098. 'title' => 'My title',
  1099. 'comments' => [
  1100. new Entity(['comment' => '']),
  1101. new Entity(['comment' => 'Second comment']),
  1102. ],
  1103. ]);
  1104. $row->comments[0]->setError('comment', ['Is required']);
  1105. $row->comments[0]->setError('article_id', ['Is required']);
  1106. $context = new EntityContext($this->request, [
  1107. 'entity' => $row,
  1108. 'table' => 'Articles',
  1109. 'validator' => 'default',
  1110. ]);
  1111. $this->assertEquals([], $context->error('title'));
  1112. $this->assertEquals([], $context->error('comments.0.user_id'));
  1113. $this->assertEquals([], $context->error('comments.0'));
  1114. $this->assertEquals(['Is required'], $context->error('comments.0.comment'));
  1115. $this->assertEquals(['Is required'], $context->error('comments.0.article_id'));
  1116. $this->assertEquals([], $context->error('comments.1'));
  1117. $this->assertEquals([], $context->error('comments.1.comment'));
  1118. $this->assertEquals([], $context->error('comments.1.article_id'));
  1119. }
  1120. /**
  1121. * Test error on associated join table entities.
  1122. *
  1123. * @return void
  1124. */
  1125. public function testErrorAssociatedJoinTable()
  1126. {
  1127. $this->_setupTables();
  1128. $row = new Article([
  1129. 'tags' => [
  1130. new Tag([
  1131. '_joinData' => new ArticlesTag([
  1132. 'article_id' => 1,
  1133. ]),
  1134. ]),
  1135. ],
  1136. ]);
  1137. $row->tags[0]->_joinData->setError('tag_id', ['Is required']);
  1138. $context = new EntityContext($this->request, [
  1139. 'entity' => $row,
  1140. 'table' => 'Articles',
  1141. ]);
  1142. $this->assertEquals([], $context->error('tags.0._joinData.article_id'));
  1143. $this->assertEquals(['Is required'], $context->error('tags.0._joinData.tag_id'));
  1144. }
  1145. /**
  1146. * Test error on nested validation
  1147. *
  1148. * @return void
  1149. */
  1150. public function testErrorNestedValidator()
  1151. {
  1152. $this->_setupTables();
  1153. $row = new Article([
  1154. 'title' => 'My title',
  1155. 'options' => ['subpages' => '']
  1156. ]);
  1157. $row->setError('options', ['subpages' => ['_empty' => 'required value']]);
  1158. $context = new EntityContext($this->request, [
  1159. 'entity' => $row,
  1160. 'table' => 'Articles',
  1161. ]);
  1162. $expected = ['_empty' => 'required value'];
  1163. $this->assertEquals($expected, $context->error('options.subpages'));
  1164. }
  1165. /**
  1166. * Test error on nested validation
  1167. *
  1168. * @return void
  1169. */
  1170. public function testErrorAssociatedNestedValidator()
  1171. {
  1172. $this->_setupTables();
  1173. $tagOne = new Tag(['name' => 'first-post']);
  1174. $tagTwo = new Tag(['name' => 'second-post']);
  1175. $tagOne->setError(
  1176. 'metadata',
  1177. ['description' => ['_empty' => 'required value']]
  1178. );
  1179. $row = new Article([
  1180. 'title' => 'My title',
  1181. 'tags' => [
  1182. $tagOne,
  1183. $tagTwo
  1184. ]
  1185. ]);
  1186. $context = new EntityContext($this->request, [
  1187. 'entity' => $row,
  1188. 'table' => 'Articles',
  1189. ]);
  1190. $expected = ['_empty' => 'required value'];
  1191. $this->assertSame([], $context->error('tags.0.notthere'));
  1192. $this->assertSame([], $context->error('tags.1.notthere'));
  1193. $this->assertEquals($expected, $context->error('tags.0.metadata.description'));
  1194. }
  1195. /**
  1196. * Setup tables for tests.
  1197. *
  1198. * @return void
  1199. */
  1200. protected function _setupTables()
  1201. {
  1202. $articles = $this->getTableLocator()->get('Articles');
  1203. $articles->belongsTo('Users');
  1204. $articles->belongsToMany('Tags');
  1205. $articles->hasMany('Comments');
  1206. $articles->setEntityClass(Article::class);
  1207. $articlesTags = $this->getTableLocator()->get('ArticlesTags');
  1208. $comments = $this->getTableLocator()->get('Comments');
  1209. $users = $this->getTableLocator()->get('Users');
  1210. $users->hasMany('Articles');
  1211. $articles->setSchema([
  1212. 'id' => ['type' => 'integer', 'length' => 11, 'null' => false],
  1213. 'title' => ['type' => 'string', 'length' => 255],
  1214. 'user_id' => ['type' => 'integer', 'length' => 11, 'null' => false],
  1215. 'body' => ['type' => 'crazy_text', 'baseType' => 'text'],
  1216. '_constraints' => ['primary' => ['type' => 'primary', 'columns' => ['id']]],
  1217. ]);
  1218. $articlesTags->setSchema([
  1219. 'article_id' => ['type' => 'integer', 'length' => 11, 'null' => false],
  1220. 'tag_id' => ['type' => 'integer', 'length' => 11, 'null' => false],
  1221. '_constraints' => ['unique_tag' => ['type' => 'primary', 'columns' => ['article_id', 'tag_id']]],
  1222. ]);
  1223. $users->setSchema([
  1224. 'id' => ['type' => 'integer', 'length' => 11],
  1225. 'username' => ['type' => 'string', 'length' => 255],
  1226. 'bio' => ['type' => 'text'],
  1227. 'rating' => ['type' => 'decimal', 'length' => 10, 'precision' => 3],
  1228. ]);
  1229. $validator = new Validator();
  1230. $validator->requirePresence('title', true, 'Don\'t forget a title!');
  1231. $validator->add('title', 'minlength', [
  1232. 'rule' => ['minlength', 10],
  1233. ])
  1234. ->add('body', 'maxlength', [
  1235. 'rule' => ['maxlength', 1000],
  1236. ])->allowEmpty('body');
  1237. $articles->setValidator('create', $validator);
  1238. $validator = new Validator();
  1239. $validator->add('username', 'length', [
  1240. 'rule' => ['minlength', 10],
  1241. ]);
  1242. $users->setValidator('custom', $validator);
  1243. $validator = new Validator();
  1244. $validator->add('comment', 'length', [
  1245. 'rule' => ['minlength', 10],
  1246. ]);
  1247. $comments->setValidator('custom', $validator);
  1248. $validator = new Validator();
  1249. $validator->requirePresence('article_id', 'create');
  1250. $validator->requirePresence('tag_id', 'create');
  1251. $articlesTags->setValidator('default', $validator);
  1252. }
  1253. /**
  1254. * Test the fieldnames method.
  1255. *
  1256. * @return void
  1257. */
  1258. public function testFieldNames()
  1259. {
  1260. $context = new EntityContext($this->request, [
  1261. 'entity' => new Entity(),
  1262. 'table' => 'Articles',
  1263. ]);
  1264. $articles = $this->getTableLocator()->get('Articles');
  1265. $this->assertEquals($articles->getSchema()->columns(), $context->fieldNames());
  1266. }
  1267. /**
  1268. * Test automatic entity provider setting
  1269. *
  1270. * @return void
  1271. */
  1272. public function testValidatorEntityProvider()
  1273. {
  1274. $row = new Article([
  1275. 'title' => 'Test entity',
  1276. 'body' => 'Something new',
  1277. ]);
  1278. $context = new EntityContext($this->request, [
  1279. 'entity' => $row,
  1280. 'table' => 'Articles',
  1281. ]);
  1282. $context->isRequired('title');
  1283. $articles = $this->getTableLocator()->get('Articles');
  1284. $this->assertSame($row, $articles->getValidator()->getProvider('entity'));
  1285. $row = new Article([
  1286. 'title' => 'First post',
  1287. 'user' => new Entity([
  1288. 'username' => 'mark',
  1289. 'fname' => 'Mark',
  1290. 'articles' => [
  1291. new Article(['title' => 'First post']),
  1292. new Article(['title' => 'Second post']),
  1293. ],
  1294. ]),
  1295. ]);
  1296. $context = new EntityContext($this->request, [
  1297. 'entity' => $row,
  1298. 'table' => 'Articles',
  1299. ]);
  1300. $validator = $articles->getValidator();
  1301. $context->isRequired('user.articles.0.title');
  1302. $this->assertSame($row->user->articles[0], $validator->getProvider('entity'));
  1303. $context->isRequired('user.articles.1.title');
  1304. $this->assertSame($row->user->articles[1], $validator->getProvider('entity'));
  1305. $context->isRequired('title');
  1306. $this->assertSame($row, $validator->getProvider('entity'));
  1307. }
  1308. }