pictcode / lib / Cake / Cache / Engine / FileEngine.php @ 9ddbf630
履歴 | 表示 | アノテート | ダウンロード (11.108 KB)
1 |
<?php
|
---|---|
2 |
/**
|
3 |
* File Storage engine for cache. Filestorage is the slowest cache storage
|
4 |
* to read and write. However, it is good for servers that don't have other storage
|
5 |
* engine available, or have content which is not performance sensitive.
|
6 |
*
|
7 |
* You can configure a FileEngine cache, using Cache::config()
|
8 |
*
|
9 |
* CakePHP(tm) : Rapid Development Framework (http://cakephp.org)
|
10 |
* Copyright (c) Cake Software Foundation, Inc. (http://cakefoundation.org)
|
11 |
*
|
12 |
* Licensed under The MIT License
|
13 |
* For full copyright and license information, please see the LICENSE.txt
|
14 |
* Redistributions of files must retain the above copyright notice.
|
15 |
*
|
16 |
* @copyright Copyright (c) Cake Software Foundation, Inc. (http://cakefoundation.org)
|
17 |
* @link http://cakephp.org CakePHP(tm) Project
|
18 |
* @since CakePHP(tm) v 1.2.0.4933
|
19 |
* @license http://www.opensource.org/licenses/mit-license.php MIT License
|
20 |
*/
|
21 |
|
22 |
/**
|
23 |
* File Storage engine for cache. Filestorage is the slowest cache storage
|
24 |
* to read and write. However, it is good for servers that don't have other storage
|
25 |
* engine available, or have content which is not performance sensitive.
|
26 |
*
|
27 |
* You can configure a FileEngine cache, using Cache::config()
|
28 |
*
|
29 |
* @package Cake.Cache.Engine
|
30 |
*/
|
31 |
class FileEngine extends CacheEngine { |
32 |
|
33 |
/**
|
34 |
* Instance of SplFileObject class
|
35 |
*
|
36 |
* @var File
|
37 |
*/
|
38 |
protected $_File = null; |
39 |
|
40 |
/**
|
41 |
* Settings
|
42 |
*
|
43 |
* - path = absolute path to cache directory, default => CACHE
|
44 |
* - prefix = string prefix for filename, default => cake_
|
45 |
* - lock = enable file locking on write, default => true
|
46 |
* - serialize = serialize the data, default => true
|
47 |
*
|
48 |
* @var array
|
49 |
* @see CacheEngine::__defaults
|
50 |
*/
|
51 |
public $settings = array(); |
52 |
|
53 |
/**
|
54 |
* True unless FileEngine::__active(); fails
|
55 |
*
|
56 |
* @var bool
|
57 |
*/
|
58 |
protected $_init = true; |
59 |
|
60 |
/**
|
61 |
* Initialize the Cache Engine
|
62 |
*
|
63 |
* Called automatically by the cache frontend
|
64 |
* To reinitialize the settings call Cache::engine('EngineName', [optional] settings = array());
|
65 |
*
|
66 |
* @param array $settings array of setting for the engine
|
67 |
* @return bool True if the engine has been successfully initialized, false if not
|
68 |
*/
|
69 |
public function init($settings = array()) { |
70 |
$settings += array( |
71 |
'engine' => 'File', |
72 |
'path' => CACHE, |
73 |
'prefix' => 'cake_', |
74 |
'lock' => true, |
75 |
'serialize' => true, |
76 |
'isWindows' => false, |
77 |
'mask' => 0664 |
78 |
); |
79 |
parent::init($settings); |
80 |
|
81 |
if (DS === '\\') { |
82 |
$this->settings['isWindows'] = true; |
83 |
} |
84 |
if (substr($this->settings['path'], -1) !== DS) { |
85 |
$this->settings['path'] .= DS; |
86 |
} |
87 |
if (!empty($this->_groupPrefix)) { |
88 |
$this->_groupPrefix = str_replace('_', DS, $this->_groupPrefix); |
89 |
} |
90 |
return $this->_active(); |
91 |
} |
92 |
|
93 |
/**
|
94 |
* Garbage collection. Permanently remove all expired and deleted data
|
95 |
*
|
96 |
* @param int $expires [optional] An expires timestamp, invalidating all data before.
|
97 |
* @return bool True if garbage collection was successful, false on failure
|
98 |
*/
|
99 |
public function gc($expires = null) { |
100 |
return $this->clear(true); |
101 |
} |
102 |
|
103 |
/**
|
104 |
* Write data for key into cache
|
105 |
*
|
106 |
* @param string $key Identifier for the data
|
107 |
* @param mixed $data Data to be cached
|
108 |
* @param int $duration How long to cache the data, in seconds
|
109 |
* @return bool True if the data was successfully cached, false on failure
|
110 |
*/
|
111 |
public function write($key, $data, $duration) { |
112 |
if ($data === '' || !$this->_init) { |
113 |
return false; |
114 |
} |
115 |
|
116 |
if ($this->_setKey($key, true) === false) { |
117 |
return false; |
118 |
} |
119 |
|
120 |
$lineBreak = "\n"; |
121 |
|
122 |
if ($this->settings['isWindows']) { |
123 |
$lineBreak = "\r\n"; |
124 |
} |
125 |
|
126 |
if (!empty($this->settings['serialize'])) { |
127 |
if ($this->settings['isWindows']) { |
128 |
$data = str_replace('\\', '\\\\\\\\', serialize($data)); |
129 |
} else {
|
130 |
$data = serialize($data); |
131 |
} |
132 |
} |
133 |
|
134 |
$expires = time() + $duration; |
135 |
$contents = $expires . $lineBreak . $data . $lineBreak; |
136 |
|
137 |
if ($this->settings['lock']) { |
138 |
$this->_File->flock(LOCK_EX); |
139 |
} |
140 |
|
141 |
$this->_File->rewind(); |
142 |
$success = $this->_File->ftruncate(0) && $this->_File->fwrite($contents) && $this->_File->fflush(); |
143 |
|
144 |
if ($this->settings['lock']) { |
145 |
$this->_File->flock(LOCK_UN); |
146 |
} |
147 |
|
148 |
return $success; |
149 |
} |
150 |
|
151 |
/**
|
152 |
* Read a key from the cache
|
153 |
*
|
154 |
* @param string $key Identifier for the data
|
155 |
* @return mixed The cached data, or false if the data doesn't exist, has expired, or if there was an error fetching it
|
156 |
*/
|
157 |
public function read($key) { |
158 |
if (!$this->_init || $this->_setKey($key) === false) { |
159 |
return false; |
160 |
} |
161 |
|
162 |
if ($this->settings['lock']) { |
163 |
$this->_File->flock(LOCK_SH); |
164 |
} |
165 |
|
166 |
$this->_File->rewind(); |
167 |
$time = time(); |
168 |
$cachetime = (int)$this->_File->current(); |
169 |
|
170 |
if ($cachetime !== false && ($cachetime < $time || ($time + $this->settings['duration']) < $cachetime)) { |
171 |
if ($this->settings['lock']) { |
172 |
$this->_File->flock(LOCK_UN); |
173 |
} |
174 |
return false; |
175 |
} |
176 |
|
177 |
$data = ''; |
178 |
$this->_File->next(); |
179 |
while ($this->_File->valid()) { |
180 |
$data .= $this->_File->current(); |
181 |
$this->_File->next(); |
182 |
} |
183 |
|
184 |
if ($this->settings['lock']) { |
185 |
$this->_File->flock(LOCK_UN); |
186 |
} |
187 |
|
188 |
$data = trim($data); |
189 |
|
190 |
if ($data !== '' && !empty($this->settings['serialize'])) { |
191 |
if ($this->settings['isWindows']) { |
192 |
$data = str_replace('\\\\\\\\', '\\', $data); |
193 |
} |
194 |
$data = unserialize((string)$data); |
195 |
} |
196 |
return $data; |
197 |
} |
198 |
|
199 |
/**
|
200 |
* Delete a key from the cache
|
201 |
*
|
202 |
* @param string $key Identifier for the data
|
203 |
* @return bool True if the value was successfully deleted, false if it didn't exist or couldn't be removed
|
204 |
*/
|
205 |
public function delete($key) { |
206 |
if ($this->_setKey($key) === false || !$this->_init) { |
207 |
return false; |
208 |
} |
209 |
$path = $this->_File->getRealPath(); |
210 |
$this->_File = null; |
211 |
|
212 |
//@codingStandardsIgnoreStart
|
213 |
return @unlink($path); |
214 |
//@codingStandardsIgnoreEnd
|
215 |
} |
216 |
|
217 |
/**
|
218 |
* Delete all values from the cache
|
219 |
*
|
220 |
* @param bool $check Optional - only delete expired cache items
|
221 |
* @return bool True if the cache was successfully cleared, false otherwise
|
222 |
*/
|
223 |
public function clear($check) { |
224 |
if (!$this->_init) { |
225 |
return false; |
226 |
} |
227 |
$this->_File = null; |
228 |
|
229 |
$threshold = $now = false; |
230 |
if ($check) { |
231 |
$now = time(); |
232 |
$threshold = $now - $this->settings['duration']; |
233 |
} |
234 |
|
235 |
$this->_clearDirectory($this->settings['path'], $now, $threshold); |
236 |
|
237 |
$directory = new RecursiveDirectoryIterator($this->settings['path']); |
238 |
$contents = new RecursiveIteratorIterator($directory, RecursiveIteratorIterator::SELF_FIRST); |
239 |
$cleared = array(); |
240 |
foreach ($contents as $path) { |
241 |
if ($path->isFile()) { |
242 |
continue;
|
243 |
} |
244 |
|
245 |
$path = $path->getRealPath() . DS; |
246 |
if (!in_array($path, $cleared)) { |
247 |
$this->_clearDirectory($path, $now, $threshold); |
248 |
$cleared[] = $path; |
249 |
} |
250 |
} |
251 |
return true; |
252 |
} |
253 |
|
254 |
/**
|
255 |
* Used to clear a directory of matching files.
|
256 |
*
|
257 |
* @param string $path The path to search.
|
258 |
* @param int $now The current timestamp
|
259 |
* @param int $threshold Any file not modified after this value will be deleted.
|
260 |
* @return void
|
261 |
*/
|
262 |
protected function _clearDirectory($path, $now, $threshold) { |
263 |
$prefixLength = strlen($this->settings['prefix']); |
264 |
|
265 |
if (!is_dir($path)) { |
266 |
return;
|
267 |
} |
268 |
|
269 |
$dir = dir($path); |
270 |
while (($entry = $dir->read()) !== false) { |
271 |
if (substr($entry, 0, $prefixLength) !== $this->settings['prefix']) { |
272 |
continue;
|
273 |
} |
274 |
|
275 |
try {
|
276 |
$file = new SplFileObject($path . $entry, 'r'); |
277 |
} catch (Exception $e) { |
278 |
continue;
|
279 |
} |
280 |
|
281 |
if ($threshold) { |
282 |
$mtime = $file->getMTime(); |
283 |
|
284 |
if ($mtime > $threshold) { |
285 |
continue;
|
286 |
} |
287 |
$expires = (int)$file->current(); |
288 |
|
289 |
if ($expires > $now) { |
290 |
continue;
|
291 |
} |
292 |
} |
293 |
if ($file->isFile()) { |
294 |
$filePath = $file->getRealPath(); |
295 |
$file = null; |
296 |
|
297 |
//@codingStandardsIgnoreStart
|
298 |
@unlink($filePath); |
299 |
//@codingStandardsIgnoreEnd
|
300 |
} |
301 |
} |
302 |
} |
303 |
|
304 |
/**
|
305 |
* Not implemented
|
306 |
*
|
307 |
* @param string $key The key to decrement
|
308 |
* @param int $offset The number to offset
|
309 |
* @return void
|
310 |
* @throws CacheException
|
311 |
*/
|
312 |
public function decrement($key, $offset = 1) { |
313 |
throw new CacheException(__d('cake_dev', 'Files cannot be atomically decremented.')); |
314 |
} |
315 |
|
316 |
/**
|
317 |
* Not implemented
|
318 |
*
|
319 |
* @param string $key The key to decrement
|
320 |
* @param int $offset The number to offset
|
321 |
* @return void
|
322 |
* @throws CacheException
|
323 |
*/
|
324 |
public function increment($key, $offset = 1) { |
325 |
throw new CacheException(__d('cake_dev', 'Files cannot be atomically incremented.')); |
326 |
} |
327 |
|
328 |
/**
|
329 |
* Sets the current cache key this class is managing, and creates a writable SplFileObject
|
330 |
* for the cache file the key is referring to.
|
331 |
*
|
332 |
* @param string $key The key
|
333 |
* @param bool $createKey Whether the key should be created if it doesn't exists, or not
|
334 |
* @return bool true if the cache key could be set, false otherwise
|
335 |
*/
|
336 |
protected function _setKey($key, $createKey = false) { |
337 |
$groups = null; |
338 |
if (!empty($this->_groupPrefix)) { |
339 |
$groups = vsprintf($this->_groupPrefix, $this->groups()); |
340 |
} |
341 |
$dir = $this->settings['path'] . $groups; |
342 |
|
343 |
if (!is_dir($dir)) { |
344 |
mkdir($dir, 0775, true); |
345 |
} |
346 |
$path = new SplFileInfo($dir . $key); |
347 |
|
348 |
if (!$createKey && !$path->isFile()) { |
349 |
return false; |
350 |
} |
351 |
if (empty($this->_File) || $this->_File->getBaseName() !== $key) { |
352 |
$exists = file_exists($path->getPathname()); |
353 |
try {
|
354 |
$this->_File = $path->openFile('c+'); |
355 |
} catch (Exception $e) { |
356 |
trigger_error($e->getMessage(), E_USER_WARNING); |
357 |
return false; |
358 |
} |
359 |
unset($path); |
360 |
|
361 |
if (!$exists && !chmod($this->_File->getPathname(), (int)$this->settings['mask'])) { |
362 |
trigger_error(__d(
|
363 |
'cake_dev', 'Could not apply permission mask "%s" on cache file "%s"', |
364 |
array($this->_File->getPathname(), $this->settings['mask'])), E_USER_WARNING); |
365 |
} |
366 |
} |
367 |
return true; |
368 |
} |
369 |
|
370 |
/**
|
371 |
* Determine is cache directory is writable
|
372 |
*
|
373 |
* @return bool
|
374 |
*/
|
375 |
protected function _active() { |
376 |
$dir = new SplFileInfo($this->settings['path']); |
377 |
if (Configure::read('debug')) { |
378 |
$path = $dir->getPathname(); |
379 |
if (!is_dir($path)) { |
380 |
mkdir($path, 0775, true); |
381 |
} |
382 |
} |
383 |
if ($this->_init && !($dir->isDir() && $dir->isWritable())) { |
384 |
$this->_init = false; |
385 |
trigger_error(__d('cake_dev', '%s is not writable', $this->settings['path']), E_USER_WARNING); |
386 |
return false; |
387 |
} |
388 |
return true; |
389 |
} |
390 |
|
391 |
/**
|
392 |
* Generates a safe key for use with cache engine storage engines.
|
393 |
*
|
394 |
* @param string $key the key passed over
|
395 |
* @return mixed string $key or false
|
396 |
*/
|
397 |
public function key($key) { |
398 |
if (empty($key)) { |
399 |
return false; |
400 |
} |
401 |
|
402 |
$key = Inflector::underscore(str_replace(array(DS, '/', '.', '<', '>', '?', ':', '|', '*', '"'), '_', strval($key))); |
403 |
return $key; |
404 |
} |
405 |
|
406 |
/**
|
407 |
* Recursively deletes all files under any directory named as $group
|
408 |
*
|
409 |
* @param string $group The group to clear.
|
410 |
* @return bool success
|
411 |
*/
|
412 |
public function clearGroup($group) { |
413 |
$this->_File = null; |
414 |
$directoryIterator = new RecursiveDirectoryIterator($this->settings['path']); |
415 |
$contents = new RecursiveIteratorIterator($directoryIterator, RecursiveIteratorIterator::CHILD_FIRST); |
416 |
foreach ($contents as $object) { |
417 |
$containsGroup = strpos($object->getPathName(), DS . $group . DS) !== false; |
418 |
$hasPrefix = true; |
419 |
if (strlen($this->settings['prefix']) !== 0) { |
420 |
$hasPrefix = strpos($object->getBaseName(), $this->settings['prefix']) === 0; |
421 |
} |
422 |
if ($object->isFile() && $containsGroup && $hasPrefix) { |
423 |
$path = $object->getPathName(); |
424 |
$object = null; |
425 |
//@codingStandardsIgnoreStart
|
426 |
@unlink($path); |
427 |
//@codingStandardsIgnoreEnd
|
428 |
} |
429 |
} |
430 |
return true; |
431 |
} |
432 |
} |