Django 處理http請求之使用session


Django 處理http請求之使用session

by:授客 QQ1033553122 歡迎加入全國軟件測試交流群:7156436

 

測試環境

Win7

Django 1.11

 

 

Django提供了匿名會話支持。允許用戶存儲和檢索任意數據。它在服務端存儲數據,並對cookie的發送和接收做了抽象。Cookie包含了一個會話ID--並非數據本身(除非使用cookie based backend)

 

開啟會話

會話是通過中間件實現的。要開啟會話功能,需要編輯settings.py文件,編輯MIDDLEWARE變量,確保變量包含 'django.contrib.sessions.middleware.SessionMiddleware'.通過django-admin startproject創建的默認settings.py已經開啟了SessionMiddleware

如果不想使用會話,需要從MIDDLEWARE變量中移除SessionMiddleware,並且移除INSTALLED_APPS中的django.contrib.sessions

 

配置會話引擎

 

默認的,Django在數據庫中存儲會話(使用django.contrib.sessions.models.Session model)。雖然很方便,但是在某些情況下,存儲會話到別的地方會更快,所以,Django 可以配置為存儲會話數據到你的文件系統上,或者是緩存中。

 

Using database-backed sessions

必須添加'django.contrib.sessions'到settings.py中的INSTALLED_APPS配置

 

 

 

Using cached sessions

 

Using file-based sessions

 

Using cookie-based sessions

 

views視圖中使用會話

當開啟會話功能,傳遞給每個Django view視圖函數的第一個參數,都會攜帶一個會話屬性--一個類似字典的對象

 

任何時候,都可以對request.session進行讀寫,可以多次編輯。

class backends.base.SessionBase

所有會話對象的基類,擁有以下標准字典方法:

__getitem__(key)

例子: fav_color = request.session['fav_color']

 

__setitem__(keyvalue)

例子: request.session['fav_color'] = 'blue'

 

__delitem__(key)

例子: del request.session['fav_color']. 如果給定key不存在會話中,會拋出KeyError。

 

__contains__(key)

例子: 'fav_color' in request.session

 

get(keydefault=None)

根據session_key獲取值,如果key不存在則返回default參數指定的值。

例子: fav_color = request.session.get('fav_color', 'red')

 

pop(keydefault=__not_given)

例子: fav_color = request.session.pop('fav_color', 'blue')

 

keys()

獲取session_key,類型<class 'dict_keys'>

 

items()

 

#setdefault()

setdefault(session_key, value)

如果會話字典的鍵session_key不存在,則設置session_key的值為value,如果session_key已經存在則不做任何操作

 

 

clear()

 

flush()

從會話中刪除當前會話數據和會話cookie。在無法再次通過用戶瀏覽器訪問之前的會話數據時這很有用。(比如調用django.contrib.auth.logout()函數將會調用它)。

 

set_test_cookie()

設置測試cookie來判斷用戶瀏覽器是否支持cookie。出於cookie的工作方式考慮,僅在用戶發起下一次頁面請求時才可以進行判斷。

 

test_cookie_worked()

根據用戶瀏覽器是否接受test cookie返回True、False。出於cookie的工作方式考慮,必須在前一個獨立頁面請求中才可以調用set_test_cookie()。查看 Setting test cookies獲取更多信息。

 

delete_test_cookie()

刪除測試cookie

 

set_expiry(value)

為session設置過期時間。

  • 如果value為整數,無活動超過那個數,會話將過期。例如,request.session.set_expiry(300) 將設定會話過期時間為5分鍾。
  • 如果 value 為一個datetime timedelta對象,那么會話將在指定的日期/時間過期。注意,datetimetimedelta值僅在使用PickleSerializer時可序列化。 
  • 如果value0,當用戶關閉瀏覽器會話cookie將會過期。
  • 如果valueNone,將使用全局會話過期策略

讀取會話並不會改變會話的活動狀態。會話過期時間是從會話上一次被修改的時間算起的。

 

