1 | <?php |
||
2 | |||
3 | declare(strict_types=1); |
||
4 | |||
5 | namespace PhpMyAdmin\MoTranslator\Cache; |
||
6 | |||
7 | use PhpMyAdmin\MoTranslator\CacheException; |
||
8 | use PhpMyAdmin\MoTranslator\MoParser; |
||
9 | |||
10 | use function apcu_enabled; |
||
11 | use function apcu_entry; |
||
12 | use function apcu_exists; |
||
13 | use function apcu_fetch; |
||
14 | use function apcu_store; |
||
15 | use function array_combine; |
||
16 | use function array_keys; |
||
17 | use function array_map; |
||
18 | use function assert; |
||
19 | use function function_exists; |
||
20 | use function is_array; |
||
21 | use function is_string; |
||
22 | |||
23 | final class ApcuCache implements CacheInterface |
||
24 | { |
||
25 | public const LOADED_KEY = '__TRANSLATIONS_LOADED__'; |
||
26 | |||
27 | public function __construct( |
||
28 | private MoParser $parser, |
||
29 | private string $locale, |
||
30 | private string $domain, |
||
31 | private int $ttl = 0, |
||
32 | private bool $reloadOnMiss = true, |
||
33 | private string $prefix = 'mo_', |
||
34 | ) { |
||
35 | if (! (function_exists('apcu_enabled') && apcu_enabled())) { |
||
36 | throw new CacheException('ACPu extension must be installed and enabled'); |
||
37 | } |
||
38 | |||
39 | $this->ensureTranslationsLoaded(); |
||
40 | 119 | } |
|
41 | |||
42 | public function get(string $msgid): string |
||
43 | { |
||
44 | $msgstr = apcu_fetch($this->getKey($msgid), $success); |
||
45 | if ($success && is_string($msgstr)) { |
||
46 | return $msgstr; |
||
47 | } |
||
48 | 119 | ||
49 | 7 | if (! $this->reloadOnMiss) { |
|
50 | return $msgid; |
||
51 | } |
||
52 | 112 | ||
53 | 112 | return $this->reloadOnMiss($msgid); |
|
54 | 112 | } |
|
55 | 112 | ||
56 | 112 | private function reloadOnMiss(string $msgid): string |
|
57 | 112 | { |
|
58 | // store original if translation is not present |
||
59 | 112 | $cached = apcu_entry($this->getKey($msgid), static function () use ($msgid) { |
|
60 | 16 | return $msgid; |
|
61 | }, $this->ttl); |
||
62 | 56 | // if another process has updated cache, return early |
|
63 | if ($cached !== $msgid && is_string($cached)) { |
||
64 | 56 | return $cached; |
|
65 | 56 | } |
|
66 | 35 | ||
67 | // reload .mo file, in case entry has been evicted |
||
68 | $this->parser->parseIntoCache($this); |
||
69 | 21 | ||
70 | 7 | $msgstr = apcu_fetch($this->getKey($msgid), $success); |
|
71 | |||
72 | return $success && is_string($msgstr) ? $msgstr : $msgid; |
||
73 | 14 | } |
|
74 | |||
75 | public function set(string $msgid, string $msgstr): void |
||
76 | 21 | { |
|
77 | apcu_store($this->getKey($msgid), $msgstr, $this->ttl); |
||
78 | } |
||
79 | 18 | ||
80 | 14 | public function has(string $msgid): bool |
|
81 | 21 | { |
|
82 | return apcu_exists($this->getKey($msgid)); |
||
0 ignored issues
–
show
Bug
Best Practice
introduced
by
![]() |
|||
83 | 21 | } |
|
84 | 7 | ||
85 | /** @inheritDoc */ |
||
86 | public function setAll(array $translations): void |
||
87 | { |
||
88 | 14 | $keys = array_map(function (string $msgid): string { |
|
89 | return $this->getKey($msgid); |
||
90 | 14 | }, array_keys($translations)); |
|
91 | $translations = array_combine($keys, $translations); |
||
92 | 14 | assert(is_array($translations)); |
|
93 | |||
94 | apcu_store($translations, null, $this->ttl); |
||
95 | 70 | } |
|
96 | |||
97 | 70 | private function getKey(string $msgid): string |
|
98 | 10 | { |
|
99 | return $this->prefix . $this->locale . '.' . $this->domain . '.' . $msgid; |
||
100 | 14 | } |
|
101 | |||
102 | 14 | private function ensureTranslationsLoaded(): void |
|
103 | { |
||
104 | // Try to prevent cache slam if multiple processes are trying to load translations. There is still a race |
||
105 | 7 | // between the exists check and creating the entry, but at least it's small |
|
106 | $key = $this->getKey(self::LOADED_KEY); |
||
107 | 6 | $loaded = apcu_exists($key) || apcu_entry($key, static function (): int { |
|
108 | 7 | return 0; |
|
109 | 7 | }, $this->ttl); |
|
110 | 7 | if ($loaded) { |
|
111 | return; |
||
112 | } |
||
113 | 7 | ||
114 | 1 | $this->parser->parseIntoCache($this); |
|
115 | apcu_store($this->getKey(self::LOADED_KEY), 1, $this->ttl); |
||
116 | 112 | } |
|
117 | } |
||
118 |