Database.php 5.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242
  1. <?php
  2. /**
  3. * Database-based session class.
  4. *
  5. * Sample schema:
  6. *
  7. * CREATE TABLE `sessions` (
  8. * `session_id` VARCHAR( 24 ) NOT NULL,
  9. * `last_active` INT UNSIGNED NOT NULL,
  10. * `contents` TEXT NOT NULL,
  11. * PRIMARY KEY ( `session_id` ),
  12. * INDEX ( `last_active` )
  13. * ) ENGINE = MYISAM ;
  14. *
  15. * @package Kohana/Database
  16. * @category Session
  17. * @author Kohana Team
  18. * @copyright (c) Kohana Team
  19. * @license https://koseven.ga/LICENSE.md
  20. */
  21. class Kohana_Session_Database extends Session {
  22. // Database instance
  23. protected $_db;
  24. // Database table name
  25. protected $_table = 'sessions';
  26. // Database column names
  27. protected $_columns = [
  28. 'session_id' => 'session_id',
  29. 'last_active' => 'last_active',
  30. 'contents' => 'contents'
  31. ];
  32. // Garbage collection requests
  33. protected $_gc = 500;
  34. // The current session id
  35. protected $_session_id;
  36. // The old session id
  37. protected $_update_id;
  38. public function __construct(array $config = NULL, $id = NULL)
  39. {
  40. if ( ! isset($config['group']))
  41. {
  42. // Use the default group
  43. $config['group'] = Database::$default;
  44. }
  45. // Load the database
  46. $this->_db = Database::instance($config['group']);
  47. if (isset($config['table']))
  48. {
  49. // Set the table name
  50. $this->_table = (string) $config['table'];
  51. }
  52. if (isset($config['gc']))
  53. {
  54. // Set the gc chance
  55. $this->_gc = (int) $config['gc'];
  56. }
  57. if (isset($config['columns']))
  58. {
  59. // Overload column names
  60. $this->_columns = $config['columns'];
  61. }
  62. parent::__construct($config, $id);
  63. if (random_int(0, $this->_gc) === $this->_gc)
  64. {
  65. // Run garbage collection
  66. // This will average out to run once every X requests
  67. $this->_gc();
  68. }
  69. }
  70. public function id()
  71. {
  72. return $this->_session_id;
  73. }
  74. protected function _read($id = NULL)
  75. {
  76. if ($id OR $id = Cookie::get($this->_name))
  77. {
  78. $result = DB::select([$this->_columns['contents'], 'contents'])
  79. ->from($this->_table)
  80. ->where($this->_columns['session_id'], '=', ':id')
  81. ->limit(1)
  82. ->param(':id', $id)
  83. ->execute($this->_db);
  84. if ($result->count())
  85. {
  86. // Set the current session id
  87. $this->_session_id = $this->_update_id = $id;
  88. // Return the contents
  89. return $result->get('contents');
  90. }
  91. }
  92. // Create a new session id
  93. $this->_regenerate();
  94. return NULL;
  95. }
  96. protected function _regenerate()
  97. {
  98. // Create the query to find an ID
  99. $query = DB::select($this->_columns['session_id'])
  100. ->from($this->_table)
  101. ->where($this->_columns['session_id'], '=', ':id')
  102. ->limit(1)
  103. ->bind(':id', $id);
  104. do
  105. {
  106. // Create a new session id
  107. $id = str_replace('.', '-', uniqid('', TRUE));
  108. // Get the the id from the database
  109. $result = $query->execute($this->_db);
  110. }
  111. while ($result->count());
  112. return $this->_session_id = $id;
  113. }
  114. protected function _write()
  115. {
  116. if ($this->_update_id === NULL)
  117. {
  118. // Insert a new row
  119. $query = DB::insert($this->_table, $this->_columns)
  120. ->values([':new_id', ':active', ':contents']);
  121. }
  122. else
  123. {
  124. // Update the row
  125. $query = DB::update($this->_table)
  126. ->value($this->_columns['last_active'], ':active')
  127. ->value($this->_columns['contents'], ':contents')
  128. ->where($this->_columns['session_id'], '=', ':old_id');
  129. if ($this->_update_id !== $this->_session_id)
  130. {
  131. // Also update the session id
  132. $query->value($this->_columns['session_id'], ':new_id');
  133. }
  134. }
  135. $query
  136. ->param(':new_id', $this->_session_id)
  137. ->param(':old_id', $this->_update_id)
  138. ->param(':active', $this->_data['last_active'])
  139. ->param(':contents', $this->__toString());
  140. // Execute the query
  141. $query->execute($this->_db);
  142. // The update and the session id are now the same
  143. $this->_update_id = $this->_session_id;
  144. // Update the cookie with the new session id
  145. Cookie::set($this->_name, $this->_session_id, $this->_lifetime);
  146. return TRUE;
  147. }
  148. /**
  149. * @return bool
  150. */
  151. protected function _restart()
  152. {
  153. $this->_regenerate();
  154. return TRUE;
  155. }
  156. protected function _destroy()
  157. {
  158. if ($this->_update_id === NULL)
  159. {
  160. // Session has not been created yet
  161. return TRUE;
  162. }
  163. // Delete the current session
  164. $query = DB::delete($this->_table)
  165. ->where($this->_columns['session_id'], '=', ':id')
  166. ->param(':id', $this->_update_id);
  167. try
  168. {
  169. // Execute the query
  170. $query->execute($this->_db);
  171. // Delete the old session id
  172. $this->_update_id = NULL;
  173. // Delete the cookie
  174. Cookie::delete($this->_name);
  175. }
  176. catch (Exception $e)
  177. {
  178. // An error occurred, the session has not been deleted
  179. return FALSE;
  180. }
  181. return TRUE;
  182. }
  183. protected function _gc()
  184. {
  185. if ($this->_lifetime)
  186. {
  187. // Expire sessions when their lifetime is up
  188. $expires = $this->_lifetime;
  189. }
  190. else
  191. {
  192. // Expire sessions after one month
  193. $expires = Date::MONTH;
  194. }
  195. // Delete all sessions that have expired
  196. DB::delete($this->_table)
  197. ->where($this->_columns['last_active'], '<', ':time')
  198. ->param(':time', time() - $expires)
  199. ->execute($this->_db);
  200. }
  201. } // End Session_Database