get_expiry_age()

獲取會話過期時間。對於那些沒有定義過期時間,或者設置為瀏覽器關閉后才過期的會話,將返回SESSION_COOKIE_AGE

該函數接受兩個可選關鍵詞參數:

modification: 表示會話最后修改時間的datetime對象,默認為當前時間。

expiry: 會話過期信息,一個datetime對象、或者是一個int(秒為單位)、或者None。默認為set_expiry()函數存儲在會話中的值,如果有的話,否則為None。

 

get_expiry_date()

返回會話過期日期。對於那些沒有定義過期時間,或者設置為瀏覽器關閉后才過期的會話,等同於從現在開始算起,加上SESSION_COOKIE_AGE秒后的日期。(For sessions with no custom expiration (or those set to expire at browser close), this will equal the date SESSION_COOKIE_AGE seconds from now)

該函數同get_expiry_age()一樣,接受統一的關鍵詞參數。

 

get_expire_at_browser_close()

根據用戶關閉瀏覽器用戶會話是否過期返回True、False。

 

clear_expired()

從會話存儲中移除過期會話。該類方法被 clearsessions調用。

cycle_key()

保留當前會話數據時創建一個新的會話key。django.contrib.auth.login()會調用該方法

 

例子

 

#-*- encoding:utf-8 -*-

 

 

__author__ = 'shouke'

 

from django.shortcuts import render

# Create your views here.

from django.template.response import TemplateResponse

 

# 可以把request.session當字典使用

def test_view(request):

print(request.session.keys())# 輸出鍵,數據類型:dict_keys

print(request.session.values())# 輸出值, 數據類型:dict_values

print(request.session.items())# 輸出鍵值對, 數據類型:dict_items

 

username = request.session.get('username', 'unknow') #獲取鍵對應的值,如果鍵不存在則返回unknow

print('username: %s ' % username)

username = request.session.get('myusername', 'unknow')

print('myusername: %s ' % username)# 輸出myusername: unknow

 

# request.session.setdefault(session_key, value)

# 設置session_key的默認值為value,如果session_key已存在則不設置

request.session.setdefault('myusername', 'shouke')

