diff --git a/docs/en/reference/second-level-cache.rst b/docs/en/reference/second-level-cache.rst index ecf682b08..e0965b429 100644 --- a/docs/en/reference/second-level-cache.rst +++ b/docs/en/reference/second-level-cache.rst @@ -71,7 +71,7 @@ A query region might be something like : Cache Regions ------------- -``Doctrine\ORM\Cache\Region\DefaultRegion`` Its the default implementation. +``Doctrine\ORM\Cache\Region\DefaultRegion`` It's the default implementation. A simplest cache region compatible with all doctrine-cache drivers but does not support locking. ``Doctrine\ORM\Cache\Region`` and ``Doctrine\ORM\Cache\ConcurrentRegion`` @@ -91,59 +91,7 @@ Defines a contract for accessing a particular region. Defines a contract for accessing a particular cache region. -.. code-block:: php - - `_. Concurrent cache region ~~~~~~~~~~~~~~~~~~~~~~~ @@ -157,29 +105,7 @@ If you want to use an ``READ_WRITE`` cache, you should consider providing your o Defines contract for concurrently managed data region. -.. code-block:: php - - `_. Cache region ~~~~~~~~~~~~ @@ -188,21 +114,7 @@ Cache region Tracks the timestamps of the most recent updates to particular entity. -.. code-block:: php - - `_. .. _reference-second-level-cache-mode: @@ -218,7 +130,7 @@ Caching mode * ``NONSTRICT_READ_WRITE`` - * Read Write Cache doesn’t employ any locks but can do reads, inserts , updates and deletes. + * Read Write Cache doesn’t employ any locks but can do reads, inserts, updates and deletes. * Good if the application needs to update data rarely. @@ -292,80 +204,7 @@ It allows you to provide a specific implementation of the following components : * ``EntityHydrator`` Transform an entity into a cache entry and cache entry into entities * ``CollectionHydrator`` Transform a collection into a cache entry and cache entry into collection -.. code-block:: php - - `_. Region Lifetime ~~~~~~~~~~~~~~~ @@ -428,81 +267,7 @@ By providing a cache logger you should be able to get information about all cach If you want to get more information you should implement ``\Doctrine\ORM\Cache\Logging\CacheLogger``. and collect all information you want. - .. code-block:: php - - `_. Entity cache definition @@ -810,6 +575,47 @@ The Cache Mode controls how a particular query interacts with the second-level c The the default query cache mode is ```Cache::MODE_NORMAL``` +DELETE / UPDATE queries +~~~~~~~~~~~~~~~~~~~~~~~ + +DQL UPDATE / DELETE statements are ported directly into a database and bypass the second-level cache, +Entities that are already cached will NOT be invalidated. +However the cached data could be evicted using the cache API or an special query hint. + + +Execute the ``UPDATE`` and invalidate ``all cache entries`` using ``Query::HINT_CACHE_EVICT`` + +.. code-block:: php + + _em->createQuery("UPDATE Entity\Country u SET u.name = 'unknown' WHERE u.id = 1") + ->setHint(Query::HINT_CACHE_EVICT, true) + ->execute(); + + +Execute the ``UPDATE`` and invalidate ``all cache entries`` using the cache API + +.. code-block:: php + + _em->createQuery("UPDATE Entity\Country u SET u.name = 'unknown' WHERE u.id = 1") + ->execute(); + // Invoke Cache API + $em->getCache()->evictEntityRegion('Entity\Country'); + + +Execute the ``UPDATE`` and invalidate ``a specific cache entry`` using the cache API + +.. code-block:: php + + _em->createQuery("UPDATE Entity\Country u SET u.name = 'unknown' WHERE u.id = 1") + ->execute(); + // Invoke Cache API + $em->getCache()->evictEntity('Entity\Country', 1); Using the repository query cache --------------------- @@ -908,45 +714,12 @@ For performance reasons the cache API does not extract from composite primary ke $id = array('source' => new Article(1), 'target' => new Article(2)); $reference = $em->find('Reference', $id); - -DELETE / UPDATE queries +Distribute environments ~~~~~~~~~~~~~~~~~~~~~~~ -DQL UPDATE / DELETE statements are ported directly into a database and bypass the second-level cache, -Entities that are already cached will NOT be invalidated. -However the cached data could be evicted using the cache API or an special query hint. +Some cache driver are not meant to be used in a distribute environment +Load-balancer for distributing workloads across multiple computing resources +should be used in conjunction with distributed caching system such as memcached, redis, riak ... - -Execute the ``UPDATE`` and invalidate ``all cache entries`` using ``Query::HINT_CACHE_EVICT`` - -.. code-block:: php - - _em->createQuery("UPDATE Entity\Country u SET u.name = 'unknown' WHERE u.id = 1") - ->setHint(Query::HINT_CACHE_EVICT, true) - ->execute(); - - -Execute the ``UPDATE`` and invalidate ``all cache entries`` using the cache API - -.. code-block:: php - - _em->createQuery("UPDATE Entity\Country u SET u.name = 'unknown' WHERE u.id = 1") - ->execute(); - // Invoke Cache API - $em->getCache()->evictEntityRegion('Entity\Country'); - - -Execute the ``UPDATE`` and invalidate ``a specific cache entry`` using the cache API - -.. code-block:: php - - _em->createQuery("UPDATE Entity\Country u SET u.name = 'unknown' WHERE u.id = 1") - ->execute(); - // Invoke Cache API - $em->getCache()->evictEntity('Entity\Country', 1); \ No newline at end of file +Caches should be used with care when using a load-balancer if you don't share the cache. +While using APC or any file based cache update occurred in a specific machine would not reflect to the cache in other machines. diff --git a/lib/Doctrine/ORM/AbstractQuery.php b/lib/Doctrine/ORM/AbstractQuery.php index 5d77b4579..3ac4cf840 100644 --- a/lib/Doctrine/ORM/AbstractQuery.php +++ b/lib/Doctrine/ORM/AbstractQuery.php @@ -346,9 +346,7 @@ abstract class AbstractQuery $parameterCollection = new ArrayCollection(); foreach ($parameters as $key => $value) { - $parameter = new Parameter($key, $value); - - $parameterCollection->add($parameter); + $parameterCollection->add(new Parameter($key, $value)); } $parameters = $parameterCollection; @@ -387,9 +385,7 @@ abstract class AbstractQuery return $this; } - $parameter = new Parameter($key, $value, $type); - - $this->parameters->add($parameter); + $this->parameters->add(new Parameter($key, $value, $type)); return $this; } @@ -453,7 +449,7 @@ abstract class AbstractQuery * * @return \Doctrine\ORM\Query\ResultSetMapping */ - public function getResultSetMapping() + protected function getResultSetMapping() { return $this->_resultSetMapping; } diff --git a/lib/Doctrine/ORM/Cache/CacheKey.php b/lib/Doctrine/ORM/Cache/CacheKey.php index ede203091..1641c9900 100644 --- a/lib/Doctrine/ORM/Cache/CacheKey.php +++ b/lib/Doctrine/ORM/Cache/CacheKey.php @@ -21,7 +21,7 @@ namespace Doctrine\ORM\Cache; /** - * Defines entity / collection key to be stored in the cache region. + * Defines entity / collection / query key to be stored in the cache region. * Allows multiple roles to be stored in the same cache region. * * @since 2.5 @@ -30,6 +30,8 @@ namespace Doctrine\ORM\Cache; abstract class CacheKey { /** + * READ-ONLY: Public only for performance reasons, it should be considered immutable. + * * @var string Unique identifier */ public $hash; diff --git a/lib/Doctrine/ORM/Cache/CollectionCacheEntry.php b/lib/Doctrine/ORM/Cache/CollectionCacheEntry.php index 4bb329a7e..58c8757e0 100644 --- a/lib/Doctrine/ORM/Cache/CollectionCacheEntry.php +++ b/lib/Doctrine/ORM/Cache/CollectionCacheEntry.php @@ -29,12 +29,14 @@ namespace Doctrine\ORM\Cache; class CollectionCacheEntry implements CacheEntry { /** - * @var array + * READ-ONLY: Public only for performance reasons, it should be considered immutable. + * + * @var array The list of entity identifiers hold by the collection */ public $identifiers; /** - * @param array $identifiers + * @param array $identifiers List of entity identifiers hold by the collection */ public function __construct(array $identifiers) { @@ -42,7 +44,11 @@ class CollectionCacheEntry implements CacheEntry } /** - * @param array $values + * Creates a new CollectionCacheEntry + * + * This method allow Doctrine\Common\Cache\PhpFileCache compatibility + * + * @param array $values array containing property values */ public static function __set_state(array $values) { diff --git a/lib/Doctrine/ORM/Cache/CollectionCacheKey.php b/lib/Doctrine/ORM/Cache/CollectionCacheKey.php index 184fa9bf2..6b6314550 100644 --- a/lib/Doctrine/ORM/Cache/CollectionCacheKey.php +++ b/lib/Doctrine/ORM/Cache/CollectionCacheKey.php @@ -29,17 +29,23 @@ namespace Doctrine\ORM\Cache; class CollectionCacheKey extends CacheKey { /** - * @var array + * READ-ONLY: Public only for performance reasons, it should be considered immutable. + * + * @var array The owner entity identifier */ public $ownerIdentifier; /** - * @var string + * READ-ONLY: Public only for performance reasons, it should be considered immutable. + * + * @var string The owner entity class */ public $entityClass; /** - * @var string + * READ-ONLY: Public only for performance reasons, it should be considered immutable. + * + * @var string The association name */ public $association; @@ -52,9 +58,9 @@ class CollectionCacheKey extends CacheKey { ksort($ownerIdentifier); - $this->entityClass = $entityClass; - $this->association = $association; $this->ownerIdentifier = $ownerIdentifier; + $this->entityClass = (string) $entityClass; + $this->association = (string) $association; $this->hash = str_replace('\\', '.', strtolower($entityClass)) . '_' . implode(' ', $ownerIdentifier) . '__' . $association; } } diff --git a/lib/Doctrine/ORM/Cache/DefaultCacheFactory.php b/lib/Doctrine/ORM/Cache/DefaultCacheFactory.php index 1add374eb..92ec80c5c 100644 --- a/lib/Doctrine/ORM/Cache/DefaultCacheFactory.php +++ b/lib/Doctrine/ORM/Cache/DefaultCacheFactory.php @@ -164,10 +164,15 @@ class DefaultCacheFactory implements CacheFactory */ public function buildQueryCache(EntityManagerInterface $em, $regionName = null) { - return new DefaultQueryCache($em, $this->getRegion(array( - 'region' => $regionName ?: Cache::DEFAULT_QUERY_REGION_NAME, - 'usage' => ClassMetadata::CACHE_USAGE_NONSTRICT_READ_WRITE - ))); + return new DefaultQueryCache( + $em, + $this->getRegion( + array( + 'region' => $regionName ?: Cache::DEFAULT_QUERY_REGION_NAME, + 'usage' => ClassMetadata::CACHE_USAGE_NONSTRICT_READ_WRITE + ) + ) + ); } /** @@ -201,7 +206,7 @@ class DefaultCacheFactory implements CacheFactory if ( ! $this->fileLockRegionDirectory) { throw new \LogicException( - 'If you want to use a "READ_WRITE" cache an implementation of "Doctrine\ORM\Cache\ConcurrentRegion" is required, ' . + 'If you what to use a "READ_WRITE" cache an implementation of "Doctrine\ORM\Cache\ConcurrentRegion" is required, ' . 'The default implementation provided by doctrine is "Doctrine\ORM\Cache\Region\FileLockRegion" if you what to use it please provide a valid directory, DefaultCacheFactory#setFileLockRegionDirectory(). ' ); } diff --git a/lib/Doctrine/ORM/Cache/DefaultEntityHydrator.php b/lib/Doctrine/ORM/Cache/DefaultEntityHydrator.php index f022ae59f..57a28e9ab 100644 --- a/lib/Doctrine/ORM/Cache/DefaultEntityHydrator.php +++ b/lib/Doctrine/ORM/Cache/DefaultEntityHydrator.php @@ -123,7 +123,6 @@ class DefaultEntityHydrator implements EntityHydrator } foreach ($metadata->associationMappings as $name => $assoc) { - if ( ! isset($assoc['cache']) || ! isset($data[$name])) { continue; } diff --git a/lib/Doctrine/ORM/Cache/EntityCacheEntry.php b/lib/Doctrine/ORM/Cache/EntityCacheEntry.php index 73ce222b3..03d3c1f24 100644 --- a/lib/Doctrine/ORM/Cache/EntityCacheEntry.php +++ b/lib/Doctrine/ORM/Cache/EntityCacheEntry.php @@ -29,12 +29,16 @@ namespace Doctrine\ORM\Cache; class EntityCacheEntry implements CacheEntry { /** - * @var array + * READ-ONLY: Public only for performance reasons, it should be considered immutable. + * + * @var array The entity map data */ public $data; /** - * @var string + * READ-ONLY: Public only for performance reasons, it should be considered immutable. + * + * @var string The entity class name */ public $class; @@ -49,7 +53,11 @@ class EntityCacheEntry implements CacheEntry } /** - * @param array $values + * Creates a new EntityCacheEntry + * + * This method allow Doctrine\Common\Cache\PhpFileCache compatibility + * + * @param array $values array containing property values */ public static function __set_state(array $values) { diff --git a/lib/Doctrine/ORM/Cache/EntityCacheKey.php b/lib/Doctrine/ORM/Cache/EntityCacheKey.php index 7729691a6..281e610fa 100644 --- a/lib/Doctrine/ORM/Cache/EntityCacheKey.php +++ b/lib/Doctrine/ORM/Cache/EntityCacheKey.php @@ -29,12 +29,16 @@ namespace Doctrine\ORM\Cache; class EntityCacheKey extends CacheKey { /** - * @var array + * READ-ONLY: Public only for performance reasons, it should be considered immutable. + * + * @var array The entity identifier */ public $identifier; /** - * @var string + * READ-ONLY: Public only for performance reasons, it should be considered immutable. + * + * @var string The entity class name */ public $entityClass; diff --git a/lib/Doctrine/ORM/Cache/QueryCacheEntry.php b/lib/Doctrine/ORM/Cache/QueryCacheEntry.php index 46e0d603f..1ba61bcf0 100644 --- a/lib/Doctrine/ORM/Cache/QueryCacheEntry.php +++ b/lib/Doctrine/ORM/Cache/QueryCacheEntry.php @@ -29,12 +29,16 @@ namespace Doctrine\ORM\Cache; class QueryCacheEntry implements CacheEntry { /** - * @var array + * READ-ONLY: Public only for performance reasons, it should be considered immutable. + * + * @var array List of entity identifiers */ public $result; /** - * @var integer + * READ-ONLY: Public only for performance reasons, it should be considered immutable. + * + * @var integer Time creation of this cache entry */ public $time; diff --git a/lib/Doctrine/ORM/Cache/QueryCacheKey.php b/lib/Doctrine/ORM/Cache/QueryCacheKey.php index 7001cc332..9a7d2b7bc 100644 --- a/lib/Doctrine/ORM/Cache/QueryCacheKey.php +++ b/lib/Doctrine/ORM/Cache/QueryCacheKey.php @@ -23,7 +23,7 @@ namespace Doctrine\ORM\Cache; use Doctrine\ORM\Cache; /** - * A key that identifies a particular query. + * A cache key that identifies a particular query. * * @since 2.5 * @author Fabio B. Silva @@ -31,12 +31,16 @@ use Doctrine\ORM\Cache; class QueryCacheKey extends CacheKey { /** - * @var integer + * READ-ONLY: Public only for performance reasons, it should be considered immutable. + * + * @var integer Cache key lifetime */ public $lifetime; /** - * @var integer + * READ-ONLY: Public only for performance reasons, it should be considered immutable. + * + * @var integer Cache mode (Doctrine\ORM\Cache::MODE_*) */ public $cacheMode; diff --git a/lib/Doctrine/ORM/Cache/Region/DefaultRegion.php b/lib/Doctrine/ORM/Cache/Region/DefaultRegion.php index ab518ff65..42e1a3471 100644 --- a/lib/Doctrine/ORM/Cache/Region/DefaultRegion.php +++ b/lib/Doctrine/ORM/Cache/Region/DefaultRegion.php @@ -56,9 +56,9 @@ class DefaultRegion implements Region */ public function __construct($name, CacheProvider $cache, $lifetime = 0) { - $this->name = $name; $this->cache = $cache; - $this->lifetime = $lifetime; + $this->name = (string) $name; + $this->lifetime = (integer) $lifetime; $this->cache->setNamespace($this->name); } diff --git a/lib/Doctrine/ORM/Cache/TimestampCacheEntry.php b/lib/Doctrine/ORM/Cache/TimestampCacheEntry.php index f7617b3bb..9d15c84b3 100644 --- a/lib/Doctrine/ORM/Cache/TimestampCacheEntry.php +++ b/lib/Doctrine/ORM/Cache/TimestampCacheEntry.php @@ -29,6 +29,8 @@ namespace Doctrine\ORM\Cache; class TimestampCacheEntry implements CacheEntry { /** + * READ-ONLY: Public only for performance reasons, it should be considered immutable. + * * @var float */ public $time; @@ -38,11 +40,15 @@ class TimestampCacheEntry implements CacheEntry */ public function __construct($time = null) { - $this->time = $time ?: microtime(true); + $this->time = $time ? (float)$time : microtime(true); } /** - * @param array $values + * Creates a new TimestampCacheEntry + * + * This method allow Doctrine\Common\Cache\PhpFileCache compatibility + * + * @param array $values array containing property values */ public static function __set_state(array $values) { diff --git a/lib/Doctrine/ORM/Cache/TimestampCacheKey.php b/lib/Doctrine/ORM/Cache/TimestampCacheKey.php index 2ae65d065..dfa72274b 100644 --- a/lib/Doctrine/ORM/Cache/TimestampCacheKey.php +++ b/lib/Doctrine/ORM/Cache/TimestampCacheKey.php @@ -33,6 +33,6 @@ class TimestampCacheKey extends CacheKey */ public function __construct($space) { - $this->hash = $space; + $this->hash = (string) $space; } } diff --git a/lib/Doctrine/ORM/Mapping/Driver/AnnotationDriver.php b/lib/Doctrine/ORM/Mapping/Driver/AnnotationDriver.php index 359f26322..a0f99ad9b 100644 --- a/lib/Doctrine/ORM/Mapping/Driver/AnnotationDriver.php +++ b/lib/Doctrine/ORM/Mapping/Driver/AnnotationDriver.php @@ -131,11 +131,12 @@ class AnnotationDriver extends AbstractAnnotationDriver // Evaluate @Cache annotation if (isset($classAnnotations['Doctrine\ORM\Mapping\Cache'])) { $cacheAnnot = $classAnnotations['Doctrine\ORM\Mapping\Cache']; + $cacheMap = array( + 'region' => $cacheAnnot->region, + 'usage' => constant('Doctrine\ORM\Mapping\ClassMetadata::CACHE_USAGE_' . $cacheAnnot->usage), + ); - $metadata->enableCache(array( - 'usage' => constant('Doctrine\ORM\Mapping\ClassMetadata::CACHE_USAGE_' . $cacheAnnot->usage), - 'region' => $cacheAnnot->region, - )); + $metadata->enableCache($cacheMap); } // Evaluate NamedNativeQueries annotation diff --git a/lib/Doctrine/ORM/Query.php b/lib/Doctrine/ORM/Query.php index 7be9dbd5b..71ddb6a3c 100644 --- a/lib/Doctrine/ORM/Query.php +++ b/lib/Doctrine/ORM/Query.php @@ -215,7 +215,7 @@ final class Query extends AbstractQuery /** * {@inheritdoc} */ - public function getResultSetMapping() + protected function getResultSetMapping() { // parse query or load from cache if ($this->_resultSetMapping === null) { diff --git a/tests/Doctrine/Tests/ORM/Cache/DefaultCacheFactoryTest.php b/tests/Doctrine/Tests/ORM/Cache/DefaultCacheFactoryTest.php index 30cc3ca76..5e72fd0d6 100644 --- a/tests/Doctrine/Tests/ORM/Cache/DefaultCacheFactoryTest.php +++ b/tests/Doctrine/Tests/ORM/Cache/DefaultCacheFactoryTest.php @@ -253,7 +253,7 @@ class DefaultCacheFactoryTest extends OrmTestCase /** * @expectedException LogicException - * @expectedExceptionMessage If you want to use a "READ_WRITE" cache an implementation of "Doctrine\ORM\Cache\ConcurrentRegion" is required, The default implementation provided by doctrine is "Doctrine\ORM\Cache\Region\FileLockRegion" if you what to use it please provide a valid directory + * @expectedExceptionMessage If you what to use a "READ_WRITE" cache an implementation of "Doctrine\ORM\Cache\ConcurrentRegion" is required, The default implementation provided by doctrine is "Doctrine\ORM\Cache\Region\FileLockRegion" if you what to use it please provide a valid directory */ public function testInvalidFileLockRegionDirectoryException() {