mirror of
https://github.com/django/django.git
synced 2025-10-25 14:46:09 +00:00
Added optional kwargs to get_expiry_age/date.
This change allows for cleaner tests: we can test the exact output. Refs #18194: this change makes it possible to compute session expiry dates at times other than when the session is saved. Fixed #18458: the existence of the `modification` kwarg implies that you must pass it to get_expiry_age/date if you call these functions outside of a short request - response cycle (the intended use case).
This commit is contained in:
@@ -170,28 +170,52 @@ class SessionBase(object):
|
||||
|
||||
_session = property(_get_session)
|
||||
|
||||
def get_expiry_age(self, expiry=None):
|
||||
def get_expiry_age(self, **kwargs):
|
||||
"""Get the number of seconds until the session expires.
|
||||
|
||||
expiry is an optional parameter specifying the datetime of expiry.
|
||||
Optionally, this function accepts `modification` and `expiry` keyword
|
||||
arguments specifying the modification and expiry of the session.
|
||||
"""
|
||||
if expiry is None:
|
||||
try:
|
||||
modification = kwargs['modification']
|
||||
except KeyError:
|
||||
modification = timezone.now()
|
||||
# Make the difference between "expiry=None passed in kwargs" and
|
||||
# "expiry not passed in kwargs", in order to guarantee not to trigger
|
||||
# self.load() when expiry is provided.
|
||||
try:
|
||||
expiry = kwargs['expiry']
|
||||
except KeyError:
|
||||
expiry = self.get('_session_expiry')
|
||||
|
||||
if not expiry: # Checks both None and 0 cases
|
||||
return settings.SESSION_COOKIE_AGE
|
||||
if not isinstance(expiry, datetime):
|
||||
return expiry
|
||||
delta = expiry - timezone.now()
|
||||
delta = expiry - modification
|
||||
return delta.days * 86400 + delta.seconds
|
||||
|
||||
def get_expiry_date(self):
|
||||
"""Get session the expiry date (as a datetime object)."""
|
||||
expiry = self.get('_session_expiry')
|
||||
def get_expiry_date(self, **kwargs):
|
||||
"""Get session the expiry date (as a datetime object).
|
||||
|
||||
Optionally, this function accepts `modification` and `expiry` keyword
|
||||
arguments specifying the modification and expiry of the session.
|
||||
"""
|
||||
try:
|
||||
modification = kwargs['modification']
|
||||
except KeyError:
|
||||
modification = timezone.now()
|
||||
# Same comment as in get_expiry_age
|
||||
try:
|
||||
expiry = kwargs['expiry']
|
||||
except KeyError:
|
||||
expiry = self.get('_session_expiry')
|
||||
|
||||
if isinstance(expiry, datetime):
|
||||
return expiry
|
||||
if not expiry: # Checks both None and 0 cases
|
||||
expiry = settings.SESSION_COOKIE_AGE
|
||||
return timezone.now() + timedelta(seconds=expiry)
|
||||
return modification + timedelta(seconds=expiry)
|
||||
|
||||
def set_expiry(self, value):
|
||||
"""
|
||||
|
||||
@@ -40,7 +40,7 @@ class SessionStore(DBStore):
|
||||
)
|
||||
data = self.decode(s.session_data)
|
||||
cache.set(self.cache_key, data,
|
||||
self.get_expiry_age(s.expire_date))
|
||||
self.get_expiry_age(expiry=s.expire_date))
|
||||
except (Session.DoesNotExist, SuspiciousOperation):
|
||||
self.create()
|
||||
data = {}
|
||||
|
||||
@@ -197,31 +197,43 @@ class SessionTestsMixin(object):
|
||||
self.assertEqual(self.session.get_expiry_age(), settings.SESSION_COOKIE_AGE)
|
||||
|
||||
def test_custom_expiry_seconds(self):
|
||||
# Using seconds
|
||||
self.session.set_expiry(10)
|
||||
delta = self.session.get_expiry_date() - timezone.now()
|
||||
self.assertIn(delta.seconds, (9, 10))
|
||||
modification = timezone.now()
|
||||
|
||||
age = self.session.get_expiry_age()
|
||||
self.assertIn(age, (9, 10))
|
||||
self.session.set_expiry(10)
|
||||
|
||||
date = self.session.get_expiry_date(modification=modification)
|
||||
self.assertEqual(date, modification + timedelta(seconds=10))
|
||||
|
||||
age = self.session.get_expiry_age(modification=modification)
|
||||
self.assertEqual(age, 10)
|
||||
|
||||
def test_custom_expiry_timedelta(self):
|
||||
# Using timedelta
|
||||
self.session.set_expiry(timedelta(seconds=10))
|
||||
delta = self.session.get_expiry_date() - timezone.now()
|
||||
self.assertIn(delta.seconds, (9, 10))
|
||||
modification = timezone.now()
|
||||
|
||||
age = self.session.get_expiry_age()
|
||||
self.assertIn(age, (9, 10))
|
||||
# Mock timezone.now, because set_expiry calls it on this code path.
|
||||
original_now = timezone.now
|
||||
try:
|
||||
timezone.now = lambda: modification
|
||||
self.session.set_expiry(timedelta(seconds=10))
|
||||
finally:
|
||||
timezone.now = original_now
|
||||
|
||||
date = self.session.get_expiry_date(modification=modification)
|
||||
self.assertEqual(date, modification + timedelta(seconds=10))
|
||||
|
||||
age = self.session.get_expiry_age(modification=modification)
|
||||
self.assertEqual(age, 10)
|
||||
|
||||
def test_custom_expiry_datetime(self):
|
||||
# Using fixed datetime
|
||||
self.session.set_expiry(timezone.now() + timedelta(seconds=10))
|
||||
delta = self.session.get_expiry_date() - timezone.now()
|
||||
self.assertIn(delta.seconds, (9, 10))
|
||||
modification = timezone.now()
|
||||
|
||||
age = self.session.get_expiry_age()
|
||||
self.assertIn(age, (9, 10))
|
||||
self.session.set_expiry(modification + timedelta(seconds=10))
|
||||
|
||||
date = self.session.get_expiry_date(modification=modification)
|
||||
self.assertEqual(date, modification + timedelta(seconds=10))
|
||||
|
||||
age = self.session.get_expiry_age(modification=modification)
|
||||
self.assertEqual(age, 10)
|
||||
|
||||
def test_custom_expiry_reset(self):
|
||||
self.session.set_expiry(None)
|
||||
|
||||
Reference in New Issue
Block a user