Django 表单验证、clean() 和文件上传

Posted

技术标签:

【中文标题】Django 表单验证、clean() 和文件上传【英文标题】:Django form validation, clean(), and file upload 【发布时间】:2011-06-29 11:40:15 【问题描述】:

有人能告诉我上传的文件何时实际写入 FileField 中“upload_to”返回的位置,特别是关于字段、模型和表单验证和清理的顺序吗?

现在我的模型上有一个“干净”的方法,它假设上传的文件已经到位,所以它可以对其进行一些验证。看起来该文件尚未保存,可能只是保存在临时位置或内存中。如果是这种情况,如果我需要执行一些外部进程/程序来验证文件,我该如何“打开”它或找到它的路径?

谢谢,

伊恩

【问题讨论】:

【参考方案1】:

表单清理与实际保存文件或保存任何其他数据无关。在您运行模型实例的save() 方法之前,文件不会保存(请注意,如果您使用ModelName.objects.create(),则会自动为您调用此save() 方法)。

绑定的表单将包含一个打开的File 对象,因此您应该能够直接对该对象进行任何验证。例如:

form = MyForm(request.POST, request.FILES)
if form.is_valid():
    file_object = form.cleaned_data['myFile']
    #run any validation on the file_object, or define a clean_myFile() method 
    #  that will be run automatically when you call form.is_valid()

    model_inst = MyModel('my_file' = file_object,
                     #assign other attributes here....
                     )
    model_inst.save() #file is saved to disk here

【讨论】:

有什么方法可以在模型验证而不是表单验证中做到这一点?【参考方案2】:

你需要做什么?如果您的验证将在没有临时文件的情况下工作,您可以通过在文件字段返回的内容上调用 read() 来访问数据。

def clean_field(self):
    _file = self.cleaned_data.get('filefield')
    contents = _file.read()

如果你确实需要它在磁盘上,你知道从这里到哪里去 :) 把它写到一个临时位置并在它上面做一些魔法!

【讨论】:

我希望这个验证进入 Model.clean() 方法,而不是表单验证,但似乎这是不可能的,是对吗?上传的文件验证只能在表单级别发生,就像你所说的那样。 等等,没关系,我记得读过一篇关于如何做到这一点的 SO 帖子……让我去挖掘。抱歉找不到。如果模型已经保存,您可以访问self.field.file.read()【参考方案3】:

或将其写为自定义表单字段。这是我使用“诱变剂”库验证 MP3 文件的基本思路。

注意事项:

首先检查文件大小,如果大小正确,则写入 tmp 位置。 将文件写入 SETTINGS 中指定的临时位置检查其 MP3,然后将其删除。

代码:

from django import forms

import os
from mutagen.mp3 import MP3, HeaderNotFoundError, InvalidMPEGHeader

from django.conf import settings

class MP3FileField(forms.FileField):

    def clean(self, *args, **kwargs):
        super(MP3FileField, self).clean(*args, **kwargs)
        tmp_file = args[0]

        if tmp_file.size > 6600000:
            raise forms.ValidationError("File is too large.")

        file_path = getattr(settings,'FILE_UPLOAD_TEMP_DIR')+'/'+tmp_file.name

        destination = open(file_path, 'wb+')
        for chunk in tmp_file.chunks():
            destination.write(chunk)
        destination.close()

        try:
            audio = MP3(file_path)
            if audio.info.length > 300:
                os.remove(file_path)
                raise forms.ValidationError("MP3 is too long.")

        except (HeaderNotFoundError, InvalidMPEGHeader):
            os.remove(file_path)
            raise forms.ValidationError("File is not valid MP3 CBR/VBR format.")
        os.remove(file_path)
        return args

【讨论】:

以上是关于Django 表单验证、clean() 和文件上传的主要内容,如果未能解决你的问题,请参考以下文章

即使缺少所需的值,Django 验证也会调用 clean()

django_5:表单1——文件上传

is_valid() vs clean() django 表单

django 表单验证和字段验证

django Form验证

Django:上传多个文件。 clean_data['file'] 中需要的文件列表