2010-06-22 88 views
18
form = AddItemForm(request.POST, request.FILES) 

if form.is_valid() 

    do_stuff 

return render_to_response(blah.html, {'form':form}) 

现已形成将与字段的原始值沿着错误信息后保留一个文件,但它不保留选定的文件 如何保持选定的文件如果表单验证失败?如何使一个Django表单验证失败

+0

我有同样的问题。如果我使用模型字段,当验证失败时是否真的必须保存文件,以便验证成功时可以上传它?然后,我必须做一些垃圾回收,以删除已上传的文件,但用户从未成功完成表单。我确信Django有这样做的好方法! – PhoebeB 2010-09-07 09:35:59

回答

2

如果您使用的是modelForm,并且您成功保存了具有fileField的该模型的新实例,则Django将仅将文件保存到磁盘上。

在你的情况下,你需要做的是从request.FILES字典中获取文件,并将其保存到磁盘上。它应该看起来像这样。

input_file = request.FILES['fileformfieldname'] 
new_file = open('/path/to/file.xxx') 
new_file.write(input_file.read()) 

现在你已经保存到磁盘上的文件,你只需要记住的文件路径,这样当用户重新提交失败的形式,你可以再次打开它。

9

你想做什么的问题是,出于安全原因,浏览器将不允许文件输入框在页面加载时具有预选值。即使它仅保留来自同一页面的前一个实例的值,情况也是如此。 Django没有办法改变这一点。

如果您想避免要求用户重新选择并重新上传文件,即使验证失败,也需要保存上传的文件,然后用某些内容替换文件输入字段以表明您已经有数据。如果用户想要放入不同的文件,您可能还需要一个运行某个JavaScript的按钮来重新启用文件字段。我不认为Django为此提供了任何机制,因此您必须自己编写代码。

1

有点棘手,但这里是逻辑。

  1. 随着您的HTML输入文件,有另一个隐藏的字段。
  2. 在此隐藏字段中写入一些JavaScript以保存选定的文件路径+名称。
  3. 页面加载期间,您的JavaScript应检查隐藏字段的值,并将其设置为“文件”字段(如果有)。

在第一次加载时,隐藏字段为空,因此没有任何反应。
在选择文件时,它会将路径+名称保存在隐藏字段中。
在第二负载,隐藏字段不为空,那么它将设置文件路径

这样一来,所有的脏活累活发生在浏览器,除非你拥有有效的形式,你不必保存文件上服务器。

+6

这不会起作用。 Javascript无法更改输入类型=文件值,或者恶意用户可以将随机猜测的文件名从计算机下载。 – 2011-04-28 22:13:19

3

好的,所以我首先提出并实现了Narendra的解决方案,然后才意识到它无法工作,因为JavaScript无法设置输入类型=“文件”值。请参阅: http://www.w3schools.com/jsref/prop_fileupload_value.asp

但这里有一个可行的解决方案:形式

  1. 独立的休息,从文件需要被上传。
  2. 始终保存文件(或运行验证以查看文件是否应保存)并保存到临时模型。
  3. 在模板中,有一个隐藏字段说明临时模型实例的ID,因此如果更改传播更改文件。您可能会在服务器上保存额外的文件,但您可以在外部进行清理。
  4. 当表格减去任何文件可以保存时,保存它,将保存的文件绑定到最初的模型,并删除中间上传的文件模型。

好的,这里是一个过程的草图,它适用于我试图用一个作者的电子邮件地址(选作电子邮件有时不验证)保存书籍pdf的例子。

models.py

class Book(models.Model): 
    pdf = models.FileField("PDF", upload_to="books/") 
    author_email = models.EmailField("Author Email") 

class TempPDF(models.Model): 
    pdf = models.FileField("PDF", upload_to="books/") 

forms.py

from project_name.app_name.models import Book, TempPDF 
from django.forms import ModelForm 
from django.contrib.admin.widgets import AdminFileWidget 

class BookForm(ModelForm): 
    class Meta: 
     model = Book 
     exclude = ['pdf',] 

class TempPDFForm(ModelForm): 
    class Meta: 
     model = TempPDF 
     widgets = dict(pdf = AdminFileWidget) 
     # The AdminFileWidget will give a link to the saved file as well as give a prompt to change the file. 
     # Note: be safe and don't let malicious users upload php/cgi scripts that your webserver may try running. 

views.py

def new_book_form(request): 
    if request.method == 'POST': 
     ## Always try saving file. 
     try: 
      temp_pdf_inst = TempPDF.objects.get(id=request.POST.has_key('temp_pdf_id')) 
     except: ## should really catch specific errors, but being quick 
      temp_pdf_inst = None 
     temp_pdf_form = TempPDFForm(request.POST, request.FILES, instance=temp_pdf_inst, prefix='temp_pdf') 
     if temp_pdf_form.is_valid() and len(request.FILES) > 0: 
      temp_pdf_inst = temp_pdf_form.save() 
      temp_pdf_id = temp_pdf_inst.id 
     book_form = BookForm(request.POST, prefix='book') 

     if book_form.is_valid(): # All validation rules pass 
      book = book_form.save(commit=False) 
      book.pdf = temp_pdf_inst.pdf 
      book.save() 
      if temp_pdf_inst != None: 
       temp_pdf_inst.delete() 
      return HttpResponseRedirect('/thanks/') # Redirect after POST 
    else: 
     book_form = BookForm() # An unbound form 
     temp_pdf_form = TempPDFForm() 
     temp_pdf_id = None 
    return render_to_response('bookform_template.html', 
           dict(book_form = book_form, 
            temp_pdf_form = temp_pdf_form, 
            temp_pdf_id = temp_pdf_id) 
          ) 

bookform_template.html

<table> 
    {{ book_form }} 
    {{ temp_pdf_form }} 
    <input type="hidden" name="temp_pdf_id" value="{{ temp_pdf_id }}"> 
</table> 
9

尝试django-file-resubmit

insallation

pip install django-file-resubmit 

settings.py

INSTALLED_APPS = { 
    ... 
    'sorl.thumbnail', 
    'file_resubmit', 
    ... 
} 

CACHES = { 
    'default': { 
     'BACKEND': 'django.core.cache.backends.locmem.LocMemCache', 
    }, 
    "file_resubmit": { 
     'BACKEND': 'django.core.cache.backends.filebased.FileBasedCache', 
     "LOCATION": project_path('data/cache/file_resubmit') 
    }, 
} 

使用

from django.contrib import admin 
from file_resubmit.admin import AdminResubmitMixin 

class ModelAdmin(AdminResubmitMixin, ModelAdmin): 
    pass 

from django.forms import ModelForm 
from file_resubmit.admin import AdminResubmitImageWidget, AdminResubmitFileWidget 

class MyModelForm(forms.ModelForm) 

    class Meta: 
     model = MyModel 
     widgets = { 
      'picture': AdminResubmitImageWidget, 
      'file': AdminResubmitFileWidget, 
     } 
+0

我第一次尝试上传时,文件消失。第二次尝试是成功的。你有什么解释吗? – Benjamin 2016-11-08 10:24:21