defprocess_response(self, request, response): """ If request.session was modified, or if the configuration is to save the session every time, save the changes and set a session cookie or delete the session cookie if the session has been emptied. """ try: accessed = request.session.accessed modified = request.session.modified empty = request.session.is_empty() except AttributeError: return response # First check if we need to delete this cookie. # The session should be deleted only if the session is entirely empty. if settings.SESSION_COOKIE_NAME in request.COOKIES and empty: response.delete_cookie( settings.SESSION_COOKIE_NAME, path=settings.SESSION_COOKIE_PATH, domain=settings.SESSION_COOKIE_DOMAIN, ) patch_vary_headers(response, ('Cookie',)) else: if accessed: patch_vary_headers(response, ('Cookie',)) if (modified or settings.SESSION_SAVE_EVERY_REQUEST) andnot empty: if request.session.get_expire_at_browser_close(): max_age = None expires = None else: max_age = request.session.get_expiry_age() expires_time = time.time() + max_age expires = http_date(expires_time) # Save the session data and refresh the client cookie. # Skip session save for 500 responses, refs #3881. if response.status_code != 500: try: request.session.save() except UpdateError: raise SuspiciousOperation( "The request's session was deleted before the " "request completed. The user may have logged " "out in a concurrent request, for example." ) response.set_cookie( settings.SESSION_COOKIE_NAME, request.session.session_key, max_age=max_age, expires=expires, domain=settings.SESSION_COOKIE_DOMAIN, path=settings.SESSION_COOKIE_PATH, secure=settings.SESSION_COOKIE_SECURE orNone, httponly=settings.SESSION_COOKIE_HTTPONLY orNone, samesite=settings.SESSION_COOKIE_SAMESITE, ) return response
@classmethod defget_model_class(cls): # Avoids a circular import and allows importing SessionStore when # django.contrib.sessions is not in INSTALLED_APPS. from django.contrib.sessions.models import Session return Session
defcreate(self): whileTrue: self._session_key = self._get_new_session_key() try: # Save immediately to ensure we have a unique entry in the # database. self.save(must_create=True) except CreateError: # Key wasn't unique. Try again. continue self.modified = True return
defcreate_model_instance(self, data): """ Return a new instance of the session model object, which represents the current session state. Intended to be used for saving the session data to the database. """ return self.model( session_key=self._get_or_create_session_key(), session_data=self.encode(data), expire_date=self.get_expiry_date(), )
defsave(self, must_create=False): """ Save the current session data to the database. If 'must_create' is True, raise a database error if the saving operation doesn't create a new entry (as opposed to possibly updating an existing entry). """ if self.session_key isNone: return self.create() data = self._get_session(no_load=must_create) obj = self.create_model_instance(data) using = router.db_for_write(self.model, instance=obj) try: with transaction.atomic(using=using): obj.save(force_insert=must_create, force_update=not must_create, using=using) except IntegrityError: if must_create: raise CreateError raise except DatabaseError: ifnot must_create: raise UpdateError raise
defdelete(self, session_key=None): if session_key isNone: if self.session_key isNone: return session_key = self.session_key try: self.model.objects.get(session_key=session_key).delete() except self.model.DoesNotExist: pass
defprocess_response(self, request, response): """ If request.session was modified, or if the configuration is to save the session every time, save the changes and set a session cookie or delete the session cookie if the session has been emptied. """ try: accessed = request.session.accessed # True modified = request.session.modified # True empty = request.session.is_empty() # False except AttributeError: return response # First check if we need to delete this cookie. # The session should be deleted only if the session is entirely empty. if settings.SESSION_COOKIE_NAME in request.COOKIES and empty: response.delete_cookie( settings.SESSION_COOKIE_NAME, path=settings.SESSION_COOKIE_PATH, domain=settings.SESSION_COOKIE_DOMAIN, ) patch_vary_headers(response, ('Cookie',)) else: if accessed: patch_vary_headers(response, ('Cookie',)) if (modified or settings.SESSION_SAVE_EVERY_REQUEST) andnot empty: if request.session.get_expire_at_browser_close(): max_age = None expires = None else: max_age = request.session.get_expiry_age() expires_time = time.time() + max_age expires = http_date(expires_time) # Save the session data and refresh the client cookie. # Skip session save for 500 responses, refs #3881. if response.status_code != 500: try: request.session.save() except UpdateError: raise SuspiciousOperation( "The request's session was deleted before the " "request completed. The user may have logged " "out in a concurrent request, for example." ) response.set_cookie( settings.SESSION_COOKIE_NAME, request.session.session_key, max_age=max_age, expires=expires, domain=settings.SESSION_COOKIE_DOMAIN, path=settings.SESSION_COOKIE_PATH, secure=settings.SESSION_COOKIE_SECURE orNone, httponly=settings.SESSION_COOKIE_HTTPONLY orNone, samesite=settings.SESSION_COOKIE_SAMESITE, ) return response
a. accessed和modified为True是在process_request方法中改成True了,现在看一下empty为什么等于False
empty = request.session.is_empty():
1 2 3 4 5 6 7 8 9 10 11 12
classSessionBase: ... ... defis_empty(self): "Return True when there is no session_key and the session is empty." try: returnnot self._session_key andnot self._session_cache # self._session_key是没有的,所以为False, # self._session_cache 是有值的,所以为True # 因为是and,False and True = False except AttributeError: returnTrue ... ...
b. 然后走下去,是一系列的判断,判断是否关闭浏览器等等。。。走到下面这段代码,是判断页面有无报错,无报错则调用db模块下的SessionStore类下的save()方法
classSessionStore(SessionBase): ... ... defcreate(self): whileTrue: self._session_key = self._get_new_session_key() try: # Save immediately to ensure we have a unique entry in the # database. self.save(must_create=True) except CreateError: # Key wasn't unique. Try again. continue self.modified = True return
... ...
defsave(self, must_create=False): """ Save the current session data to the database. If 'must_create' is True, raise a database error if the saving operation doesn't create a new entry (as opposed to possibly updating an existing entry). """ if self.session_key isNone: return self.create() data = self._get_session(no_load=must_create) obj = self.create_model_instance(data) using = router.db_for_write(self.model, instance=obj) try: with transaction.atomic(using=using): obj.save(force_insert=must_create, force_update=not must_create, using=using) except IntegrityError: if must_create: raise CreateError raise except DatabaseError: ifnot must_create: raise UpdateError raise ... ...
d. 继续看create()方法: self._session_key = self._get_new_session_key():
e. 然后继续执行create()方法里面的save()方法:self.save(must_create=True)
f. 然后save()继续走,到data = self._get_session(no_load=must_create):在create()方法调用save()方法的时候传了一个must_create=True的参数,所以no_load=True; 总结:data = self._get_session(no_load=must_create):将视图中给session赋值的字典,赋值给data
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17
classSessionBase: ... ... def_get_session(self, no_load=False): """ Lazily load session from storage (unless "no_load" is True, when only an empty dict is stored) and store it in the current instance. """ self.accessed = True try: return self._session_cache # 在process_request的时候赋值了 except AttributeError: if self.session_key isNoneor no_load: self._session_cache = {} else: self._session_cache = self.load() return self._session_cache ... ...
g. obj = self.create_model_instance(data):将session_key,session_data,expire_date保存到model属性中,这个model就是Session模型,然后obj.save(force_insert=must_create, force_update=not must_create, using=using)将数据同步到数据库中;
1 2 3 4 5 6 7 8 9 10 11 12 13 14
classSessionStore(SessionBase): ... ... defcreate_model_instance(self, data): """ Return a new instance of the session model object, which represents the current session state. Intended to be used for saving the session data to the database. """ return self.model( # 这个model就是Session模型 session_key=self._get_or_create_session_key(), session_data=self.encode(data), expire_date=self.get_expiry_date(), ) ... ...