django 存储 dropbox.stone_validators.ValidationError

django-storages dropbox.stone_validators.ValidationError

我正在尝试使用保管箱作为媒体存储。我正在尝试通过 django-storages 来实现。

settings.py

DEFAULT_FILE_STORAGE = 'storages.backends.dropbox.DropBoxStorage'
DROPBOX_OAUTH2_TOKEN = 'token'
DROPBOX_ROOT_PATH = '/media/'

models.py

logo = models.ImageField(upload_to=r'logo/%Y/%m/')
image = models.ImageField(upload_to=r'photos/%Y/%m/',
 help_text='Image size: Width=1080 pixel. Height=1920 pixel',)

错误

请求方式:| POST

请求URL:| http://127.0.0.1:8000/add

Django 版本: | 2.1.8

异常类型:|验证错误

异常值:| 'D:/media/10506738_10150004552801856_220367501106153455_o.jpg' 不匹配模式 '(/(.|[\r\n])|id:.)|(rev:[0-9a-f]{9,})|(ns:[0 -9]+(/.*)?)'

控制台

dropbox.stone_validators.ValidationError: 'D:/media/10506738_10150004552801856_220367501106153455_o.jpg' 不匹配模式 '(/(.|[\r\n])|id:.)|(rev:[0-9a-f] {9,})|(ns:[0-9]+(/.*)?)'

我不明白为什么会出现这个错误?

此错误消息来自 Dropbox API,表明为 Dropbox API 调用提供的 "path" 没有预期的格式。例如,如果您要上传文件,您提供的 "path" 将是您要放置上传数据的 Dropbox 帐户中的路径。

您提供的值:

D:/media/10506738_10150004552801856_220367501106153455_o.jpg

(这似乎是本地 Windows 文件系统路径。)

您提供的 Dropbox 路径应该类似于:

/media/10506738_10150004552801856_220367501106153455_o.jpg

(这将是相对于 Dropbox 根目录的路径。它没有驱动器号。)

您需要深入研究您的代码才能看到 where/why 向 Dropbox 提供的路径类型不正确。

问题出在 Windows 操作系统路径上。我在 Ubuntu 中尝试了相同的设置。它在那里完美地工作。

关于为什么会发生此错误,其他答案并不完全正确。 Dropbox 存储对象后端使用 django 实用程序 (django.utils._os.safe_join) 来验证目标操作系统的文件名。见代码here

即使您传递了 upload_to 参数,它给出了一个 unix 风格的路径(类似于 /save/path),django 实用程序确保该保存路径的根目录与操作的基本路径相匹配系统 :(。Dropbox SDK 不喜欢以所需保存路径为前缀的驱动器(又名 C:/path/file.name 不是 Dropbox 的有效保存目录)。

要在 Windows 上完成这项工作,请确保您做几件事

首先,像这样修改存储后端_full_path方法(或创建自己的子类)。

def _full_path(self, name):
    if name == '/':
        name = ''
    print('Root path in dropbox.storage : ', self.root_path)

    # If the machine is windows do not append the drive letter to file path
    if os.name == 'nt':
        final_path = os.path.join(self.root_path, name).replace('\', '/')

        # Separator on linux system
        sep = '//'
        base_path = self.root_path

        if (not os.path.normcase(final_path).startswith(os.path.normcase(base_path + sep)) and
                os.path.normcase(final_path) != os.path.normcase(base_path) and
                os.path.dirname(os.path.normcase(base_path)) != os.path.normcase(base_path)):
            raise SuspiciousFileOperation(
                'The joined path ({}) is located outside of the base path '
                'component ({})'.format(final_path, base_path))
        # TODO Testing
        print('Full file path in storage.dropbox._full_path : ', final_path)
        return final_path

    else:
        return safe_join(self.root_path, name).replace('\', '/')

其次,确保将内容和名称传递给模型 FileField。当我没有明确传递文件内容和名称时,我遇到了麻烦(关于保留文件名的上传文件上下文?)。我像这样重新实现了我的模型保存方法

def save(self, *args, **kwargs):
    # Save file

    ## Save raw entry from user ##
    # Extract files contents
    try:
        uploaded_raw_entry = kwargs['upload_raw_entry']
    except KeyError:
        raise UploadError(('No file was passed from the admin interface. ' + 
            'Make sure a ContentFile was passed when calling this models save method'))

    # Test raw_directory_path TODO Remove after testing
    print('Raw entry name from model.save : ', raw_directory_path(self, uploaded_raw_entry.name))

    with uploaded_raw_entry.open(mode='rb') as f:
        raw_entry_content = f.read()
    raw_entry_file = ContentFile(content=raw_entry_content.encode('utf-8'),
                                 name=raw_directory_path(self, uploaded_raw_entry.name))

    # Save the content file into a model field
    raw_entry_file.open(mode='rb')
    self.raw_entry.save(raw_entry_file.name, raw_entry_file, save=False)
    raw_entry_file.close()

从 Windows 开始这绝对可行,但需要额外的步骤:)

我在我的 Linux 工作站上也遇到了这个问题 - 所以 @greg 的答案的等效 Linux 答案是在你的备份名称中添加一个 "/"

import sys, os
import dropbox
from dropbox.files import WriteMode
from dropbox.exceptions import ApiError, AuthError

# Add OAuth2 access token here.
TOKEN = '<your-token>'

data_path = './data_files'
filename   = 'my-file-backup.txt'
BACKUPPATH = "/"+filename  # <----- This should do the trick!

# Uploads contents of filename to Dropbox
def backup():
    current_file_path = os.path.join(data_path, filename)
    with open(current_file_path, 'rb') as f:
        print("Uploading " + filename + " to Dropbox as " + BACKUPPATH + "...")
        try:
            dbx.files_upload(f.read(), BACKUPPATH, mode=WriteMode('overwrite'))
        except ApiError as err:
            if (err.error.is_path() and
                    err.error.get_path().reason.is_insufficient_space()):
                sys.exit("ERROR: Cannot back up; insufficient space.")
            elif err.user_message_text:
                print(err.user_message_text)
                sys.exit()
            else:
                print(err)
                sys.exit()

因此,错误消息非常混乱 - 有问题的文件名和路径不是您的本地文件名和路径,而是 Dropbox 远程文件名和路径。