溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點(diǎn)擊 登錄注冊 即表示同意《億速云用戶服務(wù)條款》

怎么解決Django提交表單報(bào)錯(cuò):CSRF token missing or incorrect的問題

發(fā)布時(shí)間:2021-05-08 09:52:11 來源:億速云 閱讀:409 作者:小新 欄目:開發(fā)技術(shù)

小編給大家分享一下怎么解決Django提交表單報(bào)錯(cuò):CSRF token missing or incorrect的問題,相信大部分人都還不怎么了解,因此分享這篇文章給大家參考一下,希望大家閱讀完這篇文章后大有收獲,下面讓我們一起去了解一下吧!

1、在Django提交表單時(shí)報(bào)錯(cuò):Django提交表單報(bào)錯(cuò):

CSRF token missing or incorrect

具體報(bào)錯(cuò)頁面如下:

怎么解決Django提交表單報(bào)錯(cuò):CSRF token missing or incorrect的問題

2、有道詞典翻譯后如下:

通常,當(dāng)存在真正的跨站點(diǎn)請求偽造時(shí),或者Django的CSRF機(jī)制沒有被正確使用時(shí),就會(huì)出現(xiàn)這種情況。至于郵遞表格,你須確保:

您的瀏覽器正在接受cookie。

視圖函數(shù)將一個(gè)請求傳遞給模板的呈現(xiàn)方法。

在模板中,每個(gè)POST表單中都有一個(gè){% csrf_token %}模板標(biāo)記,目標(biāo)是一個(gè)內(nèi)部URL。

如果您沒有使用CsrfViewMiddleware,那么您必須在任何使用csrf_token模板標(biāo)簽的視圖以及那些接受POST數(shù)據(jù)的視圖上使用csrf_protect。

該表單有一個(gè)有效的CSRF令牌。在登錄另一個(gè)瀏覽器選項(xiàng)卡或登錄后單擊back按鈕之后,您可能需要使用表單重新加載頁面,因?yàn)榈卿浐罅钆茣?huì)旋轉(zhuǎn)。

您將看到這個(gè)頁面的幫助部分,因?yàn)樵贒jango設(shè)置文件中有DEBUG = True。將其更改為False,將只顯示初始錯(cuò)誤消息。

您可以使用CSRF_FAILURE_VIEW設(shè)置自定義這個(gè)頁面。

3、解決辦法:

(1)、確定項(xiàng)目底下的settings.py文件,是否存在django.middleware.csrf.CsrfViewMiddleware:

MIDDLEWARE = [
  'django.middleware.security.SecurityMiddleware',
  'django.contrib.sessions.middleware.SessionMiddleware',
  'django.middleware.common.CommonMiddleware',
  'django.middleware.csrf.CsrfViewMiddleware',
  'django.contrib.auth.middleware.AuthenticationMiddleware',
  'django.contrib.messages.middleware.MessageMiddleware',
  'django.middleware.clickjacking.XFrameOptionsMiddleware',
]

(2)、如果確定settings.py文件有配置了,還是報(bào)錯(cuò),就要在form表單底下加一行代碼:

{% csrf_token %}

這樣應(yīng)該就不會(huì)報(bào)上面錯(cuò)誤了,以上內(nèi)容僅供學(xué)習(xí)參考,謝謝!

補(bǔ)充知識(shí):Django中csrf token驗(yàn)證原理

我多年沒維護(hù)的博客園,有一篇初學(xué)Django時(shí)的筆記,記錄了關(guān)于django-csrftoekn使用筆記,當(dāng)時(shí)幾乎是照抄官網(wǎng)的使用示例,后來工作全是用的flask。博客園也沒有維護(hù)。直到我的博客收到了如下評論,確實(shí)把我給問倒了,而且我也仔細(xì)研究了這個(gè)問題。

1. Django是怎么驗(yàn)證csrfmiddlewaretoken合法性的?

2. 每次刷新頁面的時(shí)候<input>中的csrf的value都會(huì)更新,每次重復(fù)登錄的時(shí)候cookie的csrf令牌都會(huì)刷新,那么這兩個(gè)csrf-token有什么區(qū)別?

怎么解決Django提交表單報(bào)錯(cuò):CSRF token missing or incorrect的問題

image.png
CSRF(Cross Site Request Forgery protection),中文簡稱跨站請求偽造。

django 第一次響應(yīng)來自某個(gè)客戶端的請求時(shí),會(huì)在服務(wù)器端隨機(jī)生成一個(gè) token,把這個(gè) token 放在 cookie 里。然后每次 POST 請求都會(huì)帶上這個(gè) token,
這樣就能避免被 CSRF 攻擊。

這樣子看起來似乎沒毛病,但是評論中的第三個(gè)問題,每次刷新頁面,form表單中的token都會(huì)刷新,而cookie中的token卻只在每次登錄時(shí)刷新。我對csrftoken的驗(yàn)證方式起了疑問,后來看了一段官方文檔的解釋。

When validating the ‘csrfmiddlewaretoken' field value, only the secret, not the full token, is compared with the secret in the cookie value. This allows the use of ever-changing tokens. While each request may use its own token, the secret remains common to all.
This check is done by CsrfViewMiddleware.

官方文檔中說到,檢驗(yàn)token時(shí),只比較secret是否和cookie中的secret值一樣,而不是比較整個(gè)token。

我又有疑問了,同一次登錄,form表單中的token每次都會(huì)變,而cookie中的token不便,django把那個(gè)salt存儲(chǔ)在哪里才能保證驗(yàn)證通過呢。

直到看到源碼。

def _compare_salted_tokens(request_csrf_token, csrf_token):
  # Assume both arguments are sanitized -- that is, strings of
  # length CSRF_TOKEN_LENGTH, all CSRF_ALLOWED_CHARS.
  return constant_time_compare(
    _unsalt_cipher_token(request_csrf_token),
    _unsalt_cipher_token(csrf_token),
  )

def _unsalt_cipher_token(token):
  """
  Given a token (assumed to be a string of CSRF_ALLOWED_CHARS, of length
  CSRF_TOKEN_LENGTH, and that its first half is a salt), use it to decrypt
  the second half to produce the original secret.
  """
  salt = token[:CSRF_SECRET_LENGTH]
  token = token[CSRF_SECRET_LENGTH:]
  chars = CSRF_ALLOWED_CHARS
  pairs = zip((chars.index(x) for x in token), (chars.index(x) for x in salt))
  secret = ''.join(chars[x - y] for x, y in pairs) # Note negative values are ok
  return secret

token字符串的前32位是salt, 后面是加密后的token, 通過salt能解密出唯一的secret。

django會(huì)驗(yàn)證表單中的token和cookie中token是否能解出同樣的secret,secret一樣則本次請求合法。

同樣也不難解釋,為什么ajax請求時(shí),需要從cookie中拿取token添加到請求頭中。

網(wǎng)上有不少關(guān)于django csrf token驗(yàn)證原理的文章都是錯(cuò)的,是因?yàn)樗麄兏静恢纁srf-token的結(jié)構(gòu)組成,我也是卡在第三條評論那.然后看了官方文檔,和CsrfViewMiddleware中間件源碼然后才恍然大悟。

以上是“怎么解決Django提交表單報(bào)錯(cuò):CSRF token missing or incorrect的問題”這篇文章的所有內(nèi)容,感謝各位的閱讀!相信大家都有了一定的了解,希望分享的內(nèi)容對大家有所幫助,如果還想學(xué)習(xí)更多知識(shí),歡迎關(guān)注億速云行業(yè)資訊頻道!

向AI問一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場,如果涉及侵權(quán)請聯(lián)系站長郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI