2010-04-26 11:55:43 -04:00
|
|
|
The heart of an object relational mapper is the mapping information that glues
|
|
|
|
everything together. It instructs the EntityManager how it should behave when
|
|
|
|
dealing with the different entities.
|
|
|
|
|
|
|
|
++ Core Metadata Drivers
|
|
|
|
|
|
|
|
Doctrine provides a few different ways for you to specify your metadata:
|
|
|
|
|
|
|
|
* **XML files** (XmlDriver)
|
|
|
|
* **Class DocBlock Annotations** (AnnotationDriver)
|
|
|
|
* **YAML files** (YamlDriver)
|
|
|
|
* **PHP Code in files or static functions** (PhpDriver)
|
|
|
|
|
|
|
|
Something important to note about the above drivers is they are all an intermediate
|
|
|
|
step to the same end result. The mapping information is populated to
|
|
|
|
`Doctrine\ORM\Mapping\ClassMetadata` instances. So in the end, Doctrine
|
2010-10-07 22:39:52 +01:00
|
|
|
only ever has to work with the API of the `ClassMetadata` class to get mapping
|
2010-04-26 11:55:43 -04:00
|
|
|
information for an entity.
|
|
|
|
|
|
|
|
> **TIP**
|
|
|
|
> The populated `ClassMetadata` instances are also cached so in a production
|
|
|
|
> environment the parsing and populating only ever happens once. You can configure
|
|
|
|
> the metadata cache implementation using the `setMetadataCacheImpl()` method on
|
|
|
|
> the `Doctrine\ORM\Configuration` class:
|
|
|
|
>
|
|
|
|
> [php]
|
|
|
|
> $em->getConfiguration()->setMetadataCacheImpl(new ApcCache());
|
|
|
|
|
|
|
|
If you want to use one of the included core metadata drivers you just need to
|
|
|
|
configure it. All the drivers are in the `Doctrine\ORM\Mapping\Driver` namespace:
|
|
|
|
|
|
|
|
[php]
|
|
|
|
$driver = new \Doctrine\ORM\Mapping\Driver\XmlDriver('/path/to/mapping/files');
|
|
|
|
$em->getConfiguration()->setMetadataDriverImpl($driver);
|
|
|
|
|
|
|
|
++ Implementing Metadata Drivers
|
|
|
|
|
|
|
|
In addition to the included metadata drivers you can very easily implement
|
|
|
|
your own. All you need to do is define a class which implements the `Driver`
|
|
|
|
interface:
|
|
|
|
|
|
|
|
[php]
|
|
|
|
namespace Doctrine\ORM\Mapping\Driver;
|
|
|
|
|
|
|
|
use Doctrine\ORM\Mapping\ClassMetadataInfo;
|
|
|
|
|
|
|
|
interface Driver
|
|
|
|
{
|
|
|
|
/**
|
|
|
|
* Loads the metadata for the specified class into the provided container.
|
|
|
|
*
|
|
|
|
* @param string $className
|
|
|
|
* @param ClassMetadataInfo $metadata
|
|
|
|
*/
|
|
|
|
function loadMetadataForClass($className, ClassMetadataInfo $metadata);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Gets the names of all mapped classes known to this driver.
|
|
|
|
*
|
|
|
|
* @return array The names of all mapped classes known to this driver.
|
|
|
|
*/
|
|
|
|
function getAllClassNames();
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Whether the class with the specified name should have its metadata loaded.
|
|
|
|
* This is only the case if it is either mapped as an Entity or a
|
|
|
|
* MappedSuperclass.
|
|
|
|
*
|
|
|
|
* @param string $className
|
|
|
|
* @return boolean
|
|
|
|
*/
|
|
|
|
function isTransient($className);
|
|
|
|
}
|
|
|
|
|
|
|
|
If you want to write a metadata driver to parse information from some file format
|
|
|
|
we've made your life a little easier by providing the `AbstractFileDriver`
|
|
|
|
implementation for you to extend from:
|
|
|
|
|
|
|
|
[php]
|
|
|
|
class MyMetadataDriver extends AbstractFileDriver
|
|
|
|
{
|
|
|
|
/**
|
|
|
|
* {@inheritdoc}
|
|
|
|
*/
|
|
|
|
protected $_fileExtension = '.dcm.ext';
|
|
|
|
|
|
|
|
/**
|
|
|
|
* {@inheritdoc}
|
|
|
|
*/
|
|
|
|
public function loadMetadataForClass($className, ClassMetadataInfo $metadata)
|
|
|
|
{
|
|
|
|
$data = $this->_loadMappingFile($file);
|
|
|
|
|
|
|
|
// populate ClassMetadataInfo instance from $data
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* {@inheritdoc}
|
|
|
|
*/
|
|
|
|
protected function _loadMappingFile($file)
|
|
|
|
{
|
|
|
|
// parse contents of $file and return php data structure
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
> **NOTE**
|
|
|
|
> When using the `AbstractFileDriver` it requires that you only have one entity
|
|
|
|
> defined per file and the file named after the class described inside where
|
|
|
|
> namespace separators are replaced by periods. So if you have an entity named
|
|
|
|
> `Entities\User` and you wanted to write a mapping file for your driver above
|
|
|
|
> you would need to name the file `Entities.User.dcm.ext` for it to be recognized.
|
|
|
|
|
|
|
|
Now you can use your `MyMetadataDriver` implementation by setting it with the
|
|
|
|
`setMetadataDriverImpl()` method:
|
|
|
|
|
|
|
|
[php]
|
|
|
|
$driver = new MyMetadataDriver('/path/to/mapping/files');
|
|
|
|
$em->getConfiguration()->setMetadataDriverImpl($driver);
|
|
|
|
|
|
|
|
++ ClassMetadata
|
|
|
|
|
|
|
|
The last piece you need to know and understand about metadata in Doctrine 2 is
|
|
|
|
the API of the `ClassMetadata` classes. You need to be familiar with them in order
|
|
|
|
to implement your own drivers but more importantly to retrieve mapping information
|
|
|
|
for a certain entity when needed.
|
|
|
|
|
|
|
|
You have all the methods you need to manually specify the mapping information
|
|
|
|
instead of using some mapping file to populate it from. The base `ClassMetadataInfo`
|
|
|
|
class is responsible for only data storage and is not meant for runtime use. It
|
|
|
|
does not require that the class actually exists yet so it is useful for describing some
|
|
|
|
entity before it exists and using that information to generate for example the
|
|
|
|
entities themselves. The class `ClassMetadata` extends `ClassMetadataInfo` and
|
|
|
|
adds some functionality required for runtime usage and requires that the PHP
|
|
|
|
class is present and can be autoloaded.
|
|
|
|
|
|
|
|
You can read more about the API of the `ClassMetadata` classes in the PHP Mapping
|
|
|
|
chapter.
|
|
|
|
|
|
|
|
++ Getting ClassMetadata Instances
|
|
|
|
|
|
|
|
If you want to get the `ClassMetadata` instance for an entity in your project
|
|
|
|
to programatically use some mapping information to generate some HTML or something
|
|
|
|
similar you can retrieve it through the `ClassMetadataFactory`:
|
|
|
|
|
|
|
|
[php]
|
|
|
|
$cmf = $em->getMetadataFactory();
|
|
|
|
$class = $cmf->getMetadataFor('MyEntityName');
|
|
|
|
|
|
|
|
Now you can learn about the entity and use the data stored in the `ClassMetadata`
|
|
|
|
instance to get all mapped fields for example and iterate over them:
|
|
|
|
|
|
|
|
[php]
|
|
|
|
foreach ($class->fieldMappings as $fieldMapping) {
|
|
|
|
echo $fieldMapping['fieldName'] . "\n";
|
|
|
|
}
|