261 lines
5.6 KiB
PHP
261 lines
5.6 KiB
PHP
|
<?php defined('SYSPATH') OR die('No direct access allowed.');
|
||
|
/**
|
||
|
* File-based Cache driver.
|
||
|
*
|
||
|
* $Id: File.php 4046 2009-03-05 19:23:29Z Shadowhand $
|
||
|
*
|
||
|
* @package Cache
|
||
|
* @author Kohana Team
|
||
|
* @copyright (c) 2007-2008 Kohana Team
|
||
|
* @license http://kohanaphp.com/license.html
|
||
|
*/
|
||
|
class Cache_File_Driver implements Cache_Driver {
|
||
|
|
||
|
protected $directory = '';
|
||
|
|
||
|
/**
|
||
|
* Tests that the storage location is a directory and is writable.
|
||
|
*/
|
||
|
public function __construct($directory)
|
||
|
{
|
||
|
// Find the real path to the directory
|
||
|
$directory = str_replace('\\', '/', realpath($directory)).'/';
|
||
|
|
||
|
// Make sure the cache directory is writable
|
||
|
if ( ! is_dir($directory) OR ! is_writable($directory))
|
||
|
throw new Kohana_Exception('cache.unwritable', $directory);
|
||
|
|
||
|
// Directory is valid
|
||
|
$this->directory = $directory;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Finds an array of files matching the given id or tag.
|
||
|
*
|
||
|
* @param string cache id or tag
|
||
|
* @param bool search for tags
|
||
|
* @return array of filenames matching the id or tag
|
||
|
*/
|
||
|
public function exists($id, $tag = FALSE)
|
||
|
{
|
||
|
if ($id === TRUE)
|
||
|
{
|
||
|
// Find all the files
|
||
|
return glob($this->directory.'*~*~*');
|
||
|
}
|
||
|
elseif ($tag === TRUE)
|
||
|
{
|
||
|
// Find all the files that have the tag name
|
||
|
$paths = glob($this->directory.'*~*'.$id.'*~*');
|
||
|
|
||
|
// Find all tags matching the given tag
|
||
|
$files = array();
|
||
|
foreach ($paths as $path)
|
||
|
{
|
||
|
// Split the files
|
||
|
$tags = explode('~', basename($path));
|
||
|
|
||
|
// Find valid tags
|
||
|
if (count($tags) !== 3 OR empty($tags[1]))
|
||
|
continue;
|
||
|
|
||
|
// Split the tags by plus signs, used to separate tags
|
||
|
$tags = explode('+', $tags[1]);
|
||
|
|
||
|
if (in_array($tag, $tags))
|
||
|
{
|
||
|
// Add the file to the array, it has the requested tag
|
||
|
$files[] = $path;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return $files;
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
// Find the file matching the given id
|
||
|
return glob($this->directory.$id.'~*');
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Sets a cache item to the given data, tags, and lifetime.
|
||
|
*
|
||
|
* @param string cache id to set
|
||
|
* @param string data in the cache
|
||
|
* @param array cache tags
|
||
|
* @param integer lifetime
|
||
|
* @return bool
|
||
|
*/
|
||
|
public function set($id, $data, array $tags = NULL, $lifetime)
|
||
|
{
|
||
|
// Remove old cache files
|
||
|
$this->delete($id);
|
||
|
|
||
|
// Cache File driver expects unix timestamp
|
||
|
if ($lifetime !== 0)
|
||
|
{
|
||
|
$lifetime += time();
|
||
|
}
|
||
|
|
||
|
if ( ! empty($tags))
|
||
|
{
|
||
|
// Convert the tags into a string list
|
||
|
$tags = implode('+', $tags);
|
||
|
}
|
||
|
|
||
|
// Write out a serialized cache
|
||
|
return (bool) file_put_contents($this->directory.$id.'~'.$tags.'~'.$lifetime, serialize($data));
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Finds an array of ids for a given tag.
|
||
|
*
|
||
|
* @param string tag name
|
||
|
* @return array of ids that match the tag
|
||
|
*/
|
||
|
public function find($tag)
|
||
|
{
|
||
|
// An array will always be returned
|
||
|
$result = array();
|
||
|
|
||
|
if ($paths = $this->exists($tag, TRUE))
|
||
|
{
|
||
|
// Length of directory name
|
||
|
$offset = strlen($this->directory);
|
||
|
|
||
|
// Find all the files with the given tag
|
||
|
foreach ($paths as $path)
|
||
|
{
|
||
|
// Get the id from the filename
|
||
|
list($id, $junk) = explode('~', basename($path), 2);
|
||
|
|
||
|
if (($data = $this->get($id)) !== FALSE)
|
||
|
{
|
||
|
// Add the result to the array
|
||
|
$result[$id] = $data;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return $result;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Fetches a cache item. This will delete the item if it is expired or if
|
||
|
* the hash does not match the stored hash.
|
||
|
*
|
||
|
* @param string cache id
|
||
|
* @return mixed|NULL
|
||
|
*/
|
||
|
public function get($id)
|
||
|
{
|
||
|
if ($file = $this->exists($id))
|
||
|
{
|
||
|
// Use the first file
|
||
|
$file = current($file);
|
||
|
|
||
|
// Validate that the cache has not expired
|
||
|
if ($this->expired($file))
|
||
|
{
|
||
|
// Remove this cache, it has expired
|
||
|
$this->delete($id);
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
// Turn off errors while reading the file
|
||
|
$ER = error_reporting(0);
|
||
|
|
||
|
if (($data = file_get_contents($file)) !== FALSE)
|
||
|
{
|
||
|
// Unserialize the data
|
||
|
$data = unserialize($data);
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
// Delete the data
|
||
|
unset($data);
|
||
|
}
|
||
|
|
||
|
// Turn errors back on
|
||
|
error_reporting($ER);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// Return NULL if there is no data
|
||
|
return isset($data) ? $data : NULL;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Deletes a cache item by id or tag
|
||
|
*
|
||
|
* @param string cache id or tag, or TRUE for "all items"
|
||
|
* @param boolean use tags
|
||
|
* @return boolean
|
||
|
*/
|
||
|
public function delete($id, $tag = FALSE)
|
||
|
{
|
||
|
$files = $this->exists($id, $tag);
|
||
|
|
||
|
if (empty($files))
|
||
|
return FALSE;
|
||
|
|
||
|
// Disable all error reporting while deleting
|
||
|
$ER = error_reporting(0);
|
||
|
|
||
|
foreach ($files as $file)
|
||
|
{
|
||
|
// Remove the cache file
|
||
|
if ( ! unlink($file))
|
||
|
Kohana::log('error', 'Cache: Unable to delete cache file: '.$file);
|
||
|
}
|
||
|
|
||
|
// Turn on error reporting again
|
||
|
error_reporting($ER);
|
||
|
|
||
|
return TRUE;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Deletes all cache files that are older than the current time.
|
||
|
*
|
||
|
* @return void
|
||
|
*/
|
||
|
public function delete_expired()
|
||
|
{
|
||
|
if ($files = $this->exists(TRUE))
|
||
|
{
|
||
|
// Disable all error reporting while deleting
|
||
|
$ER = error_reporting(0);
|
||
|
|
||
|
foreach ($files as $file)
|
||
|
{
|
||
|
if ($this->expired($file))
|
||
|
{
|
||
|
// The cache file has already expired, delete it
|
||
|
if ( ! unlink($file))
|
||
|
Kohana::log('error', 'Cache: Unable to delete cache file: '.$file);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// Turn on error reporting again
|
||
|
error_reporting($ER);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Check if a cache file has expired by filename.
|
||
|
*
|
||
|
* @param string filename
|
||
|
* @return bool
|
||
|
*/
|
||
|
protected function expired($file)
|
||
|
{
|
||
|
// Get the expiration time
|
||
|
$expires = (int) substr($file, strrpos($file, '~') + 1);
|
||
|
|
||
|
// Expirations of 0 are "never expire"
|
||
|
return ($expires !== 0 AND $expires <= time());
|
||
|
}
|
||
|
|
||
|
} // End Cache File Driver
|