123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186 |
- <?php
- /*
- * $Id$
- *
- * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
- * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
- * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
- * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
- * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
- * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
- * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
- * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
- * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
- * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
- * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
- *
- * This software consists of voluntary contributions made by many individuals
- * and is licensed under the LGPL. For more information, see
- * <http://www.doctrine-project.org>.
- */
- namespace Doctrine\ORM\Tools\Console\Command;
- use Symfony\Component\Console\Input\InputArgument,
- Symfony\Component\Console\Input\InputOption,
- Symfony\Component\Console,
- Doctrine\ORM\Tools\Console\MetadataFilter,
- Doctrine\ORM\Tools\Export\ClassMetadataExporter,
- Doctrine\ORM\Tools\EntityGenerator,
- Doctrine\ORM\Tools\DisconnectedClassMetadataFactory;
- /**
- * Command 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 Benjamin Eberlei <kontakt@beberlei.de>
- * @author Guilherme Blanco <guilhermeblanco@hotmail.com>
- * @author Jonathan Wage <jonwage@gmail.com>
- * @author Roman Borschel <roman@code-factory.org>
- */
- class ConvertMappingCommand extends Console\Command\Command
- {
- /**
- * @see Console\Command\Command
- */
- protected function configure()
- {
- $this
- ->setName('orm:convert-mapping')
- ->setDescription('Convert mapping information between supported formats.')
- ->setDefinition(array(
- new InputOption(
- 'filter', null, InputOption::VALUE_REQUIRED | InputOption::VALUE_IS_ARRAY,
- 'A string pattern used to match entities that should be processed.'
- ),
- new InputArgument(
- 'to-type', InputArgument::REQUIRED, 'The mapping type to be converted.'
- ),
- new InputArgument(
- 'dest-path', InputArgument::REQUIRED,
- 'The path to generate your entities classes.'
- ),
- new InputOption(
- 'force', null, InputOption::VALUE_NONE,
- 'Force to overwrite existing mapping files.'
- ),
- new InputOption(
- 'from-database', null, null, 'Whether or not to convert mapping information from existing database.'
- ),
- new InputOption(
- 'extend', null, InputOption::VALUE_OPTIONAL,
- 'Defines a base class to be extended by generated entity classes.'
- ),
- new InputOption(
- 'num-spaces', null, InputOption::VALUE_OPTIONAL,
- 'Defines the number of indentation spaces', 4
- ),
- new InputOption(
- 'namespace', null, InputOption::VALUE_OPTIONAL,
- 'Defines a namespace for the generated entity classes, if converted from database.'
- ),
- ))
- ->setHelp(<<<EOT
- Convert mapping information between supported formats.
- This is an execute <info>one-time</info> command. It should not be necessary for
- you to call this method multiple times, escpecially when using the <comment>--from-database</comment>
- flag.
- Converting an existing databsae schema into mapping files only solves about 70-80%
- of the necessary mapping information. Additionally the detection from an existing
- database cannot detect inverse associations, inheritance types,
- entities with foreign keys as primary keys and many of the
- semantical operations on associations such as cascade.
- <comment>Hint:</comment> There is no need to convert YAML or XML mapping files to annotations
- every time you make changes. All mapping drivers are first class citizens
- in Doctrine 2 and can be used as runtime mapping for the ORM.
- EOT
- );
- }
- /**
- * @see Console\Command\Command
- */
- protected function execute(Console\Input\InputInterface $input, Console\Output\OutputInterface $output)
- {
- $em = $this->getHelper('em')->getEntityManager();
- if ($input->getOption('from-database') === true) {
- $databaseDriver = new \Doctrine\ORM\Mapping\Driver\DatabaseDriver(
- $em->getConnection()->getSchemaManager()
- );
- $em->getConfiguration()->setMetadataDriverImpl(
- $databaseDriver
- );
- if (($namespace = $input->getOption('namespace')) !== null) {
- $databaseDriver->setNamespace($namespace);
- }
- }
- $cmf = new DisconnectedClassMetadataFactory();
- $cmf->setEntityManager($em);
- $metadata = $cmf->getAllMetadata();
- $metadata = MetadataFilter::filter($metadata, $input->getOption('filter'));
- // Process destination directory
- if ( ! is_dir($destPath = $input->getArgument('dest-path'))) {
- mkdir($destPath, 0777, true);
- }
- $destPath = realpath($destPath);
- if ( ! file_exists($destPath)) {
- throw new \InvalidArgumentException(
- sprintf("Mapping destination directory '<info>%s</info>' does not exist.", $input->getArgument('dest-path'))
- );
- } else if ( ! is_writable($destPath)) {
- throw new \InvalidArgumentException(
- sprintf("Mapping destination directory '<info>%s</info>' does not have write permissions.", $destPath)
- );
- }
- $toType = strtolower($input->getArgument('to-type'));
- $exporter = $this->getExporter($toType, $destPath);
- $exporter->setOverwriteExistingFiles( ($input->getOption('force') !== false) );
- if ($toType == 'annotation') {
- $entityGenerator = new EntityGenerator();
- $exporter->setEntityGenerator($entityGenerator);
- $entityGenerator->setNumSpaces($input->getOption('num-spaces'));
- if (($extend = $input->getOption('extend')) !== null) {
- $entityGenerator->setClassToExtend($extend);
- }
- }
- if (count($metadata)) {
- foreach ($metadata as $class) {
- $output->write(sprintf('Processing entity "<info>%s</info>"', $class->name) . PHP_EOL);
- }
- $exporter->setMetadata($metadata);
- $exporter->export();
- $output->write(PHP_EOL . sprintf(
- 'Exporting "<info>%s</info>" mapping information to "<info>%s</info>"' . PHP_EOL, $toType, $destPath
- ));
- } else {
- $output->write('No Metadata Classes to process.' . PHP_EOL);
- }
- }
- protected function getExporter($toType, $destPath)
- {
- $cme = new ClassMetadataExporter();
- return $cme->getExporter($toType, $destPath);
- }
- }
|