Ecosyste.ms: Awesome
An open API service indexing awesome lists of open source software.
https://github.com/twtrubiks/csrf-tutorial
Use Django To Introduce CSRF and Cookies , Session 📝
https://github.com/twtrubiks/csrf-tutorial
cookie csrf django double-cookie-submit session synchronizer-token-pattern
Last synced: about 2 months ago
JSON representation
Use Django To Introduce CSRF and Cookies , Session 📝
- Host: GitHub
- URL: https://github.com/twtrubiks/csrf-tutorial
- Owner: twtrubiks
- License: mit
- Created: 2017-10-09T14:52:06.000Z (about 7 years ago)
- Default Branch: master
- Last Pushed: 2018-03-27T03:24:18.000Z (almost 7 years ago)
- Last Synced: 2024-10-30T19:03:31.527Z (2 months ago)
- Topics: cookie, csrf, django, double-cookie-submit, session, synchronizer-token-pattern
- Language: Python
- Size: 32.2 KB
- Stars: 50
- Watchers: 3
- Forks: 4
- Open Issues: 1
-
Metadata Files:
- Readme: README.md
- License: LICENSE
Awesome Lists containing this project
README
# CSRF-tutorial 📝
Use Django To Introduce CSRF and Cookies , Session 📝
* [Youtube Tutorial](https://youtu.be/J075bvFA5-Q)
* [Youtube Tutorial - Django 防範 CSRF 原理](https://youtu.be/0BOFeQCEfGU)## 前言
大家在看這篇文篇時,建議先看我之前的 [Same-Origin Policy and CORS Tutorial 📝](https://github.com/twtrubiks/CORS-tutorial),
因為這篇文章算是 [Same-Origin Policy and CORS Tutorial 📝](https://github.com/twtrubiks/CORS-tutorial) 的後續介紹,建議大家有
空的話先看一下 :smile:,本篇文章我將會說明 CSRF 並且透過 [Django](https://github.com/django/django) 簡單實作介紹,也會
順便帶大家認識 Cookies 和 Session。
## CSRF
什麼是 CSRF? 他可以吃嗎 :joy:
CSRF 全名為 Cross Site Request Forgery 也被稱為 One-Click Attack ,又稱 跨站請求偽造,
為什麼我要特別提到 CSRF ?
先來思考一個問題,我們之前有介紹過 [Same-Origin Policy and CORS Tutorial 📝](https://github.com/twtrubiks/CORS-tutorial),
而且 CORS 預設也是不能跨站存取,那這樣還會有安全上的問題嗎?
答案是有 :flushed: 就是我們現在要介紹的 CSRF :open_mouth:
先說明一下 CSRF 的攻擊行為,使用者( 受害者 )在不知情的情況下,被其他網域
借用身份來完成受害者未經同意的 HTTP request( 也就是偽造出使用者本人發出的
request )。為什麼會這樣,原因是因為瀏覽器的機制,你只要發送 request 給某個
網域,就會把關聯的 cookie 一起帶上去。如果使用者是登入狀態,那這個 request
當然就包含了他的資訊( 例如說 session id ),這 request 看起來就像是使用者本
人發出去的。
由於這裡提到了 Cookies 和 Session,不管各位懂不懂,請允許我任性的簡單介紹一下 :stuck_out_tongue_closed_eyes:
## 介紹 Cookies 和 Session
先簡單解釋一下 Cookies 和 Session 的不同,**Cookie 儲存在 client 端,Session 儲存在 server 端。**
### Cookies
最常看到的 Cookie 應用是在填寫表單,大家一定都有在網頁上填寫資料到一半( 或
曾經填寫過這個表單 ),然後不小心關掉或下一次有再進到同樣的網頁填表單時,他
會幫你自動填入,這就是透過 Cookie 完成。另外一個常見的就是使用者帳號密碼的保
存,下次登入時可以幫你自動填入。
Cookies 的特性:
* Domain specific,只對同一個 domain 起作用。舉個例子,在 *.twtrubiks.com 存入的 cookie,不會出現在 *.not-twtrubiks.com。
* 生命週期,預設為當你關閉瀏覽器時,這些資料就會被刪除,不過,我們可以透過 cookie 的 expires 屬性設定 `max-age` 來決定保存多久後會刪除。
### Session
Session 是搭配 Cookie 的一種技術,Cookie 是在 Client 端建立一個文件用來暫存一些資
料或是網頁的狀態,但因為某些敏感資料存在 Client 端會有安全性問題,就是資料容易被
偽造,因為 Cookie 中的所有資料在 Client 端都可以被修改 ( 雖然也可以透過加密防範 ),
所以一些重要的資料就不適合放在 Cookie 中,而且 Cookie 如果資料太多也會影響傳輸效
率,因此才把敏感資料或狀態儲存在 Server 端,也就是 Session 。
當你瀏覽一個網頁時,Server 端會隨機產生一段字串給你,然後存在你的 Cookie 中,通常
是 session id,當你下一次瀏覽時,Cookie 會帶上 session id,然後經過 Server 端的資料
比對,就知道你是哪個使用者。
Session 儲存方式有下列幾種:
* 記憶體:MemoryStore,適合開發( development )時,因為會有 no cross-process caching 的問題。
* Cookie:將 Session 存在 Cookie 中,缺點是會增加傳輸量。
* Cache 快取:常見的有 [Redis](https://redis.io/) 或 [Memcached](https://memcached.org/),這個方法是比較常見的方法。
* 資料庫:速度比 Cache 慢。( [Django](https://github.com/django/django) 預設是存在 DB 中 )詳細的 Django Session 可參考 [https://docs.djangoproject.com/en/1.11/topics/http/sessions/](https://docs.djangoproject.com/en/1.11/topics/http/sessions/)
## 使用 Django 介紹 CSRF 攻擊情境
先介紹一下裡面的資料夾
csrf_tutorial_backed 為 backed,run 起來為 [http://127.0.0.1:8000/](http://127.0.0.1:8000/)
django_attack_site 為模擬攻擊( 被加料 )的網站,run 起來為 [http://127.0.0.1:8002/](http://127.0.0.1:8002/)
我們今天的主角是阿鬼 :satisfied: 阿鬼是一個管理員,管理文章評論是否可以刪除,如下圖
![](https://i.imgur.com/n5PaM90.png)
阿鬼登入了 [http://127.0.0.1:8000/](http://127.0.0.1:8000/),並且阿鬼**沒登出**網站,
( 登入的部份我借用 **Django admin** 幫我完成 ,我的 帳號/密碼 設定為 twtrubiks/password123 )
溫馨小提醒:heart:
這裡我們先把 csrf_tutorial_backed 裡面內建的防禦機制關閉,也就是先註解掉 MIDDLEWARE 中的
`django.middleware.csrf.CsrfViewMiddleware`,詳細的後面會再做解釋。
阿鬼此時又瀏覽了其他的網站 [http://127.0.0.1:8002/]( http://127.0.0.1:8002/),這個網頁中包含了惡意注入的 js,
( 也有可能是一封信或網頁中的一段惡意程式 )
惡意 js 程式碼如下
```html
document.getElementById("csrf-form").submit()
```上面 js 的使用方法以及來源可參考 [example-of-silently-submitting-a-post-form-csrf](https://stackoverflow.com/questions/17940811/example-of-silently-submitting-a-post-form-csrf)
這個 js 很可怕,你不點他,他都會自動幫你送出 :scream:
當阿鬼點了 [http://127.0.0.1:8002/]( http://127.0.0.1:8002/) 之後,阿鬼的某個檔案( id 為 69)就莫名奇妙的被刪除了,
[http://127.0.0.1:8002/]( http://127.0.0.1:8002/) 只有一個 submit ,就算沒點,資料也被砍了
![](https://i.imgur.com/wlszNK1.png)
我們觀察一下 Cookie,你會發現 sessionid 在裡面,Server 端當然認為你是本人,
所以合理的把檔案砍了
![](https://i.imgur.com/AW4G1MH.png)
但阿鬼根本不知道阿~
我們來確認一下阿鬼的資料還在不在,id 為 69 的資料的確被砍了
![](https://i.imgur.com/aZaprXU.png)
這也是為什麼 CSRF 會被叫做 One-Click Attack 的關係,甚至可以讓你連點都不用點,
你的資料就不知不覺的消失了 :scream:
## 防範 CSRF 攻擊
剛剛介紹了 CSRF 的攻擊行為,大家一定覺得很可怕 :grimacing:
但其實不用太擔心,也是有防範的方法 :relaxed:
現在大部分的 Framework 都有內建防禦 CSRF 的功能,要開啟也非常簡單。
像是在 Django 中的 MIDDLEWARE 裡就有防禦機制
```python
MIDDLEWARE = [
......
'django.middleware.csrf.CsrfViewMiddleware',
......
]
```我們剛剛在前面模擬 CSRF 攻擊成功,就是將 `django.middleware.csrf.CsrfViewMiddleware` 註解掉,
詳細原理以及步驟可參考 [https://docs.djangoproject.com/en/1.11/ref/csrf/](https://docs.djangoproject.com/en/1.11/ref/csrf/)
如果把這個打開 ( Django 預設也是開啟的,而且不建議關閉 ) ,就可以成功
阻擋 CSRF 攻擊,如果你再試一次剛剛的流程,這次你會發現資料沒有被刪
除,並且透過 Server 端 Console 看到 CSRF token missing or incorrect ( 403 )
![](https://i.imgur.com/X5dvCc1.png)
題外話,那使用者該如何保護自己避免 CSRF 攻擊呢 ?
最簡單的就是每次使用完網站就登出,這樣就可以避免掉 CSRF,
不過也不能把避免 CSRF 的攻擊,都交給使用者防範,因為我知道
大家都和阿鬼一樣很懶不喜歡登出 :kissing:
所以,Server 的防範(雖然很簡單)還是要做好 :grinning:
如果看到這邊你還是不太了解,建議可以參考影片 [Youtube Tutorial](https://youtu.be/J075bvFA5-Q)
## Django 防範 CSRF 原理
因為認為還是有必要了解原理,所以增加這段來說明原理 :memo:
* [Youtube Tutorial - Django 防範 CSRF 原理](https://youtu.be/0BOFeQCEfGU)
Django 主要有用到下面兩個原理來防範 CSRF
* Synchronizer Token Pattern ( STP )
* Double Submit Cookie下面我將介紹 Django 防範 CSRF 的原理
***Synchronizer Token Pattern ( STP )***
首先,server 會產生一組隨機的 token,並且加在 form 上面,
這個欄位是 hidden,名稱為 `csrfmiddlewaretoken`,如下面程式碼
( `csrfmiddlewaretoken` 這個值每次都會不一樣 )
```html
Name
Text
POST
```
***Double Submit Cookie***
另一個 token 是當你成功登入時,會儲存在 client side ( 使用者端 ) 的 Cookies 中,名稱為 `csrftoken`,
這個 token 是唯一的,如果登出再登入,此 token 也會不一樣。
![](https://i.imgur.com/JDeRwZ2.png)
接著當使用者按下 submit 時,server 會將 cookie 裡面的 `csrftoken` 以及 form 裡面的 `csrfmiddlewaretoken` 進行比對
( **注意,會將 `csrftoken` 以及 `csrfmiddlewaretoken` 這兩個值解密後再比對** ),
如果相同,代表這個請求合法,否則,會返回 403 Forbidden。
這裡我先解釋一下解密的部分,
我們先看一下 cookie 裡面的 `csrftoken`,值為 `Bsi2Aqys1l8jUnyvSPZHAQaEAI5ucul2eAEqgYU4yARbz6O0MsvnURfMGjxhWyNm`,
再看 form 裡面的 `csrfmiddlewaretoken` ,值為 `MPCHeo4bEhSu9ivqvyb7KJbRnqiXJ7kapXY5UWqNbwBmO1LVpbHN4KgZt1KKtbMu`,
咦 ? 這兩個值根本不一樣啊 :confused:
這就是我說的解密的部分,這兩個 token 都有進行加密,
你可以將這兩個值丟進去 [csrf.py](https://github.com/django/django/blob/master/django/middleware/csrf.py#L56) 裡的 `_unsalt_cipher_token` function,
你會發現結果都是 `NiwyQIwMHpT2PTqF4NGQubfigLCXUeCu` :open_mouth:
如果你想玩玩看,我將 `_unsalt_cipher_token` function 放在 [decrypt_token.py](https://github.com/twtrubiks/CSRF-tutorial/blob/master/csrf_tutorial_backed/decrypt_token.py) 這邊 :relaxed:
以上就是整個 Django 防範 CSRF 的原理 :smirk:
更多詳細的介紹可參考 [https://docs.djangoproject.com/en/1.11/ref/csrf/#how-it-works](https://docs.djangoproject.com/en/1.11/ref/csrf/#how-it-works)
## 結論
這次透過 [Django](https://github.com/django/django) 介紹 CSRF,相信大家以後聽到這個名詞一定不陌生了,其實,多數框架都有
防禦機制,不用太擔心,但是儘管都有內建防禦機制,我認為還是有必要了解一下什麼是 CSRF
,免得以後看到或被問到的時候,產生了一堆:question::question::question::question:
最後,因為文章內容很多是我去網路上查資料,自己再加以整理的,如果有介紹不清楚或有錯誤
的地方,歡迎大家 issuse 給我,希望大家會喜歡,謝謝大家 :relaxed:
## 執行環境
* Python 3.6.2
## Reference
* [Same-Origin Policy, CORS and CSRF](https://hackmd.io/s/H1cY3TTYe#same-origin-policy-cors-and-csrf)
* [TechBridge-讓我們來談談 CSRF](http://blog.techbridge.cc/2017/02/25/csrf-introduction/)## Donation
如果有幫助到您,也想鼓勵我的話,歡迎請我喝一杯咖啡:laughing:
![alt tag](https://i.imgur.com/LRct9xa.png)
[贊助者付款](https://payment.opay.tw/Broadcaster/Donate/9E47FDEF85ABE383A0F5FC6A218606F8)
## License
MIT license