12
12
namespace Symfony \Component \HttpFoundation \Session \Storage ;
13
13
14
14
use Symfony \Component \HttpFoundation \Session \SessionBagInterface ;
15
+ use Symfony \Component \HttpFoundation \Session \SessionUtils ;
15
16
use Symfony \Component \HttpFoundation \Session \Storage \Handler \StrictSessionHandler ;
16
17
use Symfony \Component \HttpFoundation \Session \Storage \Proxy \AbstractProxy ;
17
18
use Symfony \Component \HttpFoundation \Session \Storage \Proxy \SessionHandlerProxy ;
@@ -48,6 +49,11 @@ class NativeSessionStorage implements SessionStorageInterface
48
49
*/
49
50
protected $ metadataBag ;
50
51
52
+ /**
53
+ * @var string|null
54
+ */
55
+ private $ emulateSameSite ;
56
+
51
57
/**
52
58
* Depending on how you want the storage driver to behave you probably
53
59
* want to override this constructor entirely.
@@ -67,6 +73,7 @@ class NativeSessionStorage implements SessionStorageInterface
67
73
* cookie_lifetime, "0"
68
74
* cookie_path, "/"
69
75
* cookie_secure, ""
76
+ * cookie_samesite, null
70
77
* entropy_file, ""
71
78
* entropy_length, "0"
72
79
* gc_divisor, "100"
@@ -94,9 +101,7 @@ class NativeSessionStorage implements SessionStorageInterface
94
101
* trans_sid_hosts, $_SERVER['HTTP_HOST']
95
102
* trans_sid_tags, "a=href,area=href,frame=src,form="
96
103
*
97
- * @param array $options Session configuration options
98
- * @param \SessionHandlerInterface|null $handler
99
- * @param MetadataBag $metaBag MetadataBag
104
+ * @param AbstractProxy|\SessionHandlerInterface|null $handler
100
105
*/
101
106
public function __construct (array $ options = [], $ handler = null , MetadataBag $ metaBag = null )
102
107
{
@@ -150,6 +155,13 @@ public function start()
150
155
throw new \RuntimeException ('Failed to start the session ' );
151
156
}
152
157
158
+ if (null !== $ this ->emulateSameSite ) {
159
+ $ originalCookie = SessionUtils::popSessionCookie (session_name (), session_id ());
160
+ if (null !== $ originalCookie ) {
161
+ header (sprintf ('%s; SameSite=%s ' , $ originalCookie , $ this ->emulateSameSite ), false );
162
+ }
163
+ }
164
+
153
165
$ this ->loadSession ();
154
166
155
167
return true ;
@@ -215,6 +227,13 @@ public function regenerate($destroy = false, $lifetime = null)
215
227
// @see https://bugs.php.net/70013
216
228
$ this ->loadSession ();
217
229
230
+ if (null !== $ this ->emulateSameSite ) {
231
+ $ originalCookie = SessionUtils::popSessionCookie (session_name (), session_id ());
232
+ if (null !== $ originalCookie ) {
233
+ header (sprintf ('%s; SameSite=%s ' , $ originalCookie , $ this ->emulateSameSite ), false );
234
+ }
235
+ }
236
+
218
237
return $ isRegenerated ;
219
238
}
220
239
@@ -223,6 +242,7 @@ public function regenerate($destroy = false, $lifetime = null)
223
242
*/
224
243
public function save ()
225
244
{
245
+ // Store a copy so we can restore the bags in case the session was not left empty
226
246
$ session = $ _SESSION ;
227
247
228
248
foreach ($ this ->bags as $ bag ) {
@@ -248,7 +268,11 @@ public function save()
248
268
session_write_close ();
249
269
} finally {
250
270
restore_error_handler ();
251
- $ _SESSION = $ session ;
271
+
272
+ // Restore only if not empty
273
+ if ($ _SESSION ) {
274
+ $ _SESSION = $ session ;
275
+ }
252
276
}
253
277
254
278
$ this ->closed = true ;
@@ -347,7 +371,7 @@ public function setOptions(array $options)
347
371
348
372
$ validOptions = array_flip ([
349
373
'cache_expire ' , 'cache_limiter ' , 'cookie_domain ' , 'cookie_httponly ' ,
350
- 'cookie_lifetime ' , 'cookie_path ' , 'cookie_secure ' ,
374
+ 'cookie_lifetime ' , 'cookie_path ' , 'cookie_secure ' , ' cookie_samesite ' ,
351
375
'entropy_file ' , 'entropy_length ' , 'gc_divisor ' ,
352
376
'gc_maxlifetime ' , 'gc_probability ' , 'hash_bits_per_character ' ,
353
377
'hash_function ' , 'lazy_write ' , 'name ' , 'referer_check ' ,
@@ -360,6 +384,12 @@ public function setOptions(array $options)
360
384
361
385
foreach ($ options as $ key => $ value ) {
362
386
if (isset ($ validOptions [$ key ])) {
387
+ if ('cookie_samesite ' === $ key && \PHP_VERSION_ID < 70300 ) {
388
+ // PHP < 7.3 does not support same_site cookies. We will emulate it in
389
+ // the start() method instead.
390
+ $ this ->emulateSameSite = $ value ;
391
+ continue ;
392
+ }
363
393
ini_set ('url_rewriter.tags ' !== $ key ? 'session. ' .$ key : $ key , $ value );
364
394
}
365
395
}
@@ -381,7 +411,7 @@ public function setOptions(array $options)
381
411
* @see https://php.net/sessionhandlerinterface
382
412
* @see https://php.net/sessionhandler
383
413
*
384
- * @param \SessionHandlerInterface|null $saveHandler
414
+ * @param AbstractProxy| \SessionHandlerInterface|null $saveHandler
385
415
*
386
416
* @throws \InvalidArgumentException
387
417
*/
0 commit comments