| 1 | <?php |
||
| 23 | class I18nCacheUtils |
||
| 24 | { |
||
| 25 | |||
| 26 | /** |
||
| 27 | * doctrine cache |
||
| 28 | * |
||
| 29 | * @var CacheProvider |
||
| 30 | */ |
||
| 31 | private $cache; |
||
| 32 | |||
| 33 | /** |
||
| 34 | * full path to translations cache |
||
| 35 | * |
||
| 36 | * @var string |
||
| 37 | */ |
||
| 38 | private $cacheDirTranslations; |
||
| 39 | |||
| 40 | /** |
||
| 41 | * the cache key we use for our addition array |
||
| 42 | * |
||
| 43 | * @var string |
||
| 44 | */ |
||
| 45 | private $cacheKey; |
||
| 46 | |||
| 47 | /** |
||
| 48 | * the cache key we use for the full resource map |
||
| 49 | * |
||
| 50 | * @var string |
||
| 51 | */ |
||
| 52 | private $cacheKeyFinalResource; |
||
| 53 | |||
| 54 | /** |
||
| 55 | * the loader id suffix (like 'odm') |
||
| 56 | * |
||
| 57 | * @var string |
||
| 58 | */ |
||
| 59 | private $loaderId; |
||
| 60 | |||
| 61 | /** |
||
| 62 | * full path to the bundle resource dir |
||
| 63 | * |
||
| 64 | * @var string |
||
| 65 | */ |
||
| 66 | private $resourceDir; |
||
| 67 | |||
| 68 | /** |
||
| 69 | * this map contains all added resources (added to all translator knows already) |
||
| 70 | * |
||
| 71 | * @var array |
||
| 72 | */ |
||
| 73 | private $addedResources = array(); |
||
| 74 | |||
| 75 | /** |
||
| 76 | * if the postpersistListener invalidates something, it will be put here |
||
| 77 | * |
||
| 78 | * @var array |
||
| 79 | */ |
||
| 80 | private $invalidations = array(); |
||
| 81 | |||
| 82 | /** |
||
| 83 | * a boolean flag telling us if a new map persist is necessary |
||
| 84 | * (that is, when some new addition has been added that needs a resource map regeneration) |
||
| 85 | * |
||
| 86 | * @var boolean |
||
| 87 | */ |
||
| 88 | private $isDirty = false; |
||
| 89 | |||
| 90 | /** |
||
| 91 | * Constructor |
||
| 92 | * |
||
| 93 | * @param CacheProvider $cache cache |
||
| 94 | * @param string $cacheDir full path to cache dir |
||
| 95 | * @param string $loaderId loader id suffix |
||
| 96 | */ |
||
| 97 | 188 | public function __construct( |
|
| 98 | CacheProvider $cache, |
||
| 99 | $cacheDir, |
||
| 100 | $loaderId |
||
| 101 | ) { |
||
| 102 | 188 | $this->cache = $cache; |
|
| 103 | 188 | $this->cacheDirTranslations = $cacheDir . '/translations'; |
|
| 104 | |||
| 105 | // cache keys |
||
| 106 | 188 | $this->cacheKey = 'i18n.addedTranslations'; |
|
| 107 | 188 | $this->cacheKeyFinalResource = 'i18n.finalResources'; |
|
| 108 | |||
| 109 | 188 | $this->loaderId = $loaderId; |
|
| 110 | 188 | $this->resourceDir = __DIR__.'/../Resources/translations/'; |
|
| 111 | |||
| 112 | // do we have existing resources? |
||
| 113 | 188 | if ($this->cache->contains($this->cacheKey)) { |
|
| 114 | 185 | $this->addedResources = $this->cache->fetch($this->cacheKey); |
|
|
|
|||
| 115 | 185 | } |
|
| 116 | 188 | } |
|
| 117 | |||
| 118 | /** |
||
| 119 | * Gets the cache instance |
||
| 120 | * |
||
| 121 | * @return CacheProvider cache |
||
| 122 | */ |
||
| 123 | public function getCache() |
||
| 124 | { |
||
| 125 | return $this->cache; |
||
| 126 | } |
||
| 127 | |||
| 128 | /** |
||
| 129 | * sets the resource dir |
||
| 130 | * |
||
| 131 | * @param string $dir resource dir |
||
| 132 | * |
||
| 133 | * @return void |
||
| 134 | */ |
||
| 135 | 6 | public function setResourceDir($dir) |
|
| 136 | { |
||
| 137 | 6 | $this->resourceDir = $dir; |
|
| 138 | } |
||
| 139 | |||
| 140 | /** |
||
| 141 | * this shall be called by a Translator. |
||
| 142 | * it adds our additions to the already existent ones in the Translator and returns it. |
||
| 143 | * as this is called quite often, we cache the final result (the full map including the translator resources) |
||
| 144 | * and only regenerate that when a *new* domain has been added. basic invalidations by the PostPersistListener |
||
| 145 | * will *not* result in a rebuild here - only if a new domain has been added. |
||
| 146 | * |
||
| 147 | * @param array $resources the resources array of the translator |
||
| 148 | * |
||
| 149 | * @return array the finalized map containing translator resources and our additions |
||
| 150 | */ |
||
| 151 | 186 | public function getResources($resources) |
|
| 152 | { |
||
| 153 | 186 | $finalResources = null; |
|
| 154 | |||
| 155 | // do we have a full resource map in the cache already? (full = translator + additions) |
||
| 156 | 186 | if ($this->cache->contains($this->cacheKeyFinalResource)) { |
|
| 157 | 183 | $finalResources = $this->cache->fetch($this->cacheKeyFinalResource); |
|
| 158 | 183 | } |
|
| 159 | |||
| 160 | // do we have cached additions? |
||
| 161 | 186 | if (!is_array($finalResources) && $this->cache->contains($this->cacheKey)) { |
|
| 162 | // merge the two together, always keep an eye to not duplicate (paths are different!) |
||
| 163 | 6 | $finalResources = $this->mergeResourcesWithAdditions($resources); |
|
| 164 | |||
| 165 | // cache it |
||
| 166 | 6 | $this->cache->save($this->cacheKeyFinalResource, $finalResources); |
|
| 167 | 6 | } |
|
| 168 | |||
| 169 | // so, did we got anything? |
||
| 170 | 186 | if (is_array($finalResources)) { |
|
| 171 | 183 | $resources = $finalResources; |
|
| 172 | 183 | } |
|
| 173 | |||
| 174 | 186 | return $resources; |
|
| 175 | } |
||
| 176 | |||
| 177 | /** |
||
| 178 | * will be executed on the event dispatched by PostPersistTranslatableListener. |
||
| 179 | * if someone invalidates a locale & domain pair, this will lead to: |
||
| 180 | * - removal of the symfony translation cache files |
||
| 181 | * (if this pair has never been seen) |
||
| 182 | * - creation of the resource files ("trigger files") |
||
| 183 | * - a regeneration of the full resource map for the translator |
||
| 184 | * |
||
| 185 | * please note that calling invalidate() will do the above mentioned in a lazy way |
||
| 186 | * when the kernel.terminate event fires. |
||
| 187 | * |
||
| 188 | * @param string $locale locale (de,en,fr) |
||
| 189 | * @param string $domain domain |
||
| 190 | * |
||
| 191 | * @return void |
||
| 192 | */ |
||
| 193 | 61 | public function invalidate($locale, $domain) |
|
| 204 | |||
| 205 | /** |
||
| 206 | * Merges the cached additions with the one the Translator already has. |
||
| 207 | * I need to use preg_grep() here as I'm unable to compose an absolute path that is |
||
| 208 | * identical to the one the Translator would have already as I have to deal with relative |
||
| 209 | * paths here (and rightfully so). This shouldn't hurt too much as the end result is cached |
||
| 210 | * and only redone if something changes. |
||
| 211 | * |
||
| 212 | * @param array $resources resources |
||
| 213 | * |
||
| 214 | * @return array finalized full map |
||
| 215 | */ |
||
| 216 | 6 | private function mergeResourcesWithAdditions($resources) |
|
| 217 | { |
||
| 218 | 6 | foreach ($this->addedResources as $locale => $files) { |
|
| 219 | 6 | foreach ($files as $file) { |
|
| 220 | 6 | $isExistent = false; |
|
| 221 | 6 | if (isset($resources[$locale])) { |
|
| 222 | 6 | $hits = preg_grep('/\/'.str_replace('.', '\\.', $file).'$/', $resources[$locale]); |
|
| 223 | 6 | if (count($hits) > 0) { |
|
| 224 | 2 | $isExistent = true; |
|
| 225 | 2 | } |
|
| 226 | 6 | } |
|
| 227 | |||
| 228 | 6 | if (!$isExistent) { |
|
| 229 | 6 | $resourceFile = $this->resourceDir.$file; |
|
| 230 | |||
| 231 | // make sure the file exists |
||
| 232 | 6 | $fs = new Filesystem(); |
|
| 233 | 6 | $fs->touch($resourceFile); |
|
| 234 | |||
| 235 | 6 | $resources[$locale][] = $resourceFile; |
|
| 236 | 6 | } |
|
| 237 | 6 | } |
|
| 238 | 6 | } |
|
| 239 | 6 | return $resources; |
|
| 240 | } |
||
| 241 | |||
| 242 | /** |
||
| 243 | * saves our addition array to our cache and removes the full map from the cache |
||
| 244 | * leading to a regeneration of the map. |
||
| 245 | * |
||
| 246 | * @return void |
||
| 247 | */ |
||
| 248 | 6 | private function persistAdditions() |
|
| 255 | |||
| 256 | /** |
||
| 257 | * processes all queued cache invalidations for the symfony translation cache. |
||
| 258 | * this is now only 1 Finder search for a single request. |
||
| 259 | * |
||
| 260 | * @return void |
||
| 261 | */ |
||
| 262 | 186 | private function processInvalidations() |
|
| 263 | { |
||
| 264 | 186 | if (empty($this->invalidations)) { |
|
| 265 | 182 | return; |
|
| 266 | } |
||
| 267 | |||
| 268 | 19 | $fs = new Filesystem(); |
|
| 269 | 19 | $localesToClean = array_keys($this->invalidations); |
|
| 270 | 19 | $deleteRegex = '/^catalogue\.(['.implode('|', $localesToClean).'])/'; |
|
| 271 | |||
| 272 | try { |
||
| 273 | 19 | $finder = new Finder(); |
|
| 274 | $finder |
||
| 275 | 19 | ->files() |
|
| 276 | 19 | ->in($this->cacheDirTranslations) |
|
| 277 | 19 | ->name($deleteRegex); |
|
| 278 | |||
| 279 | 19 | foreach ($finder as $file) { |
|
| 280 | 19 | $fs->remove($file->getRealPath()); |
|
| 281 | 19 | } |
|
| 282 | 19 | } catch (\InvalidArgumentException $e) { |
|
|
1 ignored issue
–
show
|
|||
| 283 | // happens when cache is non-existent |
||
| 284 | } |
||
| 285 | 19 | } |
|
| 286 | |||
| 287 | /** |
||
| 288 | * processes all pending operations |
||
| 289 | * |
||
| 290 | * @return void |
||
| 291 | */ |
||
| 292 | 186 | public function processPending() |
|
| 300 | } |
||
| 301 |
Our type inference engine has found an assignment to a property that is incompatible with the declared type of that property.
Either this assignment is in error or the assigned type should be added to the documentation/type hint for that property..