2018-01-08 16:24:26 +00:00
|
|
|
import time
|
2023-01-16 04:37:25 +00:00
|
|
|
from datetime import date, datetime, timedelta, timezone
|
|
|
|
from email.utils import format_datetime as format_datetime_rfc5322
|
2018-01-08 16:24:26 +00:00
|
|
|
from http import cookies
|
|
|
|
|
|
|
|
from django.http import HttpResponse
|
|
|
|
from django.test import SimpleTestCase
|
|
|
|
from django.test.utils import freeze_time
|
|
|
|
from django.utils.http import http_date
|
|
|
|
|
|
|
|
|
|
|
|
class SetCookieTests(SimpleTestCase):
|
|
|
|
def test_near_expiration(self):
|
|
|
|
"""Cookie will expire when a near expiration time is provided."""
|
|
|
|
response = HttpResponse()
|
|
|
|
# There's a timing weakness in this test; The expected result for
|
|
|
|
# max-age requires that there be a very slight difference between the
|
|
|
|
# evaluated expiration time and the time evaluated in set_cookie(). If
|
|
|
|
# this difference doesn't exist, the cookie time will be 1 second
|
|
|
|
# larger. The sleep guarantees that there will be a time difference.
|
2022-03-23 11:15:36 +00:00
|
|
|
expires = datetime.now(tz=timezone.utc).replace(tzinfo=None) + timedelta(
|
|
|
|
seconds=10
|
|
|
|
)
|
2018-01-08 16:24:26 +00:00
|
|
|
time.sleep(0.001)
|
2022-02-03 19:24:19 +00:00
|
|
|
response.set_cookie("datetime", expires=expires)
|
|
|
|
datetime_cookie = response.cookies["datetime"]
|
|
|
|
self.assertEqual(datetime_cookie["max-age"], 10)
|
2018-01-08 16:24:26 +00:00
|
|
|
|
|
|
|
def test_aware_expiration(self):
|
|
|
|
"""set_cookie() accepts an aware datetime as expiration time."""
|
|
|
|
response = HttpResponse()
|
2022-03-23 11:15:36 +00:00
|
|
|
expires = datetime.now(tz=timezone.utc) + timedelta(seconds=10)
|
2018-01-08 16:24:26 +00:00
|
|
|
time.sleep(0.001)
|
2022-02-03 19:24:19 +00:00
|
|
|
response.set_cookie("datetime", expires=expires)
|
|
|
|
datetime_cookie = response.cookies["datetime"]
|
|
|
|
self.assertEqual(datetime_cookie["max-age"], 10)
|
2018-01-08 16:24:26 +00:00
|
|
|
|
|
|
|
def test_create_cookie_after_deleting_cookie(self):
|
|
|
|
"""Setting a cookie after deletion clears the expiry date."""
|
|
|
|
response = HttpResponse()
|
2022-02-03 19:24:19 +00:00
|
|
|
response.set_cookie("c", "old-value")
|
|
|
|
self.assertEqual(response.cookies["c"]["expires"], "")
|
|
|
|
response.delete_cookie("c")
|
|
|
|
self.assertEqual(
|
|
|
|
response.cookies["c"]["expires"], "Thu, 01 Jan 1970 00:00:00 GMT"
|
|
|
|
)
|
|
|
|
response.set_cookie("c", "new-value")
|
|
|
|
self.assertEqual(response.cookies["c"]["expires"], "")
|
2018-01-08 16:24:26 +00:00
|
|
|
|
|
|
|
def test_far_expiration(self):
|
|
|
|
"""Cookie will expire when a distant expiration time is provided."""
|
|
|
|
response = HttpResponse()
|
2023-01-16 04:37:25 +00:00
|
|
|
future_datetime = datetime(
|
|
|
|
date.today().year + 2, 1, 1, 4, 5, 6, tzinfo=timezone.utc
|
|
|
|
)
|
|
|
|
response.set_cookie("datetime", expires=future_datetime)
|
2022-02-03 19:24:19 +00:00
|
|
|
datetime_cookie = response.cookies["datetime"]
|
2018-01-08 16:24:26 +00:00
|
|
|
self.assertIn(
|
2022-02-03 19:24:19 +00:00
|
|
|
datetime_cookie["expires"],
|
2018-01-08 16:24:26 +00:00
|
|
|
# assertIn accounts for slight time dependency (#23450)
|
2023-01-16 04:37:25 +00:00
|
|
|
(
|
|
|
|
format_datetime_rfc5322(future_datetime, usegmt=True),
|
|
|
|
format_datetime_rfc5322(future_datetime.replace(second=7), usegmt=True),
|
|
|
|
),
|
2018-01-08 16:24:26 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
def test_max_age_expiration(self):
|
|
|
|
"""Cookie will expire if max_age is provided."""
|
|
|
|
response = HttpResponse()
|
|
|
|
set_cookie_time = time.time()
|
|
|
|
with freeze_time(set_cookie_time):
|
2022-02-03 19:24:19 +00:00
|
|
|
response.set_cookie("max_age", max_age=10)
|
|
|
|
max_age_cookie = response.cookies["max_age"]
|
|
|
|
self.assertEqual(max_age_cookie["max-age"], 10)
|
|
|
|
self.assertEqual(max_age_cookie["expires"], http_date(set_cookie_time + 10))
|
2018-01-08 16:24:26 +00:00
|
|
|
|
2020-09-04 12:04:21 +00:00
|
|
|
def test_max_age_int(self):
|
|
|
|
response = HttpResponse()
|
2022-02-03 19:24:19 +00:00
|
|
|
response.set_cookie("max_age", max_age=10.6)
|
|
|
|
self.assertEqual(response.cookies["max_age"]["max-age"], 10)
|
2020-09-04 12:04:21 +00:00
|
|
|
|
2022-03-04 12:57:10 +00:00
|
|
|
def test_max_age_timedelta(self):
|
|
|
|
response = HttpResponse()
|
|
|
|
response.set_cookie("max_age", max_age=timedelta(hours=1))
|
|
|
|
self.assertEqual(response.cookies["max_age"]["max-age"], 3600)
|
|
|
|
|
2022-03-04 13:05:07 +00:00
|
|
|
def test_max_age_with_expires(self):
|
|
|
|
response = HttpResponse()
|
|
|
|
msg = "'expires' and 'max_age' can't be used together."
|
|
|
|
with self.assertRaisesMessage(ValueError, msg):
|
|
|
|
response.set_cookie(
|
|
|
|
"max_age", expires=datetime(2000, 1, 1), max_age=timedelta(hours=1)
|
|
|
|
)
|
|
|
|
|
2018-01-08 16:24:26 +00:00
|
|
|
def test_httponly_cookie(self):
|
|
|
|
response = HttpResponse()
|
2022-02-03 19:24:19 +00:00
|
|
|
response.set_cookie("example", httponly=True)
|
|
|
|
example_cookie = response.cookies["example"]
|
|
|
|
self.assertIn(
|
|
|
|
"; %s" % cookies.Morsel._reserved["httponly"], str(example_cookie)
|
|
|
|
)
|
|
|
|
self.assertIs(example_cookie["httponly"], True)
|
2018-01-08 16:24:26 +00:00
|
|
|
|
|
|
|
def test_unicode_cookie(self):
|
2020-04-18 14:46:05 +00:00
|
|
|
"""HttpResponse.set_cookie() works with Unicode data."""
|
2018-01-08 16:24:26 +00:00
|
|
|
response = HttpResponse()
|
2022-02-03 19:24:19 +00:00
|
|
|
cookie_value = "清風"
|
|
|
|
response.set_cookie("test", cookie_value)
|
|
|
|
self.assertEqual(response.cookies["test"].value, cookie_value)
|
2018-01-08 16:24:26 +00:00
|
|
|
|
2018-04-14 00:58:31 +00:00
|
|
|
def test_samesite(self):
|
|
|
|
response = HttpResponse()
|
2022-02-03 19:24:19 +00:00
|
|
|
response.set_cookie("example", samesite="None")
|
|
|
|
self.assertEqual(response.cookies["example"]["samesite"], "None")
|
|
|
|
response.set_cookie("example", samesite="Lax")
|
|
|
|
self.assertEqual(response.cookies["example"]["samesite"], "Lax")
|
|
|
|
response.set_cookie("example", samesite="strict")
|
|
|
|
self.assertEqual(response.cookies["example"]["samesite"], "strict")
|
2018-04-14 00:58:31 +00:00
|
|
|
|
|
|
|
def test_invalid_samesite(self):
|
2019-10-09 11:42:55 +00:00
|
|
|
msg = 'samesite must be "lax", "none", or "strict".'
|
|
|
|
with self.assertRaisesMessage(ValueError, msg):
|
2022-02-03 19:24:19 +00:00
|
|
|
HttpResponse().set_cookie("example", samesite="invalid")
|
2018-04-14 00:58:31 +00:00
|
|
|
|
2018-01-08 16:24:26 +00:00
|
|
|
|
|
|
|
class DeleteCookieTests(SimpleTestCase):
|
|
|
|
def test_default(self):
|
|
|
|
response = HttpResponse()
|
2022-02-03 19:24:19 +00:00
|
|
|
response.delete_cookie("c")
|
|
|
|
cookie = response.cookies["c"]
|
|
|
|
self.assertEqual(cookie["expires"], "Thu, 01 Jan 1970 00:00:00 GMT")
|
|
|
|
self.assertEqual(cookie["max-age"], 0)
|
|
|
|
self.assertEqual(cookie["path"], "/")
|
|
|
|
self.assertEqual(cookie["secure"], "")
|
|
|
|
self.assertEqual(cookie["domain"], "")
|
|
|
|
self.assertEqual(cookie["samesite"], "")
|
2018-01-04 17:53:35 +00:00
|
|
|
|
|
|
|
def test_delete_cookie_secure_prefix(self):
|
|
|
|
"""
|
|
|
|
delete_cookie() sets the secure flag if the cookie name starts with
|
|
|
|
__Host- or __Secure- (without that, browsers ignore cookies with those
|
|
|
|
prefixes).
|
|
|
|
"""
|
|
|
|
response = HttpResponse()
|
2022-02-03 19:24:19 +00:00
|
|
|
for prefix in ("Secure", "Host"):
|
2018-01-04 17:53:35 +00:00
|
|
|
with self.subTest(prefix=prefix):
|
2022-02-03 19:24:19 +00:00
|
|
|
cookie_name = "__%s-c" % prefix
|
2018-01-04 17:53:35 +00:00
|
|
|
response.delete_cookie(cookie_name)
|
2022-02-03 19:24:19 +00:00
|
|
|
self.assertIs(response.cookies[cookie_name]["secure"], True)
|
2020-07-16 06:16:58 +00:00
|
|
|
|
|
|
|
def test_delete_cookie_secure_samesite_none(self):
|
|
|
|
# delete_cookie() sets the secure flag if samesite='none'.
|
|
|
|
response = HttpResponse()
|
2022-02-03 19:24:19 +00:00
|
|
|
response.delete_cookie("c", samesite="none")
|
|
|
|
self.assertIs(response.cookies["c"]["secure"], True)
|
2020-07-16 06:16:58 +00:00
|
|
|
|
|
|
|
def test_delete_cookie_samesite(self):
|
|
|
|
response = HttpResponse()
|
2022-02-03 19:24:19 +00:00
|
|
|
response.delete_cookie("c", samesite="lax")
|
|
|
|
self.assertEqual(response.cookies["c"]["samesite"], "lax")
|