|  |  |  | <?php
 | 
					
						
							|  |  |  | /**
 | 
					
						
							|  |  |  |  * DirectoryDependency class file.
 | 
					
						
							|  |  |  |  *
 | 
					
						
							|  |  |  |  * @link http://www.yiiframework.com/
 | 
					
						
							|  |  |  |  * @copyright Copyright © 2008-2012 Yii Software LLC
 | 
					
						
							|  |  |  |  * @license http://www.yiiframework.com/license/
 | 
					
						
							|  |  |  |  */
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | namespace yii\caching;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /**
 | 
					
						
							|  |  |  |  * DirectoryDependency represents a dependency based on change of a directory.
 | 
					
						
							|  |  |  |  *
 | 
					
						
							|  |  |  |  * DirectoryDependency performs dependency checking based on the
 | 
					
						
							|  |  |  |  * modification time of the files contained in the specified directory.
 | 
					
						
							|  |  |  |  * The directory being checked is specified via {@link directory}.
 | 
					
						
							|  |  |  |  *
 | 
					
						
							|  |  |  |  * By default, all files under the specified directory and subdirectories
 | 
					
						
							|  |  |  |  * will be checked. If the last modification time of any of them is changed
 | 
					
						
							|  |  |  |  * or if different number of files are contained in a directory, the dependency
 | 
					
						
							|  |  |  |  * is reported as changed. By specifying {@link recursiveLevel},
 | 
					
						
							|  |  |  |  * one can limit the checking to a certain depth of the directory.
 | 
					
						
							|  |  |  |  *
 | 
					
						
							|  |  |  |  * Note, dependency checking for a directory is expensive because it involves
 | 
					
						
							|  |  |  |  * accessing modification time of multiple files under the directory.
 | 
					
						
							|  |  |  |  *
 | 
					
						
							|  |  |  |  * @author Qiang Xue <qiang.xue@gmail.com>
 | 
					
						
							|  |  |  |  * @since 2.0
 | 
					
						
							|  |  |  |  */
 | 
					
						
							|  |  |  | class DirectoryDependency extends Dependency
 | 
					
						
							|  |  |  | {
 | 
					
						
							|  |  |  | 	/**
 | 
					
						
							|  |  |  | 	 * @var string the directory whose change is used to determine if the dependency has been changed.
 | 
					
						
							|  |  |  | 	 * If any of the files under the directory is changed, the dependency is considered as changed.
 | 
					
						
							|  |  |  | 	 */
 | 
					
						
							|  |  |  | 	public $directory;
 | 
					
						
							|  |  |  | 	/**
 | 
					
						
							|  |  |  | 	 * @var integer the depth of the subdirectories to be recursively checked.
 | 
					
						
							|  |  |  | 	 * If the value is less than 0, it means unlimited depth.
 | 
					
						
							|  |  |  | 	 * If the value is 0, it means checking the files directly under the specified directory.
 | 
					
						
							|  |  |  | 	 */
 | 
					
						
							|  |  |  | 	public $recursiveLevel = -1;
 | 
					
						
							|  |  |  | 	/**
 | 
					
						
							|  |  |  | 	 * @var string the regular expression matching valid file/directory names.
 | 
					
						
							|  |  |  | 	 * Only the matching files or directories will be checked for changes.
 | 
					
						
							|  |  |  | 	 * Defaults to null, meaning all files/directories will qualify.
 | 
					
						
							|  |  |  | 	 */
 | 
					
						
							|  |  |  | 	public $namePattern;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/**
 | 
					
						
							|  |  |  | 	 * Constructor.
 | 
					
						
							|  |  |  | 	 * @param string $directory the directory to be checked
 | 
					
						
							|  |  |  | 	 */
 | 
					
						
							|  |  |  | 	public function __construct($directory = null)
 | 
					
						
							|  |  |  | 	{
 | 
					
						
							|  |  |  | 		$this->directory = $directory;
 | 
					
						
							|  |  |  | 	}
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/**
 | 
					
						
							|  |  |  | 	 * Generates the data needed to determine if dependency has been changed.
 | 
					
						
							|  |  |  | 	 * This method returns the modification timestamps for files under the directory.
 | 
					
						
							|  |  |  | 	 * @return mixed the data needed to determine if dependency has been changed.
 | 
					
						
							|  |  |  | 	 */
 | 
					
						
							|  |  |  | 	protected function generateDependencyData()
 | 
					
						
							|  |  |  | 	{
 | 
					
						
							|  |  |  | 		if ($this->directory !== null) {
 | 
					
						
							|  |  |  | 			return $this->generateTimestamps($this->directory);
 | 
					
						
							|  |  |  | 		} else {
 | 
					
						
							|  |  |  | 			throw new CException(Yii::t('yii', 'DirectoryDependency.directory cannot be empty.'));
 | 
					
						
							|  |  |  | 		}
 | 
					
						
							|  |  |  | 	}
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/**
 | 
					
						
							|  |  |  | 	 * Determines the last modification time for files under the directory.
 | 
					
						
							|  |  |  | 	 * This method may go recursively into subdirectories if {@link recursiveLevel} is not 0.
 | 
					
						
							|  |  |  | 	 * @param string $directory the directory name
 | 
					
						
							|  |  |  | 	 * @param integer $level level of the recursion
 | 
					
						
							|  |  |  | 	 * @return array list of file modification time indexed by the file path
 | 
					
						
							|  |  |  | 	 */
 | 
					
						
							|  |  |  | 	protected function generateTimestamps($directory, $level = 0)
 | 
					
						
							|  |  |  | 	{
 | 
					
						
							|  |  |  | 		if (($dir = @opendir($directory)) === false) {
 | 
					
						
							|  |  |  | 			throw new CException(Yii::t('yii', '"{path}" is not a valid directory.',
 | 
					
						
							|  |  |  | 				array('{path}' => $directory)));
 | 
					
						
							|  |  |  | 		}
 | 
					
						
							|  |  |  | 		$timestamps = array();
 | 
					
						
							|  |  |  | 		while (($file = readdir($dir)) !== false) {
 | 
					
						
							|  |  |  | 			$path = $directory . DIRECTORY_SEPARATOR . $file;
 | 
					
						
							|  |  |  | 			if ($file === '.' || $file === '..') {
 | 
					
						
							|  |  |  | 				continue;
 | 
					
						
							|  |  |  | 			}
 | 
					
						
							|  |  |  | 			if ($this->namePattern !== null && !preg_match($this->namePattern, $file)) {
 | 
					
						
							|  |  |  | 				continue;
 | 
					
						
							|  |  |  | 			}
 | 
					
						
							|  |  |  | 			if (is_file($path)) {
 | 
					
						
							|  |  |  | 				if ($this->validateFile($path)) {
 | 
					
						
							|  |  |  | 					$timestamps[$path] = filemtime($path);
 | 
					
						
							|  |  |  | 				}
 | 
					
						
							|  |  |  | 			} else {
 | 
					
						
							|  |  |  | 				if (($this->recursiveLevel < 0 || $level < $this->recursiveLevel) && $this->validateDirectory($path)) {
 | 
					
						
							|  |  |  | 					$timestamps = array_merge($timestamps, $this->generateTimestamps($path, $level + 1));
 | 
					
						
							|  |  |  | 				}
 | 
					
						
							|  |  |  | 			}
 | 
					
						
							|  |  |  | 		}
 | 
					
						
							|  |  |  | 		closedir($dir);
 | 
					
						
							|  |  |  | 		return $timestamps;
 | 
					
						
							|  |  |  | 	}
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/**
 | 
					
						
							|  |  |  | 	 * Checks to see if the file should be checked for dependency.
 | 
					
						
							|  |  |  | 	 * This method is invoked when dependency of the whole directory is being checked.
 | 
					
						
							|  |  |  | 	 * By default, it always returns true, meaning the file should be checked.
 | 
					
						
							|  |  |  | 	 * You may override this method to check only certain files.
 | 
					
						
							|  |  |  | 	 * @param string $fileName the name of the file that may be checked for dependency.
 | 
					
						
							|  |  |  | 	 * @return boolean whether this file should be checked.
 | 
					
						
							|  |  |  | 	 */
 | 
					
						
							|  |  |  | 	protected function validateFile($fileName)
 | 
					
						
							|  |  |  | 	{
 | 
					
						
							|  |  |  | 		return true;
 | 
					
						
							|  |  |  | 	}
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/**
 | 
					
						
							|  |  |  | 	 * Checks to see if the specified subdirectory should be checked for dependency.
 | 
					
						
							|  |  |  | 	 * This method is invoked when dependency of the whole directory is being checked.
 | 
					
						
							|  |  |  | 	 * By default, it always returns true, meaning the subdirectory should be checked.
 | 
					
						
							|  |  |  | 	 * You may override this method to check only certain subdirectories.
 | 
					
						
							|  |  |  | 	 * @param string $directory the name of the subdirectory that may be checked for dependency.
 | 
					
						
							|  |  |  | 	 * @return boolean whether this subdirectory should be checked.
 | 
					
						
							|  |  |  | 	 */
 | 
					
						
							|  |  |  | 	protected function validateDirectory($directory)
 | 
					
						
							|  |  |  | 	{
 | 
					
						
							|  |  |  | 		return true;
 | 
					
						
							|  |  |  | 	}
 | 
					
						
							|  |  |  | }
 |