2014-08-31 09:27:16 +02:00
|
|
|
<?php
|
|
|
|
/*
|
|
|
|
* 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 MIT license. For more information, see
|
|
|
|
* <http://www.doctrine-project.org>.
|
|
|
|
*/
|
|
|
|
|
|
|
|
namespace Doctrine\ORM\Tools\Console\Command;
|
|
|
|
|
2014-10-19 18:14:33 +02:00
|
|
|
use Doctrine\Common\Persistence\Mapping\MappingException;
|
2014-10-19 17:49:28 +02:00
|
|
|
use Doctrine\ORM\EntityManagerInterface;
|
2014-08-31 09:27:16 +02:00
|
|
|
use Symfony\Component\Console\Command\Command;
|
2014-10-19 18:13:01 +02:00
|
|
|
use Symfony\Component\Console\Helper\Table;
|
2014-10-19 17:49:28 +02:00
|
|
|
use Symfony\Component\Console\Input\InputArgument;
|
|
|
|
use Symfony\Component\Console\Input\InputInterface;
|
|
|
|
use Symfony\Component\Console\Output\OutputInterface;
|
2014-08-31 09:27:16 +02:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Show information about mapped entities.
|
|
|
|
*
|
|
|
|
* @link www.doctrine-project.org
|
|
|
|
* @since 2.4
|
|
|
|
* @author Daniel Leech <daniel@dantleech.com>
|
|
|
|
*/
|
2014-10-19 17:43:32 +02:00
|
|
|
final class MappingDescribeCommand extends Command
|
2014-08-31 09:27:16 +02:00
|
|
|
{
|
|
|
|
/**
|
|
|
|
* @var OutputInterface
|
|
|
|
*/
|
|
|
|
private $output;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @var array
|
|
|
|
*/
|
2014-10-19 18:13:01 +02:00
|
|
|
private $out = array();
|
2014-08-31 09:27:16 +02:00
|
|
|
|
|
|
|
/**
|
|
|
|
* {@inheritdoc}
|
|
|
|
*/
|
|
|
|
protected function configure()
|
|
|
|
{
|
|
|
|
$this
|
|
|
|
->setName('orm:mapping:describe')
|
|
|
|
->addArgument('entityName', InputArgument::REQUIRED, 'Full or partial name of entity')
|
|
|
|
->setDescription('Display information about mapped objects')
|
|
|
|
->setHelp(<<<EOT
|
|
|
|
The %command.full_name% command describes the metadata for the given full or partial entity class name.
|
|
|
|
|
|
|
|
<info>%command.full_name%</info> My\Namespace\Entity\MyEntity
|
|
|
|
|
|
|
|
Or:
|
|
|
|
|
|
|
|
<info>%command.full_name%</info> MyEntity
|
|
|
|
EOT
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* {@inheritdoc}
|
|
|
|
*/
|
|
|
|
protected function execute(InputInterface $input, OutputInterface $output)
|
|
|
|
{
|
|
|
|
$entityName = $input->getArgument('entityName');
|
|
|
|
|
|
|
|
/* @var $entityManager \Doctrine\ORM\EntityManager */
|
|
|
|
$entityManager = $this->getHelper('em')->getEntityManager();
|
|
|
|
|
|
|
|
$this->output = $output;
|
|
|
|
|
2014-10-19 18:13:01 +02:00
|
|
|
$this->displayEntity($entityName, $entityManager, $output);
|
2014-08-31 09:27:16 +02:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Display all the mapping information for a single Entity.
|
|
|
|
*
|
2014-10-19 17:49:28 +02:00
|
|
|
* @param string $entityName Full or partial entity class name
|
|
|
|
* @param EntityManagerInterface $entityManager
|
2014-10-19 18:23:50 +02:00
|
|
|
* @param OutputInterface $output
|
2014-08-31 09:27:16 +02:00
|
|
|
*/
|
2014-10-19 18:13:01 +02:00
|
|
|
private function displayEntity($entityName, EntityManagerInterface $entityManager, OutputInterface $output)
|
2014-08-31 09:27:16 +02:00
|
|
|
{
|
2014-10-19 18:32:12 +02:00
|
|
|
$metadata = $this->getClassMetadata($entityName, $entityManager);
|
|
|
|
|
|
|
|
$rows = array(
|
|
|
|
$this->formatField('Name', $metadata->name),
|
|
|
|
$this->formatField('Root entity name', $metadata->rootEntityName),
|
|
|
|
$this->formatField('Custom generator definition', $metadata->customGeneratorDefinition),
|
|
|
|
$this->formatField('Custom repository class', $metadata->customRepositoryClassName),
|
|
|
|
$this->formatField('Mapped super class?', $metadata->isMappedSuperclass),
|
|
|
|
$this->formatField('Embedded class?', $metadata->isEmbeddedClass),
|
|
|
|
$this->formatField('Parent classes', $metadata->parentClasses),
|
|
|
|
$this->formatField('Sub classes', $metadata->subClasses),
|
|
|
|
$this->formatField('Embedded classes', $metadata->subClasses),
|
|
|
|
$this->formatField('Named queries', $metadata->namedQueries),
|
|
|
|
$this->formatField('Named native queries', $metadata->namedNativeQueries),
|
|
|
|
$this->formatField('SQL result set mappings', $metadata->sqlResultSetMappings),
|
|
|
|
$this->formatField('Identifier', $metadata->identifier),
|
|
|
|
$this->formatField('Inheritance type', $metadata->inheritanceType),
|
|
|
|
$this->formatField('Discriminator column', $metadata->discriminatorColumn),
|
|
|
|
$this->formatField('Discriminator value', $metadata->discriminatorValue),
|
|
|
|
$this->formatField('Discriminator map', $metadata->discriminatorMap),
|
|
|
|
$this->formatField('Generator type', $metadata->generatorType),
|
|
|
|
$this->formatField('Table', $metadata->table),
|
|
|
|
$this->formatField('Composite identifier?', $metadata->isIdentifierComposite),
|
|
|
|
$this->formatField('Foreign identifier?', $metadata->containsForeignIdentifier),
|
|
|
|
$this->formatField('Sequence generator definition', $metadata->sequenceGeneratorDefinition),
|
|
|
|
$this->formatField('Table generator definition', $metadata->tableGeneratorDefinition),
|
|
|
|
$this->formatField('Change tracking policy', $metadata->changeTrackingPolicy),
|
|
|
|
$this->formatField('Versioned?', $metadata->isVersioned),
|
|
|
|
$this->formatField('Version field', $metadata->versionField),
|
|
|
|
$this->formatField('Read only?', $metadata->isReadOnly),
|
|
|
|
|
|
|
|
$this->formatEntityListeners($metadata->entityListeners),
|
|
|
|
);
|
|
|
|
|
|
|
|
$rows = array_merge(
|
|
|
|
$rows,
|
|
|
|
$this->formatAssociationMappings($metadata->associationMappings),
|
|
|
|
$this->formatFieldMappings($metadata->fieldMappings)
|
|
|
|
);
|
2014-08-31 09:27:16 +02:00
|
|
|
|
2014-10-19 18:13:01 +02:00
|
|
|
$table = new Table($output);
|
|
|
|
|
|
|
|
$table->setHeaders(array('Field', 'Value'));
|
|
|
|
|
2014-10-19 18:32:12 +02:00
|
|
|
array_map(array($table, 'addRow'), $rows);
|
2014-10-19 18:13:01 +02:00
|
|
|
|
|
|
|
$table->render();
|
2014-08-31 09:27:16 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Return all mapped entity class names
|
|
|
|
*
|
2014-10-19 17:49:28 +02:00
|
|
|
* @param EntityManagerInterface $entityManager
|
|
|
|
*
|
2014-10-19 18:21:45 +02:00
|
|
|
* @return string[]
|
2014-08-31 09:27:16 +02:00
|
|
|
*/
|
2014-10-19 17:49:28 +02:00
|
|
|
private function getMappedEntities(EntityManagerInterface $entityManager)
|
2014-08-31 09:27:16 +02:00
|
|
|
{
|
2014-10-19 17:49:28 +02:00
|
|
|
$entityClassNames = $entityManager
|
|
|
|
->getConfiguration()
|
|
|
|
->getMetadataDriverImpl()
|
|
|
|
->getAllClassNames();
|
2014-08-31 09:27:16 +02:00
|
|
|
|
2014-10-19 17:49:28 +02:00
|
|
|
if ( ! $entityClassNames) {
|
2014-08-31 09:27:16 +02:00
|
|
|
throw new \InvalidArgumentException(
|
|
|
|
'You do not have any mapped Doctrine ORM entities according to the current configuration. '.
|
|
|
|
'If you have entities or mapping files you should check your mapping configuration for errors.'
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
return $entityClassNames;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Return the class metadata for the given entity
|
|
|
|
* name
|
|
|
|
*
|
2014-10-19 17:49:28 +02:00
|
|
|
* @param string $entityName Full or partial entity name
|
|
|
|
* @param EntityManagerInterface $entityManager
|
|
|
|
*
|
|
|
|
* @return \Doctrine\ORM\Mapping\ClassMetadata
|
2014-08-31 09:27:16 +02:00
|
|
|
*/
|
2014-10-19 17:49:28 +02:00
|
|
|
private function getClassMetadata($entityName, EntityManagerInterface $entityManager)
|
2014-08-31 09:27:16 +02:00
|
|
|
{
|
|
|
|
try {
|
2014-10-19 18:14:33 +02:00
|
|
|
return $entityManager->getClassMetadata($entityName);
|
|
|
|
} catch (MappingException $e) {
|
2014-10-19 18:21:45 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
$matches = array_filter(
|
|
|
|
$this->getMappedEntities($entityManager),
|
|
|
|
function ($mappedEntity) use ($entityName) {
|
2014-08-31 09:27:16 +02:00
|
|
|
if (preg_match('{' . preg_quote($entityName) . '}', $mappedEntity)) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
2014-10-19 18:21:45 +02:00
|
|
|
);
|
2014-08-31 09:27:16 +02:00
|
|
|
|
2014-10-19 18:21:45 +02:00
|
|
|
if (! $matches) {
|
|
|
|
throw new \InvalidArgumentException(sprintf(
|
|
|
|
'Could not find any mapped Entity classes matching "%s"',
|
|
|
|
$entityName
|
|
|
|
));
|
|
|
|
}
|
2014-08-31 09:27:16 +02:00
|
|
|
|
2014-10-19 18:21:45 +02:00
|
|
|
if (count($matches) > 1) {
|
|
|
|
throw new \InvalidArgumentException(sprintf(
|
|
|
|
'Entity name "%s" is ambigous, possible matches: "%s"',
|
|
|
|
$entityName, implode(', ', $matches)
|
|
|
|
));
|
2014-10-19 18:14:33 +02:00
|
|
|
}
|
2014-10-19 18:21:45 +02:00
|
|
|
|
|
|
|
return $entityManager->getClassMetadata(current($matches));
|
2014-08-31 09:27:16 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Format the given value for console output
|
|
|
|
*
|
|
|
|
* @param mixed $value
|
2014-10-19 17:49:28 +02:00
|
|
|
*
|
|
|
|
* @return string
|
2014-08-31 09:27:16 +02:00
|
|
|
*/
|
|
|
|
private function formatValue($value)
|
|
|
|
{
|
|
|
|
if ('' === $value) {
|
|
|
|
return '';
|
|
|
|
}
|
|
|
|
|
|
|
|
if (null === $value) {
|
|
|
|
return '<comment>Null</comment>';
|
|
|
|
}
|
|
|
|
|
|
|
|
if (is_bool($value)) {
|
|
|
|
return '<comment>' . ($value ? 'True' : 'False') . '</comment>';
|
|
|
|
}
|
|
|
|
|
|
|
|
if (empty($value)) {
|
|
|
|
return '<comment>Empty</comment>';
|
|
|
|
}
|
|
|
|
|
|
|
|
if (is_array($value)) {
|
2014-10-19 17:53:40 +02:00
|
|
|
if (defined('JSON_UNESCAPED_UNICODE') && defined('JSON_UNESCAPED_SLASHES')) {
|
2014-10-19 18:21:45 +02:00
|
|
|
return json_encode($value, JSON_UNESCAPED_UNICODE | JSON_UNESCAPED_SLASHES);
|
2014-08-31 09:27:16 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
return json_encode($value);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (is_object($value)) {
|
|
|
|
return sprintf('<%s>', get_class($value));
|
|
|
|
}
|
|
|
|
|
|
|
|
if (is_scalar($value)) {
|
|
|
|
return $value;
|
|
|
|
}
|
|
|
|
|
|
|
|
throw new \InvalidArgumentException(sprintf('Do not know how to format value "%s"', print_r($value, true)));
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2014-10-19 17:49:28 +02:00
|
|
|
* Add the given label and value to the two column table output
|
2014-08-31 09:27:16 +02:00
|
|
|
*
|
|
|
|
* @param string $label Label for the value
|
2014-10-19 17:49:28 +02:00
|
|
|
* @param mixed $value A Value to show
|
2014-10-19 18:32:12 +02:00
|
|
|
*
|
|
|
|
* @return array
|
2014-08-31 09:27:16 +02:00
|
|
|
*/
|
|
|
|
private function formatField($label, $value)
|
|
|
|
{
|
|
|
|
if (null === $value) {
|
|
|
|
$value = '<comment>None</comment>';
|
|
|
|
}
|
|
|
|
|
2014-10-19 18:32:12 +02:00
|
|
|
return array(sprintf('<info>%s</info>', $label), $this->formatValue($value));
|
2014-08-31 09:27:16 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Format the association mappings
|
|
|
|
*
|
|
|
|
* @param array
|
2014-10-19 18:32:12 +02:00
|
|
|
*
|
|
|
|
* @return array
|
2014-08-31 09:27:16 +02:00
|
|
|
*/
|
|
|
|
private function formatAssociationMappings($associationMappings)
|
|
|
|
{
|
2014-10-19 18:32:12 +02:00
|
|
|
$output = array();
|
|
|
|
$output[] = $this->formatField('Association mappings:', '');
|
2014-10-19 17:53:40 +02:00
|
|
|
|
2014-08-31 09:27:16 +02:00
|
|
|
foreach ($associationMappings as $associationName => $mapping) {
|
2014-10-19 18:32:12 +02:00
|
|
|
$output[] = $this->formatField(sprintf(' %s', $associationName), '');
|
2014-10-19 17:53:40 +02:00
|
|
|
|
2014-08-31 09:27:16 +02:00
|
|
|
foreach ($mapping as $field => $value) {
|
2014-10-19 18:32:12 +02:00
|
|
|
$output[] = $this->formatField(sprintf(' %s', $field), $this->formatValue($value));
|
2014-08-31 09:27:16 +02:00
|
|
|
}
|
|
|
|
}
|
2014-10-19 18:32:12 +02:00
|
|
|
|
|
|
|
return $output;
|
2014-08-31 09:27:16 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Format the entity listeners
|
|
|
|
*
|
|
|
|
* @param array $entityListeners
|
2014-10-19 18:32:12 +02:00
|
|
|
*
|
|
|
|
* @return array
|
2014-08-31 09:27:16 +02:00
|
|
|
*/
|
2014-10-19 18:34:06 +02:00
|
|
|
private function formatEntityListeners(array $entityListeners)
|
2014-08-31 09:27:16 +02:00
|
|
|
{
|
2014-10-19 18:34:06 +02:00
|
|
|
return $this->formatField(
|
|
|
|
'Entity listeners',
|
|
|
|
array_map(
|
|
|
|
function ($entityListener) {
|
|
|
|
return get_class($entityListener);
|
|
|
|
},
|
|
|
|
$entityListeners
|
|
|
|
)
|
|
|
|
);
|
2014-08-31 09:27:16 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Form the field mappings
|
|
|
|
*
|
|
|
|
* @param array $fieldMappings
|
2014-10-19 18:32:12 +02:00
|
|
|
*
|
|
|
|
* @return array
|
2014-08-31 09:27:16 +02:00
|
|
|
*/
|
|
|
|
private function formatFieldMappings($fieldMappings)
|
|
|
|
{
|
2014-10-19 18:32:12 +02:00
|
|
|
$output = array();
|
|
|
|
$output[] = $this->formatField('Field mappings:', '');
|
2014-10-19 18:21:45 +02:00
|
|
|
|
2014-08-31 09:27:16 +02:00
|
|
|
foreach ($fieldMappings as $fieldName => $mapping) {
|
2014-10-19 18:32:12 +02:00
|
|
|
$output[] = $this->formatField(sprintf(' %s',$fieldName), '');
|
2014-10-19 18:21:45 +02:00
|
|
|
|
2014-08-31 09:27:16 +02:00
|
|
|
foreach ($mapping as $field => $value) {
|
2014-10-19 18:32:12 +02:00
|
|
|
$output[] = $this->formatField(sprintf(' %s', $field), $this->formatValue($value));
|
2014-08-31 09:27:16 +02:00
|
|
|
}
|
|
|
|
}
|
2014-10-19 18:32:12 +02:00
|
|
|
|
|
|
|
return $output;
|
2014-08-31 09:27:16 +02:00
|
|
|
}
|
|
|
|
}
|