Session_redis_driver.php 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459
  1. <?php
  2. /**
  3. * CodeIgniter
  4. *
  5. * An open source application development framework for PHP
  6. *
  7. * This content is released under the MIT License (MIT)
  8. *
  9. * Copyright (c) 2014 - 2019, British Columbia Institute of Technology
  10. *
  11. * Permission is hereby granted, free of charge, to any person obtaining a copy
  12. * of this software and associated documentation files (the "Software"), to deal
  13. * in the Software without restriction, including without limitation the rights
  14. * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
  15. * copies of the Software, and to permit persons to whom the Software is
  16. * furnished to do so, subject to the following conditions:
  17. *
  18. * The above copyright notice and this permission notice shall be included in
  19. * all copies or substantial portions of the Software.
  20. *
  21. * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
  22. * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
  23. * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
  24. * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
  25. * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
  26. * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
  27. * THE SOFTWARE.
  28. *
  29. * @package CodeIgniter
  30. * @author EllisLab Dev Team
  31. * @copyright Copyright (c) 2008 - 2014, EllisLab, Inc. (https://ellislab.com/)
  32. * @copyright Copyright (c) 2014 - 2019, British Columbia Institute of Technology (https://bcit.ca/)
  33. * @license https://opensource.org/licenses/MIT MIT License
  34. * @link https://codeigniter.com
  35. * @since Version 3.0.0
  36. * @filesource
  37. */
  38. defined('BASEPATH') OR exit('No direct script access allowed');
  39. /**
  40. * CodeIgniter Session Redis Driver
  41. *
  42. * @package CodeIgniter
  43. * @subpackage Libraries
  44. * @category Sessions
  45. * @author Andrey Andreev
  46. * @link https://codeigniter.com/user_guide/libraries/sessions.html
  47. */
  48. class CI_Session_redis_driver extends CI_Session_driver implements SessionHandlerInterface {
  49. /**
  50. * phpRedis instance
  51. *
  52. * @var Redis
  53. */
  54. protected $_redis;
  55. /**
  56. * Key prefix
  57. *
  58. * @var string
  59. */
  60. protected $_key_prefix = 'ci_session:';
  61. /**
  62. * Lock key
  63. *
  64. * @var string
  65. */
  66. protected $_lock_key;
  67. /**
  68. * Key exists flag
  69. *
  70. * @var bool
  71. */
  72. protected $_key_exists = FALSE;
  73. /**
  74. * Name of setTimeout() method in phpRedis
  75. *
  76. * Due to some deprecated methods in phpRedis, we need to call the
  77. * specific methods depending on the version of phpRedis.
  78. *
  79. * @var string
  80. */
  81. protected $_setTimeout_name;
  82. /**
  83. * Name of delete() method in phpRedis
  84. *
  85. * Due to some deprecated methods in phpRedis, we need to call the
  86. * specific methods depending on the version of phpRedis.
  87. *
  88. * @var string
  89. */
  90. protected $_delete_name;
  91. /**
  92. * Success return value of ping() method in phpRedis
  93. *
  94. * @var mixed
  95. */
  96. protected $_ping_success;
  97. // ------------------------------------------------------------------------
  98. /**
  99. * Class constructor
  100. *
  101. * @param array $params Configuration parameters
  102. * @return void
  103. */
  104. public function __construct(&$params)
  105. {
  106. parent::__construct($params);
  107. // Detect the names of some methods in phpRedis instance
  108. if (version_compare(phpversion('redis'), '5', '>='))
  109. {
  110. $this->_setTimeout_name = 'expire';
  111. $this->_delete_name = 'del';
  112. $this->_ping_success = TRUE;
  113. }
  114. else
  115. {
  116. $this->_setTimeout_name = 'setTimeout';
  117. $this->_delete_name = 'delete';
  118. $this->_ping_success = '+PONG';
  119. }
  120. if (empty($this->_config['save_path']))
  121. {
  122. log_message('error', 'Session: No Redis save path configured.');
  123. }
  124. elseif (preg_match('#(?:tcp://)?([^:?]+)(?:\:(\d+))?(\?.+)?#', $this->_config['save_path'], $matches))
  125. {
  126. isset($matches[3]) OR $matches[3] = ''; // Just to avoid undefined index notices below
  127. $this->_config['save_path'] = array(
  128. 'host' => $matches[1],
  129. 'port' => empty($matches[2]) ? NULL : $matches[2],
  130. 'password' => preg_match('#auth=([^\s&]+)#', $matches[3], $match) ? $match[1] : NULL,
  131. 'database' => preg_match('#database=(\d+)#', $matches[3], $match) ? (int) $match[1] : NULL,
  132. 'timeout' => preg_match('#timeout=(\d+\.\d+)#', $matches[3], $match) ? (float) $match[1] : NULL
  133. );
  134. preg_match('#prefix=([^\s&]+)#', $matches[3], $match) && $this->_key_prefix = $match[1];
  135. }
  136. else
  137. {
  138. log_message('error', 'Session: Invalid Redis save path format: '.$this->_config['save_path']);
  139. }
  140. if ($this->_config['match_ip'] === TRUE)
  141. {
  142. $this->_key_prefix .= $_SERVER['REMOTE_ADDR'].':';
  143. }
  144. }
  145. // ------------------------------------------------------------------------
  146. /**
  147. * Open
  148. *
  149. * Sanitizes save_path and initializes connection.
  150. *
  151. * @param string $save_path Server path
  152. * @param string $name Session cookie name, unused
  153. * @return bool
  154. */
  155. public function open($save_path, $name)
  156. {
  157. if (empty($this->_config['save_path']))
  158. {
  159. return $this->_failure;
  160. }
  161. $redis = new Redis();
  162. if ( ! $redis->connect($this->_config['save_path']['host'], $this->_config['save_path']['port'], $this->_config['save_path']['timeout']))
  163. {
  164. log_message('error', 'Session: Unable to connect to Redis with the configured settings.');
  165. }
  166. elseif (isset($this->_config['save_path']['password']) && ! $redis->auth($this->_config['save_path']['password']))
  167. {
  168. log_message('error', 'Session: Unable to authenticate to Redis instance.');
  169. }
  170. elseif (isset($this->_config['save_path']['database']) && ! $redis->select($this->_config['save_path']['database']))
  171. {
  172. log_message('error', 'Session: Unable to select Redis database with index '.$this->_config['save_path']['database']);
  173. }
  174. else
  175. {
  176. $this->_redis = $redis;
  177. $this->php5_validate_id();
  178. return $this->_success;
  179. }
  180. return $this->_failure;
  181. }
  182. // ------------------------------------------------------------------------
  183. /**
  184. * Read
  185. *
  186. * Reads session data and acquires a lock
  187. *
  188. * @param string $session_id Session ID
  189. * @return string Serialized session data
  190. */
  191. public function read($session_id)
  192. {
  193. if (isset($this->_redis) && $this->_get_lock($session_id))
  194. {
  195. // Needed by write() to detect session_regenerate_id() calls
  196. $this->_session_id = $session_id;
  197. $session_data = $this->_redis->get($this->_key_prefix.$session_id);
  198. is_string($session_data)
  199. ? $this->_key_exists = TRUE
  200. : $session_data = '';
  201. $this->_fingerprint = md5($session_data);
  202. return $session_data;
  203. }
  204. return $this->_failure;
  205. }
  206. // ------------------------------------------------------------------------
  207. /**
  208. * Write
  209. *
  210. * Writes (create / update) session data
  211. *
  212. * @param string $session_id Session ID
  213. * @param string $session_data Serialized session data
  214. * @return bool
  215. */
  216. public function write($session_id, $session_data)
  217. {
  218. if ( ! isset($this->_redis, $this->_lock_key))
  219. {
  220. return $this->_failure;
  221. }
  222. // Was the ID regenerated?
  223. elseif ($session_id !== $this->_session_id)
  224. {
  225. if ( ! $this->_release_lock() OR ! $this->_get_lock($session_id))
  226. {
  227. return $this->_failure;
  228. }
  229. $this->_key_exists = FALSE;
  230. $this->_session_id = $session_id;
  231. }
  232. $this->_redis->{$this->_setTimeout_name}($this->_lock_key, 300);
  233. if ($this->_fingerprint !== ($fingerprint = md5($session_data)) OR $this->_key_exists === FALSE)
  234. {
  235. if ($this->_redis->set($this->_key_prefix.$session_id, $session_data, $this->_config['expiration']))
  236. {
  237. $this->_fingerprint = $fingerprint;
  238. $this->_key_exists = TRUE;
  239. return $this->_success;
  240. }
  241. return $this->_failure;
  242. }
  243. return ($this->_redis->{$this->_setTimeout_name}($this->_key_prefix.$session_id, $this->_config['expiration']))
  244. ? $this->_success
  245. : $this->_failure;
  246. }
  247. // ------------------------------------------------------------------------
  248. /**
  249. * Close
  250. *
  251. * Releases locks and closes connection.
  252. *
  253. * @return bool
  254. */
  255. public function close()
  256. {
  257. if (isset($this->_redis))
  258. {
  259. try {
  260. if ($this->_redis->ping() === $this->_ping_success)
  261. {
  262. $this->_release_lock();
  263. if ($this->_redis->close() === FALSE)
  264. {
  265. return $this->_failure;
  266. }
  267. }
  268. }
  269. catch (RedisException $e)
  270. {
  271. log_message('error', 'Session: Got RedisException on close(): '.$e->getMessage());
  272. }
  273. $this->_redis = NULL;
  274. return $this->_success;
  275. }
  276. return $this->_success;
  277. }
  278. // ------------------------------------------------------------------------
  279. /**
  280. * Destroy
  281. *
  282. * Destroys the current session.
  283. *
  284. * @param string $session_id Session ID
  285. * @return bool
  286. */
  287. public function destroy($session_id)
  288. {
  289. if (isset($this->_redis, $this->_lock_key))
  290. {
  291. if (($result = $this->_redis->{$this->_delete_name}($this->_key_prefix.$session_id)) !== 1)
  292. {
  293. log_message('debug', 'Session: Redis::'.$this->_delete_name.'() expected to return 1, got '.var_export($result, TRUE).' instead.');
  294. }
  295. $this->_cookie_destroy();
  296. return $this->_success;
  297. }
  298. return $this->_failure;
  299. }
  300. // ------------------------------------------------------------------------
  301. /**
  302. * Garbage Collector
  303. *
  304. * Deletes expired sessions
  305. *
  306. * @param int $maxlifetime Maximum lifetime of sessions
  307. * @return bool
  308. */
  309. public function gc($maxlifetime)
  310. {
  311. // Not necessary, Redis takes care of that.
  312. return $this->_success;
  313. }
  314. // --------------------------------------------------------------------
  315. /**
  316. * Validate ID
  317. *
  318. * Checks whether a session ID record exists server-side,
  319. * to enforce session.use_strict_mode.
  320. *
  321. * @param string $id
  322. * @return bool
  323. */
  324. public function validateSessionId($id)
  325. {
  326. return (bool) $this->_redis->exists($this->_key_prefix.$id);
  327. }
  328. // ------------------------------------------------------------------------
  329. /**
  330. * Get lock
  331. *
  332. * Acquires an (emulated) lock.
  333. *
  334. * @param string $session_id Session ID
  335. * @return bool
  336. */
  337. protected function _get_lock($session_id)
  338. {
  339. // PHP 7 reuses the SessionHandler object on regeneration,
  340. // so we need to check here if the lock key is for the
  341. // correct session ID.
  342. if ($this->_lock_key === $this->_key_prefix.$session_id.':lock')
  343. {
  344. return $this->_redis->{$this->_setTimeout_name}($this->_lock_key, 300);
  345. }
  346. // 30 attempts to obtain a lock, in case another request already has it
  347. $lock_key = $this->_key_prefix.$session_id.':lock';
  348. $attempt = 0;
  349. do
  350. {
  351. if (($ttl = $this->_redis->ttl($lock_key)) > 0)
  352. {
  353. sleep(1);
  354. continue;
  355. }
  356. if ($ttl === -2 && ! $this->_redis->set($lock_key, time(), array('nx', 'ex' => 300)))
  357. {
  358. // Sleep for 1s to wait for lock releases.
  359. sleep(1);
  360. continue;
  361. }
  362. elseif ( ! $this->_redis->setex($lock_key, 300, time()))
  363. {
  364. log_message('error', 'Session: Error while trying to obtain lock for '.$this->_key_prefix.$session_id);
  365. return FALSE;
  366. }
  367. $this->_lock_key = $lock_key;
  368. break;
  369. }
  370. while (++$attempt < 30);
  371. if ($attempt === 30)
  372. {
  373. log_message('error', 'Session: Unable to obtain lock for '.$this->_key_prefix.$session_id.' after 30 attempts, aborting.');
  374. return FALSE;
  375. }
  376. elseif ($ttl === -1)
  377. {
  378. log_message('debug', 'Session: Lock for '.$this->_key_prefix.$session_id.' had no TTL, overriding.');
  379. }
  380. $this->_lock = TRUE;
  381. return TRUE;
  382. }
  383. // ------------------------------------------------------------------------
  384. /**
  385. * Release lock
  386. *
  387. * Releases a previously acquired lock
  388. *
  389. * @return bool
  390. */
  391. protected function _release_lock()
  392. {
  393. if (isset($this->_redis, $this->_lock_key) && $this->_lock)
  394. {
  395. if ( ! $this->_redis->{$this->_delete_name}($this->_lock_key))
  396. {
  397. log_message('error', 'Session: Error while trying to free lock for '.$this->_lock_key);
  398. return FALSE;
  399. }
  400. $this->_lock_key = NULL;
  401. $this->_lock = FALSE;
  402. }
  403. return TRUE;
  404. }
  405. }