I am trying to send Cookies to a PHP Script within a javascript fetch CORS request. The Request starts on and contains the following options:
let response = await fetch('.php', {
method: "POST",
headers: headers,
body: formData,
mode: 'cors',
credentials: 'include',
cache: 'no-store'
});
The corresponding PHP Script sets the following Headers:
header('Access-Control-Allow-Origin: ');
header('Access-Control-Allow-Methods: POST, OPTIONS');
header('Access-Control-Allow-Credentials: true');
header('Access-Control-Allow-Headers: Origin, Content-Type, Accept, Authorization, X-Request-With, Set-Cookie, Cookie, Bearer');
But the Cookie Header is not send with the request. I also tried:
let headers = new Headers();
headers.set('Cookie', document.cookie);
That also had no effect. What exactly am I doing wrong here?
I checked the Network Tab in the Development Tools. Also $_COOKIE
in the PHP Script is empty. There is absolutely no error. I can also see that the Cookie Header is sent in any not CORS fetch
request.
EDIT: Here are the Settings of one of the Cookies:
Name: PHPSESSID
Path: /
Secure: true
SameSite: none
I can't share the Domain because it's not public. But the Cookie Domain has the same Value as the Origin in the Request Header (Minus the https://).
EDIT 2: Changed the fetch URL to make clearer what's happening.
I am trying to send Cookies to a PHP Script within a javascript fetch CORS request. The Request starts on https://sub1.example.com
and contains the following options:
let response = await fetch('https://sub2.example.com/target.php', {
method: "POST",
headers: headers,
body: formData,
mode: 'cors',
credentials: 'include',
cache: 'no-store'
});
The corresponding PHP Script sets the following Headers:
header('Access-Control-Allow-Origin: https://www.example.com');
header('Access-Control-Allow-Methods: POST, OPTIONS');
header('Access-Control-Allow-Credentials: true');
header('Access-Control-Allow-Headers: Origin, Content-Type, Accept, Authorization, X-Request-With, Set-Cookie, Cookie, Bearer');
But the Cookie Header is not send with the request. I also tried:
let headers = new Headers();
headers.set('Cookie', document.cookie);
That also had no effect. What exactly am I doing wrong here?
I checked the Network Tab in the Development Tools. Also $_COOKIE
in the PHP Script is empty. There is absolutely no error. I can also see that the Cookie Header is sent in any not CORS fetch
request.
EDIT: Here are the Settings of one of the Cookies:
Name: PHPSESSID
Path: /
Secure: true
SameSite: none
I can't share the Domain because it's not public. But the Cookie Domain has the same Value as the Origin in the Request Header (Minus the https://).
EDIT 2: Changed the fetch URL to make clearer what's happening.
Share Improve this question edited Jan 5, 2022 at 12:24 jub0bs 66.2k27 gold badges193 silver badges196 bronze badges asked Jan 5, 2022 at 9:23 KHansenKHansen 9301 gold badge6 silver badges28 bronze badges 12 | Show 7 more comments3 Answers
Reset to default 7Problem
Be aware that, depending on
- the value of the cookie's
Path
attribute, - the effective value of the cookie's
Domain
attribute, - the value of the cookie's
Secure
attribute, - the effective value of the cookie's
SameSite
attribute, - the request's issuing and destination origins,
a cookie may or may not be attached to the request. Of particular relevance to your case is the Domain
attribute; check out MDN's page on the topic:
The
Domain
attribute specifies which hosts can receive a cookie. If unspecified, the attribute defaults to the same host that set the cookie, excluding subdomains. IfDomain
is specified, then subdomains are always included. Therefore, specifyingDomain
is less restrictive than omitting it. However, it can be helpful when subdomains need to share information about a user.
You're setting the cookie as follows on origin https://sub1.example.com
:
Set-Cookie: PHPSESSID=whatever; Path=/; SameSite=None; Secure
Therefore, that cookie will get attached to (credentialed) requests whose destination origin is https://sub1.example.com
, and no other.
Solution
If you want your cookie to be sent to all secure origins whose domain is an example.com
subdomain, you need to explicitly set its Domain
to example.com
.
About sending cookies with fetch
The Fetch standard specifies a list of forbidden header names; Cookie
is one of them. You cannot set a header named Cookie
on a request sent with fetch
; the standard simply forbids it. If you want to attach existing cookies to a cross-origin request, use the 'include'
value for the credentials
parameter passed in fetch
options.
These are the conditions that need to be met in order for the browser to save and then use cookies initiated using fetch
:
- Client initializes asynchronously a fetch request with
credentials: 'include'
. See [here][1] for more details.- To do CORS, server response header must contain
Access-Control-Allow-Origin
explicitly set to a domain, could be different from the server domain. For example, in a Single-Page-App architecture, your frontend site is temporarily hosted at localhost:3000 and your backend server hosted at localhost:8000, then the header should beAccess-Control-Allow-Origin: http://localhost:3000
. See [here][2] and [here][3].- To allow client to process cookies, which is obviously a sensitive resource, server response header must further contain
Access-Control-Allow-Credentials: true
. See [here][4]. Note that this enforces a non-wildcard setting forAccess-Control-Allow-Origin
. See [here][6] - that's why in point 2 above, it has to be explicitly set to something likehttp://localhost:3000
rather than*
- When server sets the cookie, it has to include
SameSite=None; Secure; HttpOnly
. So overall something likeSet-Cookie: session_id=12345; SameSite=None; Secure; HttpOnly
.SameSite
seems to be a relatively [new requirement][5] in latest browsers, and must be used withSecure
together whenSameSite
is set toNone
.- With regard to
HttpOnly
, I haven't found relevant materials, but in my experiment, omitting it caused the browser to ignore theSet-Cookie
header.- Further requests to the backend server also must have
credentials: 'include'
set.
Source: https://stackoverflow.com/a/67001424/368691
Cookies normally are not supposed to be attached to preflight requests in CORS mode. You might want to check this out.
Note: Browsers should not send credentials in preflight requests irrespective of this setting. For more information see: CORS > Requests with credentials.
https://developer.mozilla.org/en-US/docs/Web/API/Fetch_API/Using_Fetch
Cookie Header is not send
...how do you know? It's not clear what you've tested, or what consequences you're seeing. is there an error? – ADyson Commented Jan 5, 2022 at 9:28$_COOKIE
in the PHP Script is empty. There is absolutely no error. I can also see, that the Cookie Header is send in none CORS fetch Requests. – KHansen Commented Jan 5, 2022 at 9:33Name
,Domain
,Path
,Secure
, andSameSite
attributes of the cookie in question. – jub0bs Commented Jan 5, 2022 at 10:28https://
)? Does the cookie actually get set / exist in your browser? – jub0bs Commented Jan 5, 2022 at 10:47Domain
attribute issub1.example.com
tosub2.example.com
. If you want your cookie to be sent to subdomains, you need to explicitly set itsDomain
attribute to a common parent domain, e.g.example.com
. See developer.mozilla.org/en-US/docs/Web/HTTP/… – jub0bs Commented Jan 5, 2022 at 11:19