| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800 |
- <?php
- /**
- * CakePHP(tm) : Rapid Development Framework (https://cakephp.org)
- * Copyright (c) Cake Software Foundation, Inc. (https://cakefoundation.org)
- *
- * Licensed under The MIT License
- * For full copyright and license information, please see the LICENSE.txt
- * Redistributions of files must retain the above copyright notice.
- *
- * @copyright Copyright (c) Cake Software Foundation, Inc. (https://cakefoundation.org)
- * @link https://cakephp.org CakePHP(tm) Project
- * @since 3.0.0
- * @license https://opensource.org/licenses/mit-license.php MIT License
- */
- namespace Cake\Test\TestCase\ORM;
- use Cake\Database\Driver\Sqlite;
- use Cake\Database\Driver\Sqlserver;
- use Cake\Datasource\ConnectionManager;
- use Cake\ORM\Entity;
- use Cake\ORM\Marshaller;
- use Cake\ORM\Query;
- use Cake\ORM\Table;
- use Cake\TestSuite\TestCase;
- use PDO;
- /**
- * Test entity for mass assignment.
- */
- class OpenArticleEntity extends Entity
- {
- protected $_accessible = [
- '*' => true,
- ];
- }
- /**
- * Integration tests for table operations involving composite keys
- */
- class CompositeKeyTest extends TestCase
- {
- /**
- * Fixture to be used
- *
- * @var array
- */
- public $fixtures = [
- 'core.CompositeIncrements',
- 'core.SiteArticles',
- 'core.SiteArticlesTags',
- 'core.SiteAuthors',
- 'core.SiteTags',
- ];
- /**
- * setUp method
- *
- * @return void
- */
- public function setUp()
- {
- parent::setUp();
- $this->connection = ConnectionManager::get('test');
- }
- /**
- * Data provider for the two types of strategies HasOne implements
- *
- * @return array
- */
- public function strategiesProviderHasOne()
- {
- return [['join'], ['select']];
- }
- /**
- * Data provider for the two types of strategies HasMany implements
- *
- * @return array
- */
- public function strategiesProviderHasMany()
- {
- return [['subquery'], ['select']];
- }
- /**
- * Data provider for the two types of strategies BelongsTo implements
- *
- * @return array
- */
- public function strategiesProviderBelongsTo()
- {
- return [['join'], ['select']];
- }
- /**
- * Data provider for the two types of strategies BelongsToMany implements
- *
- * @return array
- */
- public function strategiesProviderBelongsToMany()
- {
- return [['subquery'], ['select']];
- }
- /**
- * Test that you cannot save rows with composite keys if some columns are missing.
- *
- * @group save
- * @return void
- */
- public function testSaveNewErrorCompositeKeyNoIncrement()
- {
- $this->expectException(\RuntimeException::class);
- $this->expectExceptionMessage('Cannot insert row, some of the primary key values are missing');
- $articles = $this->getTableLocator()->get('SiteArticles');
- $article = $articles->newEntity(['site_id' => 1, 'author_id' => 1, 'title' => 'testing']);
- $articles->save($article);
- }
- /**
- * Test that saving into composite primary keys where one column is missing & autoIncrement works.
- *
- * SQLite is skipped because it doesn't support autoincrement composite keys.
- *
- * @group save
- * @return void
- */
- public function testSaveNewCompositeKeyIncrement()
- {
- $this->skipIfSqlite();
- $table = $this->getTableLocator()->get('CompositeIncrements');
- $thing = $table->newEntity(['account_id' => 3, 'name' => 'new guy']);
- $this->assertSame($thing, $table->save($thing));
- $this->assertNotEmpty($thing->id, 'Primary key should have been populated');
- $this->assertSame(3, $thing->account_id);
- }
- /**
- * Tests that HasMany associations are correctly eager loaded and results
- * correctly nested when multiple foreignKeys are used
- *
- * @dataProvider strategiesProviderHasMany
- * @return void
- */
- public function testHasManyEager($strategy)
- {
- $table = $this->getTableLocator()->get('SiteAuthors');
- $table->hasMany('SiteArticles', [
- 'propertyName' => 'articles',
- 'strategy' => $strategy,
- 'sort' => ['SiteArticles.id' => 'asc'],
- 'foreignKey' => ['author_id', 'site_id'],
- ]);
- $query = new Query($this->connection, $table);
- $results = $query->select()
- ->contain('SiteArticles')
- ->enableHydration(false)
- ->toArray();
- $expected = [
- [
- 'id' => 1,
- 'name' => 'mark',
- 'site_id' => 1,
- 'articles' => [
- [
- 'id' => 1,
- 'title' => 'First Article',
- 'body' => 'First Article Body',
- 'author_id' => 1,
- 'site_id' => 1,
- ],
- ],
- ],
- [
- 'id' => 2,
- 'name' => 'juan',
- 'site_id' => 2,
- 'articles' => [],
- ],
- [
- 'id' => 3,
- 'name' => 'jose',
- 'site_id' => 2,
- 'articles' => [
- [
- 'id' => 2,
- 'title' => 'Second Article',
- 'body' => 'Second Article Body',
- 'author_id' => 3,
- 'site_id' => 2,
- ],
- ],
- ],
- [
- 'id' => 4,
- 'name' => 'andy',
- 'site_id' => 1,
- 'articles' => [],
- ],
- ];
- $this->assertEquals($expected, $results);
- $results = $query->repository($table)
- ->select()
- ->contain(['SiteArticles' => ['conditions' => ['SiteArticles.id' => 2]]])
- ->enableHydration(false)
- ->toArray();
- $expected[0]['articles'] = [];
- $this->assertEquals($expected, $results);
- $this->assertEquals($table->getAssociation('SiteArticles')->getStrategy(), $strategy);
- }
- /**
- * Tests that BelongsToMany associations are correctly eager loaded when multiple
- * foreignKeys are used
- *
- * @dataProvider strategiesProviderBelongsToMany
- * @return void
- */
- public function testBelongsToManyEager($strategy)
- {
- $articles = $this->getTableLocator()->get('SiteArticles');
- $tags = $this->getTableLocator()->get('SiteTags');
- $junction = $this->getTableLocator()->get('SiteArticlesTags');
- $articles->belongsToMany('SiteTags', [
- 'strategy' => $strategy,
- 'targetTable' => $tags,
- 'propertyName' => 'tags',
- 'through' => 'SiteArticlesTags',
- 'sort' => ['SiteTags.id' => 'asc'],
- 'foreignKey' => ['article_id', 'site_id'],
- 'targetForeignKey' => ['tag_id', 'site_id'],
- ]);
- $query = new Query($this->connection, $articles);
- $results = $query->select()->contain('SiteTags')->enableHydration(false)->toArray();
- $expected = [
- [
- 'id' => 1,
- 'author_id' => 1,
- 'title' => 'First Article',
- 'body' => 'First Article Body',
- 'site_id' => 1,
- 'tags' => [
- [
- 'id' => 1,
- 'name' => 'tag1',
- '_joinData' => ['article_id' => 1, 'tag_id' => 1, 'site_id' => 1],
- 'site_id' => 1,
- ],
- [
- 'id' => 3,
- 'name' => 'tag3',
- '_joinData' => ['article_id' => 1, 'tag_id' => 3, 'site_id' => 1],
- 'site_id' => 1,
- ],
- ],
- ],
- [
- 'id' => 2,
- 'title' => 'Second Article',
- 'body' => 'Second Article Body',
- 'author_id' => 3,
- 'site_id' => 2,
- 'tags' => [
- [
- 'id' => 4,
- 'name' => 'tag4',
- '_joinData' => ['article_id' => 2, 'tag_id' => 4, 'site_id' => 2],
- 'site_id' => 2,
- ],
- ],
- ],
- [
- 'id' => 3,
- 'title' => 'Third Article',
- 'body' => 'Third Article Body',
- 'author_id' => 1,
- 'site_id' => 2,
- 'tags' => [],
- ],
- [
- 'id' => 4,
- 'title' => 'Fourth Article',
- 'body' => 'Fourth Article Body',
- 'author_id' => 3,
- 'site_id' => 1,
- 'tags' => [
- [
- 'id' => 1,
- 'name' => 'tag1',
- '_joinData' => ['article_id' => 4, 'tag_id' => 1, 'site_id' => 1],
- 'site_id' => 1,
- ],
- ],
- ],
- ];
- $this->assertEquals($expected, $results);
- $this->assertEquals($articles->getAssociation('SiteTags')->getStrategy(), $strategy);
- }
- /**
- * Tests loading belongsTo with composite keys
- *
- * @dataProvider strategiesProviderBelongsTo
- * @return void
- */
- public function testBelongsToEager($strategy)
- {
- $table = $this->getTableLocator()->get('SiteArticles');
- $table->belongsTo('SiteAuthors', [
- 'propertyName' => 'author',
- 'strategy' => $strategy,
- 'foreignKey' => ['author_id', 'site_id'],
- ]);
- $query = new Query($this->connection, $table);
- $results = $query->select()
- ->where(['SiteArticles.id IN' => [1, 2]])
- ->contain('SiteAuthors')
- ->enableHydration(false)
- ->toArray();
- $expected = [
- [
- 'id' => 1,
- 'author_id' => 1,
- 'site_id' => 1,
- 'title' => 'First Article',
- 'body' => 'First Article Body',
- 'author' => [
- 'id' => 1,
- 'name' => 'mark',
- 'site_id' => 1,
- ],
- ],
- [
- 'id' => 2,
- 'author_id' => 3,
- 'site_id' => 2,
- 'title' => 'Second Article',
- 'body' => 'Second Article Body',
- 'author' => [
- 'id' => 3,
- 'name' => 'jose',
- 'site_id' => 2,
- ],
- ],
- ];
- $this->assertEquals($expected, $results);
- }
- /**
- * Tests loading hasOne with composite keys
- *
- * @dataProvider strategiesProviderHasOne
- * @return void
- */
- public function testHasOneEager($strategy)
- {
- $table = $this->getTableLocator()->get('SiteAuthors');
- $table->hasOne('SiteArticles', [
- 'propertyName' => 'first_article',
- 'strategy' => $strategy,
- 'foreignKey' => ['author_id', 'site_id'],
- ]);
- $query = new Query($this->connection, $table);
- $results = $query->select()
- ->where(['SiteAuthors.id IN' => [1, 3]])
- ->contain('SiteArticles')
- ->enableHydration(false)
- ->toArray();
- $expected = [
- [
- 'id' => 1,
- 'name' => 'mark',
- 'site_id' => 1,
- 'first_article' => [
- 'id' => 1,
- 'author_id' => 1,
- 'site_id' => 1,
- 'title' => 'First Article',
- 'body' => 'First Article Body',
- ],
- ],
- [
- 'id' => 3,
- 'name' => 'jose',
- 'site_id' => 2,
- 'first_article' => [
- 'id' => 2,
- 'author_id' => 3,
- 'site_id' => 2,
- 'title' => 'Second Article',
- 'body' => 'Second Article Body',
- ],
- ],
- ];
- $this->assertEquals($expected, $results);
- }
- /**
- * Tests that it is possible to insert a new row using the save method
- * if the entity has composite primary key
- *
- * @group save
- * @return void
- */
- public function testSaveNewEntity()
- {
- $entity = new Entity([
- 'id' => 5,
- 'site_id' => 1,
- 'title' => 'Fifth Article',
- 'body' => 'Fifth Article Body',
- 'author_id' => 3,
- ]);
- $table = $this->getTableLocator()->get('SiteArticles');
- $this->assertSame($entity, $table->save($entity));
- $this->assertEquals($entity->id, 5);
- $row = $table->find('all')->where(['id' => 5, 'site_id' => 1])->first();
- $this->assertEquals($entity->toArray(), $row->toArray());
- }
- /**
- * Tests that it is possible to insert a new row using the save method
- * if the entity has composite primary key
- *
- * @group save
- * @return void
- */
- public function testSaveNewEntityMissingKey()
- {
- $this->expectException(\RuntimeException::class);
- $this->expectExceptionMessage('Cannot insert row, some of the primary key values are missing. Got (5, ), expecting (id, site_id)');
- $entity = new Entity([
- 'id' => 5,
- 'title' => 'Fifth Article',
- 'body' => 'Fifth Article Body',
- 'author_id' => 3,
- ]);
- $table = $this->getTableLocator()->get('SiteArticles');
- $table->save($entity);
- }
- /**
- * Test simple delete with composite primary key
- *
- * @return void
- */
- public function testDelete()
- {
- $table = $this->getTableLocator()->get('SiteAuthors');
- $table->save(new Entity(['id' => 1, 'site_id' => 2]));
- $entity = $table->get([1, 1]);
- $result = $table->delete($entity);
- $this->assertTrue($result);
- $this->assertEquals(4, $table->find('all')->count());
- $this->assertEmpty($table->find()->where(['id' => 1, 'site_id' => 1])->first());
- }
- /**
- * Test delete with dependent records having composite keys
- *
- * @return void
- */
- public function testDeleteDependent()
- {
- $table = $this->getTableLocator()->get('SiteAuthors');
- $table->hasMany('SiteArticles', [
- 'foreignKey' => ['author_id', 'site_id'],
- 'dependent' => true,
- ]);
- $entity = $table->get([3, 2]);
- $result = $table->delete($entity);
- $query = $table->getAssociation('SiteArticles')->find('all', [
- 'conditions' => [
- 'author_id' => $entity->id,
- 'site_id' => $entity->site_id,
- ],
- ]);
- $this->assertNull($query->all()->first(), 'Should not find any rows.');
- }
- /**
- * Test generating a list of entities from a list of composite ids
- *
- * @return void
- */
- public function testOneGenerateBelongsToManyEntitiesFromIds()
- {
- $articles = $this->getTableLocator()->get('SiteArticles');
- $articles->setEntityClass(__NAMESPACE__ . '\OpenArticleEntity');
- $tags = $this->getTableLocator()->get('SiteTags');
- $junction = $this->getTableLocator()->get('SiteArticlesTags');
- $articles->belongsToMany('SiteTags', [
- 'targetTable' => $tags,
- 'propertyName' => 'tags',
- 'through' => 'SiteArticlesTags',
- 'foreignKey' => ['article_id', 'site_id'],
- 'targetForeignKey' => ['tag_id', 'site_id'],
- ]);
- $data = [
- 'title' => 'Haz tags',
- 'body' => 'Some content here',
- 'tags' => ['_ids' => [[1, 1], [2, 2], [3, 1]]],
- ];
- $marshall = new Marshaller($articles);
- $result = $marshall->one($data, ['associated' => ['SiteTags']]);
- $this->assertCount(3, $result->tags);
- $this->assertInstanceOf('Cake\ORM\Entity', $result->tags[0]);
- $this->assertInstanceOf('Cake\ORM\Entity', $result->tags[1]);
- $this->assertInstanceOf('Cake\ORM\Entity', $result->tags[2]);
- $data = [
- 'title' => 'Haz tags',
- 'body' => 'Some content here',
- 'tags' => ['_ids' => [1, 2, 3]],
- ];
- $marshall = new Marshaller($articles);
- $result = $marshall->one($data, ['associated' => ['SiteTags']]);
- $this->assertEmpty($result->tags);
- }
- /**
- * Tests find('list') with composite keys
- *
- * @return void
- */
- public function testFindListCompositeKeys()
- {
- $table = new Table([
- 'table' => 'site_authors',
- 'connection' => $this->connection,
- ]);
- $table->setDisplayField('name');
- $query = $table->find('list')
- ->enableHydration(false)
- ->order('id');
- $expected = [
- '1;1' => 'mark',
- '2;2' => 'juan',
- '3;2' => 'jose',
- '4;1' => 'andy',
- ];
- $this->assertEquals($expected, $query->toArray());
- $table->setDisplayField(['name', 'site_id']);
- $query = $table->find('list')
- ->enableHydration(false)
- ->order('id');
- $expected = [
- '1;1' => 'mark;1',
- '2;2' => 'juan;2',
- '3;2' => 'jose;2',
- '4;1' => 'andy;1',
- ];
- $this->assertEquals($expected, $query->toArray());
- $query = $table->find('list', ['groupField' => ['site_id', 'site_id']])
- ->enableHydration(false)
- ->order('id');
- $expected = [
- '1;1' => [
- '1;1' => 'mark;1',
- '4;1' => 'andy;1',
- ],
- '2;2' => [
- '2;2' => 'juan;2',
- '3;2' => 'jose;2',
- ],
- ];
- $this->assertEquals($expected, $query->toArray());
- }
- /**
- * Tests find('threaded') with composite keys
- *
- * @return void
- */
- public function testFindThreadedCompositeKeys()
- {
- $table = $this->getTableLocator()->get('SiteAuthors');
- $query = $this->getMockBuilder('\Cake\ORM\Query')
- ->setMethods(['_addDefaultFields', 'execute'])
- ->setConstructorArgs([null, $table])
- ->getMock();
- $items = new \Cake\Datasource\ResultSetDecorator([
- ['id' => 1, 'name' => 'a', 'site_id' => 1, 'parent_id' => null],
- ['id' => 2, 'name' => 'a', 'site_id' => 2, 'parent_id' => null],
- ['id' => 3, 'name' => 'a', 'site_id' => 1, 'parent_id' => 1],
- ['id' => 4, 'name' => 'a', 'site_id' => 2, 'parent_id' => 2],
- ['id' => 5, 'name' => 'a', 'site_id' => 2, 'parent_id' => 4],
- ['id' => 6, 'name' => 'a', 'site_id' => 1, 'parent_id' => 2],
- ['id' => 7, 'name' => 'a', 'site_id' => 1, 'parent_id' => 3],
- ['id' => 8, 'name' => 'a', 'site_id' => 2, 'parent_id' => 4],
- ]);
- $query->find('threaded', ['parentField' => ['parent_id', 'site_id']]);
- $formatter = $query->getResultFormatters()[0];
- $expected = [
- [
- 'id' => 1,
- 'name' => 'a',
- 'site_id' => 1,
- 'parent_id' => null,
- 'children' => [
- [
- 'id' => 3,
- 'name' => 'a',
- 'site_id' => 1,
- 'parent_id' => 1,
- 'children' => [
- [
- 'id' => 7,
- 'name' => 'a',
- 'site_id' => 1,
- 'parent_id' => 3,
- 'children' => [],
- ],
- ],
- ],
- ],
- ],
- [
- 'id' => 2,
- 'name' => 'a',
- 'site_id' => 2,
- 'parent_id' => null,
- 'children' => [
- [
- 'id' => 4,
- 'name' => 'a',
- 'site_id' => 2,
- 'parent_id' => 2,
- 'children' => [
- [
- 'id' => 5,
- 'name' => 'a',
- 'site_id' => 2,
- 'parent_id' => 4,
- 'children' => [],
- ],
- [
- 'id' => 8,
- 'name' => 'a',
- 'site_id' => 2,
- 'parent_id' => 4,
- 'children' => [],
- ],
- ],
- ],
- ],
- ],
- [
- 'id' => 6,
- 'name' => 'a',
- 'site_id' => 1,
- 'parent_id' => 2,
- 'children' => [],
- ],
- ];
- $this->assertEquals($expected, $formatter($items)->toArray());
- }
- /**
- * Tests that loadInto() is capable of handling composite primary keys
- *
- * @return void
- */
- public function testLoadInto()
- {
- $table = $this->getTableLocator()->get('SiteAuthors');
- $tags = $this->getTableLocator()->get('SiteTags');
- $table->hasMany('SiteArticles', [
- 'foreignKey' => ['author_id', 'site_id'],
- ]);
- $author = $table->get([1, 1]);
- $result = $table->loadInto($author, ['SiteArticles']);
- $this->assertSame($author, $result);
- $this->assertNotEmpty($result->site_articles);
- $expected = $table->get([1, 1], ['contain' => ['SiteArticles']]);
- $this->assertEquals($expected, $result);
- }
- /**
- * Tests that loadInto() is capable of handling composite primary keys
- * when loading belongsTo associations
- *
- * @return void
- */
- public function testLoadIntoWithBelongsTo()
- {
- $table = $this->getTableLocator()->get('SiteArticles');
- $table->belongsTo('SiteAuthors', [
- 'foreignKey' => ['author_id', 'site_id'],
- ]);
- $author = $table->get([2, 2]);
- $result = $table->loadInto($author, ['SiteAuthors']);
- $this->assertSame($author, $result);
- $this->assertNotEmpty($result->site_author);
- $expected = $table->get([2, 2], ['contain' => ['SiteAuthors']]);
- $this->assertEquals($expected, $result);
- }
- /**
- * Tests that loadInto() is capable of handling composite primary keys
- * when loading into multiple entities
- *
- * @return void
- */
- public function testLoadIntoMany()
- {
- $table = $this->getTableLocator()->get('SiteAuthors');
- $tags = $this->getTableLocator()->get('SiteTags');
- $table->hasMany('SiteArticles', [
- 'foreignKey' => ['author_id', 'site_id'],
- ]);
- $authors = $table->find()->toList();
- $result = $table->loadInto($authors, ['SiteArticles']);
- foreach ($authors as $k => $v) {
- $this->assertSame($result[$k], $v);
- }
- $expected = $table->find('all', ['contain' => ['SiteArticles']])->toList();
- $this->assertEquals($expected, $result);
- }
- /**
- * Tests notMatching() with a belongsToMany association
- *
- * @return void
- */
- public function testNotMatchingBelongsToMany()
- {
- $driver = $this->connection->getDriver();
- if ($driver instanceof Sqlserver) {
- $this->markTestSkipped('Sqlserver does not support the requirements of this test.');
- } elseif ($driver instanceof Sqlite) {
- $serverVersion = $driver->getConnection()->getAttribute(PDO::ATTR_SERVER_VERSION);
- if (version_compare($serverVersion, '3.15.0', '<')) {
- $this->markTestSkipped("Sqlite ($serverVersion) does not support the requirements of this test.");
- }
- }
- $articles = $this->getTableLocator()->get('SiteArticles');
- $articles->belongsToMany('SiteTags', [
- 'through' => 'SiteArticlesTags',
- 'foreignKey' => ['article_id', 'site_id'],
- 'targetForeignKey' => ['tag_id', 'site_id'],
- ]);
- $results = $articles->find()
- ->enableHydration(false)
- ->notMatching('SiteTags')
- ->toArray();
- $expected = [
- [
- 'id' => 3,
- 'author_id' => 1,
- 'site_id' => 2,
- 'title' => 'Third Article',
- 'body' => 'Third Article Body',
- ],
- ];
- $this->assertEquals($expected, $results);
- }
- /**
- * Helper method to skip tests when connection is SQLite.
- *
- * @return void
- */
- public function skipIfSqlite()
- {
- $this->skipIf(
- $this->connection->getDriver() instanceof Sqlite,
- 'SQLite does not support the requirements of this test.'
- );
- }
- }
|