您好,登錄后才能下訂單哦!
今天就跟大家聊聊有關(guān)Django中和時區(qū)相關(guān)的安全問題有哪些,可能很多人都不太了解,為了讓大家更加了解,小編給大家總結(jié)了以下內(nèi)容,希望大家根據(jù)這篇文章可以有所收獲。
在開發(fā)國際化網(wǎng)站的時候,難免會與時區(qū)打交道,通用CMS更是如此,畢竟其潛在用戶可能是來自于全球各地的。Django在時區(qū)這個問題上下了不少功夫,但是很多資深的開發(fā)者都有可能尚未完全屢清楚Django中各種時間的實際意義和使用方法,導(dǎo)致寫出錯誤的代碼;作為安全研究人員,時區(qū)問題也可能和一些安全問題掛鉤,比如優(yōu)惠券的過期時間、訂單的下單與取消時間等,如果沒有考慮時區(qū)問題,有可能將導(dǎo)致一些邏輯漏洞。
本文就從多個常用模塊開始,了解一下Django中的時區(qū)究竟是怎么回事,以及在時間的比較中可能出現(xiàn)的一些邏輯錯誤。
從“兩種時間”說起
我們都知道,在Python中表示“時間”的對象是datetime.datetime
。
其實在Python中,這個對象被分成了兩個類型:
他們的區(qū)別是:如果datetime
對象的tzinfo
屬性有設(shè)置時區(qū)值,則這個對象是一個aware datime;否則它是一個naive datetime。
舉個例子,我們平時在編寫Python腳本的時候,使用下面這行代碼獲取當(dāng)前時間:
from datetime import datetime t = datetime.now()
此時,t是一個naive datetime,因為我們沒有給他設(shè)置時區(qū):
naive的中文意思大家應(yīng)該都很熟悉,這里的大概意思就是“simple”,這是一個很簡單、原始的時間對象。實際上就是指,計算機不知道這個時間,他的時區(qū)究竟是什么,它可能代表著北京時間,也可能是UTC時間,因為我們沒有指定時區(qū),我們無法“假設(shè)”其是計算機系統(tǒng)所在的時區(qū),也無法“假設(shè)”其是UTC時區(qū)。也就是說,計算機拿到了一個naive datetime,是無法準確地定位到某一個時間點的,也無法直接轉(zhuǎn)換成一個unix時間戳。
那么相對的,aware datetime就是計算機能準確知道其時區(qū)的時間對象,他是一個準確的時間點,就落在時間軸上的某個地方,不管從哪個時區(qū)看,這個點都是絕對固定的。所以,我們可以將一個aware datetime轉(zhuǎn)換成unix時間戳。
有的同學(xué)可能比較好奇,你說naive datetime無法轉(zhuǎn)換成時間戳,那么為什么這個對象有一個timestamp()
方法呢:
原因我們查文檔可以得出結(jié)論,如果對象是naive datetime,則會以當(dāng)前系統(tǒng)本地時區(qū)為準。
Django的時區(qū)配置
回到Django。由于Django是一個國際化框架,時區(qū)相關(guān)處理自然是其必不可少的組成部分。Django的配置項中,有下面兩個選項與時區(qū)相關(guān):
USE_TZ
TIME_ZONE
USE_TZ
用來指定整個項目是否使用時區(qū),TIME_ZONE
是默認時區(qū)的值。
如果USE_TZ
的值設(shè)置為False,那么Django項目中所有時間都使用naive datetime(除非有明確指定時區(qū)的情況)。也就是說,網(wǎng)站內(nèi)存儲和使用的時間全部是TIME_ZONE
的值所指定的時區(qū)。
這樣做有一些弊端:
默認情況下,用django-admin
生成的項目,其設(shè)置中USE_TZ
等于True,這也是Django官方建議的配置。此時,在網(wǎng)站內(nèi)部存儲與使用的是UTC時間,而與用戶交互時使用TIME_ZONE
或手工的時區(qū)。
我們后文中也以Django的默認配置USE_TZ=True
為前提條件,否則也沒有討論的必要了。
Django的時間函數(shù)
Django的包django.utils.timezone
中有下面幾個常用的時間相關(guān)函數(shù):
因為開啟了USE_TZ
,Django內(nèi)部操作時間時都應(yīng)該使用aware時間,否則會出現(xiàn)異常。所以,我們在獲取當(dāng)前時間的時候,一定要使用Django自帶的now()
或localtime()
函數(shù),而不能使用Python的datetime.datetime.now()
函數(shù)。
數(shù)據(jù)庫存儲的時間
我們在使用ORM的DatetimeField時,常常會有這樣的疑慮:我們究竟應(yīng)該給DatetimeField傳入哪個時區(qū)的時間呢?
可以做個試驗,編寫下面這個model:
class Archive(models.Model): title = models.CharField('title', max_length=256) now_time = models.DateTimeField(default=timezone.now) local_time = models.DateTimeField(default=timezone.localtime)
這個model有三個屬性,title是他的名字,now_time和local_time是兩個時間,他們的默認值分別是timezone.now和timezone.localtime。
也就是說,默認情況下,now_time字段傳入的是UTC時區(qū)的當(dāng)前時間,local_time字段傳入的是本地時區(qū)的當(dāng)前時間,我這里是Asia/Shanghai
。
然后,我們創(chuàng)建一個Archive對象:
可以發(fā)現(xiàn),不管我們使用a.now_time
還是a.local_time
,讀取到的datetime對象的tzinfo都是UTC。
這也印證了Django文檔中說到的,不管傳入的時間對象時區(qū)是什么,其內(nèi)部存儲的時間均為UTC時區(qū)。但是,值得注意的是,如果我們傳入了一個不帶時區(qū)的naive datetime,將會出現(xiàn)一個警告,并使用默認時區(qū)填充其tzinfo:
模板中展示的時間
對于網(wǎng)站的用戶來說,他們想看到的時間顯然不是UTC時間,而是某一個具體時區(qū)的時間。比如,我的網(wǎng)站幾乎全部是中國用戶,那么展示時使用的時區(qū)應(yīng)該是Asia/Shanghai
。
這一部分的轉(zhuǎn)換,Django放在的模板引擎中。
Django在渲染模板變量時,將會遇到兩種與時間有關(guān)的情況:
<p>origin value: {{ object.now_time }}</p> <p>date filter: {{ object.now_time | date:'Y-m-d H:i:s' }}</p>
前者是直接將時間渲染到頁面中,后者是通過date這樣的模板filter處理后渲染在頁面中。這兩種情況在內(nèi)部處理方式略有不同此處不細表,總體而言,任意模板中變量的渲染,都會被轉(zhuǎn)換時區(qū)。
那么,脫離模板引擎,我們會得到怎樣的結(jié)果呢?
在流行的前后端分離架構(gòu)中,后端服務(wù)器通常只提供JSON格式的接口給前端,那么,我們編寫下面這樣一個view,看看返回值是什么:
from django.shortcuts import get_object_or_404 from django.http.response import JsonResponse from django.utils import timezone from . import models def json(request): object = get_object_or_404(models.Archive, pk=1) data = dict( id=object.pk, now_time=object.now_time, local_time=timezone.localtime(object.local_time) ) return JsonResponse(data=data)
返回對象的now_time,我直接將object.now_time
返回;返回對象的local_time,我將數(shù)據(jù)庫值轉(zhuǎn)換成本地時間timezone.localtime(object.local_time)
返回。
我前文說過,這兩個值在數(shù)據(jù)庫中的值是完全相等的,不過在json返回中,now_time是UTC時間,而local_time是北京時間:
也就是說,在前后端分離的網(wǎng)站中,如果直接使用Model的字段,那么前端需要負責(zé)進行時區(qū)的轉(zhuǎn)換,否則將會出現(xiàn)時間的偏差。
時間的校驗和比較
在一些業(yè)務(wù)場景下,我們可能會涉及到時間的校驗和比較,如:
我們就以付費用戶為例:用戶購買了30天的VIP會員,我們需要給用戶表中設(shè)置一個過期時間,比如下面這個model。
from django.db import models from django.utils import timezone class Account(models.Model): username = models.CharField(max_length=256) password = models.CharField(max_length=64) created_time = models.DateTimeField(default=timezone.now) expired_time = models.DateTimeField()
如果某個用戶某一個時刻對網(wǎng)站進行訪問,我們?nèi)绾闻袛嗨欠窬哂蠽IP權(quán)限呢?
通常情況下我們有兩種常見的判斷方法。一是,用戶訪問時,直接從model中取出這個對象,然后和now()
進行比較:
這種情況下,當(dāng)前時間不管是now()
還是localtime()
都不影響比較的結(jié)果,因為兩個datetime對象在比較時會考慮時差。
另一種情況是,通過ORM的queryset進行比較,等于在數(shù)據(jù)庫層面進行操作:
if models.Account.objects.filter(expired_time__gt=timezone.now()).exists(): # doing sth
Django也幫我們考慮過這種情況,即使此時我們使用本地時間timezone.localtime()
進行查詢,系統(tǒng)也會將其轉(zhuǎn)換成UTC時間傳入SQL語句:
但是,如果我們使用到了和日期、時間有關(guān)的lookups,將產(chǎn)生相反的結(jié)果。
怎么理解這個問題呢,我們還是來舉個例子。比如,網(wǎng)站以用戶注冊當(dāng)天的日子作為“會員日”(比如1月2日注冊的會員,以后每月的2日都是他的會員日),會員日這一天會給這個用戶贈送優(yōu)惠券。
那么,發(fā)送優(yōu)惠券時,我們?nèi)绾魏Y選網(wǎng)站內(nèi)會員日是今日的所有用戶?
下面這個filter是否正確?
models.Account.objects.filter(created_time__day=timezone.now().day).all()
答案是否定的,我們應(yīng)該使用timezone.localtime()
表示今天,而非timezone.now()
:
models.Account.objects.filter(created_time__day=timezone.localtime().day).all()
這是為什么呢?你不是說數(shù)據(jù)庫中存儲的都是UTC時間嗎,為何會使用到timezone.localtime()
?
原因是,Django在使用日期、時間有關(guān)的lookups時,會在數(shù)據(jù)庫層面對時間進行時區(qū)的轉(zhuǎn)換再進行比較,所以我們需要使用本地時間而不是UTC時間。
可以看看原始的SQL語句:
可見,SQL語句中使用了django_datetime_extract('day', "sample_account"."created_time", 'Asia/Shanghai', 'UTC')
將UTC時間轉(zhuǎn)換成了北京時間,因此后面比較的時候,也應(yīng)該使用北京時間。
這一點需要格外注意。時間比較的不謹慎,說小點是一個Bug,說大點就是漏洞,畢竟很多涉及到時間比較的情景,都是非常需要嚴謹?shù)摹?/p>
所以,我們總結(jié)一下:
看完上述內(nèi)容,你們對Django中和時區(qū)相關(guān)的安全問題有哪些有進一步的了解嗎?如果還想了解更多知識或者相關(guān)內(nèi)容,請關(guān)注億速云行業(yè)資訊頻道,感謝大家的支持。
免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點不代表本網(wǎng)站立場,如果涉及侵權(quán)請聯(lián)系站長郵箱:is@yisu.com進行舉報,并提供相關(guān)證據(jù),一經(jīng)查實,將立刻刪除涉嫌侵權(quán)內(nèi)容。