request.session.setdefault('myusername', 'shouke2'

username = request.session.get('myusername')

print('myusername: %s ' % username)# 輸出myusername: shouke

 

request.session['myusername'] = 'keshou'# 設置鍵的值

username = request.session['myusername']

print('myusername: %s ' % username)# 輸出myusername: keshou

 

# 刪除鍵值對

username = request.session.pop('myusername')

print('myusername: %s ' % username)

# 等價實現

# delrequest.session['myusername']

 

print(request.session.session_key)# 會話ID,# 即SESSION_COOKIE_NAME,存放在數據表.字段:django_session.session_key

print(request.session.exists(request.session.session_key))# 判斷會話ID是否存在

 

request.session.clear_expired()# 刪除過期會話(過期時間小於當前時間的會話)

request.session.delete(request.session.session_key)# 刪除指定會話

 

request.session.clear() # 清除所有會話數據

# request.session.flush() # 清除所有會話數據並刪除會話cookie

print(request.session.keys()) # 輸出[]

 

return render(request, 'website/pages/mytest.html',{})

 

注意:使用request.session.get('username') 比使用#request.session['username']安全,前者如果不存在名為username的字典key,則返回None,后者則會報錯。

會話序列化

默認的Django使用JSON序列化session數據。可以使用SESSION_SERIALIZER 配置自定義會話序列化格式。強烈推薦使用JSON序列化,特別是使用backend cookie的情況下(存在安全問題)。

Bundled serializers

class serializers.JSONSerializer

注意:保存字典類型的內容到會話中,django會對被保存的字典內容執行序列化,這會導致字典中的鍵值對失去原有的順序解決方案

所以得先采用json.dumps把內容轉為字符串,再次從會話中取出該字符串時,使用json.loads轉字符串為字典,這樣可以保持字典內容的順序不變

其它,略

class serializers.PickleSerializer

支持任意python對象,但是存在安全問題。略

 

Write your own serializer

 

會話對象指南

  • 使用普通python字符串作為request.session字典的key。這更是一種約定,而非硬性規則.
  • 下划線開頭的會話字典key保留為Django內部使用.
  • 不要使用新的對象覆蓋request.session,並且不要訪問、設置其屬性。像字典一樣使用它。

例子

用戶提交一個評論后設置has_commented為True,即限制用戶只提交一次:

def post_comment(request, new_comment):

if request.session.get('has_commented', False):

return HttpResponse("You've already commented.")

    c = comments.Comment(comment=new_comment)

c.save()

request.session['has_commented'] = True

return HttpResponse('Thanks for your comment!')

 

設置測試cookies

典型用法:

from django.http import HttpResponse
from django.shortcuts import render
 
def login(request):
if request.method=='POST':
if request.session.test_cookie_worked():
request.session.delete_test_cookie()
return HttpResponse("You're logged in.")
else:
return HttpResponse("Please enable cookies and try again.")
request.session.set_test_cookie()
return render(request,'foo/login_form.html')

注意:實踐驗證,要按以下形式寫才可以

request.session.set_test_cookie()

if request.session.test_cookie_worked():

request.session.delete_test_cookie()

else:

# do something

 

view視圖之外使用會話

注意:

以下例子,直接從django.contrib.sessions.backends.db引入SessionStore 對象,實際編碼中需要結合實際,從SESSION_ENGINE設置的對應會話引擎中引入 SessionStore:

>>> from importlib import import_module
>>> from django.conf import settings
>>> SessionStore=import_module(settings.SESSION_ENGINE).SessionStore

 

提供了在視圖之外維護session數據的api

>>> from django.contrib.sessions.backends.db import SessionStore
>>> s=SessionStore()
>>> # 由於datetime不支持按JSON格式序列化,所以存儲為對應的秒數
>>> s['last_login']=1376587691
>>> s.create()
>>> s.session_key
'2b1189a188b44ad18c35e113ac6ceead'
>>> s=SessionStore(session_key='2b1189a188b44ad18c35e113ac6ceead')
>>> s['last_login']
1376587691

SessionStore.create() 用於創建一個新的會話(比如,一個不是從會話存儲中加載並且攜帶session_key=None的會話)。 save() 用於保存一個已有會話(比如,一個從會話存儲中加載的會話)。對一個新會話執行save()調用,也可起作用,生成一個和已存在session_key沖突的session_key的機率更小。 create() 調用save()方法,假如生成的session_key已經存在,會循環調用直到生成一個未使用session_key.

如果當前使用的是django.contrib.sessions.backends.db 每個會話都是一個普通的Django model實例。Sessionmodel定義在django/contrib/sessions/models.py,因為它是一個普通的model,所以可以通過普通的Django數據庫api訪問會話:

>>>from django.contrib.sessions.models import Session
>>>s=Session.objects.get(pk='2b1189a188b44ad18c35e113ac6ceead')
>>>s.expire_date
datetime.datetime(2005, 8, 20, 13, 35, 12)

注意,必須調用 get_decoded() 來獲取session字典,因為session字典是按一定格式編碼后存儲的。

>>>s.session_data
'KGRwMQpTJ19hdXRoX3VzZXJfaWQnCnAyCkkxCnMuMTExY2ZjODI2Yj...'
>>>s.get_decoded()
{'user_id': 42}

何時存儲會話

默認的,Django僅在會話被修改后存儲到會話數據庫--也就是說,只要任意會話字典值被賦值或者刪除:

#修改會話
request.session['foo']='bar'
del request.session['foo']
request.session['foo']={}
 
# 會話未被修改,因為它更改的是request.session['foo'],而非request.session.
request.session['foo']['bar']='baz'

上述最后一種情況下,如果我們想顯示的告訴django會話已經被修改,需要增加以下語句:

request.session.modified=True

如果想要改變這種默認行為,可以設置SESSION_SAVE_EVERY_REQUEST 配置為True。這樣以后,Django將會針對每個請求存儲會話到數據庫。

注意,默認的,會話cookie僅在會話被修改、創建時才會發送給客戶端。如果SESSION_SAVE_EVERY_REQUEST被設置True,針對每個請求,都會發送會話cookie

類似的,每次發送會話cookie,都會更新會話cookie組成部分expires

如果響應狀態碼為500,不會保存會話。

 

Browser-length sessions vs. persistent sessions

可以通過修改SESSION_EXPIRE_AT_BROWSER_CLOSE配置,控制會話框架使用browser-length sessions 還是 persistent sessions 。

 

默認的,SESSION_EXPIRE_AT_BROWSER_CLOSE被設置為False,也就是說會話cookie會存儲在用戶瀏覽器中,存儲時長和SESSION_COOKIE_AGE一樣久。這樣就重新打開瀏覽器不需要重新登錄

 

如果SESSION_EXPIRE_AT_BROWSER_CLOSE被設置為TrueDjango將使用browser-length cookies – 用戶一關閉瀏覽器,cookie就失效。這樣,每次重新打開瀏覽器,用戶都要重新登錄。

 

該配置是全局配置的,可以通過調用 request.sessionset_expiry()的方法對單個視圖中的會話進行配置。

注意:一些瀏覽器,比如chrome,提供允許用戶關閉瀏覽器,重新打開瀏覽器后用戶可以不用重新登錄繼續會話操作。這會影響SESSION_EXPIRE_AT_BROWSER_CLOSE配置

 

清除會話存儲

會話后端當使用數據庫時,用戶登錄時,Django會添加一行記錄到django_session數據表,每次會話被修改時,Django會更新對應記錄,如果用戶手動退出,Django會刪除該行數據,但是如果用戶一直不退出,該行記錄不會被刪除。會話后端使用文件時也是類似這個處理過程。

 

Django 不提供過期會話自動清理,所以,需要自己定期清理過期會話。出於這個目的,Django 提供了一個清理管理命令: clearsessions. 推薦定期執行這個命令

注意,假如會話后端使用緩存,則不存這個問題,因為緩存會自動清理過期數據。

 

Settings.py中的配置

設置session cookie失效日期(默認值:2周,1209600毫秒)

SESSION_COOKIE_AGE = 1209600                             #默認

設置session cookie的域名,例子:

SESSION_COOKIE_DOMAIN = None            # 默認

設置session cookie是否只支持http傳輸,例子:

SESSION_COOKIE_HTTPONLY = True                           # 默認

設置session cookie名稱,即session的cookie保存在瀏覽器上時的key,例子:

SESSION_COOKIE_NAME = 'sessionid' #默認

設置session cookie路徑,例子:

session_cookie_name = '/'  # 默認

設置是否Https傳輸cookie,例子:

SESSION_COOKIE_SECURE = False    # 默認

配置會話引擎,例子:

SESSION_ENGINE = 'django.contrib.sessions.backends.db'   #默認

設置是否關閉瀏器session立即過期

SESSION_EXPIRE_AT_BROWSER_CLOSE = False                  #默認

設置是否每次請求都保存session,例子:

SESSION_SAVE_EVERY_REQUEST = False                       #默認

 

會話安全

  • 盡量使用JSONSerializer,而不是PickleSerializer
  • 存儲會話數據到django_session數據表。
  • 僅在需要時發送cookie

SessionStore 對象

當在內部處理會話時,Django使用來自相應會話引擎的會話存儲對象。按約定,會話存儲對象類名為SessionStore,並且位於SESSION_ENGINE指定的模塊中。

Django中可獲取的SessionStore對象繼承與SessionBase ,並實現了以下數據維護方法:

 

Extending database-backed session engines

Session IDs in URLs

 

 

參考鏈接

https://docs.djangoproject.com/en/dev/topics/http/sessions/

 


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM