Advanced Configuration ====================== The configuration of the EntityManager requires a ``Doctrine\ORM\Configuration`` instance as well as some database connection parameters. This example shows all the potential steps of configuration. .. code-block:: php setMetadataCacheImpl($cache); $driverImpl = $config->newDefaultAnnotationDriver('/path/to/lib/MyProject/Entities'); $config->setMetadataDriverImpl($driverImpl); $config->setQueryCacheImpl($cache); $config->setProxyDir('/path/to/myproject/lib/MyProject/Proxies'); $config->setProxyNamespace('MyProject\Proxies'); if ($applicationMode == "development") { $config->setAutoGenerateProxyClasses(true); } else { $config->setAutoGenerateProxyClasses(false); } $connectionOptions = array( 'driver' => 'pdo_sqlite', 'path' => 'database.sqlite' ); $em = EntityManager::create($connectionOptions, $config); .. note:: Do not use Doctrine without a metadata and query cache! Doctrine is optimized for working with caches. The main parts in Doctrine that are optimized for caching are the metadata mapping information with the metadata cache and the DQL to SQL conversions with the query cache. These 2 caches require only an absolute minimum of memory yet they heavily improve the runtime performance of Doctrine. The recommended cache driver to use with Doctrine is `APC `_. APC provides you with an opcode-cache (which is highly recommended anyway) and a very fast in-memory cache storage that you can use for the metadata and query caches as seen in the previous code snippet. Configuration Options --------------------- The following sections describe all the configuration options available on a ``Doctrine\ORM\Configuration`` instance. Proxy Directory (***REQUIRED***) ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ .. code-block:: php setProxyDir($dir); $config->getProxyDir(); Gets or sets the directory where Doctrine generates any proxy classes. For a detailed explanation on proxy classes and how they are used in Doctrine, refer to the "Proxy Objects" section further down. Proxy Namespace (***REQUIRED***) ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ .. code-block:: php setProxyNamespace($namespace); $config->getProxyNamespace(); Gets or sets the namespace to use for generated proxy classes. For a detailed explanation on proxy classes and how they are used in Doctrine, refer to the "Proxy Objects" section further down. Metadata Driver (***REQUIRED***) ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ .. code-block:: php setMetadataDriverImpl($driver); $config->getMetadataDriverImpl(); Gets or sets the metadata driver implementation that is used by Doctrine to acquire the object-relational metadata for your classes. There are currently 4 available implementations: - ``Doctrine\ORM\Mapping\Driver\AnnotationDriver`` - ``Doctrine\ORM\Mapping\Driver\XmlDriver`` - ``Doctrine\ORM\Mapping\Driver\YamlDriver`` - ``Doctrine\ORM\Mapping\Driver\DriverChain`` Throughout the most part of this manual the AnnotationDriver is used in the examples. For information on the usage of the XmlDriver or YamlDriver please refer to the dedicated chapters ``XML Mapping`` and ``YAML Mapping``. The annotation driver can be configured with a factory method on the ``Doctrine\ORM\Configuration``: .. code-block:: php newDefaultAnnotationDriver('/path/to/lib/MyProject/Entities'); $config->setMetadataDriverImpl($driverImpl); The path information to the entities is required for the annotation driver, because otherwise mass-operations on all entities through the console could not work correctly. All of metadata drivers accept either a single directory as a string or an array of directories. With this feature a single driver can support multiple directories of Entities. Metadata Cache (***RECOMMENDED***) ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ .. code-block:: php setMetadataCacheImpl($cache); $config->getMetadataCacheImpl(); Gets or sets the cache implementation to use for caching metadata information, that is, all the information you supply via annotations, xml or yaml, so that they do not need to be parsed and loaded from scratch on every single request which is a waste of resources. The cache implementation must implement the ``Doctrine\Common\Cache\Cache`` interface. Usage of a metadata cache is highly recommended. The recommended implementations for production are: - ``Doctrine\Common\Cache\ApcCache`` - ``Doctrine\Common\Cache\MemcacheCache`` - ``Doctrine\Common\Cache\XcacheCache`` - ``Doctrine\Common\Cache\RedisCache`` For development you should use the ``Doctrine\Common\Cache\ArrayCache`` which only caches data on a per-request basis. Query Cache (***RECOMMENDED***) ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ .. code-block:: php setQueryCacheImpl($cache); $config->getQueryCacheImpl(); Gets or sets the cache implementation to use for caching DQL queries, that is, the result of a DQL parsing process that includes the final SQL as well as meta information about how to process the SQL result set of a query. Note that the query cache does not affect query results. You do not get stale data. This is a pure optimization cache without any negative side-effects (except some minimal memory usage in your cache). Usage of a query cache is highly recommended. The recommended implementations for production are: - ``Doctrine\Common\Cache\ApcCache`` - ``Doctrine\Common\Cache\MemcacheCache`` - ``Doctrine\Common\Cache\XcacheCache`` - ``Doctrine\Common\Cache\RedisCache`` For development you should use the ``Doctrine\Common\Cache\ArrayCache`` which only caches data on a per-request basis. SQL Logger (***Optional***) ~~~~~~~~~~~~~~~~~~~~~~~~~~~ .. code-block:: php setSQLLogger($logger); $config->getSQLLogger(); Gets or sets the logger to use for logging all SQL statements executed by Doctrine. The logger class must implement the ``Doctrine\DBAL\Logging\SQLLogger`` interface. A simple default implementation that logs to the standard output using ``echo`` and ``var_dump`` can be found at ``Doctrine\DBAL\Logging\EchoSQLLogger``. Auto-generating Proxy Classes (***OPTIONAL***) ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ Proxy classes can either be generated manually through the Doctrine Console or automatically at runtime by Doctrine. The configuration option that controls this behavior is: .. code-block:: php setAutoGenerateProxyClasses($mode); Possible values for ``$mode`` are: - ``Doctrine\Common\Proxy\AbstractProxyFactory::AUTOGENERATE_NEVER`` Never autogenerate a proxy. You will need to generate the proxies manually, for this use the Doctrine Console like so: .. code-block:: php $ ./doctrine orm:generate-proxies When you do this in a development environment, be aware that you may get class/file not found errors if certain proxies are not yet generated. You may also get failing lazy-loads if new methods were added to the entity class that are not yet in the proxy class. In such a case, simply use the Doctrine Console to (re)generate the proxy classes. - ``Doctrine\Common\Proxy\AbstractProxyFactory::AUTOGENERATE_ALWAYS`` Always generates a new proxy in every request and writes it to disk. - ``Doctrine\Common\Proxy\AbstractProxyFactory::AUTOGENERATE_FILE_NOT_EXISTS`` Generate the proxy class when the proxy file does not exist. This strategy causes a file exists call whenever any proxy is used the first time in a request. - ``Doctrine\Common\Proxy\AbstractProxyFactory::AUTOGENERATE_EVAL`` Generate the proxy classes and evaluate them on the fly via eval(), avoiding writing the proxies to disk. This strategy is only sane for development. In a production environment, it is highly recommended to use AUTOGENERATE_NEVER to allow for optimal performances. The other options are interesting in development environment. Before v2.4, ``setAutoGenerateProxyClasses`` would accept a boolean value. This is still possible, ``FALSE`` being equivalent to AUTOGENERATE_NEVER and ``TRUE`` to AUTOGENERATE_ALWAYS. Development vs Production Configuration --------------------------------------- You should code your Doctrine2 bootstrapping with two different runtime models in mind. There are some serious benefits of using APC or Memcache in production. In development however this will frequently give you fatal errors, when you change your entities and the cache still keeps the outdated metadata. That is why we recommend the ``ArrayCache`` for development. Furthermore you should have the Auto-generating Proxy Classes option to true in development and to false in production. If this option is set to ``TRUE`` it can seriously hurt your script performance if several proxy classes are re-generated during script execution. Filesystem calls of that magnitude can even slower than all the database queries Doctrine issues. Additionally writing a proxy sets an exclusive file lock which can cause serious performance bottlenecks in systems with regular concurrent requests. Connection Options ------------------ The ``$connectionOptions`` passed as the first argument to ``EntityManager::create()`` has to be either an array or an instance of ``Doctrine\DBAL\Connection``. If an array is passed it is directly passed along to the DBAL Factory ``Doctrine\DBAL\DriverManager::getConnection()``. The DBAL configuration is explained in the `DBAL section <./../../../../../dbal/latest/docs/reference/configuration/en>`_. Proxy Objects ------------- A proxy object is an object that is put in place or used instead of the "real" object. A proxy object can add behavior to the object being proxied without that object being aware of it. In Doctrine 2, proxy objects are used to realize several features but mainly for transparent lazy-loading. Proxy objects with their lazy-loading facilities help to keep the subset of objects that are already in memory connected to the rest of the objects. This is an essential property as without it there would always be fragile partial objects at the outer edges of your object graph. Doctrine 2 implements a variant of the proxy pattern where it generates classes that extend your entity classes and adds lazy-loading capabilities to them. Doctrine can then give you an instance of such a proxy class whenever you request an object of the class being proxied. This happens in two situations: Reference Proxies ~~~~~~~~~~~~~~~~~ The method ``EntityManager#getReference($entityName, $identifier)`` lets you obtain a reference to an entity for which the identifier is known, without loading that entity from the database. This is useful, for example, as a performance enhancement, when you want to establish an association to an entity for which you have the identifier. You could simply do this: .. code-block:: php getReference('MyProject\Model\Item', $itemId); $cart->addItem($item); Here, we added an Item to a Cart without loading the Item from the database. If you invoke any method on the Item instance, it would fully initialize its state transparently from the database. Here $item is actually an instance of the proxy class that was generated for the Item class but your code does not need to care. In fact it **should not care**. Proxy objects should be transparent to your code. Association proxies ~~~~~~~~~~~~~~~~~~~ The second most important situation where Doctrine uses proxy objects is when querying for objects. Whenever you query for an object that has a single-valued association to another object that is configured LAZY, without joining that association in the same query, Doctrine puts proxy objects in place where normally the associated object would be. Just like other proxies it will transparently initialize itself on first access. .. note:: Joining an association in a DQL or native query essentially means eager loading of that association in that query. This will override the 'fetch' option specified in the mapping for that association, but only for that query. Generating Proxy classes ~~~~~~~~~~~~~~~~~~~~~~~~ In a production environment, it is highly recommended to use ``AUTOGENERATE_NEVER`` to allow for optimal performances. However you will be required to generate the proxies manually using the Doctrine Console: .. code-block:: php $ ./doctrine orm:generate-proxies The other options are interesting in development environment: - ``AUTOGENERATE_ALWAYS`` will require you to create and configure a proxy directory. Proxies will be generated and written to file on each request, so any modification to your code will be acknowledged. - ``AUTOGENERATE_FILE_NOT_EXISTS`` will not overwrite an existing proxy file. If your code changes, you will need to regenerate the proxies manually. - ``AUTOGENERATE_EVAL`` will regenerate each proxy on each request, but without writing them to disk. Autoloading Proxies ------------------- When you deserialize proxy objects from the session or any other storage it is necessary to have an autoloading mechanism in place for these classes. For implementation reasons Proxy class names are not PSR-0 compliant. This means that you have to register a special autoloader for these classes: .. code-block:: php addDriver($xmlDriver, 'Doctrine\Tests\Models\Company'); $chain->addDriver($yamlDriver, 'Doctrine\Tests\ORM\Mapping'); Based on the namespace of the entity the loading of entities is delegated to the appropriate driver. The chain semantics come from the fact that the driver loops through all namespaces and matches the entity class name against the namespace using a ``strpos() === 0`` call. This means you need to order the drivers correctly if sub-namespaces use different metadata driver implementations. Default Repository (***OPTIONAL***) ----------------------------------- Specifies the FQCN of a subclass of the EntityRepository. That will be available for all entities without a custom repository class. .. code-block:: php setDefaultRepositoryClassName($fqcn); $config->getDefaultRepositoryClassName(); The default value is ``Doctrine\ORM\EntityRepository``. Any repository class must be a subclass of EntityRepository otherwise you got an ORMException Setting up the Console ---------------------- Doctrine uses the Symfony Console component for generating the command line interface. You can take a look at the ``vendor/bin/doctrine.php`` script and the ``Doctrine\ORM\Tools\Console\ConsoleRunner`` command for inspiration how to setup the cli. In general the required code looks like this: .. code-block:: php setCatchExceptions(true); $cli->setHelperSet($helperSet); Doctrine\ORM\Tools\Console\ConsoleRunner::addCommands($cli); $cli->run();