您好,登錄后才能下訂單哦!
Django默認(rèn)使用的文件存儲(chǔ)系統(tǒng)'django.core.files.storage.FileSystemStorage'是一個(gè)本地存儲(chǔ)系統(tǒng),由settings中的DEFAULT_FILE_STORAGE值確定。
class FileSystemStorage(location=None, base_url=None, file_permissions_mode=None, directory_permissions_mode=None)
FileSystemStorage類(lèi)繼承自Storage類(lèi),location是存儲(chǔ)文件的絕對(duì)路徑,默認(rèn)值是settings中的MEDIA_ROOT值,base_url默認(rèn)值是settings中的MEDIA_URL值。
當(dāng)定義location參數(shù)時(shí),可以無(wú)視MEDIA_ROOT值來(lái)存儲(chǔ)文件:
from django.db import models from django.core.files.storage import FileSystemStorage fs = FileSystemStorage(location='/media/photos') class Car(models.Model): ... photo = models.ImageField(storage=fs)
這樣文件會(huì)存儲(chǔ)在/media/photos文件夾。
可以直接使用Django的文件存儲(chǔ)系統(tǒng)來(lái)存儲(chǔ)文件:
>>> from django.core.files.storage import default_storage >>> from django.core.files.base import ContentFile >>> path = default_storage.save('/path/to/file', ContentFile('new content')) >>> path '/path/to/file' >>> default_storage.size(path) 11 >>> default_storage.open(path).read() 'new content' >>> default_storage.delete(path) >>> default_storage.exists(path) False
可以從FileSystemStorage類(lèi)的_save方法看下上傳文件是怎么存儲(chǔ)的:
def _save(self, name, content): full_path = self.path(name) # Create any intermediate directories that do not exist. # Note that there is a race between os.path.exists and os.makedirs: # if os.makedirs fails with EEXIST, the directory was created # concurrently, and we can continue normally. Refs #16082. directory = os.path.dirname(full_path) if not os.path.exists(directory): try: if self.directory_permissions_mode is not None: # os.makedirs applies the global umask, so we reset it, # for consistency with file_permissions_mode behavior. old_umask = os.umask(0) try: os.makedirs(directory, self.directory_permissions_mode) finally: os.umask(old_umask) else: os.makedirs(directory) except OSError as e: if e.errno != errno.EEXIST: raise if not os.path.isdir(directory): raise IOError("%s exists and is not a directory." % directory) # There's a potential race condition between get_available_name and # saving the file; it's possible that two threads might return the # same name, at which point all sorts of fun happens. So we need to # try to create the file, but if it already exists we have to go back # to get_available_name() and try again. while True: try: # This file has a file path that we can move. if hasattr(content, 'temporary_file_path'): file_move_safe(content.temporary_file_path(), full_path) # This is a normal uploadedfile that we can stream. else: # This fun binary flag incantation makes os.open throw an # OSError if the file already exists before we open it. flags = (os.O_WRONLY | os.O_CREAT | os.O_EXCL | getattr(os, 'O_BINARY', 0)) # The current umask value is masked out by os.open! fd = os.open(full_path, flags, 0o666) _file = None try: locks.lock(fd, locks.LOCK_EX) for chunk in content.chunks(): if _file is None: mode = 'wb' if isinstance(chunk, bytes) else 'wt' _file = os.fdopen(fd, mode) _file.write(chunk) finally: locks.unlock(fd) if _file is not None: _file.close() else: os.close(fd) except OSError as e: if e.errno == errno.EEXIST: # Ooops, the file exists. We need a new file name. name = self.get_available_name(name) full_path = self.path(name) else: raise else: # OK, the file save worked. Break out of the loop. break if self.file_permissions_mode is not None: os.chmod(full_path, self.file_permissions_mode) # Store filenames with forward slashes, even on Windows. return force_text(name.replace('\\', '/'))
方法中可以看出,先判斷文件存儲(chǔ)的目錄是否存在,如果不存在,使用os.mkdirs()依次創(chuàng)建目錄。
根據(jù)directory_permissions_mode參數(shù)來(lái)確定創(chuàng)建的目錄的權(quán)限,應(yīng)該為(0777 &~umask)。
然后使用os.open()創(chuàng)建文件,flags參數(shù)為(os.O_WRONLY | os.O_CREAT | os.O_EXCL | getattr(os, 'O_BINARY', 0)),
這樣當(dāng)文件已存在時(shí),則報(bào)EEXIST異常,使用get_available_name()方法重新確定文件的名字。
mode為0o666,權(quán)限為(0666 &~umask)。
content為FILE對(duì)象,如一切正常,使用FILE.chunks()依次將內(nèi)容寫(xiě)入文件。
最后,根據(jù)file_permissions_mode參數(shù),修改創(chuàng)建文件的權(quán)限。
免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如果涉及侵權(quán)請(qǐng)聯(lián)系站長(zhǎng)郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。