ILIAS  trunk Revision v11.0_alpha-1702-gfd3ecb7f852
All Data Structures Namespaces Files Functions Variables Enumerations Enumerator Modules Pages
ActiveContainer.php
Go to the documentation of this file.
1 <?php
2 
19 declare(strict_types=1);
20 
22 
28 
32 final class ActiveContainer implements Container
33 {
37  private const LOCK_UNTIL = '_lock_until';
41  private const GLUE = '|||||';
45  private const STRING_PREFIX = 'string';
49  private const ARRAY_PREFIX = 'array';
53  private const INT_PREFIX = 'int';
57  private const BOOL_PREFIX = 'bool';
61  private const NULL_PREFIX = 'null';
65  private const TRUE = 'true';
69  private const FALSE = 'false';
70 
72  private $null_trafo;
73  private string $prefix_pattern;
74 
75  public function __construct(
76  private Request $request,
77  private Adaptor $adaptor,
78  private Config $config
79  ) {
80  $this->data_factory = new Factory();
81  // see comment in buildFinalTransformation why this is not a good solution.
82  $this->null_trafo = new \ILIAS\Refinery\Custom\Transformation(fn($value): null => null);
83  $this->prefix_pattern = '(' . implode('|', [
84  preg_quote(self::STRING_PREFIX, '/'),
85  preg_quote(self::ARRAY_PREFIX, '/'),
86  preg_quote(self::INT_PREFIX, '/'),
87  preg_quote(self::BOOL_PREFIX, '/'),
88  preg_quote(self::NULL_PREFIX, '/')
89  ]) . ')';
90  }
91 
92  private function pack(mixed $value): string
93  {
94  if (is_string($value)) {
95  return self::STRING_PREFIX . self::GLUE . $value;
96  }
97  if (is_array($value)) {
98  $value = $this->packRecursive($value);
99 
100  return self::ARRAY_PREFIX . self::GLUE . json_encode($value, JSON_THROW_ON_ERROR);
101  }
102  if (is_int($value)) {
103  return self::INT_PREFIX . self::GLUE . $value;
104  }
105  if (is_bool($value)) {
106  return self::BOOL_PREFIX . self::GLUE . ($value ? self::TRUE : self::FALSE);
107  }
108  if (is_null($value)) {
109  return self::NULL_PREFIX . self::GLUE;
110  }
111 
112  throw new \InvalidArgumentException(
113  'Only strings, integers and arrays containing those values are allowed, ' . gettype($value) . ' given.'
114  );
115  }
116 
117  private function packRecursive(array $value): array
118  {
119  array_walk($value, function (&$item): void {
120  $item = is_array($item) ? $this->packRecursive($item) : $this->pack($item);
121  });
122  return $value;
123  }
124 
125  private function unprefix(string $value): array
126  {
127  $str = '/^' . $this->prefix_pattern . preg_quote(self::GLUE, '/') . '(.*)/is';
128  if (!preg_match($str, $value, $matches)) {
129  return [self::NULL_PREFIX, null];
130  }
131 
132  return [$matches[1], $matches[2]];
133  }
134 
135  private function unpack(?string $value): string|int|array|bool|null
136  {
137  // simple detection
138  if ($value === null) {
139  return null;
140  }
141  if ($value === self::NULL_PREFIX . self::GLUE) {
142  return null;
143  }
144 
145  // type detection
146  [$type, $unprefixed_value] = $this->unprefix($value);
147 
148  switch ($type) {
149  case self::STRING_PREFIX:
150  return $unprefixed_value;
151  case self::BOOL_PREFIX:
152  return $unprefixed_value === self::TRUE;
153  case self::ARRAY_PREFIX:
154  $unprefixed_value = json_decode((string) $unprefixed_value, true, 512);
155  if (!is_array($unprefixed_value)) {
156  return null;
157  }
158 
159  return $this->unpackRecursive($unprefixed_value);
160  case self::INT_PREFIX:
161  return (int) $unprefixed_value;
162  default:
163  return null;
164  }
165 
166  return null;
167  }
168 
169  private function unpackRecursive(array $value): array
170  {
171  array_walk($value, function (&$item): void {
172  $item = is_array($item) ? $this->unpackRecursive($item) : $this->unpack($item);
173  });
174  return $value;
175  }
176 
177  protected function buildFinalTransformation(Transformation $transformation): ByTrying
178  {
179  // This is a workaround for the fact that the ByTrying transformation cannot be created by
180  // $DIC->refinery()->byTrying() since we are in a hell of dependencies. E.g. we cant instantiate the
181  // caching service with $DIC->refinery() since the Refinery needs ilLanguage, but ilLanguage
182  // needs the caching service...
183  return new ByTrying([$transformation, $this->getNullFallback()], $this->data_factory);
184  }
185 
186  protected function getNullFallback(): Transformation
187  {
188  return $this->null_trafo;
189  }
190 
191  public function isLocked(): bool
192  {
193  // see comment in buildFinalTransformation why this is not a good solution.
194  $lock_until = $this->adaptor->get($this->request->getContainerKey(), self::LOCK_UNTIL);
195  $lock_until = $lock_until === null ? null : (float) $lock_until;
196 
197  return $lock_until !== null && $lock_until > microtime(true);
198  }
199 
200  public function lock(float $seconds): void
201  {
202  if ($seconds > 300.0 || $seconds < 0.0) {
203  throw new \InvalidArgumentException('Locking for more than 5 minutes is not allowed.');
204  }
205  $lock_until = (string) (microtime(true) + $seconds);
206  $this->adaptor->set($this->request->getContainerKey(), self::LOCK_UNTIL, $lock_until, 300);
207  }
208 
209  public function has(string $key): bool
210  {
211  if ($this->isLocked()) {
212  return false;
213  }
214 
215  return $this->adaptor->has($this->request->getContainerKey(), $key);
216  }
217 
218  public function get(string $key, Transformation $transformation): string|int|array|bool|null
219  {
220  if ($this->isLocked()) {
221  return null;
222  }
223  $unpacked_values = $this->unpack(
224  $this->adaptor->get($this->request->getContainerKey(), $key)
225  );
226 
227  return $this->buildFinalTransformation($transformation)->transform($unpacked_values);
228  }
229 
230  public function set(string $key, string|int|array|bool|null $value, ?int $ttl = null): void
231  {
232  if ($this->isLocked()) {
233  return;
234  }
235  $ttl = $ttl ?? $this->config->getDefaultTTL();
236 
237  $this->adaptor->set(
238  $this->request->getContainerKey(),
239  $key,
240  $this->pack($value),
241  $ttl
242  );
243  }
244 
245  public function delete(string $key): void
246  {
247  if ($this->isLocked()) {
248  return;
249  }
250  $this->adaptor->delete($this->request->getContainerKey(), $key);
251  }
252 
253  public function flush(): void
254  {
255  $this->adaptor->flushContainer($this->request->getContainerKey());
256  }
257 
258  public function getAdaptorName(): string
259  {
260  return $this->config->getAdaptorName();
261  }
262 
263  public function getContainerName(): string
264  {
265  return $this->request->getContainerKey();
266  }
267 }
has(string $key)
Returns true if the container contains a value for the given key.
flush()
Deletes all values in the container.
__construct(private Request $request, private Adaptor $adaptor, private Config $config)
while($session_entry=$r->fetchRow(ilDBConstants::FETCHMODE_ASSOC)) return null
lock(float $seconds)
Locks the container for a given amount of seconds (max 300), in this time, get() will return null and...
isLocked()
Returns true if the container is locked.
getAdaptorName()
Returns the name of the adaptop used (such as apc, memcache, phpstatic)
This file is part of ILIAS, a powerful learning management system published by ILIAS open source e-Le...
buildFinalTransformation(Transformation $transformation)
Builds data types.
Definition: Factory.php:35
A transformation is a function from one datatype to another.
getContainerName()
Returns the name of the container.