This repository has been archived on 2023-08-20. You can view files and clone it, but cannot push or open issues or pull requests.
symfony/src/Symfony/Component/PropertyInfo/PropertyInfoCacheExtractor.php

153 lines
3.8 KiB
PHP
Raw Normal View History

2015-12-09 07:14:29 +00:00
<?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\PropertyInfo;
use Psr\Cache\CacheItemPoolInterface;
/**
* Adds a PSR-6 cache layer on top of an extractor.
*
* @author Kévin Dunglas <dunglas@gmail.com>
*/
class PropertyInfoCacheExtractor implements PropertyInfoExtractorInterface
{
/**
* @var PropertyInfoExtractorInterface
*/
private $propertyInfoExtractor;
/**
* @var CacheItemPoolInterface
*/
private $cacheItemPool;
/**
* @var array
*/
private $arrayCache = array();
public function __construct(PropertyInfoExtractorInterface $propertyInfoExtractor, CacheItemPoolInterface $cacheItemPool)
{
$this->propertyInfoExtractor = $propertyInfoExtractor;
$this->cacheItemPool = $cacheItemPool;
}
/**
* {@inheritdoc}
*/
public function isReadable($class, $property, array $context = array())
{
return $this->extract('isReadable', array($class, $property, $context));
}
/**
* {@inheritdoc}
*/
public function isWritable($class, $property, array $context = array())
{
return $this->extract('isWritable', array($class, $property, $context));
}
/**
* {@inheritdoc}
*/
public function getShortDescription($class, $property, array $context = array())
{
return $this->extract('getShortDescription', array($class, $property, $context));
}
/**
* {@inheritdoc}
*/
public function getLongDescription($class, $property, array $context = array())
{
return $this->extract('getLongDescription', array($class, $property, $context));
}
/**
* {@inheritdoc}
*/
public function getProperties($class, array $context = array())
{
return $this->extract('getProperties', array($class, $context));
}
/**
* {@inheritdoc}
*/
public function getTypes($class, $property, array $context = array())
{
return $this->extract('getTypes', array($class, $context));
}
/**
* Retrieves the cached data if applicable or delegates to the decorated extractor.
*
* @param string $method
* @param array $arguments
*
* @return mixed
*/
private function extract($method, array $arguments)
{
try {
$serializedArguments = serialize($arguments);
} catch (\Exception $exception) {
// If arguments are not serializable, skip the cache
return call_user_func_array(array($this->propertyInfoExtractor, $method), $arguments);
}
$key = $this->escape($method.'.'.$serializedArguments);
if (isset($this->arrayCache[$key])) {
return $this->arrayCache[$key];
}
$item = $this->cacheItemPool->getItem($key);
if ($item->isHit()) {
return $this->arrayCache[$key] = $item->get();
}
$value = call_user_func_array(array($this->propertyInfoExtractor, $method), $arguments);
$item->set($value);
$this->cacheItemPool->save($item);
return $this->arrayCache[$key] = $value;
}
/**
* Escapes a key according to PSR-6.
*
* Replaces characters forbidden by PSR-6 and the _ char by the _ char followed by the ASCII
* code of the escaped char.
*
* @param string $key
*
* @return string
*/
private function escape($key)
{
return strtr($key, array(
'{' => '_123',
'}' => '_125',
'(' => '_40',
')' => '_41',
'/' => '_47',
'\\' => '_92',
'@' => '_64',
':' => '_58',
'_' => '_95',
));
}
}