. */ namespace Doctrine\ORM\Tools\Cli\Tasks; use Doctrine\Common\Cli\Tasks\AbstractTask, Doctrine\Common\Cli\CliException, Doctrine\Common\Cli\Option, Doctrine\Common\Cli\OptionGroup, Doctrine\ORM\Tools\Export\ClassMetadataExporter, Doctrine\ORM\Mapping\Driver\DriverChain, Doctrine\ORM\Mapping\Driver\AnnotationDriver, Doctrine\ORM\Mapping\Driver\Driver; /** * CLI Task to convert your mapping information between the various formats * * @license http://www.opensource.org/licenses/lgpl-license.php LGPL * @link www.doctrine-project.org * @since 2.0 * @version $Revision$ * @author Guilherme Blanco * @author Jonathan Wage * @author Roman Borschel */ class ConvertMappingTask extends AbstractTask { /** * @inheritdoc */ public function buildDocumentation() { $convertOptions = new OptionGroup(OptionGroup::CARDINALITY_N_N, array( new OptionGroup(OptionGroup::CARDINALITY_1_1, array( new Option('from', '', 'The path to the mapping information to convert from (yml, xml, php, annotation).'), new Option('from-database', null, 'Use this option if you wish to reverse engineer your database to a set of Doctrine mapping files.') )), new Option('to', '', 'The format to convert to (yml, xml, php, annotation).'), new Option('dest', '', 'The path to write the converted mapping information.') )); $doc = $this->getDocumentation(); $doc->setName('convert-mapping') ->setDescription('Convert mapping information between supported formats.') ->getOptionGroup() ->addOption($convertOptions); } /** * @inheritdoc */ public function validate() { $arguments = $this->getArguments(); if (isset($arguments['from-database']) && $arguments['from-database']) { $arguments['from'] = 'database'; $this->setArguments($arguments); } if (!(isset($arguments['from']) && isset($arguments['to']) && isset($arguments['dest']))) { throw new CliException( 'You must include a value for all three options: --from, --to and --dest.' ); } if (strtolower($arguments['to']) != 'annotation' && isset($arguments['extend'])) { throw new CliException( 'You can only use the --extend argument when converting to annotations.' ); } if (strtolower($arguments['from']) == 'database') { // Check if we have an active EntityManager $em = $this->getConfiguration()->getAttribute('em'); if ($em === null) { throw new CliException( "Attribute 'em' of CLI Configuration is not defined or it is not a valid EntityManager." ); } $config = $em->getConfiguration(); $config->setMetadataDriverImpl( new \Doctrine\ORM\Mapping\Driver\DatabaseDriver( $em->getConnection()->getSchemaManager() ) ); } return true; } public function run() { $arguments = $this->getArguments(); $cme = new ClassMetadataExporter(); $printer = $this->getPrinter(); // Get exporter and configure it $exporter = $cme->getExporter($arguments['to'], $arguments['dest']); if (isset($arguments['extend']) && $arguments['extend']) { $exporter->setClassToExtend($arguments['extend']); } if (isset($arguments['num-spaces']) && $arguments['extend']) { $exporter->setNumSpaces($arguments['num-spaces']); } $from = (array) $arguments['from']; foreach ($from as $source) { $sourceArg = $source; $type = $this->_determineSourceType($sourceArg); if ( ! $type) { throw new CliException( "Invalid mapping source type '$sourceArg'." ); } $source = $this->_getSourceByType($type, $sourceArg); $printer->writeln( sprintf( 'Adding "%s" mapping source which contains the "%s" format', $printer->format($sourceArg, 'KEYWORD'), $printer->format($type, 'KEYWORD') ) ); $cme->addMappingSource($source, $type); } $metadatas = $cme->getMetadatasForMappingSources(); foreach ($metadatas as $metadata) { $printer->writeln( sprintf('Processing entity "%s"', $printer->format($metadata->name, 'KEYWORD')) ); } $printer->writeln( sprintf( 'Exporting "%s" mapping information to directory "%s"', $printer->format($arguments['to'], 'KEYWORD'), $printer->format($arguments['dest'], 'KEYWORD') ) ); $exporter->setMetadatas($metadatas); $exporter->export(); } protected function _determineSourceType($source) { // If the --from= is a directory lets determine if it is // annotations, yaml, xml, etc. if (is_dir($source)) { // Find the files in the directory $files = glob($source . '/*.*'); if ( ! $files) { throw new \InvalidArgumentException( sprintf('No mapping files found in "%s"', $source) ); } // Get the contents of the first file $contents = file_get_contents($files[0]); // Check if it has a class definition in it for annotations if (preg_match("/class (.*)/", $contents)) { return 'annotation'; // Otherwise lets determine the type based on the extension of the // first file in the directory (yml, xml, etc) } else { $info = pathinfo($files[0]); return $info['extension']; } // Nothing special for database } else if ($source == 'database') { return 'database'; } } protected function _getSourceByType($type, $source) { // If --from==database then the source is an instance of SchemaManager // for the current EntityMAnager if ($type == 'database') { $em = $this->getConfiguration()->getAttribute('em'); return $em->getConnection()->getSchemaManager(); // If source is annotation then lets try and find the existing annotation // driver for the source instead of re-creating a new instance } else if ($type == 'annotation') { $em = $this->getConfiguration()->getAttribute('em'); $metadataDriverImpl = $em->getConfiguration()->getMetadataDriverImpl(); // Find the annotation driver in the chain of drivers if ($metadataDriverImpl instanceof DriverChain) { foreach ($metadataDriverImpl->getDrivers() as $namespace => $driver) { if ($this->_isAnnotationDriverForPath($driver, $source)) { return $driver; } } } else if ($this->_isAnnotationDriverForPath($metadataDriverImpl, $source)) { return $metadataDriverImpl; } else { return $source; } } else { return $source; } } /** * Check to see if the given metadata driver is the annotation driver for the * given directory path * * @param Driver $driver * @param string $path * @return boolean */ private function _isAnnotationDriverForPath(Driver $driver, $path) { if ( ! $driver instanceof AnnotationDriver) { return false; } if (in_array(realpath($path), $driver->getPaths())) { return true; } else { return false; } } }