django - 如何解析上传的文件并将结果发送到不同的形式

django - how to parse an uploaded file and send the result to a different form

我是 django 的新手,在我的第一个应用程序中,
我想上传一个行数未知的 csv 文件,解析它的内容并显示解析的数据,然后操作这些数据并将其保存在 table.

实现此目标的最佳方法是什么?

目前我有一个视图和一个用于上传文件的表单,
我的第一次尝试是在该视图中解析上传的文件并将解析后的数据发送到第二个视图:

def UploadFileView(request):
    if request.method == 'POST':
        form = UploadFileForm(request.POST, request.FILES)
        if form.is_valid():
            upload_list = handle_uploaded_file(request.FILES['file'])
            return HttpResponseRedirect(reverse('preview_file', ?????))
    else:
        form = UploadFileForm()

    return render(request, 'upload_file.html', {'form': form,})

upload_list - 描述文件中一行的对象列表。
?????? - 为了将列表发送到第二个视图,在这里放什么?

我知道我需要以某种方式序列化此列表,以便能够将其发送到第二个视图,即 preview_file.

  1. 如何序列化它?将其设为 JSON 字符串 ?
  2. 如何将此序列化数据发送到第二个视图?

csv 文件的行数未知,可以是 3、10 或 500。

在发布这个问题时,我想到了另一种选择,想知道它是否可行以及是否是更好的选择:
不是在第一个视图中解析文件,而是将其发送到第二个视图并在该视图中调用handle_uploaded_file

将列表保存到会话:

import random
from django.shortcuts import redirect

key = 'preview%s' % random.randint(0, 999999)
request.session[key] = upload_list
return redirect('preview_file', key)

并且在 prefiew_file() 中:

from django.http import Http404

def preview_file(request, key):
    upload_list = request.session.get(key)
    if not upload_list:
        raise Http404

更新:会话大小限制取决于使用的SESSION_ENGINE。默认 django.contrib.sessions.backends.db 引擎将会话数据存储在数据库 TEXT 字段中。在 Postgres/SQLite 的情况下,此类字段最多可包含 1Gb 的数据。对于 MySQL,限制为 64Kb。

您可以为此使用 Django 表单集 (https://docs.djangoproject.com/en/dev/topics/forms/formsets/)。表单集是相同表单实例的集合。您可以定义一个显示 CSV 文件一行内容的表单,并为其创建一个表单集以显示整个文件。

在下面的示例中,我使用了以下 CSV 文件:

Name,Email
Bob,bob@example.com
Alice,alice@example.com

表示此数据的形式如下(非常简单):

class PersonForm(forms.Form):
    name = forms.CharField()
    email = forms.EmailField()

要为此创建一个表单集,请执行以下操作:

PersonFormSet = formset_factory(PersonForm)

要填充表单集,您应该设置初始数据。在视图中实例化表单时可以这样做:

def some_view(request):
    # ....
    form = PersonFormSet(
               initial=[
                   {'name': "Bob", 'email': "bob@example.com"},
                   {'name': "Alice", 'email': "alice@example.com"},
               ]
           )

或者,使用基于 class 的视图:

class SomeView(FormView):
    form_class = PersonFormSet

    def get_initial(self):
        return [
                   {'name': "Bob", 'email': "bob@example.com"},
                   {'name': "Alice", 'email': "alice@example.com"},
               ]

如您所见,一个字典列表就足够了,您需要对 CSV 数据进行最少的操作。

剩下的挑战是获取表单中的数据。这里有几种方法,但我建议您在同一视图中处理所有内容,或者保存文件并传递文件名(尽管后者可能具有安全隐患)。