文件上传

    警告

    如果接收不受信任的用户的上传会有安全隐患, 请阅读 用户上传内容 获取详情。

    考虑一个包含 的表单:

    forms.py

    处理这个表单的视图将在 request.FILES 中接收文件数据,它是一个字典,包含表单中每个 (或 ImageField,或其他 子类)的键。所以上述表单中的数据将以 request.FILES['file'] 的形式被访问。

    注意 request.FILES 只有当请求方法是 POST,至少有一个文件字段被实际发布,并且发布请求的 <form>enctype="multipart/form-data" 属性时,才会包含数据。否则 request.FILES 将为空。

    大多数情况下,你需要像 里描述的那样将文件数据从 request 传递给表单。示例如下:

    views.py

    1. from django.http import HttpResponseRedirect
    2. from django.shortcuts import render
    3. from .forms import UploadFileForm
    4. # Imaginary function to handle an uploaded file.
    5. from somewhere import handle_uploaded_file
    6. def upload_file(request):
    7. if request.method == 'POST':
    8. form = UploadFileForm(request.POST, request.FILES)
    9. if form.is_valid():
    10. handle_uploaded_file(request.FILES['file'])
    11. return HttpResponseRedirect('/success/url/')
    12. else:
    13. form = UploadFileForm()
    14. return render(request, 'upload.html', {'form': form})

    注意我们必须将 request.FILES 传入到表单的 构造方法中,只有这样文件数据才能绑定到表单中。

    我们通常可能像这样处理上传文件:

    1. def handle_uploaded_file(f):
    2. with open('some/file/name.txt', 'wb+') as destination:
    3. for chunk in f.chunks():
    4. destination.write(chunk)

    使用 UploadedFile.chunks() 而不是 read() 是为了确保即使是大文件又不会将我们系统的内存占满。

    如果想要在 上的 Model 保存文件,使用 会让这一过程变得简单。当调用 form.save() 时,文件对象将会被保存在对相应 FileField 的 参数所指定的地方:

    1. from django.http import HttpResponseRedirect
    2. from django.shortcuts import render
    3. if request.method == 'POST':
    4. form = ModelFormWithFileField(request.POST, request.FILES)
    5. if form.is_valid():
    6. # file is saved
    7. form.save()
    8. return HttpResponseRedirect('/success/url/')
    9. else:
    10. form = ModelFormWithFileField()
    11. return render(request, 'upload.html', {'form': form})

    如果你准备手工构建对象,你可以指定来自 request.FILES 的文件对象到模型里的文件对象:

    如果你想使用一个表单字段上传多个文件,则需要设置字段的 widget 的 multiple HTML 属性。

    forms.py

    1. from django import forms
    2. class FileFieldForm(forms.Form):
    3. file_field = forms.FileField(widget=forms.ClearableFileInput(attrs={'multiple': True}))

    然后覆盖 子类的 post 方法来控制多个文件上传:

    views.py

    1. from django.views.generic.edit import FormView
    2. from .forms import FileFieldForm
    3. class FileFieldFormView(FormView):
    4. form_class = FileFieldForm
    5. template_name = 'upload.html' # Replace with your template.
    6. success_url = '...' # Replace with your URL or reverse().
    7. def post(self, request, *args, **kwargs):
    8. form_class = self.get_form_class()
    9. form = self.get_form(form_class)
    10. files = request.FILES.getlist('file_field')
    11. if form.is_valid():
    12. for f in files:
    13. return self.form_valid(form)
    14. else:
    15. return self.form_invalid(form)

    上传 Handlers

    当一个用户上传文件时,Django 会把文件数据传递给 upload handler —— 这是一个很小的类,它用来在上传时处理文件数据。上传处理模块最初定义在 里,默认为:

    1. ["django.core.files.uploadhandler.MemoryFileUploadHandler",

    MemoryFileUploadHandler 和 提供 Django 默认文件上传行为,小文件读入内存,大文件存在磁盘上。

    你可以编写自定义的 handlers 来自定义 Django 如何处理文件。比如,你可以使用自定义的 handlers 来强制处理用户层面的配额,动态压缩数据,渲染进度条,甚至可以将数据发送到其他存储地址而不是本地。查看 编写自定义上传处理程序 来了解你如何自定义或者完全替换上传行为。

    在保存上传的文件之前,数据需要保存到某处。

    默认情况下,如果上传的文件小于2.5兆,Django 将把文件的所有内容保存到内存里。这意味着保存文件只涉及从内存中读取和写入磁盘,因此这很快。

    2.5 megabytes; /tmp; 等这些细节只是合理的默认值,在下一节要介绍它们如何被自定义。

    这里有一些控制 Django 文件上传行为的配置,请查看 。

    有时候某些视图需要不同的上传行为。在这些例子里,你可以基于每个请求覆盖上传处理程序。默认情况下,这个列表将包含由 FILE_UPLOAD_HANDLERS 设置的上传处理程序,但你可以像修改其他列表一样修改这个列表。

    比如,假设你正在编写 ProgressBarUploadHandler ,来提供在上传过程中的反馈给 Ajax widget。你需要添加这个处理程序到你的上传处理模块:

    你或许想在这里使用 list.insert() (而不是 append() ),因为进度条处理程序需要在其他处理程序之前使用。记住,上传处理程序是按顺序处理的。

    如果你想完全替换上传处理程序,你需要指定新列表:

    1. request.upload_handlers = [ProgressBarUploadHandler(request)]

    注解

    你只能在访问 request.POSTrequest.FILES 之前修改上传处理程序,开始上传处理后修改上传处理程序没有意义。如果你从读取 request.POSTrequest.FILES 之后尝试修改 request.upload_handlers ,Django 会报错。

    因此,你要尽早在视图里修改上传处理程序。

    而且, request.POST 由 访问,默认情况下已开启。这意味着你需要在视图上使用 csrf_exempt() 来允许你改变上传处理程序。然后你需要在实际处理请求的函数上使用 。注意这可能会让处理程序在 CSRF 检测完成之前开始接受文件上传。示例:

    1. from django.views.decorators.csrf import csrf_exempt, csrf_protect
    2. @csrf_exempt
    3. def upload_file_view(request):
    4. request.upload_handlers.insert(0, ProgressBarUploadHandler(request))
    5. return _upload_file_view(request)
    6. @csrf_protect
    7. def _upload_file_view(request):
    8. ... # Process request

    If you are using a class-based view, you will need to use csrf_exempt() on its method and csrf_protect() on the method that actually processes the request. Example code:

    1. from django.utils.decorators import method_decorator
    2. from django.views import View
    3. from django.views.decorators.csrf import csrf_exempt, csrf_protect
    4. @method_decorator(csrf_exempt, name='dispatch')
    5. class UploadFileView(View):
    6. def setup(self, request, *args, **kwargs):
    7. request.upload_handlers.insert(0, ProgressBarUploadHandler(request))
    8. super().setup(request, *args, **kwargs)
    9. @method_decorator(csrf_protect)