diff options
Diffstat (limited to 'phpBB/phpbb/cache')
| -rw-r--r-- | phpBB/phpbb/cache/driver/apc.php | 70 | ||||
| -rw-r--r-- | phpBB/phpbb/cache/driver/base.php | 18 | ||||
| -rw-r--r-- | phpBB/phpbb/cache/driver/driver_interface.php | 167 | ||||
| -rw-r--r-- | phpBB/phpbb/cache/driver/eaccelerator.php | 105 | ||||
| -rw-r--r-- | phpBB/phpbb/cache/driver/file.php | 787 | ||||
| -rw-r--r-- | phpBB/phpbb/cache/driver/memcache.php | 122 | ||||
| -rw-r--r-- | phpBB/phpbb/cache/driver/memory.php | 440 | ||||
| -rw-r--r-- | phpBB/phpbb/cache/driver/null.php | 151 | ||||
| -rw-r--r-- | phpBB/phpbb/cache/driver/redis.php | 158 | ||||
| -rw-r--r-- | phpBB/phpbb/cache/driver/wincache.php | 73 | ||||
| -rw-r--r-- | phpBB/phpbb/cache/driver/xcache.php | 107 | ||||
| -rw-r--r-- | phpBB/phpbb/cache/service.php | 396 | 
12 files changed, 2594 insertions, 0 deletions
| diff --git a/phpBB/phpbb/cache/driver/apc.php b/phpBB/phpbb/cache/driver/apc.php new file mode 100644 index 0000000000..521d5d41ea --- /dev/null +++ b/phpBB/phpbb/cache/driver/apc.php @@ -0,0 +1,70 @@ +<?php +/** +* +* This file is part of the phpBB Forum Software package. +* +* @copyright (c) phpBB Limited <https://www.phpbb.com> +* @license GNU General Public License, version 2 (GPL-2.0) +* +* For full copyright and license information, please see +* the docs/CREDITS.txt file. +* +*/ + +namespace phpbb\cache\driver; + +/** +* ACM for APC +*/ +class apc extends \phpbb\cache\driver\memory +{ +	var $extension = 'apc'; + +	/** +	* {@inheritDoc} +	*/ +	function purge() +	{ +		apc_clear_cache('user'); + +		parent::purge(); +	} + +	/** +	* Fetch an item from the cache +	* +	* @access protected +	* @param string $var Cache key +	* @return mixed Cached data +	*/ +	function _read($var) +	{ +		return apc_fetch($this->key_prefix . $var); +	} + +	/** +	* Store data in the cache +	* +	* @access protected +	* @param string $var Cache key +	* @param mixed $data Data to store +	* @param int $ttl Time-to-live of cached data +	* @return bool True if the operation succeeded +	*/ +	function _write($var, $data, $ttl = 2592000) +	{ +		return apc_store($this->key_prefix . $var, $data, $ttl); +	} + +	/** +	* Remove an item from the cache +	* +	* @access protected +	* @param string $var Cache key +	* @return bool True if the operation succeeded +	*/ +	function _delete($var) +	{ +		return apc_delete($this->key_prefix . $var); +	} +} diff --git a/phpBB/phpbb/cache/driver/base.php b/phpBB/phpbb/cache/driver/base.php new file mode 100644 index 0000000000..685cdc4e57 --- /dev/null +++ b/phpBB/phpbb/cache/driver/base.php @@ -0,0 +1,18 @@ +<?php +/** +* +* This file is part of the phpBB Forum Software package. +* +* @copyright (c) phpBB Limited <https://www.phpbb.com> +* @license GNU General Public License, version 2 (GPL-2.0) +* +* For full copyright and license information, please see +* the docs/CREDITS.txt file. +* +*/ + +namespace phpbb\cache\driver; + +abstract class base implements \phpbb\cache\driver\driver_interface +{ +} diff --git a/phpBB/phpbb/cache/driver/driver_interface.php b/phpBB/phpbb/cache/driver/driver_interface.php new file mode 100644 index 0000000000..9ac9ca0c59 --- /dev/null +++ b/phpBB/phpbb/cache/driver/driver_interface.php @@ -0,0 +1,167 @@ +<?php +/** +* +* This file is part of the phpBB Forum Software package. +* +* @copyright (c) phpBB Limited <https://www.phpbb.com> +* @license GNU General Public License, version 2 (GPL-2.0) +* +* For full copyright and license information, please see +* the docs/CREDITS.txt file. +* +*/ + +namespace phpbb\cache\driver; + +/** +* An interface that all cache drivers must implement +*/ +interface driver_interface +{ +	/** +	* Load global cache +	* +	* @return mixed False if an error was encountered, otherwise the data type of the cached data +	*/ +	public function load(); + +	/** +	* Unload cache object +	* +	* @return null +	*/ +	public function unload(); + +	/** +	* Save modified objects +	* +	* @return null +	*/ +	public function save(); + +	/** +	* Tidy cache +	* +	* @return null +	*/ +	public function tidy(); + +	/** +	* Get saved cache object +	* +	* @param string $var_name 		Cache key +	* @return mixed 				False if an error was encountered, otherwise the saved cached object +	*/ +	public function get($var_name); + +	/** +	* Put data into cache +	* +	* @param string $var_name 		Cache key +	* @param mixed $var 			Cached data to store +	* @param int $ttl 				Time-to-live of cached data +	* @return null +	*/ +	public function put($var_name, $var, $ttl = 0); + +	/** +	* Purge cache data +	* +	* @return null +	*/ +	public function purge(); + +	/** +	* Destroy cache data +	* +	* @param string $var_name 		Cache key +	* @param string $table 			Table name +	* @return null +	*/ +	public function destroy($var_name, $table = ''); + +	/** +	* Check if a given cache entry exists +	* +	* @param string $var_name 		Cache key +	* +	* @return bool 					True if cache file exists and has not expired. +	*								False otherwise. +	*/ +	public function _exists($var_name); + +	/** +	* Load result of an SQL query from cache. +	* +	* @param string $query			SQL query +	* +	* @return int|bool				Query ID (integer) if cache contains a rowset +	*								for the specified query. +	*								False otherwise. +	*/ +	public function sql_load($query); + +	/** +	* Save result of an SQL query in cache. +	* +	* In persistent cache stores, this function stores the query +	* result to persistent storage. In other words, there is no need +	* to call save() afterwards. +	* +	* @param \phpbb\db\driver\driver_interface $db	Database connection +	* @param string $query			SQL query, should be used for generating storage key +	* @param mixed $query_result	The result from \dbal::sql_query, to be passed to +	* 								\dbal::sql_fetchrow to get all rows and store them +	* 								in cache. +	* @param int $ttl				Time to live, after this timeout the query should +	*								expire from the cache. +	* @return int|mixed				If storing in cache succeeded, an integer $query_id +	* 								representing the query should be returned. Otherwise +	* 								the original $query_result should be returned. +	*/ +	public function sql_save(\phpbb\db\driver\driver_interface $db, $query, $query_result, $ttl); + +	/** +	* Check if result for a given SQL query exists in cache. +	* +	* @param int $query_id +	* @return bool +	*/ +	public function sql_exists($query_id); + +	/** +	* Fetch row from cache (database) +	* +	* @param int $query_id +	* @return array|bool 			The query result if found in the cache, otherwise +	* 								false. +	*/ +	public function sql_fetchrow($query_id); + +	/** +	* Fetch a field from the current row of a cached database result (database) +	* +	* @param int $query_id +	* @param string $field 			The name of the column. +	* @return string|bool 			The field of the query result if found in the cache, +	* 								otherwise false. +	*/ +	public function sql_fetchfield($query_id, $field); + +	/** +	* Seek a specific row in an a cached database result (database) +	* +	* @param int $rownum 			Row to seek to. +	* @param int $query_id +	* @return bool +	*/ +	public function sql_rowseek($rownum, $query_id); + +	/** +	* Free memory used for a cached database result (database) +	* +	* @param int $query_id +	* @return bool +	*/ +	public function sql_freeresult($query_id); +} diff --git a/phpBB/phpbb/cache/driver/eaccelerator.php b/phpBB/phpbb/cache/driver/eaccelerator.php new file mode 100644 index 0000000000..1697758acc --- /dev/null +++ b/phpBB/phpbb/cache/driver/eaccelerator.php @@ -0,0 +1,105 @@ +<?php +/** +* +* This file is part of the phpBB Forum Software package. +* +* @copyright (c) phpBB Limited <https://www.phpbb.com> +* @license GNU General Public License, version 2 (GPL-2.0) +* +* For full copyright and license information, please see +* the docs/CREDITS.txt file. +* +*/ + +namespace phpbb\cache\driver; + +/** +* ACM for eAccelerator +* @todo Missing locks from destroy() talk with David +*/ +class eaccelerator extends \phpbb\cache\driver\memory +{ +	var $extension = 'eaccelerator'; +	var $function = 'eaccelerator_get'; + +	var $serialize_header = '#phpbb-serialized#'; + +	/** +	* {@inheritDoc} +	*/ +	function purge() +	{ +		foreach (eaccelerator_list_keys() as $var) +		{ +			// @todo Check why the substr() +			// @todo Only unset vars matching $this->key_prefix +			eaccelerator_rm(substr($var['name'], 1)); +		} + +		parent::purge(); +	} + +	/** +	* {@inheritDoc} +	*/ +	function tidy() +	{ +		eaccelerator_gc(); + +		set_config('cache_last_gc', time(), true); +	} + +	/** +	* Fetch an item from the cache +	* +	* @access protected +	* @param string $var Cache key +	* @return mixed Cached data +	*/ +	function _read($var) +	{ +		$result = eaccelerator_get($this->key_prefix . $var); + +		if ($result === null) +		{ +			return false; +		} + +		// Handle serialized objects +		if (is_string($result) && strpos($result, $this->serialize_header . 'O:') === 0) +		{ +			$result = unserialize(substr($result, strlen($this->serialize_header))); +		} + +		return $result; +	} + +	/** +	* Store data in the cache +	* +	* @access protected +	* @param string $var Cache key +	* @param mixed $data Data to store +	* @param int $ttl Time-to-live of cached data +	* @return bool True if the operation succeeded +	*/ +	function _write($var, $data, $ttl = 2592000) +	{ +		// Serialize objects and make them easy to detect +		$data = (is_object($data)) ? $this->serialize_header . serialize($data) : $data; + +		return eaccelerator_put($this->key_prefix . $var, $data, $ttl); +	} + +	/** +	* Remove an item from the cache +	* +	* @access protected +	* @param string $var Cache key +	* @return bool True if the operation succeeded +	*/ +	function _delete($var) +	{ +		return eaccelerator_rm($this->key_prefix . $var); +	} +} diff --git a/phpBB/phpbb/cache/driver/file.php b/phpBB/phpbb/cache/driver/file.php new file mode 100644 index 0000000000..b32af32d25 --- /dev/null +++ b/phpBB/phpbb/cache/driver/file.php @@ -0,0 +1,787 @@ +<?php +/** +* +* This file is part of the phpBB Forum Software package. +* +* @copyright (c) phpBB Limited <https://www.phpbb.com> +* @license GNU General Public License, version 2 (GPL-2.0) +* +* For full copyright and license information, please see +* the docs/CREDITS.txt file. +* +*/ + +namespace phpbb\cache\driver; + +/** +* ACM File Based Caching +*/ +class file extends \phpbb\cache\driver\base +{ +	var $vars = array(); +	var $var_expires = array(); +	var $is_modified = false; + +	var $sql_rowset = array(); +	var $sql_row_pointer = array(); +	var $cache_dir = ''; + +	/** +	* Set cache path +	* +	* @param string $cache_dir Define the path to the cache directory (default: $phpbb_root_path . 'cache/') +	*/ +	function __construct($cache_dir = null) +	{ +		global $phpbb_root_path; +		$this->cache_dir = !is_null($cache_dir) ? $cache_dir : $phpbb_root_path . 'cache/'; +	} + +	/** +	* {@inheritDoc} +	*/ +	function load() +	{ +		return $this->_read('data_global'); +	} + +	/** +	* {@inheritDoc} +	*/ +	function unload() +	{ +		$this->save(); +		unset($this->vars); +		unset($this->var_expires); +		unset($this->sql_rowset); +		unset($this->sql_row_pointer); + +		$this->vars = array(); +		$this->var_expires = array(); +		$this->sql_rowset = array(); +		$this->sql_row_pointer = array(); +	} + +	/** +	* {@inheritDoc} +	*/ +	function save() +	{ +		if (!$this->is_modified) +		{ +			return; +		} + +		global $phpEx; + +		if (!$this->_write('data_global')) +		{ +			if (!function_exists('phpbb_is_writable')) +			{ +				global $phpbb_root_path; +				include($phpbb_root_path . 'includes/functions.' . $phpEx); +			} + +			// Now, this occurred how often? ... phew, just tell the user then... +			if (!phpbb_is_writable($this->cache_dir)) +			{ +				// We need to use die() here, because else we may encounter an infinite loop (the message handler calls $cache->unload()) +				die('Fatal: ' . $this->cache_dir . ' is NOT writable.'); +				exit; +			} + +			die('Fatal: Not able to open ' . $this->cache_dir . 'data_global.' . $phpEx); +			exit; +		} + +		$this->is_modified = false; +	} + +	/** +	* {@inheritDoc} +	*/ +	function tidy() +	{ +		global $phpEx; + +		$dir = @opendir($this->cache_dir); + +		if (!$dir) +		{ +			return; +		} + +		$time = time(); + +		while (($entry = readdir($dir)) !== false) +		{ +			if (!preg_match('/^(sql_|data_(?!global))/', $entry)) +			{ +				continue; +			} + +			if (!($handle = @fopen($this->cache_dir . $entry, 'rb'))) +			{ +				continue; +			} + +			// Skip the PHP header +			fgets($handle); + +			// Skip expiration +			$expires = (int) fgets($handle); + +			fclose($handle); + +			if ($time >= $expires) +			{ +				$this->remove_file($this->cache_dir . $entry); +			} +		} +		closedir($dir); + +		if (file_exists($this->cache_dir . 'data_global.' . $phpEx)) +		{ +			if (!sizeof($this->vars)) +			{ +				$this->load(); +			} + +			foreach ($this->var_expires as $var_name => $expires) +			{ +				if ($time >= $expires) +				{ +					$this->destroy($var_name); +				} +			} +		} + +		set_config('cache_last_gc', time(), true); +	} + +	/** +	* {@inheritDoc} +	*/ +	function get($var_name) +	{ +		if ($var_name[0] == '_') +		{ +			global $phpEx; + +			if (!$this->_exists($var_name)) +			{ +				return false; +			} + +			return $this->_read('data' . $var_name); +		} +		else +		{ +			return ($this->_exists($var_name)) ? $this->vars[$var_name] : false; +		} +	} + +	/** +	* {@inheritDoc} +	*/ +	function put($var_name, $var, $ttl = 31536000) +	{ +		if ($var_name[0] == '_') +		{ +			$this->_write('data' . $var_name, $var, time() + $ttl); +		} +		else +		{ +			$this->vars[$var_name] = $var; +			$this->var_expires[$var_name] = time() + $ttl; +			$this->is_modified = true; +		} +	} + +	/** +	* {@inheritDoc} +	*/ +	function purge() +	{ +		// Purge all phpbb cache files +		try +		{ +			$iterator = new \DirectoryIterator($this->cache_dir); +		} +		catch (Exception $e) +		{ +			return; +		} + +		foreach ($iterator as $fileInfo) +		{ +			if ($fileInfo->isDot()) +			{ +				continue; +			} +			$filename = $fileInfo->getFilename(); +			if ($fileInfo->isDir()) +			{ +				$this->remove_dir($fileInfo->getPathname()); +			} +			else if (strpos($filename, 'container_') === 0 || +				strpos($filename, 'url_matcher') === 0 || +				strpos($filename, 'sql_') === 0 || +				strpos($filename, 'data_') === 0) +			{ +				$this->remove_file($fileInfo->getPathname()); +			} +		} + +		unset($this->vars); +		unset($this->var_expires); +		unset($this->sql_rowset); +		unset($this->sql_row_pointer); + +		$this->vars = array(); +		$this->var_expires = array(); +		$this->sql_rowset = array(); +		$this->sql_row_pointer = array(); + +		$this->is_modified = false; +	} + +	/** +	* Remove directory +	* +	* @param string $dir Directory to remove +	* +	* @return null +	*/ +	protected function remove_dir($dir) +	{ +		try +		{ +			$iterator = new \DirectoryIterator($dir); +		} +		catch (Exception $e) +		{ +			return; +		} + +		foreach ($iterator as $fileInfo) +		{ +			if ($fileInfo->isDot()) +			{ +				continue; +			} + +			if ($fileInfo->isDir()) +			{ +				$this->remove_dir($fileInfo->getPathname()); +			} +			else +			{ +				$this->remove_file($fileInfo->getPathname()); +			} +		} + +		@rmdir($dir); +	} + +	/** +	* {@inheritDoc} +	*/ +	function destroy($var_name, $table = '') +	{ +		global $phpEx; + +		if ($var_name == 'sql' && !empty($table)) +		{ +			if (!is_array($table)) +			{ +				$table = array($table); +			} + +			$dir = @opendir($this->cache_dir); + +			if (!$dir) +			{ +				return; +			} + +			while (($entry = readdir($dir)) !== false) +			{ +				if (strpos($entry, 'sql_') !== 0) +				{ +					continue; +				} + +				if (!($handle = @fopen($this->cache_dir . $entry, 'rb'))) +				{ +					continue; +				} + +				// Skip the PHP header +				fgets($handle); + +				// Skip expiration +				fgets($handle); + +				// Grab the query, remove the LF +				$query = substr(fgets($handle), 0, -1); + +				fclose($handle); + +				foreach ($table as $check_table) +				{ +					// Better catch partial table names than no table names. ;) +					if (strpos($query, $check_table) !== false) +					{ +						$this->remove_file($this->cache_dir . $entry); +						break; +					} +				} +			} +			closedir($dir); + +			return; +		} + +		if (!$this->_exists($var_name)) +		{ +			return; +		} + +		if ($var_name[0] == '_') +		{ +			$this->remove_file($this->cache_dir . 'data' . $var_name . ".$phpEx", true); +		} +		else if (isset($this->vars[$var_name])) +		{ +			$this->is_modified = true; +			unset($this->vars[$var_name]); +			unset($this->var_expires[$var_name]); + +			// We save here to let the following cache hits succeed +			$this->save(); +		} +	} + +	/** +	* {@inheritDoc} +	*/ +	function _exists($var_name) +	{ +		if ($var_name[0] == '_') +		{ +			global $phpEx; +			return file_exists($this->cache_dir . 'data' . $var_name . ".$phpEx"); +		} +		else +		{ +			if (!sizeof($this->vars)) +			{ +				$this->load(); +			} + +			if (!isset($this->var_expires[$var_name])) +			{ +				return false; +			} + +			return (time() > $this->var_expires[$var_name]) ? false : isset($this->vars[$var_name]); +		} +	} + +	/** +	* {@inheritDoc} +	*/ +	function sql_load($query) +	{ +		// Remove extra spaces and tabs +		$query = preg_replace('/[\n\r\s\t]+/', ' ', $query); + +		if (($rowset = $this->_read('sql_' . md5($query))) === false) +		{ +			return false; +		} + +		$query_id = sizeof($this->sql_rowset); +		$this->sql_rowset[$query_id] = $rowset; +		$this->sql_row_pointer[$query_id] = 0; + +		return $query_id; +	} + +	/** +	* {@inheritDoc} +	*/ +	function sql_save(\phpbb\db\driver\driver_interface $db, $query, $query_result, $ttl) +	{ +		// Remove extra spaces and tabs +		$query = preg_replace('/[\n\r\s\t]+/', ' ', $query); + +		$query_id = sizeof($this->sql_rowset); +		$this->sql_rowset[$query_id] = array(); +		$this->sql_row_pointer[$query_id] = 0; + +		while ($row = $db->sql_fetchrow($query_result)) +		{ +			$this->sql_rowset[$query_id][] = $row; +		} +		$db->sql_freeresult($query_result); + +		if ($this->_write('sql_' . md5($query), $this->sql_rowset[$query_id], $ttl + time(), $query)) +		{ +			return $query_id; +		} + +		return $query_result; +	} + +	/** +	* {@inheritDoc} +	*/ +	function sql_exists($query_id) +	{ +		return isset($this->sql_rowset[$query_id]); +	} + +	/** +	* {@inheritDoc} +	*/ +	function sql_fetchrow($query_id) +	{ +		if ($this->sql_row_pointer[$query_id] < sizeof($this->sql_rowset[$query_id])) +		{ +			return $this->sql_rowset[$query_id][$this->sql_row_pointer[$query_id]++]; +		} + +		return false; +	} + +	/** +	* {@inheritDoc} +	*/ +	function sql_fetchfield($query_id, $field) +	{ +		if ($this->sql_row_pointer[$query_id] < sizeof($this->sql_rowset[$query_id])) +		{ +			return (isset($this->sql_rowset[$query_id][$this->sql_row_pointer[$query_id]][$field])) ? $this->sql_rowset[$query_id][$this->sql_row_pointer[$query_id]++][$field] : false; +		} + +		return false; +	} + +	/** +	* {@inheritDoc} +	*/ +	function sql_rowseek($rownum, $query_id) +	{ +		if ($rownum >= sizeof($this->sql_rowset[$query_id])) +		{ +			return false; +		} + +		$this->sql_row_pointer[$query_id] = $rownum; +		return true; +	} + +	/** +	* {@inheritDoc} +	*/ +	function sql_freeresult($query_id) +	{ +		if (!isset($this->sql_rowset[$query_id])) +		{ +			return false; +		} + +		unset($this->sql_rowset[$query_id]); +		unset($this->sql_row_pointer[$query_id]); + +		return true; +	} + +	/** +	* Read cached data from a specified file +	* +	* @access private +	* @param string $filename Filename to write +	* @return mixed False if an error was encountered, otherwise the data type of the cached data +	*/ +	function _read($filename) +	{ +		global $phpEx; + +		$file = "{$this->cache_dir}$filename.$phpEx"; + +		$type = substr($filename, 0, strpos($filename, '_')); + +		if (!file_exists($file)) +		{ +			return false; +		} + +		if (!($handle = @fopen($file, 'rb'))) +		{ +			return false; +		} + +		// Skip the PHP header +		fgets($handle); + +		if ($filename == 'data_global') +		{ +			$this->vars = $this->var_expires = array(); + +			$time = time(); + +			while (($expires = (int) fgets($handle)) && !feof($handle)) +			{ +				// Number of bytes of data +				$bytes = substr(fgets($handle), 0, -1); + +				if (!is_numeric($bytes) || ($bytes = (int) $bytes) === 0) +				{ +					// We cannot process the file without a valid number of bytes +					// so we discard it +					fclose($handle); + +					$this->vars = $this->var_expires = array(); +					$this->is_modified = false; + +					$this->remove_file($file); + +					return false; +				} + +				if ($time >= $expires) +				{ +					fseek($handle, $bytes, SEEK_CUR); + +					continue; +				} + +				$var_name = substr(fgets($handle), 0, -1); + +				// Read the length of bytes that consists of data. +				$data = fread($handle, $bytes - strlen($var_name)); +				$data = @unserialize($data); + +				// Don't use the data if it was invalid +				if ($data !== false) +				{ +					$this->vars[$var_name] = $data; +					$this->var_expires[$var_name] = $expires; +				} + +				// Absorb the LF +				fgets($handle); +			} + +			fclose($handle); + +			$this->is_modified = false; + +			return true; +		} +		else +		{ +			$data = false; +			$line = 0; + +			while (($buffer = fgets($handle)) && !feof($handle)) +			{ +				$buffer = substr($buffer, 0, -1); // Remove the LF + +				// $buffer is only used to read integers +				// if it is non numeric we have an invalid +				// cache file, which we will now remove. +				if (!is_numeric($buffer)) +				{ +					break; +				} + +				if ($line == 0) +				{ +					$expires = (int) $buffer; + +					if (time() >= $expires) +					{ +						break; +					} + +					if ($type == 'sql') +					{ +						// Skip the query +						fgets($handle); +					} +				} +				else if ($line == 1) +				{ +					$bytes = (int) $buffer; + +					// Never should have 0 bytes +					if (!$bytes) +					{ +						break; +					} + +					// Grab the serialized data +					$data = fread($handle, $bytes); + +					// Read 1 byte, to trigger EOF +					fread($handle, 1); + +					if (!feof($handle)) +					{ +						// Somebody tampered with our data +						$data = false; +					} +					break; +				} +				else +				{ +					// Something went wrong +					break; +				} +				$line++; +			} +			fclose($handle); + +			// unserialize if we got some data +			$data = ($data !== false) ? @unserialize($data) : $data; + +			if ($data === false) +			{ +				$this->remove_file($file); +				return false; +			} + +			return $data; +		} +	} + +	/** +	* Write cache data to a specified file +	* +	* 'data_global' is a special case and the generated format is different for this file: +	* <code> +	* <?php exit; ?> +	* (expiration) +	* (length of var and serialised data) +	* (var) +	* (serialised data) +	* ... (repeat) +	* </code> +	* +	* The other files have a similar format: +	* <code> +	* <?php exit; ?> +	* (expiration) +	* (query) [SQL files only] +	* (length of serialised data) +	* (serialised data) +	* </code> +	* +	* @access private +	* @param string $filename Filename to write +	* @param mixed $data Data to store +	* @param int $expires Timestamp when the data expires +	* @param string $query Query when caching SQL queries +	* @return bool True if the file was successfully created, otherwise false +	*/ +	function _write($filename, $data = null, $expires = 0, $query = '') +	{ +		global $phpEx; + +		$file = "{$this->cache_dir}$filename.$phpEx"; + +		$lock = new \phpbb\lock\flock($file); +		$lock->acquire(); + +		if ($handle = @fopen($file, 'wb')) +		{ +			// File header +			fwrite($handle, '<' . '?php exit; ?' . '>'); + +			if ($filename == 'data_global') +			{ +				// Global data is a different format +				foreach ($this->vars as $var => $data) +				{ +					if (strpos($var, "\r") !== false || strpos($var, "\n") !== false) +					{ +						// CR/LF would cause fgets() to read the cache file incorrectly +						// do not cache test entries, they probably won't be read back +						// the cache keys should really be alphanumeric with a few symbols. +						continue; +					} +					$data = serialize($data); + +					// Write out the expiration time +					fwrite($handle, "\n" . $this->var_expires[$var] . "\n"); + +					// Length of the remaining data for this var (ignoring two LF's) +					fwrite($handle, strlen($data . $var) . "\n"); +					fwrite($handle, $var . "\n"); +					fwrite($handle, $data); +				} +			} +			else +			{ +				fwrite($handle, "\n" . $expires . "\n"); + +				if (strpos($filename, 'sql_') === 0) +				{ +					fwrite($handle, $query . "\n"); +				} +				$data = serialize($data); + +				fwrite($handle, strlen($data) . "\n"); +				fwrite($handle, $data); +			} + +			fclose($handle); + +			if (!function_exists('phpbb_chmod')) +			{ +				global $phpbb_root_path; +				include($phpbb_root_path . 'includes/functions.' . $phpEx); +			} + +			phpbb_chmod($file, CHMOD_READ | CHMOD_WRITE); + +			$return_value = true; +		} +		else +		{ +			$return_value = false; +		} + +		$lock->release(); + +		return $return_value; +	} + +	/** +	* Removes/unlinks file +	* +	* @param string $filename Filename to remove +	* @param bool $check Check file permissions +	* @return bool True if the file was successfully removed, otherwise false +	*/ +	function remove_file($filename, $check = false) +	{ +		if (!function_exists('phpbb_is_writable')) +		{ +			global $phpbb_root_path, $phpEx; +			include($phpbb_root_path . 'includes/functions.' . $phpEx); +		} + +		if ($check && !phpbb_is_writable($this->cache_dir)) +		{ +			// E_USER_ERROR - not using language entry - intended. +			trigger_error('Unable to remove files within ' . $this->cache_dir . '. Please check directory permissions.', E_USER_ERROR); +		} + +		return @unlink($filename); +	} +} diff --git a/phpBB/phpbb/cache/driver/memcache.php b/phpBB/phpbb/cache/driver/memcache.php new file mode 100644 index 0000000000..406ab11ddd --- /dev/null +++ b/phpBB/phpbb/cache/driver/memcache.php @@ -0,0 +1,122 @@ +<?php +/** +* +* This file is part of the phpBB Forum Software package. +* +* @copyright (c) phpBB Limited <https://www.phpbb.com> +* @license GNU General Public License, version 2 (GPL-2.0) +* +* For full copyright and license information, please see +* the docs/CREDITS.txt file. +* +*/ + +namespace phpbb\cache\driver; + +if (!defined('PHPBB_ACM_MEMCACHE_PORT')) +{ +	define('PHPBB_ACM_MEMCACHE_PORT', 11211); +} + +if (!defined('PHPBB_ACM_MEMCACHE_COMPRESS')) +{ +	define('PHPBB_ACM_MEMCACHE_COMPRESS', false); +} + +if (!defined('PHPBB_ACM_MEMCACHE_HOST')) +{ +	define('PHPBB_ACM_MEMCACHE_HOST', 'localhost'); +} + +if (!defined('PHPBB_ACM_MEMCACHE')) +{ +	//can define multiple servers with host1/port1,host2/port2 format +	define('PHPBB_ACM_MEMCACHE', PHPBB_ACM_MEMCACHE_HOST . '/' . PHPBB_ACM_MEMCACHE_PORT); +} + +/** +* ACM for Memcached +*/ +class memcache extends \phpbb\cache\driver\memory +{ +	var $extension = 'memcache'; + +	var $memcache; +	var $flags = 0; + +	function __construct() +	{ +		// Call the parent constructor +		parent::__construct(); + +		$this->memcache = new \Memcache; +		foreach(explode(',', PHPBB_ACM_MEMCACHE) as $u) +		{ +			$parts = explode('/', $u); +			$this->memcache->addServer(trim($parts[0]), trim($parts[1])); +		} +		$this->flags = (PHPBB_ACM_MEMCACHE_COMPRESS) ? MEMCACHE_COMPRESSED : 0; +	} + +	/** +	* {@inheritDoc} +	*/ +	function unload() +	{ +		parent::unload(); + +		$this->memcache->close(); +	} + +	/** +	* {@inheritDoc} +	*/ +	function purge() +	{ +		$this->memcache->flush(); + +		parent::purge(); +	} + +	/** +	* Fetch an item from the cache +	* +	* @access protected +	* @param string $var Cache key +	* @return mixed Cached data +	*/ +	function _read($var) +	{ +		return $this->memcache->get($this->key_prefix . $var); +	} + +	/** +	* Store data in the cache +	* +	* @access protected +	* @param string $var Cache key +	* @param mixed $data Data to store +	* @param int $ttl Time-to-live of cached data +	* @return bool True if the operation succeeded +	*/ +	function _write($var, $data, $ttl = 2592000) +	{ +		if (!$this->memcache->replace($this->key_prefix . $var, $data, $this->flags, $ttl)) +		{ +			return $this->memcache->set($this->key_prefix . $var, $data, $this->flags, $ttl); +		} +		return true; +	} + +	/** +	* Remove an item from the cache +	* +	* @access protected +	* @param string $var Cache key +	* @return bool True if the operation succeeded +	*/ +	function _delete($var) +	{ +		return $this->memcache->delete($this->key_prefix . $var); +	} +} diff --git a/phpBB/phpbb/cache/driver/memory.php b/phpBB/phpbb/cache/driver/memory.php new file mode 100644 index 0000000000..5dee375192 --- /dev/null +++ b/phpBB/phpbb/cache/driver/memory.php @@ -0,0 +1,440 @@ +<?php +/** +* +* This file is part of the phpBB Forum Software package. +* +* @copyright (c) phpBB Limited <https://www.phpbb.com> +* @license GNU General Public License, version 2 (GPL-2.0) +* +* For full copyright and license information, please see +* the docs/CREDITS.txt file. +* +*/ + +namespace phpbb\cache\driver; + +/** +* ACM Abstract Memory Class +*/ +abstract class memory extends \phpbb\cache\driver\base +{ +	var $key_prefix; + +	var $vars = array(); +	var $is_modified = false; + +	var $sql_rowset = array(); +	var $sql_row_pointer = array(); +	var $cache_dir = ''; + +	/** +	* Set cache path +	*/ +	function __construct() +	{ +		global $phpbb_root_path, $dbname, $table_prefix; + +		$this->cache_dir	= $phpbb_root_path . 'cache/'; +		$this->key_prefix	= substr(md5($dbname . $table_prefix), 0, 8) . '_'; + +		if (!isset($this->extension) || !extension_loaded($this->extension)) +		{ +			global $acm_type; + +			trigger_error("Could not find required extension [{$this->extension}] for the ACM module $acm_type.", E_USER_ERROR); +		} + +		if (isset($this->function) && !function_exists($this->function)) +		{ +			global $acm_type; + +			trigger_error("The required function [{$this->function}] is not available for the ACM module $acm_type.", E_USER_ERROR); +		} +	} + +	/** +	* {@inheritDoc} +	*/ +	function load() +	{ +		// grab the global cache +		$this->vars = $this->_read('global'); + +		if ($this->vars !== false) +		{ +			return true; +		} + +		return false; +	} + +	/** +	* {@inheritDoc} +	*/ +	function unload() +	{ +		$this->save(); +		unset($this->vars); +		unset($this->sql_rowset); +		unset($this->sql_row_pointer); + +		$this->vars = array(); +		$this->sql_rowset = array(); +		$this->sql_row_pointer = array(); +	} + +	/** +	* {@inheritDoc} +	*/ +	function save() +	{ +		if (!$this->is_modified) +		{ +			return; +		} + +		$this->_write('global', $this->vars, 2592000); + +		$this->is_modified = false; +	} + +	/** +	* {@inheritDoc} +	*/ +	function tidy() +	{ +		// cache has auto GC, no need to have any code here :) + +		set_config('cache_last_gc', time(), true); +	} + +	/** +	* {@inheritDoc} +	*/ +	function get($var_name) +	{ +		if ($var_name[0] == '_') +		{ +			if (!$this->_exists($var_name)) +			{ +				return false; +			} + +			return $this->_read($var_name); +		} +		else +		{ +			return ($this->_exists($var_name)) ? $this->vars[$var_name] : false; +		} +	} + +	/** +	* {@inheritDoc} +	*/ +	function put($var_name, $var, $ttl = 2592000) +	{ +		if ($var_name[0] == '_') +		{ +			$this->_write($var_name, $var, $ttl); +		} +		else +		{ +			$this->vars[$var_name] = $var; +			$this->is_modified = true; +		} +	} + +	/** +	* {@inheritDoc} +	*/ +	function purge() +	{ +		// Purge all phpbb cache files +		$dir = @opendir($this->cache_dir); + +		if (!$dir) +		{ +			return; +		} + +		while (($entry = readdir($dir)) !== false) +		{ +			if (strpos($entry, 'container_') !== 0 && +				strpos($entry, 'url_matcher') !== 0 && +				strpos($entry, 'sql_') !== 0 && +				strpos($entry, 'data_') !== 0 && +				strpos($entry, 'ctpl_') !== 0 && +				strpos($entry, 'tpl_') !== 0) +			{ +				continue; +			} + +			$this->remove_file($this->cache_dir . $entry); +		} +		closedir($dir); + +		unset($this->vars); +		unset($this->sql_rowset); +		unset($this->sql_row_pointer); + +		$this->vars = array(); +		$this->sql_rowset = array(); +		$this->sql_row_pointer = array(); + +		$this->is_modified = false; +	} + + +	/** +	* {@inheritDoc} +	*/ +	function destroy($var_name, $table = '') +	{ +		if ($var_name == 'sql' && !empty($table)) +		{ +			if (!is_array($table)) +			{ +				$table = array($table); +			} + +			foreach ($table as $table_name) +			{ +				// gives us the md5s that we want +				$temp = $this->_read('sql_' . $table_name); + +				if ($temp === false) +				{ +					continue; +				} + +				// delete each query ref +				foreach ($temp as $md5_id => $void) +				{ +					$this->_delete('sql_' . $md5_id); +				} + +				// delete the table ref +				$this->_delete('sql_' . $table_name); +			} + +			return; +		} + +		if (!$this->_exists($var_name)) +		{ +			return; +		} + +		if ($var_name[0] == '_') +		{ +			$this->_delete($var_name); +		} +		else if (isset($this->vars[$var_name])) +		{ +			$this->is_modified = true; +			unset($this->vars[$var_name]); + +			// We save here to let the following cache hits succeed +			$this->save(); +		} +	} + +	/** +	* {@inheritDoc} +	*/ +	function _exists($var_name) +	{ +		if ($var_name[0] == '_') +		{ +			return $this->_isset($var_name); +		} +		else +		{ +			if (!sizeof($this->vars)) +			{ +				$this->load(); +			} + +			return isset($this->vars[$var_name]); +		} +	} + +	/** +	* {@inheritDoc} +	*/ +	function sql_load($query) +	{ +		// Remove extra spaces and tabs +		$query = preg_replace('/[\n\r\s\t]+/', ' ', $query); +		$query_id = sizeof($this->sql_rowset); + +		if (($result = $this->_read('sql_' . md5($query))) === false) +		{ +			return false; +		} + +		$this->sql_rowset[$query_id] = $result; +		$this->sql_row_pointer[$query_id] = 0; + +		return $query_id; +	} + +	/** +	* {@inheritDoc} +	*/ +	function sql_save(\phpbb\db\driver\driver_interface $db, $query, $query_result, $ttl) +	{ +		// Remove extra spaces and tabs +		$query = preg_replace('/[\n\r\s\t]+/', ' ', $query); +		$hash = md5($query); + +		// determine which tables this query belongs to +		// Some queries use backticks, namely the get_database_size() query +		// don't check for conformity, the SQL would error and not reach here. +		if (!preg_match('/FROM \\(?(`?\\w+`?(?: \\w+)?(?:, ?`?\\w+`?(?: \\w+)?)*)\\)?/', $query, $regs)) +		{ +			// Bail out if the match fails. +			return $query_result; +		} +		$tables = array_map('trim', explode(',', $regs[1])); + +		foreach ($tables as $table_name) +		{ +			// Remove backticks +			$table_name = ($table_name[0] == '`') ? substr($table_name, 1, -1) : $table_name; + +			if (($pos = strpos($table_name, ' ')) !== false) +			{ +				$table_name = substr($table_name, 0, $pos); +			} + +			$temp = $this->_read('sql_' . $table_name); + +			if ($temp === false) +			{ +				$temp = array(); +			} + +			$temp[$hash] = true; + +			// This must never expire +			$this->_write('sql_' . $table_name, $temp, 0); +		} + +		// store them in the right place +		$query_id = sizeof($this->sql_rowset); +		$this->sql_rowset[$query_id] = array(); +		$this->sql_row_pointer[$query_id] = 0; + +		while ($row = $db->sql_fetchrow($query_result)) +		{ +			$this->sql_rowset[$query_id][] = $row; +		} +		$db->sql_freeresult($query_result); + +		$this->_write('sql_' . $hash, $this->sql_rowset[$query_id], $ttl); + +		return $query_id; +	} + +	/** +	* {@inheritDoc} +	*/ +	function sql_exists($query_id) +	{ +		return isset($this->sql_rowset[$query_id]); +	} + +	/** +	* {@inheritDoc} +	*/ +	function sql_fetchrow($query_id) +	{ +		if ($this->sql_row_pointer[$query_id] < sizeof($this->sql_rowset[$query_id])) +		{ +			return $this->sql_rowset[$query_id][$this->sql_row_pointer[$query_id]++]; +		} + +		return false; +	} + +	/** +	* {@inheritDoc} +	*/ +	function sql_fetchfield($query_id, $field) +	{ +		if ($this->sql_row_pointer[$query_id] < sizeof($this->sql_rowset[$query_id])) +		{ +			return (isset($this->sql_rowset[$query_id][$this->sql_row_pointer[$query_id]][$field])) ? $this->sql_rowset[$query_id][$this->sql_row_pointer[$query_id]++][$field] : false; +		} + +		return false; +	} + +	/** +	* {@inheritDoc} +	*/ +	function sql_rowseek($rownum, $query_id) +	{ +		if ($rownum >= sizeof($this->sql_rowset[$query_id])) +		{ +			return false; +		} + +		$this->sql_row_pointer[$query_id] = $rownum; +		return true; +	} + +	/** +	* {@inheritDoc} +	*/ +	function sql_freeresult($query_id) +	{ +		if (!isset($this->sql_rowset[$query_id])) +		{ +			return false; +		} + +		unset($this->sql_rowset[$query_id]); +		unset($this->sql_row_pointer[$query_id]); + +		return true; +	} + +	/** +	* Removes/unlinks file +	* +	* @param string $filename Filename to remove +	* @param bool $check Check file permissions +	* @return bool True if the file was successfully removed, otherwise false +	*/ +	function remove_file($filename, $check = false) +	{ +		if (!function_exists('phpbb_is_writable')) +		{ +			global $phpbb_root_path, $phpEx; +			include($phpbb_root_path . 'includes/functions.' . $phpEx); +		} + +		if ($check && !phpbb_is_writable($this->cache_dir)) +		{ +			// E_USER_ERROR - not using language entry - intended. +			trigger_error('Unable to remove files within ' . $this->cache_dir . '. Please check directory permissions.', E_USER_ERROR); +		} + +		return @unlink($filename); +	} + +	/** +	* Check if a cache var exists +	* +	* @access protected +	* @param string $var Cache key +	* @return bool True if it exists, otherwise false +	*/ +	function _isset($var) +	{ +		// Most caches don't need to check +		return true; +	} +} diff --git a/phpBB/phpbb/cache/driver/null.php b/phpBB/phpbb/cache/driver/null.php new file mode 100644 index 0000000000..a45cf97862 --- /dev/null +++ b/phpBB/phpbb/cache/driver/null.php @@ -0,0 +1,151 @@ +<?php +/** +* +* This file is part of the phpBB Forum Software package. +* +* @copyright (c) phpBB Limited <https://www.phpbb.com> +* @license GNU General Public License, version 2 (GPL-2.0) +* +* For full copyright and license information, please see +* the docs/CREDITS.txt file. +* +*/ + +namespace phpbb\cache\driver; + +/** +* ACM Null Caching +*/ +class null extends \phpbb\cache\driver\base +{ +	/** +	* Set cache path +	*/ +	function __construct() +	{ +	} + +	/** +	* {@inheritDoc} +	*/ +	function load() +	{ +		return true; +	} + +	/** +	* {@inheritDoc} +	*/ +	function unload() +	{ +	} + +	/** +	* {@inheritDoc} +	*/ +	function save() +	{ +	} + +	/** +	* {@inheritDoc} +	*/ +	function tidy() +	{ +		// This cache always has a tidy room. +		set_config('cache_last_gc', time(), true); +	} + +	/** +	* {@inheritDoc} +	*/ +	function get($var_name) +	{ +		return false; +	} + +	/** +	* {@inheritDoc} +	*/ +	function put($var_name, $var, $ttl = 0) +	{ +	} + +	/** +	* {@inheritDoc} +	*/ +	function purge() +	{ +	} + +	/** +	* {@inheritDoc} +	*/ +	function destroy($var_name, $table = '') +	{ +	} + +	/** +	* {@inheritDoc} +	*/ +	function _exists($var_name) +	{ +		return false; +	} + +	/** +	* {@inheritDoc} +	*/ +	function sql_load($query) +	{ +		return false; +	} + +	/** +	* {@inheritDoc} +	*/ +	function sql_save(\phpbb\db\driver\driver_interface $db, $query, $query_result, $ttl) +	{ +		return $query_result; +	} + +	/** +	* {@inheritDoc} +	*/ +	function sql_exists($query_id) +	{ +		return false; +	} + +	/** +	* {@inheritDoc} +	*/ +	function sql_fetchrow($query_id) +	{ +		return false; +	} + +	/** +	* {@inheritDoc} +	*/ +	function sql_fetchfield($query_id, $field) +	{ +		return false; +	} + +	/** +	* {@inheritDoc} +	*/ +	function sql_rowseek($rownum, $query_id) +	{ +		return false; +	} + +	/** +	* {@inheritDoc} +	*/ +	function sql_freeresult($query_id) +	{ +		return false; +	} +} diff --git a/phpBB/phpbb/cache/driver/redis.php b/phpBB/phpbb/cache/driver/redis.php new file mode 100644 index 0000000000..eda774491c --- /dev/null +++ b/phpBB/phpbb/cache/driver/redis.php @@ -0,0 +1,158 @@ +<?php +/** +* +* This file is part of the phpBB Forum Software package. +* +* @copyright (c) phpBB Limited <https://www.phpbb.com> +* @license GNU General Public License, version 2 (GPL-2.0) +* +* For full copyright and license information, please see +* the docs/CREDITS.txt file. +* +*/ + +namespace phpbb\cache\driver; + +if (!defined('PHPBB_ACM_REDIS_PORT')) +{ +	define('PHPBB_ACM_REDIS_PORT', 6379); +} + +if (!defined('PHPBB_ACM_REDIS_HOST')) +{ +	define('PHPBB_ACM_REDIS_HOST', 'localhost'); +} + +/** +* ACM for Redis +* +* Compatible with the php extension phpredis available +* at https://github.com/nicolasff/phpredis +* +*/ +class redis extends \phpbb\cache\driver\memory +{ +	var $extension = 'redis'; + +	var $redis; + +	/** +	* Creates a redis cache driver. +	* +	* The following global constants affect operation: +	* +	* PHPBB_ACM_REDIS_HOST +	* PHPBB_ACM_REDIS_PORT +	* PHPBB_ACM_REDIS_PASSWORD +	* PHPBB_ACM_REDIS_DB +	* +	* There are no publicly documented constructor parameters. +	*/ +	function __construct() +	{ +		// Call the parent constructor +		parent::__construct(); + +		$this->redis = new \Redis(); + +		$args = func_get_args(); +		if (!empty($args)) +		{ +			$ok = call_user_func_array(array($this->redis, 'connect'), $args); +		} +		else +		{ +			$ok = $this->redis->connect(PHPBB_ACM_REDIS_HOST, PHPBB_ACM_REDIS_PORT); +		} + +		if (!$ok) +		{ +			trigger_error('Could not connect to redis server'); +		} + +		if (defined('PHPBB_ACM_REDIS_PASSWORD')) +		{ +			if (!$this->redis->auth(PHPBB_ACM_REDIS_PASSWORD)) +			{ +				global $acm_type; + +				trigger_error("Incorrect password for the ACM module $acm_type.", E_USER_ERROR); +			} +		} + +		$this->redis->setOption(\Redis::OPT_SERIALIZER, \Redis::SERIALIZER_PHP); +		$this->redis->setOption(\Redis::OPT_PREFIX, $this->key_prefix); + +		if (defined('PHPBB_ACM_REDIS_DB')) +		{ +			if (!$this->redis->select(PHPBB_ACM_REDIS_DB)) +			{ +				global $acm_type; + +				trigger_error("Incorrect database for the ACM module $acm_type.", E_USER_ERROR); +			} +		} +	} + +	/** +	* {@inheritDoc} +	*/ +	function unload() +	{ +		parent::unload(); + +		$this->redis->close(); +	} + +	/** +	* {@inheritDoc} +	*/ +	function purge() +	{ +		$this->redis->flushDB(); + +		parent::purge(); +	} + +	/** +	* Fetch an item from the cache +	* +	* @access protected +	* @param string $var Cache key +	* @return mixed Cached data +	*/ +	function _read($var) +	{ +		return $this->redis->get($var); +	} + +	/** +	* Store data in the cache +	* +	* @access protected +	* @param string $var Cache key +	* @param mixed $data Data to store +	* @param int $ttl Time-to-live of cached data +	* @return bool True if the operation succeeded +	*/ +	function _write($var, $data, $ttl = 2592000) +	{ +		return $this->redis->setex($var, $ttl, $data); +	} + +	/** +	* Remove an item from the cache +	* +	* @access protected +	* @param string $var Cache key +	* @return bool True if the operation succeeded +	*/ +	function _delete($var) +	{ +		if ($this->redis->delete($var) > 0) +		{ +			return true; +		} +		return false; +	} +} diff --git a/phpBB/phpbb/cache/driver/wincache.php b/phpBB/phpbb/cache/driver/wincache.php new file mode 100644 index 0000000000..632b534362 --- /dev/null +++ b/phpBB/phpbb/cache/driver/wincache.php @@ -0,0 +1,73 @@ +<?php +/** +* +* This file is part of the phpBB Forum Software package. +* +* @copyright (c) phpBB Limited <https://www.phpbb.com> +* @license GNU General Public License, version 2 (GPL-2.0) +* +* For full copyright and license information, please see +* the docs/CREDITS.txt file. +* +*/ + +namespace phpbb\cache\driver; + +/** +* ACM for WinCache +*/ +class wincache extends \phpbb\cache\driver\memory +{ +	var $extension = 'wincache'; + +	/** +	* {@inheritDoc} +	*/ +	function purge() +	{ +		wincache_ucache_clear(); + +		parent::purge(); +	} + +	/** +	* Fetch an item from the cache +	* +	* @access protected +	* @param string $var Cache key +	* @return mixed Cached data +	*/ +	function _read($var) +	{ +		$success = false; +		$result = wincache_ucache_get($this->key_prefix . $var, $success); + +		return ($success) ? $result : false; +	} + +	/** +	* Store data in the cache +	* +	* @access protected +	* @param string $var Cache key +	* @param mixed $data Data to store +	* @param int $ttl Time-to-live of cached data +	* @return bool True if the operation succeeded +	*/ +	function _write($var, $data, $ttl = 2592000) +	{ +		return wincache_ucache_set($this->key_prefix . $var, $data, $ttl); +	} + +	/** +	* Remove an item from the cache +	* +	* @access protected +	* @param string $var Cache key +	* @return bool True if the operation succeeded +	*/ +	function _delete($var) +	{ +		return wincache_ucache_delete($this->key_prefix . $var); +	} +} diff --git a/phpBB/phpbb/cache/driver/xcache.php b/phpBB/phpbb/cache/driver/xcache.php new file mode 100644 index 0000000000..0c845a6a8d --- /dev/null +++ b/phpBB/phpbb/cache/driver/xcache.php @@ -0,0 +1,107 @@ +<?php +/** +* +* This file is part of the phpBB Forum Software package. +* +* @copyright (c) phpBB Limited <https://www.phpbb.com> +* @license GNU General Public License, version 2 (GPL-2.0) +* +* For full copyright and license information, please see +* the docs/CREDITS.txt file. +* +*/ + +namespace phpbb\cache\driver; + +/** +* ACM for XCache +* +* To use this module you need ini_get() enabled and the following INI settings configured as follows: +* - xcache.var_size > 0 +* - xcache.admin.enable_auth = off (or xcache.admin.user and xcache.admin.password set) +* +*/ +class xcache extends \phpbb\cache\driver\memory +{ +	var $extension = 'XCache'; + +	function __construct() +	{ +		parent::__construct(); + +		if (!function_exists('ini_get') || (int) ini_get('xcache.var_size') <= 0) +		{ +			trigger_error('Increase xcache.var_size setting above 0 or enable ini_get() to use this ACM module.', E_USER_ERROR); +		} +	} + +	/** +	* {@inheritDoc} +	*/ +	function purge() +	{ +		// Run before for XCache, if admin functions are disabled it will terminate execution +		parent::purge(); + +		// If the admin authentication is enabled but not set up, this will cause a nasty error. +		// Not much we can do about it though. +		$n = xcache_count(XC_TYPE_VAR); + +		for ($i = 0; $i < $n; $i++) +		{ +			xcache_clear_cache(XC_TYPE_VAR, $i); +		} +	} + +	/** +	* Fetch an item from the cache +	* +	* @access protected +	* @param string $var Cache key +	* @return mixed Cached data +	*/ +	function _read($var) +	{ +		$result = xcache_get($this->key_prefix . $var); + +		return ($result !== null) ? $result : false; +	} + +	/** +	* Store data in the cache +	* +	* @access protected +	* @param string $var Cache key +	* @param mixed $data Data to store +	* @param int $ttl Time-to-live of cached data +	* @return bool True if the operation succeeded +	*/ +	function _write($var, $data, $ttl = 2592000) +	{ +		return xcache_set($this->key_prefix . $var, $data, $ttl); +	} + +	/** +	* Remove an item from the cache +	* +	* @access protected +	* @param string $var Cache key +	* @return bool True if the operation succeeded +	*/ +	function _delete($var) +	{ +		return xcache_unset($this->key_prefix . $var); +	} + +	/** +	* Check if a cache var exists +	* +	* @access protected +	* @param string $var Cache key +	* @return bool True if it exists, otherwise false +	*/ +	function _isset($var) +	{ +		return xcache_isset($this->key_prefix . $var); +	} +} diff --git a/phpBB/phpbb/cache/service.php b/phpBB/phpbb/cache/service.php new file mode 100644 index 0000000000..e47177758a --- /dev/null +++ b/phpBB/phpbb/cache/service.php @@ -0,0 +1,396 @@ +<?php +/** +* +* This file is part of the phpBB Forum Software package. +* +* @copyright (c) phpBB Limited <https://www.phpbb.com> +* @license GNU General Public License, version 2 (GPL-2.0) +* +* For full copyright and license information, please see +* the docs/CREDITS.txt file. +* +*/ + +namespace phpbb\cache; + +/** +* Class for grabbing/handling cached entries +*/ +class service +{ +	/** +	* Cache driver. +	* +	* @var \phpbb\cache\driver\driver_interface +	*/ +	protected $driver; + +	/** +	* The config. +	* +	* @var \phpbb\config\config +	*/ +	protected $config; + +	/** +	* Database connection. +	* +	* @var \phpbb\db\driver\driver_interface +	*/ +	protected $db; + +	/** +	* Root path. +	* +	* @var string +	*/ +	protected $phpbb_root_path; + +	/** +	* PHP extension. +	* +	* @var string +	*/ +	protected $php_ext; + +	/** +	* Creates a cache service around a cache driver +	* +	* @param \phpbb\cache\driver\driver_interface $driver The cache driver +	* @param \phpbb\config\config $config The config +	* @param \phpbb\db\driver\driver_interface $db Database connection +	* @param string $phpbb_root_path Root path +	* @param string $php_ext PHP extension +	*/ +	public function __construct(\phpbb\cache\driver\driver_interface $driver, \phpbb\config\config $config, \phpbb\db\driver\driver_interface $db, $phpbb_root_path, $php_ext) +	{ +		$this->set_driver($driver); +		$this->config = $config; +		$this->db = $db; +		$this->phpbb_root_path = $phpbb_root_path; +		$this->php_ext = $php_ext; +	} + +	/** +	* Returns the cache driver used by this cache service. +	* +	* @return \phpbb\cache\driver\driver_interface The cache driver +	*/ +	public function get_driver() +	{ +		return $this->driver; +	} + +	/** +	* Replaces the cache driver used by this cache service. +	* +	* @param \phpbb\cache\driver\driver_interface $driver The cache driver +	*/ +	public function set_driver(\phpbb\cache\driver\driver_interface $driver) +	{ +		$this->driver = $driver; +	} + +	public function __call($method, $arguments) +	{ +		return call_user_func_array(array($this->driver, $method), $arguments); +	} + +	/** +	* Obtain list of naughty words and build preg style replacement arrays for use by the +	* calling script +	*/ +	function obtain_word_list() +	{ +		if (($censors = $this->driver->get('_word_censors')) === false) +		{ +			$sql = 'SELECT word, replacement +				FROM ' . WORDS_TABLE; +			$result = $this->db->sql_query($sql); + +			$censors = array(); +			while ($row = $this->db->sql_fetchrow($result)) +			{ +				$censors['match'][] = get_censor_preg_expression($row['word']); +				$censors['replace'][] = $row['replacement']; +			} +			$this->db->sql_freeresult($result); + +			$this->driver->put('_word_censors', $censors); +		} + +		return $censors; +	} + +	/** +	* Obtain currently listed icons +	*/ +	function obtain_icons() +	{ +		if (($icons = $this->driver->get('_icons')) === false) +		{ +			// Topic icons +			$sql = 'SELECT * +				FROM ' . ICONS_TABLE . ' +				ORDER BY icons_order'; +			$result = $this->db->sql_query($sql); + +			$icons = array(); +			while ($row = $this->db->sql_fetchrow($result)) +			{ +				$icons[$row['icons_id']]['img'] = $row['icons_url']; +				$icons[$row['icons_id']]['width'] = (int) $row['icons_width']; +				$icons[$row['icons_id']]['height'] = (int) $row['icons_height']; +				$icons[$row['icons_id']]['display'] = (bool) $row['display_on_posting']; +			} +			$this->db->sql_freeresult($result); + +			$this->driver->put('_icons', $icons); +		} + +		return $icons; +	} + +	/** +	* Obtain ranks +	*/ +	function obtain_ranks() +	{ +		if (($ranks = $this->driver->get('_ranks')) === false) +		{ +			$sql = 'SELECT * +				FROM ' . RANKS_TABLE . ' +				ORDER BY rank_min DESC'; +			$result = $this->db->sql_query($sql); + +			$ranks = array(); +			while ($row = $this->db->sql_fetchrow($result)) +			{ +				if ($row['rank_special']) +				{ +					$ranks['special'][$row['rank_id']] = array( +						'rank_title'	=>	$row['rank_title'], +						'rank_image'	=>	$row['rank_image'] +					); +				} +				else +				{ +					$ranks['normal'][] = array( +						'rank_title'	=>	$row['rank_title'], +						'rank_min'		=>	$row['rank_min'], +						'rank_image'	=>	$row['rank_image'] +					); +				} +			} +			$this->db->sql_freeresult($result); + +			$this->driver->put('_ranks', $ranks); +		} + +		return $ranks; +	} + +	/** +	* Obtain allowed extensions +	* +	* @param mixed $forum_id If false then check for private messaging, if int then check for forum id. If true, then only return extension informations. +	* +	* @return array allowed extensions array. +	*/ +	function obtain_attach_extensions($forum_id) +	{ +		if (($extensions = $this->driver->get('_extensions')) === false) +		{ +			$extensions = array( +				'_allowed_post'	=> array(), +				'_allowed_pm'	=> array(), +			); + +			// The rule is to only allow those extensions defined. ;) +			$sql = 'SELECT e.extension, g.* +				FROM ' . EXTENSIONS_TABLE . ' e, ' . EXTENSION_GROUPS_TABLE . ' g +				WHERE e.group_id = g.group_id +					AND (g.allow_group = 1 OR g.allow_in_pm = 1)'; +			$result = $this->db->sql_query($sql); + +			while ($row = $this->db->sql_fetchrow($result)) +			{ +				$extension = strtolower(trim($row['extension'])); + +				$extensions[$extension] = array( +					'display_cat'	=> (int) $row['cat_id'], +					'download_mode'	=> (int) $row['download_mode'], +					'upload_icon'	=> trim($row['upload_icon']), +					'max_filesize'	=> (int) $row['max_filesize'], +					'allow_group'	=> $row['allow_group'], +					'allow_in_pm'	=> $row['allow_in_pm'], +					'group_name'	=> $row['group_name'], +				); + +				$allowed_forums = ($row['allowed_forums']) ? unserialize(trim($row['allowed_forums'])) : array(); + +				// Store allowed extensions forum wise +				if ($row['allow_group']) +				{ +					$extensions['_allowed_post'][$extension] = (!sizeof($allowed_forums)) ? 0 : $allowed_forums; +				} + +				if ($row['allow_in_pm']) +				{ +					$extensions['_allowed_pm'][$extension] = 0; +				} +			} +			$this->db->sql_freeresult($result); + +			$this->driver->put('_extensions', $extensions); +		} + +		// Forum post +		if ($forum_id === false) +		{ +			// We are checking for private messages, therefore we only need to get the pm extensions... +			$return = array('_allowed_' => array()); + +			foreach ($extensions['_allowed_pm'] as $extension => $check) +			{ +				$return['_allowed_'][$extension] = 0; +				$return[$extension] = $extensions[$extension]; +			} + +			$extensions = $return; +		} +		else if ($forum_id === true) +		{ +			return $extensions; +		} +		else +		{ +			$forum_id = (int) $forum_id; +			$return = array('_allowed_' => array()); + +			foreach ($extensions['_allowed_post'] as $extension => $check) +			{ +				// Check for allowed forums +				if (is_array($check)) +				{ +					$allowed = (!in_array($forum_id, $check)) ? false : true; +				} +				else +				{ +					$allowed = true; +				} + +				if ($allowed) +				{ +					$return['_allowed_'][$extension] = 0; +					$return[$extension] = $extensions[$extension]; +				} +			} + +			$extensions = $return; +		} + +		if (!isset($extensions['_allowed_'])) +		{ +			$extensions['_allowed_'] = array(); +		} + +		return $extensions; +	} + +	/** +	* Obtain active bots +	*/ +	function obtain_bots() +	{ +		if (($bots = $this->driver->get('_bots')) === false) +		{ +			switch ($this->db->sql_layer) +			{ +				case 'mssql': +				case 'mssql_odbc': +				case 'mssqlnative': +					$sql = 'SELECT user_id, bot_agent, bot_ip +						FROM ' . BOTS_TABLE . ' +						WHERE bot_active = 1 +					ORDER BY LEN(bot_agent) DESC'; +				break; + +				// LENGTH supported by MySQL, IBM DB2 and Oracle for sure... +				default: +					$sql = 'SELECT user_id, bot_agent, bot_ip +						FROM ' . BOTS_TABLE . ' +						WHERE bot_active = 1 +					ORDER BY LENGTH(bot_agent) DESC'; +				break; +			} +			$result = $this->db->sql_query($sql); + +			$bots = array(); +			while ($row = $this->db->sql_fetchrow($result)) +			{ +				$bots[] = $row; +			} +			$this->db->sql_freeresult($result); + +			$this->driver->put('_bots', $bots); +		} + +		return $bots; +	} + +	/** +	* Obtain cfg file data +	*/ +	function obtain_cfg_items($style) +	{ +		$parsed_array = $this->driver->get('_cfg_' . $style['style_path']); + +		if ($parsed_array === false) +		{ +			$parsed_array = array(); +		} + +		$filename = $this->phpbb_root_path . 'styles/' . $style['style_path'] . '/style.cfg'; + +		if (!file_exists($filename)) +		{ +			return $parsed_array; +		} + +		if (!isset($parsed_array['filetime']) || (($this->config['load_tplcompile'] && @filemtime($filename) > $parsed_array['filetime']))) +		{ +			// Re-parse cfg file +			$parsed_array = parse_cfg_file($filename); +			$parsed_array['filetime'] = @filemtime($filename); + +			$this->driver->put('_cfg_' . $style['style_path'], $parsed_array); +		} + +		return $parsed_array; +	} + +	/** +	* Obtain disallowed usernames +	*/ +	function obtain_disallowed_usernames() +	{ +		if (($usernames = $this->driver->get('_disallowed_usernames')) === false) +		{ +			$sql = 'SELECT disallow_username +				FROM ' . DISALLOW_TABLE; +			$result = $this->db->sql_query($sql); + +			$usernames = array(); +			while ($row = $this->db->sql_fetchrow($result)) +			{ +				$usernames[] = str_replace('%', '.*?', preg_quote(utf8_clean_string($row['disallow_username']), '#')); +			} +			$this->db->sql_freeresult($result); + +			$this->driver->put('_disallowed_usernames', $usernames); +		} + +		return $usernames; +	} +} | 
