forked from GNUsocial/gnu-social
		
	
		
			
				
	
	
		
			169 lines
		
	
	
		
			5.1 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
			
		
		
	
	
			169 lines
		
	
	
		
			5.1 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
<?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\Config;
 | 
						|
 | 
						|
use Symfony\Component\Config\Resource\ResourceInterface;
 | 
						|
use Symfony\Component\Filesystem\Exception\IOException;
 | 
						|
use Symfony\Component\Filesystem\Filesystem;
 | 
						|
 | 
						|
/**
 | 
						|
 * ResourceCheckerConfigCache uses instances of ResourceCheckerInterface
 | 
						|
 * to check whether cached data is still fresh.
 | 
						|
 *
 | 
						|
 * @author Matthias Pigulla <mp@webfactory.de>
 | 
						|
 */
 | 
						|
class ResourceCheckerConfigCache implements ConfigCacheInterface
 | 
						|
{
 | 
						|
    /**
 | 
						|
     * @var string
 | 
						|
     */
 | 
						|
    private $file;
 | 
						|
 | 
						|
    /**
 | 
						|
     * @var ResourceCheckerInterface[]
 | 
						|
     */
 | 
						|
    private $resourceCheckers;
 | 
						|
 | 
						|
    /**
 | 
						|
     * @param string                     $file             The absolute cache path
 | 
						|
     * @param ResourceCheckerInterface[] $resourceCheckers The ResourceCheckers to use for the freshness check
 | 
						|
     */
 | 
						|
    public function __construct($file, array $resourceCheckers = array())
 | 
						|
    {
 | 
						|
        $this->file = $file;
 | 
						|
        $this->resourceCheckers = $resourceCheckers;
 | 
						|
    }
 | 
						|
 | 
						|
    /**
 | 
						|
     * {@inheritdoc}
 | 
						|
     */
 | 
						|
    public function getPath()
 | 
						|
    {
 | 
						|
        return $this->file;
 | 
						|
    }
 | 
						|
 | 
						|
    /**
 | 
						|
     * Checks if the cache is still fresh.
 | 
						|
     *
 | 
						|
     * This implementation will make a decision solely based on the ResourceCheckers
 | 
						|
     * passed in the constructor.
 | 
						|
     *
 | 
						|
     * The first ResourceChecker that supports a given resource is considered authoritative.
 | 
						|
     * Resources with no matching ResourceChecker will silently be ignored and considered fresh.
 | 
						|
     *
 | 
						|
     * @return bool true if the cache is fresh, false otherwise
 | 
						|
     */
 | 
						|
    public function isFresh()
 | 
						|
    {
 | 
						|
        if (!is_file($this->file)) {
 | 
						|
            return false;
 | 
						|
        }
 | 
						|
 | 
						|
        if (!$this->resourceCheckers) {
 | 
						|
            return true; // shortcut - if we don't have any checkers we don't need to bother with the meta file at all
 | 
						|
        }
 | 
						|
 | 
						|
        $metadata = $this->getMetaFile();
 | 
						|
        if (!is_file($metadata)) {
 | 
						|
            return false;
 | 
						|
        }
 | 
						|
 | 
						|
        $e = null;
 | 
						|
        $meta = false;
 | 
						|
        $time = filemtime($this->file);
 | 
						|
        $signalingException = new \UnexpectedValueException();
 | 
						|
        $prevUnserializeHandler = ini_set('unserialize_callback_func', '');
 | 
						|
        $prevErrorHandler = set_error_handler(function ($type, $msg, $file, $line, $context) use (&$prevErrorHandler, $signalingException) {
 | 
						|
            if (E_WARNING === $type && 'Class __PHP_Incomplete_Class has no unserializer' === $msg) {
 | 
						|
                throw $signalingException;
 | 
						|
            }
 | 
						|
 | 
						|
            return $prevErrorHandler ? $prevErrorHandler($type, $msg, $file, $line, $context) : false;
 | 
						|
        });
 | 
						|
 | 
						|
        try {
 | 
						|
            $meta = unserialize(file_get_contents($metadata));
 | 
						|
        } catch (\Error $e) {
 | 
						|
        } catch (\Exception $e) {
 | 
						|
        }
 | 
						|
        restore_error_handler();
 | 
						|
        ini_set('unserialize_callback_func', $prevUnserializeHandler);
 | 
						|
        if (null !== $e && $e !== $signalingException) {
 | 
						|
            throw $e;
 | 
						|
        }
 | 
						|
        if (false === $meta) {
 | 
						|
            return false;
 | 
						|
        }
 | 
						|
 | 
						|
        foreach ($meta as $resource) {
 | 
						|
            /* @var ResourceInterface $resource */
 | 
						|
            foreach ($this->resourceCheckers as $checker) {
 | 
						|
                if (!$checker->supports($resource)) {
 | 
						|
                    continue; // next checker
 | 
						|
                }
 | 
						|
                if ($checker->isFresh($resource, $time)) {
 | 
						|
                    break; // no need to further check this resource
 | 
						|
                }
 | 
						|
 | 
						|
                return false; // cache is stale
 | 
						|
            }
 | 
						|
            // no suitable checker found, ignore this resource
 | 
						|
        }
 | 
						|
 | 
						|
        return true;
 | 
						|
    }
 | 
						|
 | 
						|
    /**
 | 
						|
     * Writes cache.
 | 
						|
     *
 | 
						|
     * @param string              $content  The content to write in the cache
 | 
						|
     * @param ResourceInterface[] $metadata An array of metadata
 | 
						|
     *
 | 
						|
     * @throws \RuntimeException When cache file can't be written
 | 
						|
     */
 | 
						|
    public function write($content, array $metadata = null)
 | 
						|
    {
 | 
						|
        $mode = 0666;
 | 
						|
        $umask = umask();
 | 
						|
        $filesystem = new Filesystem();
 | 
						|
        $filesystem->dumpFile($this->file, $content, null);
 | 
						|
        try {
 | 
						|
            $filesystem->chmod($this->file, $mode, $umask);
 | 
						|
        } catch (IOException $e) {
 | 
						|
            // discard chmod failure (some filesystem may not support it)
 | 
						|
        }
 | 
						|
 | 
						|
        if (null !== $metadata) {
 | 
						|
            $filesystem->dumpFile($this->getMetaFile(), serialize($metadata), null);
 | 
						|
            try {
 | 
						|
                $filesystem->chmod($this->getMetaFile(), $mode, $umask);
 | 
						|
            } catch (IOException $e) {
 | 
						|
                // discard chmod failure (some filesystem may not support it)
 | 
						|
            }
 | 
						|
        }
 | 
						|
 | 
						|
        if (\function_exists('opcache_invalidate') && filter_var(ini_get('opcache.enable'), FILTER_VALIDATE_BOOLEAN)) {
 | 
						|
            @opcache_invalidate($this->file, true);
 | 
						|
        }
 | 
						|
    }
 | 
						|
 | 
						|
    /**
 | 
						|
     * Gets the meta file path.
 | 
						|
     *
 | 
						|
     * @return string The meta file path
 | 
						|
     */
 | 
						|
    private function getMetaFile()
 | 
						|
    {
 | 
						|
        return $this->file.'.meta';
 | 
						|
    }
 | 
						|
}
 |