溫馨提示×

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

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

Django文件存儲(chǔ) 默認(rèn)存儲(chǔ)系統(tǒng)解析

發(fā)布時(shí)間:2020-09-21 05:41:02 來(lái)源:腳本之家 閱讀:159 作者:再見(jiàn)紫羅蘭 欄目:開(kāi)發(fā)技術(shù)

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)限。

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

免責(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)容。

AI