123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330 |
- <?php
- /*
- * This file is part of the Symfony package.
- *
- * (c) Fabien Potencier <fabien@symfony.com>
- *
- * For the full copyright and license information, please view the LICENSE
- * file that was distributed with this source code.
- */
- namespace Symfony\Component\Cache\Adapter;
- use Symfony\Component\Cache\Exception\CacheException;
- use Symfony\Component\Cache\Exception\InvalidArgumentException;
- use Symfony\Component\Cache\PruneableInterface;
- use Symfony\Component\Cache\Traits\FilesystemCommonTrait;
- use Symfony\Component\VarExporter\VarExporter;
- /**
- * @author Piotr Stankowski <git@trakos.pl>
- * @author Nicolas Grekas <p@tchwork.com>
- * @author Rob Frawley 2nd <rmf@src.run>
- */
- class PhpFilesAdapter extends AbstractAdapter implements PruneableInterface
- {
- use FilesystemCommonTrait {
- doClear as private doCommonClear;
- doDelete as private doCommonDelete;
- }
- private $includeHandler;
- private $appendOnly;
- private $values = [];
- private $files = [];
- private static $startTime;
- private static $valuesCache = [];
- /**
- * @param $appendOnly Set to `true` to gain extra performance when the items stored in this pool never expire.
- * Doing so is encouraged because it fits perfectly OPcache's memory model.
- *
- * @throws CacheException if OPcache is not enabled
- */
- public function __construct(string $namespace = '', int $defaultLifetime = 0, ?string $directory = null, bool $appendOnly = false)
- {
- $this->appendOnly = $appendOnly;
- self::$startTime = self::$startTime ?? $_SERVER['REQUEST_TIME'] ?? time();
- parent::__construct('', $defaultLifetime);
- $this->init($namespace, $directory);
- $this->includeHandler = static function ($type, $msg, $file, $line) {
- throw new \ErrorException($msg, 0, $type, $file, $line);
- };
- }
- public static function isSupported()
- {
- self::$startTime = self::$startTime ?? $_SERVER['REQUEST_TIME'] ?? time();
- return \function_exists('opcache_invalidate') && filter_var(\ini_get('opcache.enable'), \FILTER_VALIDATE_BOOLEAN) && (!\in_array(\PHP_SAPI, ['cli', 'phpdbg'], true) || filter_var(\ini_get('opcache.enable_cli'), \FILTER_VALIDATE_BOOLEAN));
- }
- /**
- * @return bool
- */
- public function prune()
- {
- $time = time();
- $pruned = true;
- $getExpiry = true;
- set_error_handler($this->includeHandler);
- try {
- foreach ($this->scanHashDir($this->directory) as $file) {
- try {
- if (\is_array($expiresAt = include $file)) {
- $expiresAt = $expiresAt[0];
- }
- } catch (\ErrorException $e) {
- $expiresAt = $time;
- }
- if ($time >= $expiresAt) {
- $pruned = ($this->doUnlink($file) || !file_exists($file)) && $pruned;
- }
- }
- } finally {
- restore_error_handler();
- }
- return $pruned;
- }
- /**
- * {@inheritdoc}
- */
- protected function doFetch(array $ids)
- {
- if ($this->appendOnly) {
- $now = 0;
- $missingIds = [];
- } else {
- $now = time();
- $missingIds = $ids;
- $ids = [];
- }
- $values = [];
- begin:
- $getExpiry = false;
- foreach ($ids as $id) {
- if (null === $value = $this->values[$id] ?? null) {
- $missingIds[] = $id;
- } elseif ('N;' === $value) {
- $values[$id] = null;
- } elseif (!\is_object($value)) {
- $values[$id] = $value;
- } elseif (!$value instanceof LazyValue) {
- $values[$id] = $value();
- } elseif (false === $values[$id] = include $value->file) {
- unset($values[$id], $this->values[$id]);
- $missingIds[] = $id;
- }
- if (!$this->appendOnly) {
- unset($this->values[$id]);
- }
- }
- if (!$missingIds) {
- return $values;
- }
- set_error_handler($this->includeHandler);
- try {
- $getExpiry = true;
- foreach ($missingIds as $k => $id) {
- try {
- $file = $this->files[$id] ?? $this->files[$id] = $this->getFile($id);
- if (isset(self::$valuesCache[$file])) {
- [$expiresAt, $this->values[$id]] = self::$valuesCache[$file];
- } elseif (\is_array($expiresAt = include $file)) {
- if ($this->appendOnly) {
- self::$valuesCache[$file] = $expiresAt;
- }
- [$expiresAt, $this->values[$id]] = $expiresAt;
- } elseif ($now < $expiresAt) {
- $this->values[$id] = new LazyValue($file);
- }
- if ($now >= $expiresAt) {
- unset($this->values[$id], $missingIds[$k], self::$valuesCache[$file]);
- }
- } catch (\ErrorException $e) {
- unset($missingIds[$k]);
- }
- }
- } finally {
- restore_error_handler();
- }
- $ids = $missingIds;
- $missingIds = [];
- goto begin;
- }
- /**
- * {@inheritdoc}
- */
- protected function doHave(string $id)
- {
- if ($this->appendOnly && isset($this->values[$id])) {
- return true;
- }
- set_error_handler($this->includeHandler);
- try {
- $file = $this->files[$id] ?? $this->files[$id] = $this->getFile($id);
- $getExpiry = true;
- if (isset(self::$valuesCache[$file])) {
- [$expiresAt, $value] = self::$valuesCache[$file];
- } elseif (\is_array($expiresAt = include $file)) {
- if ($this->appendOnly) {
- self::$valuesCache[$file] = $expiresAt;
- }
- [$expiresAt, $value] = $expiresAt;
- } elseif ($this->appendOnly) {
- $value = new LazyValue($file);
- }
- } catch (\ErrorException $e) {
- return false;
- } finally {
- restore_error_handler();
- }
- if ($this->appendOnly) {
- $now = 0;
- $this->values[$id] = $value;
- } else {
- $now = time();
- }
- return $now < $expiresAt;
- }
- /**
- * {@inheritdoc}
- */
- protected function doSave(array $values, int $lifetime)
- {
- $ok = true;
- $expiry = $lifetime ? time() + $lifetime : 'PHP_INT_MAX';
- $allowCompile = self::isSupported();
- foreach ($values as $key => $value) {
- unset($this->values[$key]);
- $isStaticValue = true;
- if (null === $value) {
- $value = "'N;'";
- } elseif (\is_object($value) || \is_array($value)) {
- try {
- $value = VarExporter::export($value, $isStaticValue);
- } catch (\Exception $e) {
- throw new InvalidArgumentException(sprintf('Cache key "%s" has non-serializable "%s" value.', $key, get_debug_type($value)), 0, $e);
- }
- } elseif (\is_string($value)) {
- // Wrap "N;" in a closure to not confuse it with an encoded `null`
- if ('N;' === $value) {
- $isStaticValue = false;
- }
- $value = var_export($value, true);
- } elseif (!\is_scalar($value)) {
- throw new InvalidArgumentException(sprintf('Cache key "%s" has non-serializable "%s" value.', $key, get_debug_type($value)));
- } else {
- $value = var_export($value, true);
- }
- $encodedKey = rawurlencode($key);
- if ($isStaticValue) {
- $value = "return [{$expiry}, {$value}];";
- } elseif ($this->appendOnly) {
- $value = "return [{$expiry}, static function () { return {$value}; }];";
- } else {
- // We cannot use a closure here because of https://bugs.php.net/76982
- $value = str_replace('\Symfony\Component\VarExporter\Internal\\', '', $value);
- $value = "namespace Symfony\Component\VarExporter\Internal;\n\nreturn \$getExpiry ? {$expiry} : {$value};";
- }
- $file = $this->files[$key] = $this->getFile($key, true);
- // Since OPcache only compiles files older than the script execution start, set the file's mtime in the past
- $ok = $this->write($file, "<?php //{$encodedKey}\n\n{$value}\n", self::$startTime - 10) && $ok;
- if ($allowCompile) {
- @opcache_invalidate($file, true);
- @opcache_compile_file($file);
- }
- unset(self::$valuesCache[$file]);
- }
- if (!$ok && !is_writable($this->directory)) {
- throw new CacheException(sprintf('Cache directory is not writable (%s).', $this->directory));
- }
- return $ok;
- }
- /**
- * {@inheritdoc}
- */
- protected function doClear(string $namespace)
- {
- $this->values = [];
- return $this->doCommonClear($namespace);
- }
- /**
- * {@inheritdoc}
- */
- protected function doDelete(array $ids)
- {
- foreach ($ids as $id) {
- unset($this->values[$id]);
- }
- return $this->doCommonDelete($ids);
- }
- protected function doUnlink(string $file)
- {
- unset(self::$valuesCache[$file]);
- if (self::isSupported()) {
- @opcache_invalidate($file, true);
- }
- return @unlink($file);
- }
- private function getFileKey(string $file): string
- {
- if (!$h = @fopen($file, 'r')) {
- return '';
- }
- $encodedKey = substr(fgets($h), 8);
- fclose($h);
- return rawurldecode(rtrim($encodedKey));
- }
- }
- /**
- * @internal
- */
- class LazyValue
- {
- public $file;
- public function __construct(string $file)
- {
- $this->file = $file;
- }
- }
|