AbstractClassMetadataExporterTest.php 16 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392
  1. <?php
  2. /*
  3. * $Id$
  4. *
  5. * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
  6. * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
  7. * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
  8. * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
  9. * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
  10. * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
  11. * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
  12. * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
  13. * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
  14. * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
  15. * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
  16. *
  17. * This software consists of voluntary contributions made by many individuals
  18. * and is licensed under the LGPL. For more information, see
  19. * <http://www.doctrine-project.org>.
  20. */
  21. namespace Doctrine\Tests\ORM\Tools\Export;
  22. use Doctrine\ORM\Tools\Export\ClassMetadataExporter;
  23. use Doctrine\ORM\Mapping\ClassMetadataInfo;
  24. use Doctrine\ORM\Tools\EntityGenerator;
  25. use Doctrine\Tests\Mocks\MetadataDriverMock;
  26. use Doctrine\Tests\Mocks\DatabasePlatformMock;
  27. use Doctrine\Tests\Mocks\EntityManagerMock;
  28. use Doctrine\Tests\Mocks\ConnectionMock;
  29. use Doctrine\Tests\Mocks\DriverMock;
  30. use Doctrine\Common\EventManager;
  31. use Doctrine\ORM\Tools\DisconnectedClassMetadataFactory;
  32. use Doctrine\ORM\Mapping\ClassMetadataFactory;
  33. require_once __DIR__ . '/../../../TestInit.php';
  34. /**
  35. * Test case for ClassMetadataExporter
  36. *
  37. * @author Jonathan H. Wage <jonwage@gmail.com>
  38. * @author Roman Borschel <roman@code-factory.org
  39. * @license http://www.opensource.org/licenses/lgpl-license.php LGPL
  40. * @link http://www.phpdoctrine.org
  41. * @since 2.0
  42. * @version $Revision$
  43. */
  44. abstract class AbstractClassMetadataExporterTest extends \Doctrine\Tests\OrmTestCase
  45. {
  46. protected $_extension;
  47. abstract protected function _getType();
  48. protected function _createEntityManager($metadataDriver)
  49. {
  50. $driverMock = new DriverMock();
  51. $config = new \Doctrine\ORM\Configuration();
  52. $config->setProxyDir(__DIR__ . '/../../Proxies');
  53. $config->setProxyNamespace('Doctrine\Tests\Proxies');
  54. $eventManager = new EventManager();
  55. $conn = new ConnectionMock(array(), $driverMock, $config, $eventManager);
  56. $mockDriver = new MetadataDriverMock();
  57. $config->setMetadataDriverImpl($metadataDriver);
  58. return EntityManagerMock::create($conn, $config, $eventManager);
  59. }
  60. protected function _createMetadataDriver($type, $path)
  61. {
  62. $mappingDriver = array(
  63. 'php' => 'PHPDriver',
  64. 'annotation' => 'AnnotationDriver',
  65. 'xml' => 'XmlDriver',
  66. 'yaml' => 'YamlDriver',
  67. );
  68. $this->assertArrayHasKey($type, $mappingDriver, "There is no metadata driver for the type '" . $type . "'.");
  69. $driverName = $mappingDriver[$type];
  70. $class = 'Doctrine\ORM\Mapping\Driver\\' . $driverName;
  71. if ($type === 'annotation') {
  72. $driver = $this->createAnnotationDriver(array($path));
  73. } else {
  74. $driver = new $class($path);
  75. }
  76. return $driver;
  77. }
  78. protected function _createClassMetadataFactory($em, $type)
  79. {
  80. if ($type === 'annotation') {
  81. $factory = new ClassMetadataFactory();
  82. } else {
  83. $factory = new DisconnectedClassMetadataFactory();
  84. }
  85. $factory->setEntityManager($em);
  86. return $factory;
  87. }
  88. public function testExportDirectoryAndFilesAreCreated()
  89. {
  90. $this->_deleteDirectory(__DIR__ . '/export/'.$this->_getType());
  91. $type = $this->_getType();
  92. $metadataDriver = $this->_createMetadataDriver($type, __DIR__ . '/' . $type);
  93. $em = $this->_createEntityManager($metadataDriver);
  94. $cmf = $this->_createClassMetadataFactory($em, $type);
  95. $metadata = $cmf->getAllMetadata();
  96. $metadata[0]->name = 'Doctrine\Tests\ORM\Tools\Export\ExportedUser';
  97. $this->assertEquals('Doctrine\Tests\ORM\Tools\Export\ExportedUser', $metadata[0]->name);
  98. $type = $this->_getType();
  99. $cme = new ClassMetadataExporter();
  100. $exporter = $cme->getExporter($type, __DIR__ . '/export/' . $type);
  101. if ($type === 'annotation') {
  102. $entityGenerator = new EntityGenerator();
  103. $entityGenerator->setAnnotationPrefix("");
  104. $exporter->setEntityGenerator($entityGenerator);
  105. }
  106. $this->_extension = $exporter->getExtension();
  107. $exporter->setMetadata($metadata);
  108. $exporter->export();
  109. if ($type == 'annotation') {
  110. $this->assertTrue(file_exists(__DIR__ . '/export/' . $type . '/'.str_replace('\\', '/', 'Doctrine\Tests\ORM\Tools\Export\ExportedUser').$this->_extension));
  111. } else {
  112. $this->assertTrue(file_exists(__DIR__ . '/export/' . $type . '/Doctrine.Tests.ORM.Tools.Export.ExportedUser'.$this->_extension));
  113. }
  114. }
  115. /**
  116. * @depends testExportDirectoryAndFilesAreCreated
  117. */
  118. public function testExportedMetadataCanBeReadBackIn()
  119. {
  120. $type = $this->_getType();
  121. $metadataDriver = $this->_createMetadataDriver($type, __DIR__ . '/export/' . $type);
  122. $em = $this->_createEntityManager($metadataDriver);
  123. $cmf = $this->_createClassMetadataFactory($em, $type);
  124. $metadata = $cmf->getAllMetadata();
  125. $this->assertEquals(1, count($metadata));
  126. $class = current($metadata);
  127. $this->assertEquals('Doctrine\Tests\ORM\Tools\Export\ExportedUser', $class->name);
  128. return $class;
  129. }
  130. /**
  131. * @depends testExportedMetadataCanBeReadBackIn
  132. * @param ClassMetadataInfo $class
  133. */
  134. public function testTableIsExported($class)
  135. {
  136. $this->assertEquals('cms_users', $class->table['name']);
  137. return $class;
  138. }
  139. /**
  140. * @depends testTableIsExported
  141. * @param ClassMetadataInfo $class
  142. */
  143. public function testTypeIsExported($class)
  144. {
  145. $this->assertFalse($class->isMappedSuperclass);
  146. return $class;
  147. }
  148. /**
  149. * @depends testTypeIsExported
  150. * @param ClassMetadataInfo $class
  151. */
  152. public function testIdentifierIsExported($class)
  153. {
  154. $this->assertEquals(ClassMetadataInfo::GENERATOR_TYPE_IDENTITY, $class->generatorType);
  155. $this->assertEquals(array('id'), $class->identifier);
  156. $this->assertTrue(isset($class->fieldMappings['id']['id']) && $class->fieldMappings['id']['id'] === true);
  157. return $class;
  158. }
  159. /**
  160. * @depends testIdentifierIsExported
  161. * @param ClassMetadataInfo $class
  162. */
  163. public function testFieldsAreExported($class)
  164. {
  165. $this->assertTrue(isset($class->fieldMappings['id']['id']) && $class->fieldMappings['id']['id'] === true);
  166. $this->assertEquals('id', $class->fieldMappings['id']['fieldName']);
  167. $this->assertEquals('integer', $class->fieldMappings['id']['type']);
  168. $this->assertEquals('id', $class->fieldMappings['id']['columnName']);
  169. $this->assertEquals('name', $class->fieldMappings['name']['fieldName']);
  170. $this->assertEquals('string', $class->fieldMappings['name']['type']);
  171. $this->assertEquals(50, $class->fieldMappings['name']['length']);
  172. $this->assertEquals('name', $class->fieldMappings['name']['columnName']);
  173. $this->assertEquals('email', $class->fieldMappings['email']['fieldName']);
  174. $this->assertEquals('string', $class->fieldMappings['email']['type']);
  175. $this->assertEquals('user_email', $class->fieldMappings['email']['columnName']);
  176. $this->assertEquals('CHAR(32) NOT NULL', $class->fieldMappings['email']['columnDefinition']);
  177. return $class;
  178. }
  179. /**
  180. * @depends testFieldsAreExported
  181. * @param ClassMetadataInfo $class
  182. */
  183. public function testOneToOneAssociationsAreExported($class)
  184. {
  185. $this->assertTrue(isset($class->associationMappings['address']));
  186. $this->assertEquals('Doctrine\Tests\ORM\Tools\Export\Address', $class->associationMappings['address']['targetEntity']);
  187. $this->assertEquals('address_id', $class->associationMappings['address']['joinColumns'][0]['name']);
  188. $this->assertEquals('id', $class->associationMappings['address']['joinColumns'][0]['referencedColumnName']);
  189. $this->assertEquals('CASCADE', $class->associationMappings['address']['joinColumns'][0]['onDelete']);
  190. $this->assertTrue($class->associationMappings['address']['isCascadeRemove']);
  191. $this->assertTrue($class->associationMappings['address']['isCascadePersist']);
  192. $this->assertFalse($class->associationMappings['address']['isCascadeRefresh']);
  193. $this->assertFalse($class->associationMappings['address']['isCascadeMerge']);
  194. $this->assertFalse($class->associationMappings['address']['isCascadeDetach']);
  195. $this->assertTrue($class->associationMappings['address']['orphanRemoval']);
  196. return $class;
  197. }
  198. /**
  199. * @depends testFieldsAreExported
  200. */
  201. public function testManyToOneAssociationsAreExported($class)
  202. {
  203. $this->assertTrue(isset($class->associationMappings['mainGroup']));
  204. $this->assertEquals('Doctrine\Tests\ORM\Tools\Export\Group', $class->associationMappings['mainGroup']['targetEntity']);
  205. }
  206. /**
  207. * @depends testOneToOneAssociationsAreExported
  208. * @param ClassMetadataInfo $class
  209. */
  210. public function testOneToManyAssociationsAreExported($class)
  211. {
  212. $this->assertTrue(isset($class->associationMappings['phonenumbers']));
  213. //$this->assertInstanceOf('Doctrine\ORM\Mapping\OneToManyMapping', $class->associationMappings['phonenumbers']);
  214. $this->assertEquals('Doctrine\Tests\ORM\Tools\Export\Phonenumber', $class->associationMappings['phonenumbers']['targetEntity']);
  215. $this->assertEquals('user', $class->associationMappings['phonenumbers']['mappedBy']);
  216. $this->assertEquals(array('number' => 'ASC'), $class->associationMappings['phonenumbers']['orderBy']);
  217. $this->assertTrue($class->associationMappings['phonenumbers']['isCascadeRemove']);
  218. $this->assertTrue($class->associationMappings['phonenumbers']['isCascadePersist']);
  219. $this->assertFalse($class->associationMappings['phonenumbers']['isCascadeRefresh']);
  220. $this->assertTrue($class->associationMappings['phonenumbers']['isCascadeMerge']);
  221. $this->assertFalse($class->associationMappings['phonenumbers']['isCascadeDetach']);
  222. $this->assertTrue($class->associationMappings['phonenumbers']['orphanRemoval']);
  223. return $class;
  224. }
  225. /**
  226. * @depends testOneToManyAssociationsAreExported
  227. * @param ClassMetadataInfo $metadata
  228. */
  229. public function testManyToManyAssociationsAreExported($class)
  230. {
  231. $this->assertTrue(isset($class->associationMappings['groups']));
  232. //$this->assertInstanceOf('Doctrine\ORM\Mapping\ManyToManyMapping', $class->associationMappings['groups']);
  233. $this->assertEquals('Doctrine\Tests\ORM\Tools\Export\Group', $class->associationMappings['groups']['targetEntity']);
  234. $this->assertEquals('cms_users_groups', $class->associationMappings['groups']['joinTable']['name']);
  235. $this->assertEquals('user_id', $class->associationMappings['groups']['joinTable']['joinColumns'][0]['name']);
  236. $this->assertEquals('id', $class->associationMappings['groups']['joinTable']['joinColumns'][0]['referencedColumnName']);
  237. $this->assertEquals('group_id', $class->associationMappings['groups']['joinTable']['inverseJoinColumns'][0]['name']);
  238. $this->assertEquals('id', $class->associationMappings['groups']['joinTable']['inverseJoinColumns'][0]['referencedColumnName']);
  239. $this->assertEquals('INT NULL', $class->associationMappings['groups']['joinTable']['inverseJoinColumns'][0]['columnDefinition']);
  240. $this->assertTrue($class->associationMappings['groups']['isCascadeRemove']);
  241. $this->assertTrue($class->associationMappings['groups']['isCascadePersist']);
  242. $this->assertTrue($class->associationMappings['groups']['isCascadeRefresh']);
  243. $this->assertTrue($class->associationMappings['groups']['isCascadeMerge']);
  244. $this->assertTrue($class->associationMappings['groups']['isCascadeDetach']);
  245. return $class;
  246. }
  247. /**
  248. * @depends testManyToManyAssociationsAreExported
  249. * @param ClassMetadataInfo $class
  250. */
  251. public function testLifecycleCallbacksAreExported($class)
  252. {
  253. $this->assertTrue(isset($class->lifecycleCallbacks['prePersist']));
  254. $this->assertEquals(2, count($class->lifecycleCallbacks['prePersist']));
  255. $this->assertEquals('doStuffOnPrePersist', $class->lifecycleCallbacks['prePersist'][0]);
  256. $this->assertEquals('doOtherStuffOnPrePersistToo', $class->lifecycleCallbacks['prePersist'][1]);
  257. $this->assertTrue(isset($class->lifecycleCallbacks['postPersist']));
  258. $this->assertEquals(1, count($class->lifecycleCallbacks['postPersist']));
  259. $this->assertEquals('doStuffOnPostPersist', $class->lifecycleCallbacks['postPersist'][0]);
  260. return $class;
  261. }
  262. /**
  263. * @depends testLifecycleCallbacksAreExported
  264. * @param ClassMetadataInfo $class
  265. */
  266. public function testCascadeIsExported($class)
  267. {
  268. $this->assertTrue($class->associationMappings['phonenumbers']['isCascadePersist']);
  269. $this->assertTrue($class->associationMappings['phonenumbers']['isCascadeMerge']);
  270. $this->assertTrue($class->associationMappings['phonenumbers']['isCascadeRemove']);
  271. $this->assertFalse($class->associationMappings['phonenumbers']['isCascadeRefresh']);
  272. $this->assertFalse($class->associationMappings['phonenumbers']['isCascadeDetach']);
  273. $this->assertTrue($class->associationMappings['phonenumbers']['orphanRemoval']);
  274. return $class;
  275. }
  276. /**
  277. * @depends testCascadeIsExported
  278. * @param ClassMetadataInfo $class
  279. */
  280. public function testInversedByIsExported($class)
  281. {
  282. $this->assertEquals('user', $class->associationMappings['address']['inversedBy']);
  283. }
  284. /**
  285. * @depends testExportDirectoryAndFilesAreCreated
  286. */
  287. public function testCascadeAllCollapsed()
  288. {
  289. $type = $this->_getType();
  290. if ($type == 'xml') {
  291. $xml = simplexml_load_file(__DIR__ . '/export/'.$type.'/Doctrine.Tests.ORM.Tools.Export.ExportedUser.dcm.xml');
  292. $xml->registerXPathNamespace("d", "http://doctrine-project.org/schemas/orm/doctrine-mapping");
  293. $nodes = $xml->xpath("/d:doctrine-mapping/d:entity/d:one-to-many[@field='interests']/d:cascade/d:*");
  294. $this->assertEquals(1, count($nodes));
  295. $this->assertEquals('cascade-all', $nodes[0]->getName());
  296. } elseif ($type == 'yaml') {
  297. $yaml = new \Symfony\Component\Yaml\Parser();
  298. $value = $yaml->parse(file_get_contents(__DIR__ . '/export/'.$type.'/Doctrine.Tests.ORM.Tools.Export.ExportedUser.dcm.yml'));
  299. $this->assertTrue(isset($value['Doctrine\Tests\ORM\Tools\Export\ExportedUser']['oneToMany']['interests']['cascade']));
  300. $this->assertEquals(1, count($value['Doctrine\Tests\ORM\Tools\Export\ExportedUser']['oneToMany']['interests']['cascade']));
  301. $this->assertEquals('all', $value['Doctrine\Tests\ORM\Tools\Export\ExportedUser']['oneToMany']['interests']['cascade'][0]);
  302. } else {
  303. $this->markTestSkipped('Test aviable only for '.$type.' dirver');
  304. }
  305. }
  306. public function __destruct()
  307. {
  308. # $this->_deleteDirectory(__DIR__ . '/export/'.$this->_getType());
  309. }
  310. protected function _deleteDirectory($path)
  311. {
  312. if (is_file($path)) {
  313. return unlink($path);
  314. } else if (is_dir($path)) {
  315. $files = glob(rtrim($path,'/').'/*');
  316. foreach ($files as $file){
  317. $this->_deleteDirectory($file);
  318. }
  319. return rmdir($path);
  320. }
  321. }
  322. }
  323. class Address
  324. {
  325. }
  326. class Phonenumber
  327. {
  328. }
  329. class Group
  330. {
  331. }