payload.php 7.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327
  1. <?php namespace Laravel\Session;
  2. use Closure;
  3. use Laravel\Arr;
  4. use Laravel\Str;
  5. use Laravel\Config;
  6. use Laravel\Cookie;
  7. use Laravel\Session\Drivers\Driver;
  8. use Laravel\Session\Drivers\Sweeper;
  9. if (Config::$items['application']['key'] === '')
  10. {
  11. throw new \LogicException("An application key is required to use sessions.");
  12. }
  13. class Payload {
  14. /**
  15. * The session array that is stored by the driver.
  16. *
  17. * @var array
  18. */
  19. public $session;
  20. /**
  21. * Indicates if the session already exists in storage.
  22. *
  23. * @var bool
  24. */
  25. protected $exists = true;
  26. /**
  27. * The session driver used to retrieve and store the session payload.
  28. *
  29. * @var Driver
  30. */
  31. protected $driver;
  32. /**
  33. * The string name of the CSRF token stored in the session.
  34. *
  35. * @var string
  36. */
  37. const token = 'csrf_token';
  38. /**
  39. * Create a new session payload instance.
  40. *
  41. * @param Driver $driver
  42. * @return void
  43. */
  44. public function __construct(Driver $driver)
  45. {
  46. $this->driver = $driver;
  47. }
  48. /**
  49. * Load the session for the current request.
  50. *
  51. * @param string $id
  52. * @return void
  53. */
  54. public function load($id)
  55. {
  56. if ( ! is_null($id)) $this->session = $this->driver->load($id);
  57. // If the session doesn't exist or is invalid, we will create a new session
  58. // array and mark the session as being non-existent. Some drivers, such as
  59. // the database driver, need to know whether the session exists in storage
  60. // so they can know whether to "insert" or "update" the session.
  61. if (is_null($this->session) or $this->invalid())
  62. {
  63. $this->exists = false;
  64. $this->session = array('id' => Str::random(40), 'data' => array(
  65. ':new:' => array(),
  66. ':old:' => array(),
  67. ));
  68. }
  69. // A CSRF token is stored in every session. The token is used by the Form
  70. // class and the "csrf" filter to protect the application from cross-site
  71. // request forgery attacks. The token is simply a long, random string
  72. // which should be posted with each request.
  73. if ( ! $this->has(Payload::token))
  74. {
  75. $this->put(Payload::token, Str::random(40));
  76. }
  77. }
  78. /**
  79. * Deteremine if the session payload instance is valid.
  80. *
  81. * The session is considered valid if it exists and has not expired.
  82. *
  83. * @return bool
  84. */
  85. protected function invalid()
  86. {
  87. $lifetime = Config::$items['session']['lifetime'];
  88. return (time() - $this->session['last_activity']) > ($lifetime * 60);
  89. }
  90. /**
  91. * Determine if session handling has been started for the request.
  92. *
  93. * @return bool
  94. */
  95. public function started()
  96. {
  97. return is_array($this->session);
  98. }
  99. /**
  100. * Determine if the session or flash data contains an item.
  101. *
  102. * @param string $key
  103. * @return bool
  104. */
  105. public function has($key)
  106. {
  107. return ( ! is_null($this->get($key)));
  108. }
  109. /**
  110. * Get an item from the session.
  111. *
  112. * The session flash data will also be checked for the requested item.
  113. *
  114. * <code>
  115. * // Get an item from the session
  116. * $name = Session::get('name');
  117. *
  118. * // Return a default value if the item doesn't exist
  119. * $name = Session::get('name', 'Taylor');
  120. * </code>
  121. *
  122. * @param string $key
  123. * @param mixed $default
  124. * @return mixed
  125. */
  126. public function get($key, $default = null)
  127. {
  128. if (isset($this->session['data'][$key]))
  129. {
  130. return $this->session['data'][$key];
  131. }
  132. elseif (isset($this->session['data'][':new:'][$key]))
  133. {
  134. return $this->session['data'][':new:'][$key];
  135. }
  136. elseif (isset($this->session['data'][':old:'][$key]))
  137. {
  138. return $this->session['data'][':old:'][$key];
  139. }
  140. return ($default instanceof Closure) ? call_user_func($default) : $default;
  141. }
  142. /**
  143. * Write an item to the session.
  144. *
  145. * @param string $key
  146. * @param mixed $value
  147. * @return void
  148. */
  149. public function put($key, $value)
  150. {
  151. Arr::set($this->session['data'], $key, $value);
  152. }
  153. /**
  154. * Write an item to the session flash data.
  155. *
  156. * Flash data only exists for the next request to the application, and is
  157. * useful for storing temporary data such as error or status messages.
  158. *
  159. * @param string $key
  160. * @param mixed $value
  161. * @return void
  162. */
  163. public function flash($key, $value)
  164. {
  165. Arr::set($this->session['data'][':new:'], $key, $value);
  166. }
  167. /**
  168. * Keep the session flash data from expiring at the end of the request.
  169. *
  170. * @return void
  171. */
  172. public function reflash()
  173. {
  174. $old = $this->session['data'][':old:'];
  175. $this->session['data'][':new:'] = array_merge($this->session['data'][':new:'], $old);
  176. }
  177. /**
  178. * Keep a session flash item from expiring at the end of the request.
  179. *
  180. * @param string|array $key
  181. * @return void
  182. */
  183. public function keep($keys)
  184. {
  185. foreach ((array) $keys as $key)
  186. {
  187. $this->flash($key, $this->get($key));
  188. }
  189. }
  190. /**
  191. * Remove an item from the session data.
  192. *
  193. * @param string $key
  194. * @return void
  195. */
  196. public function forget($key)
  197. {
  198. Arr::forget($this->session['data'], $key);
  199. }
  200. /**
  201. * Remove all of the items from the session.
  202. *
  203. * @return void
  204. */
  205. public function flush()
  206. {
  207. $this->session['data'] = array();
  208. }
  209. /**
  210. * Assign a new, random ID to the session.
  211. *
  212. * @return void
  213. */
  214. public function regenerate()
  215. {
  216. $this->session['id'] = Str::random(40);
  217. $this->exists = false;
  218. }
  219. /**
  220. * Get the CSRF token that is stored in the session data.
  221. *
  222. * @return string
  223. */
  224. public function token()
  225. {
  226. return $this->get(Payload::token);
  227. }
  228. /**
  229. * Store the session payload in storage.
  230. *
  231. * The activity timestamp will be set, the flash data will be aged, and the
  232. * session cookie will be written. The driver given when the session payload
  233. * was constructed will be used to persist the session to storage.
  234. *
  235. * If the session's driver is a sweeper implementation, garbage collection
  236. * may be performed based on the probabilities set in the "sweepage" option
  237. * in the session configuration file.
  238. *
  239. * @return void
  240. */
  241. public function save()
  242. {
  243. $this->session['last_activity'] = time();
  244. $this->age();
  245. $config = Config::$items['session'];
  246. $this->driver->save($this->session, $config, $this->exists);
  247. $this->cookie();
  248. // Some session drivers implement the Sweeper interface, meaning that they
  249. // must clean up expired sessions manually. If the driver is a sweeper, we
  250. // need to determine if garbage collection should be run for the request.
  251. // Since garbage collection can be expensive, the probability of it
  252. // occuring is controlled by the "sweepage" configuration option.
  253. $sweepage = $config['sweepage'];
  254. if ($this->driver instanceof Sweeper and (mt_rand(1, $sweepage[1]) <= $sweepage[0]))
  255. {
  256. $this->driver->sweep(time() - ($config['lifetime'] * 60));
  257. }
  258. }
  259. /**
  260. * Age the session flash data.
  261. *
  262. * Session flash data is only available during the request in which it
  263. * was flashed, and the request after that. To "age" the data, we will
  264. * remove all of the :old: items and re-address the new items.
  265. *
  266. * @return void
  267. */
  268. protected function age()
  269. {
  270. $this->session['data'][':old:'] = $this->session['data'][':new:'];
  271. $this->session['data'][':new:'] = array();
  272. }
  273. /**
  274. * Send the session ID cookie to the browser.
  275. *
  276. * @return void
  277. */
  278. protected function cookie()
  279. {
  280. $config = Config::$items['session'];
  281. extract($config, EXTR_SKIP);
  282. $minutes = ( ! $expire_on_close) ? $lifetime : 0;
  283. Cookie::put($cookie, $this->session['id'], $minutes, $path, $domain, $secure);
  284. }
  285. }