blob: 4acbcf6c50de67d94a1411aa39f39b5276196395 [file] [log] [blame]
Andrey Andreev47a47fb2014-05-31 16:08:30 +03001<?php
2/**
3 * CodeIgniter
4 *
5 * An open source application development framework for PHP 5.2.4 or newer
6 *
7 * NOTICE OF LICENSE
8 *
9 * Licensed under the Open Software License version 3.0
10 *
11 * This source file is subject to the Open Software License (OSL 3.0) that is
12 * bundled with this package in the files license.txt / license.rst. It is
13 * also available through the world wide web at this URL:
14 * http://opensource.org/licenses/OSL-3.0
15 * If you did not receive a copy of the license and are unable to obtain it
16 * through the world wide web, please send an email to
17 * licensing@ellislab.com so we can send you a copy immediately.
18 *
19 * @package CodeIgniter
20 * @author Andrey Andreev
21 * @copyright Copyright (c) 2008 - 2014, EllisLab, Inc. (http://ellislab.com/)
22 * @license http://opensource.org/licenses/OSL-3.0 Open Software License (OSL 3.0)
23 * @link http://codeigniter.com
24 * @since Version 3.0
25 * @filesource
26 */
27defined('BASEPATH') OR exit('No direct script access allowed');
28
29/**
30 * CodeIgniter Session Files Driver
31 *
32 * @package CodeIgniter
33 * @subpackage Libraries
34 * @category Sessions
35 * @author Andrey Andreev
36 * @link http://codeigniter.com/user_guide/libraries/sessions.html
37 */
38class CI_Session_files_driver extends CI_Session_driver implements SessionHandlerInterface {
39
40 /**
41 * Save path
42 *
43 * @var string
44 */
45 protected $_save_path;
46
47 /**
48 * File handle
49 *
50 * @var resource
51 */
52 protected $_file_handle;
53
54 /**
55 * File name
56 *
57 * @var resource
58 */
59 protected $_file_path;
60
61 /**
62 * File new flag
63 *
64 * @var bool
65 */
66 protected $_file_new;
67
68 // ------------------------------------------------------------------------
69
70 /**
71 * Class constructor
72 *
73 * @param array $params Configuration parameters
74 * @return void
75 */
76 public function __construct(&$params)
77 {
78 parent::__construct($params);
79
80 if (isset($this->_save_path))
81 {
82 $this->_save_path = rtrim($this->_save_path, '/\\');
83 ini_set('session.save_path', $this->_save_path);
84 }
85 else
86 {
87 $this->_save_path = rtrim(ini_get('session.save_path'), '/\\');
88 }
89 }
90
91 // ------------------------------------------------------------------------
92
93 public function open($save_path, $name)
94 {
95 if ( ! is_dir($save_path) && ! mkdir($save_path, 0700, TRUE))
96 {
97 log_message('error', "Session: Configured save path '".$this->_save_path."' is not a directory, doesn't exist or cannot be created.");
98 return FALSE;
99 }
100
101 $this->_save_path = $save_path;
102 $this->_file_path = $this->_save_path.DIRECTORY_SEPARATOR
103 .$name // we'll use the session cookie name as a prefix to avoid collisions
104 .($this->_match_ip ? md5($_SERVER['REMOTE_ADDR']) : '');
105
106 return TRUE;
107 }
108
109 // ------------------------------------------------------------------------
110
111 public function read($session_id)
112 {
113 // This might seem weird, but PHP 5.6 introduces session_reset(),
114 // which re-reads session data
115 if ($this->_file_handle === NULL)
116 {
117 $this->_file_path .= $session_id;
118
119 // Just using fopen() with 'c+b' mode would be perfect, but it is only
120 // available since PHP 5.2.6 and we have to set permissions for new files,
121 // so we'd have to hack around this ...
122 if (($this->_file_new = ! file_exists($this->_file_path)) === TRUE)
123 {
124 if (($this->_file_handle = fopen($this->_file_path, 'w+b')) === FALSE)
125 {
126 log_message('error', "Session: File '".$this->_file_path."' doesn't exist and cannot be created.");
127 return FALSE;
128 }
129 }
130 elseif (($this->_file_handle = fopen($this->_file_path, 'r+b')) === FALSE)
131 {
132 log_message('error', "Session: Unable to open file '".$this->_file_path."'.");
133 return FALSE;
134 }
135
136 if (flock($this->_file_handle, LOCK_EX) === FALSE)
137 {
138 log_message('error', "Session: Unable to obtain lock for file '".$this->_file_path."'.");
139 fclose($this->_file_handle);
140 $this->_file_handle = NULL;
141 return FALSE;
142 }
143
144 if ($this->_file_new)
145 {
146 chmod($this->_file_path, 0600);
147 $this->_fingerprint = md5('');
148 return '';
149 }
150 }
151 else
152 {
153 rewind($this->_file_handle);
154 }
155
156 $session_data = '';
157 for ($read = 0, $length = filesize($this->_file_path); $read < $length; $read += strlen($buffer))
158 {
159 if (($buffer = fread($this->_file_handle, $length - $read)) === FALSE)
160 {
161 break;
162 }
163
164 $session_data .= $buffer;
165 }
166
167 $this->_fingerprint = md5($session_data);
168 return $session_data;
169 }
170
171 public function write($session_id, $session_data)
172 {
173 if ( ! is_resource($this->_file_handle))
174 {
175 return FALSE;
176 }
177 elseif ($this->_fingerprint === md5($session_data))
178 {
179 return ($this->_file_new)
180 ? TRUE
181 : touch($this->_file_path);
182 }
183
184 if ( ! $this->_file_new)
185 {
186 ftruncate($this->_file_handle, 0);
187 rewind($this->_file_handle);
188 }
189
190 for ($written = 0, $length = strlen($session_data); $written < $length; $written += $result)
191 {
192 if (($result = fwrite($this->_file_handle, substr($session_data, $written))) === FALSE)
193 {
194 break;
195 }
196 }
197
198 if ( ! is_int($result))
199 {
200 $this->_fingerprint = md5(substr($session_data, 0, $written));
201 log_message('error', 'Session: Unable to write data.');
202 return FALSE;
203 }
204
205 $this->_fingerprint = md5($session_data);
206 return TRUE;
207 }
208
209 // ------------------------------------------------------------------------
210
211 public function close()
212 {
213 if (is_resource($this->_file_handle))
214 {
215 flock($this->_file_handle, LOCK_UN);
216 fclose($this->_file_handle);
217
218 $this->_file_handle = $this->_file_new = NULL;
219 return TRUE;
220 }
221
222 return FALSE;
223 }
224
225 // ------------------------------------------------------------------------
226
227 public function destroy($session_id)
228 {
229 if ($this->close())
230 {
231 return unlink($this->_file_path) && $this->_cookie_destroy();
232 }
233 elseif ($this->_file_path !== NULL)
234 {
235 clearstatcache();
236 return file_exists($this->_file_path)
237 ? (unlink($this->_file_path) && $this->_cookie_destroy())
238 : TRUE;
239 }
240
241 return FALSE;
242 }
243
244 // ------------------------------------------------------------------------
245
246 public function gc($maxlifetime)
247 {
248 if ( ! is_dir($this->_save_path) OR ($files = scandir($this->_save_path)) === FALSE)
249 {
250 log_message('debug', "Session: Garbage collector couldn't list files under directory '".$this->_save_path."'.");
251 return FALSE;
252 }
253
254 $ts = time() - $maxlifetime;
255
256 foreach ($files as $file)
257 {
258 // If the filename doesn't match this pattern, it's either not a session file or is not ours
259 if ( ! preg_match('/(?:[0-9a-f]{32})?[0-9a-f]{40}$/i', $file)
260 OR ! is_file($this->_save_path.DIRECTORY_SEPARATOR.$file)
261 OR ($mtime = filemtime($file)) === FALSE
262 OR $mtime > $ts)
263 {
264 continue;
265 }
266
267 unlink($this->_save_path.DIRECTORY_SEPARATOR.$file);
268 }
269
270 return TRUE;
271 }
272
273}
274
275/* End of file Session_files_driver.php */
276/* Location: ./system/libraries/Session/drivers/Session_files_driver.php */