Duplicate code is one of the most pungent code smells. A rule that is often used is to re-structure code once it is duplicated in three or more places.
Common duplication problems, and corresponding solutions are:
| 1 | <?php | ||
| 18 | abstract class AbstractInMemoryHandler | ||
| 19 | { | ||
| 20 | /** | ||
| 21 | * NOTE: Instance of this must be InMemoryClearingProxyAdapter in order for cache clearing to affect in-memory cache. | ||
| 22 | * | ||
| 23 | * @var \eZ\Publish\Core\Persistence\Cache\Adapter\InMemoryClearingProxyAdapter | ||
| 24 | */ | ||
| 25 | protected $cache; | ||
| 26 | |||
| 27 | /** | ||
| 28 | * @var \eZ\Publish\SPI\Persistence\Handler | ||
| 29 | */ | ||
| 30 | protected $persistenceHandler; | ||
| 31 | |||
| 32 | /** | ||
| 33 | * @var \eZ\Publish\Core\Persistence\Cache\PersistenceLogger | ||
| 34 | */ | ||
| 35 | protected $logger; | ||
| 36 | |||
| 37 | /** | ||
| 38 | * @var \eZ\Publish\Core\Persistence\Cache\InMemory\InMemoryCache | ||
| 39 | */ | ||
| 40 | private $inMemory; | ||
| 41 | |||
| 42 | /** | ||
| 43 | * Setups current handler with everything needed. | ||
| 44 | * | ||
| 45 | * @param \eZ\Publish\SPI\Persistence\Handler $persistenceHandler | ||
| 46 | * @param \eZ\Publish\Core\Persistence\Cache\Adapter\InMemoryClearingProxyAdapter $cache | ||
| 47 | * @param \eZ\Publish\Core\Persistence\Cache\PersistenceLogger $logger | ||
| 48 | * @param \eZ\Publish\Core\Persistence\Cache\InMemory\InMemoryCache $inMemory | ||
| 49 | */ | ||
| 50 | public function __construct( | ||
| 60 | |||
| 61 | /** | ||
| 62 | * @api May be used from handlers that needs to inject something else then PersistenceHandler to __construct(), to | ||
| 63 | * avoid having to have intimate knowledge of internal classes used by AbstractInMemoryHandler in signature. | ||
| 64 | * | ||
| 65 | * E.g: | ||
| 66 |      *    public function __construct(MyHandler $myHandler, ...$params) { | ||
| 67 | * $this->myHandler = $myHandler; | ||
| 68 | * $this->setCacheDependencies(...$params); | ||
| 69 | * } | ||
| 70 | * | ||
| 71 | * @param \eZ\Publish\Core\Persistence\Cache\Adapter\InMemoryClearingProxyAdapter $cache | ||
| 72 | * @param \eZ\Publish\Core\Persistence\Cache\PersistenceLogger $logger | ||
| 73 | * @param \eZ\Publish\Core\Persistence\Cache\InMemory\InMemoryCache $inMemory | ||
| 74 | */ | ||
| 75 | protected function setCacheDependencies( | ||
| 84 | |||
| 85 | /** | ||
| 86 | * Optional function to initialize handler without having to overload __construct(). | ||
| 87 | */ | ||
| 88 | protected function init(): void | ||
| 92 | |||
| 93 | /** | ||
| 94 | * Load one cache item from cache and loggs the hits / misses. | ||
| 95 | * | ||
| 96 | * Load items from in-memory cache, symfony cache pool or backend in that order. | ||
| 97 | * If not cached the returned objects will be placed in cache. | ||
| 98 | * | ||
| 99 | * @param int|string $id | ||
| 100 | * @param string $keyPrefix E.g "ez-content-" | ||
| 101 | * @param callable $backendLoader Function for loading missing objects, gets array with missing id's as argument, | ||
| 102 | * expects return value to be array with id as key. Missing items should be missing. | ||
| 103 | * @param callable $cacheTagger Gets cache object as argument, return array of cache tags. | ||
| 104 | * @param callable $cacheIndexes Gets cache object as argument, return array of cache keys. | ||
| 105 | * @param string $keySuffix Optional, e.g "-by-identifier" | ||
| 106 | * | ||
| 107 | * @return object | ||
| 108 | */ | ||
| 109 | final protected function getCacheValue( | ||
| 146 | |||
| 147 | /** | ||
| 148 | * Load list of objects of some type and loggs the hits / misses. | ||
| 149 | * | ||
| 150 | * Load items from in-memory cache, symfony cache pool or backend in that order. | ||
| 151 | * If not cached the returned objects will be placed in cache. | ||
| 152 | * | ||
| 153 | * @param string $key | ||
| 154 | * @param callable $backendLoader Function for loading ALL objects, value is cached as-is. | ||
| 155 | * @param callable $cacheTagger Gets cache object as argument, return array of cache tags. | ||
| 156 | * @param callable $cacheIndexes Gets cache object as argument, return array of cache keys. | ||
| 157 | * @param callable $listTags Optional, global tags for the list cache. | ||
| 158 | * @param array $arguments Optional, arguments when parnt method takes arguments that key varies on. | ||
| 159 | * | ||
| 160 | * @return array | ||
| 161 | */ | ||
| 162 | final protected function getListCacheValue( | ||
| 209 | |||
| 210 | /** | ||
| 211 | * Load several cache items from cache and loggs the hits / misses. | ||
| 212 | * | ||
| 213 | * Load items from in-memory cache, symfony cache pool or backend in that order. | ||
| 214 | * If not cached the returned objects will be placed in cache. | ||
| 215 | * | ||
| 216 |      * Cache items must be stored with a key in the following format "${keyPrefix}${id}", like "ez-content-info-${id}", | ||
| 217 | * in order for this method to be able to prefix key on id's and also extract key prefix afterwards. | ||
| 218 | * | ||
| 219 | * @param array $ids | ||
| 220 | * @param string $keyPrefix E.g "ez-content-" | ||
| 221 | * @param callable $backendLoader Function for loading missing objects, gets array with missing id's as argument, | ||
| 222 | * expects return value to be array with id as key. Missing items should be missing. | ||
| 223 | * @param callable $cacheTagger Gets cache object as argument, return array of cache tags. | ||
| 224 | * @param callable $cacheIndexes Gets cache object as argument, return array of cache keys. | ||
| 225 | * @param string $keySuffix Optional, e.g "-by-identifier" | ||
| 226 | * | ||
| 227 | * @return array | ||
| 228 | */ | ||
| 229 | final protected function getMultipleCacheValues( | ||
| 307 | } | ||
| 308 | 
Duplicated code is one of the most pungent code smells. If you need to duplicate the same code in three or more different places, we strongly encourage you to look into extracting the code into a single class or operation.
You can also find more detailed suggestions in the “Code” section of your repository.