Django import-export-让用户导入数据

手机

我正在尝试使用django import-export来让用户导入自己的数据。我已经将其与管理员集成在一起,并且效果很好,但是我无法让用户导入端正常工作。

这是我的看法:

from .models import WordResource
from tablib import Dataset
from .models import Word
from django.contrib import messages

# Word import
def import_words(request):
    if request.method == 'POST':
        file_format = request.POST['file-format']
        word_resource = WordResource()
        dataset = Dataset()
        new_words = request.FILES['importData']

        if file_format == 'CSV':
            imported_data = dataset.load(new_words.read().decode('utf-8'),format='csv')
            result = word_resource.import_data(dataset, dry_run=True, raise_errors=True)
        elif file_format == 'XLSX':
            imported_data = dataset.load(new_words.read(),format='xlsx')
            result = word_resource.import_data(dataset, dry_run=True, raise_errors=True)

        if result.has_errors():
            messages.error(request, 'Uh oh! Something went wrong...')

        else:
            # Import now
            word_resource.import_data(dataset, dry_run=False)
            messages.success(request, 'Your words were successfully imported')


    return render(request, 'vocab/import.html')

我的WordResource:

from import_export import resources
from import_export.fields import Field
from import_export.widgets import ForeignKeyWidget

class WordResource(resources.ModelResource):
    target_word = Field(attribute='target_word', column_name='Russian')
    source_word = Field(attribute='source_word', column_name='Meaning')
    example_sentence = Field(attribute='example_sentence', column_name='Example sentence')
    fluency = Field(attribute='fluency', column_name='Fluency level')
    deck_name = Field(attribute='deck_name', column_name='Deck name')
    username = Field(attribute='username', column_name='username',widget=ForeignKeyWidget(User, 'username'))


    class Meta:
        model = Word
        fields = ("username", "target_word",'source_word','example_sentence',
        'fluency', 'deck_name',)
        import_order = fields
        skip_unchanged = True
        # exclude = ('id',)
        import_id_fields = ['username']

我的Word模型:

class Word(models.Model):
    target_word = models.CharField('Word in Russian',max_length=25,help_text="The word you want to add, in Russian")
    source_word = models.CharField('What does it mean?',max_length=25, help_text="Write down the translation in any language")
    add_to_word_list = models.BooleanField('Would you like to create a flashcard?', default=True)
    deck_name = models.CharField('Deck name', max_length=25)
    example_sentence = models.CharField(max_length=150,blank=True,help_text="It's important to learn words in context!")

    ## how well you know the word
    class Fluency(models.IntegerChoices):
        Beginner = 0
        Lower_intermediate = 1
        Upper_intermediate = 2
        Advanced = 3
    fluency = models.IntegerField(choices=Fluency.choices, help_text="How well do you know this word?",null=False)

    user = models.ForeignKey(User, related_name="words",on_delete=models.CASCADE)
    created_at = models.DateTimeField(auto_now=True)

    def __str__(self):
        return self.target_word

    def get_absolute_url(self):
        return reverse("vocab:detail",
        kwargs={"username": self.user.username, "pk": self.pk})

    class Meta:
        ordering = ["target_word"]

        constraints = [
            models.UniqueConstraint(fields=['user','target_word', 'source_word'],name='unique_word')]

还有我的import.html模板:

{% extends "vocab/vocab_base.html" %}
{% load bootstrap %}

{% block content %}

{% if messages %}
<div class="messages">
   {% for message in messages %}
   <h3  {% if message.tags %} class=" {{ message.tags }} " {% endif %}> {{ message }} </h3>
   {% endfor %}
</div>

{% else %}

<h1> Import your words</h1>
<p>Here you can import your words from a csv or excel file.</p>

  <form method="post" enctype="multipart/form-data">
      {% csrf_token %}
      <input type="file" name="importData">
      <p>Please select the format of your file</p>
      <select name="file-format" class="form-control my-3">
          <option selected>Choose format...</option>
          <option>CSV</option>
          <option>XLSX</option>
        </select>
      <button class="btn btn-primary" type="submit">Import</button>
    </form>
  <a href="{% url 'vocab:index' %}">Back</a>
{% endif %}
  {% endblock %}

奇怪的是,当我尝试使用csv文件而不是xlsx文件时,它最初似乎可以工作。然后,我仅对代码的xlsx部分进行了更改。它不仅不能解决问题,现在csv也不起作用。

注意-不是由于id列,因为我的测试上传文件中确实有一个空白的id列。

更新-我意识到我缺少用户字段,这是我的Word模型上的外键(非null)。因此,我添加了ForeignKeyWidget,但出现以下错误:NOT NULL constraint failed: vocab_word.user_id即使添加带有用户ID的列,错误仍然存​​在。我怎样才能解决这个问题?

我的csv文件如下所示:

username;Russian;Meaning;Example Sentence;Fluency level;Deck name
testuser;word1;word2;one two three;1;new

实施Matthew的更改后的回溯:

Environment:


Request Method: POST
Request URL: http://127.0.0.1:8000/vocab/import/

Django Version: 3.0.3
Python Version: 3.8.2
Installed Applications:
['django.contrib.admin',
 'django.contrib.auth',
 'django.contrib.contenttypes',
 'django.contrib.sessions',
 'django.contrib.messages',
 'django.contrib.staticfiles',
 'django.contrib.humanize',
 'rest_framework',
 'bootstrap4',
 'bootstrapform',
 'languages',
 'django_countries',
 'import_export',
 'django_tables2',
 'django_filters',
 'accounts',
 'vocab',
 'flash',
 'api',
 'django_cleanup.apps.CleanupConfig']
Installed Middleware:
['django.middleware.security.SecurityMiddleware',
 'django.contrib.sessions.middleware.SessionMiddleware',
 'django.middleware.common.CommonMiddleware',
 'django.middleware.csrf.CsrfViewMiddleware',
 'django.contrib.auth.middleware.AuthenticationMiddleware',
 'django.contrib.messages.middleware.MessageMiddleware',
 'django.middleware.clickjacking.XFrameOptionsMiddleware']



Traceback (most recent call last):
  File "C:\Users\mvren\miniconda3\envs\myRuEnv\lib\site-packages\django\db\backends\utils.py", line 86, in _execute
    return self.cursor.execute(sql, params)
  File "C:\Users\mvren\miniconda3\envs\myRuEnv\lib\site-packages\django\db\backends\sqlite3\base.py", line 396, in execute
    return Database.Cursor.execute(self, query, params)

The above exception (NOT NULL constraint failed: vocab_word.user_id) was the direct cause of the following exception:
  File "C:\Users\mvren\miniconda3\envs\myRuEnv\lib\site-packages\django\core\handlers\exception.py", line 34, in inner
    response = get_response(request)
  File "C:\Users\mvren\miniconda3\envs\myRuEnv\lib\site-packages\django\core\handlers\base.py", line 115, in _get_response
    response = self.process_exception_by_middleware(e, request)
  File "C:\Users\mvren\miniconda3\envs\myRuEnv\lib\site-packages\django\core\handlers\base.py", line 113, in _get_response
    response = wrapped_callback(request, *callback_args, **callback_kwargs)
  File "C:\Users\mvren\OneDrive\Documents\Coding\Russki\mysite\vocab\views.py", line 115, in import_words
    result = word_resource.import_data(dataset, dry_run=True, raise_errors=True)
  File "C:\Users\mvren\miniconda3\envs\myRuEnv\lib\site-packages\import_export\resources.py", line 627, in import_data
    return self.import_data_inner(dataset, dry_run, raise_errors, using_transactions, collect_failed_rows, **kwargs)
  File "C:\Users\mvren\miniconda3\envs\myRuEnv\lib\site-packages\import_export\resources.py", line 673, in import_data_inner
    raise row_result.errors[-1].error
  File "C:\Users\mvren\miniconda3\envs\myRuEnv\lib\site-packages\import_export\resources.py", line 569, in import_row
    self.save_instance(instance, using_transactions, dry_run)
  File "C:\Users\mvren\miniconda3\envs\myRuEnv\lib\site-packages\import_export\resources.py", line 352, in save_instance
    instance.save()
  File "C:\Users\mvren\miniconda3\envs\myRuEnv\lib\site-packages\django\db\models\base.py", line 745, in save
    self.save_base(using=using, force_insert=force_insert,
  File "C:\Users\mvren\miniconda3\envs\myRuEnv\lib\site-packages\django\db\models\base.py", line 782, in save_base
    updated = self._save_table(
  File "C:\Users\mvren\miniconda3\envs\myRuEnv\lib\site-packages\django\db\models\base.py", line 887, in _save_table
    results = self._do_insert(cls._base_manager, using, fields, returning_fields, raw)
  File "C:\Users\mvren\miniconda3\envs\myRuEnv\lib\site-packages\django\db\models\base.py", line 924, in _do_insert
    return manager._insert(
  File "C:\Users\mvren\miniconda3\envs\myRuEnv\lib\site-packages\django\db\models\manager.py", line 82, in manager_method
    return getattr(self.get_queryset(), name)(*args, **kwargs)
  File "C:\Users\mvren\miniconda3\envs\myRuEnv\lib\site-packages\django\db\models\query.py", line 1204, in _insert
    return query.get_compiler(using=using).execute_sql(returning_fields)
  File "C:\Users\mvren\miniconda3\envs\myRuEnv\lib\site-packages\django\db\models\sql\compiler.py", line 1384, in execute_sql
    cursor.execute(sql, params)
  File "C:\Users\mvren\miniconda3\envs\myRuEnv\lib\site-packages\django\db\backends\utils.py", line 100, in execute
    return super().execute(sql, params)
  File "C:\Users\mvren\miniconda3\envs\myRuEnv\lib\site-packages\django\db\backends\utils.py", line 68, in execute
    return self._execute_with_wrappers(sql, params, many=False, executor=self._execute)
  File "C:\Users\mvren\miniconda3\envs\myRuEnv\lib\site-packages\django\db\backends\utils.py", line 77, in _execute_with_wrappers
    return executor(sql, params, many, context)
  File "C:\Users\mvren\miniconda3\envs\myRuEnv\lib\site-packages\django\db\backends\utils.py", line 86, in _execute
    return self.cursor.execute(sql, params)
  File "C:\Users\mvren\miniconda3\envs\myRuEnv\lib\site-packages\django\db\utils.py", line 90, in __exit__
    raise dj_exc_value.with_traceback(traceback) from exc_value
  File "C:\Users\mvren\miniconda3\envs\myRuEnv\lib\site-packages\django\db\backends\utils.py", line 86, in _execute
    return self.cursor.execute(sql, params)
  File "C:\Users\mvren\miniconda3\envs\myRuEnv\lib\site-packages\django\db\backends\sqlite3\base.py", line 396, in execute
    return Database.Cursor.execute(self, query, params)

Exception Type: IntegrityError at /vocab/import/
Exception Value: NOT NULL constraint failed: vocab_word.user_id
马修·赫加蒂(Matthew Hegarty)

您的错误非常清楚-无法创建对象,因为在创建时该user_id字段为null:

NOT NULL constraint failed: vocab_word.user_id

您的csv包含:

username;Russian;Meaning;Example Sentence;Fluency level;Deck name
testuser;word1;word2;one two three;1;new

您的Word模型还定义了一个用户字段:

user = models.ForeignKey(User, related_name="words",on_delete=models.CASCADE)

这意味着Resource在django-import-export中声明a时,需要指定如何通过FK关系将csvusername映射到任何现有user实例。

您应该ForeignKeyWidget为此使用,因为它可以处理csv字段到对象的映射。

  • column_name 定义用于查找用户引用的csv列
  • attributeWord要设置模型上定义属性

另外,我们需要确保使用“正确”字段查找用户关系。文档

查找字段默认使用主键(pk)作为查找条件,但可以自定义为使用相关模型上的任何字段。

综上所述,我们的Field定义如下所示:

userid = fields.Field(column_name='username', attribute='user', widget=widgets.ForeignKeyWidget(User, "username")

因此,我认为错误的根源是您错误地将设置attributeusername如果您调试了代码,则可能会发现'testuser'User实例已被加载并分配给Word.username,该实例将被忽略,并且Word.user为null,因此出现错误。

更新资料

另一个要纠正的问题:

fields声明应引用要从csv数据设置的模型属性。

因此,该username字段应为user,因为这是要更新的模型属性。

fields = ("user", "target_word",'source_word','example_sentence', 'fluency', 'deck_name',)

本文收集自互联网,转载请注明来源。

如有侵权,请联系[email protected] 删除。

编辑于
0

我来说两句

0条评论
登录后参与评论

相关文章

来自分类Dev

Django import export扩展用户

来自分类Dev

Django import-export导入具有对象关系的数据

来自分类Dev

Django import-export导入具有对象关系的数据

来自分类Dev

django-import-export导出用户模型

来自分类Dev

使用django-import-export导入FloatField时出错

来自分类Dev

扩展django import_export的管理员导入表单

来自分类Dev

在django-import-export中处理外键的导入

来自分类Dev

Django-import-export始终导入空值

来自分类Dev

使用 tablib 和 django-import-export 导入数据,使用 excel 出错,但使用 csv 可以正常工作

来自分类Dev

无法导入“ import_export”

来自分类Dev

Django-Import-Export:导入期间不会自动递增表ID

来自分类Dev

在django-import-export中导入m2m关系

来自分类Dev

django-import-export在导入时csv标头触发异常之前的空行

来自分类Dev

Django Import-Export: Admin interface "TypeError at /"

来自分类Dev

Django Import Export:无法理解文档

来自分类Dev

Django Import-Export:管理界面“ TypeError at /”

来自分类Dev

Django import-export:仅导出

来自分类Dev

如何集成Django Import Export和Wagtail?

来自分类Dev

Django Import Export-UNIQUE约束失败

来自分类Dev

Django-import-export-如何将用户信息传递给ForeignKeyWidget?

来自分类Dev

将django import_export添加到抽象用户模型

来自分类Dev

使用django-import-export进行django迁移的外键

来自分类Dev

Django:django-import-export,更改文件名

来自分类Dev

用django-import-export流式传输响应

来自分类Dev

django-import-export ForeignKeyWidget不执行查找

来自分类Dev

如何使用django-import-export导出属性值

来自分类Dev

无法在Ubuntu 18.04上安装django-import-export

来自分类Dev

Django-import-export-从模型的功能导出吗?

来自分类Dev

如何使用django-import-export导出属性值

Related 相关文章

  1. 1

    Django import export扩展用户

  2. 2

    Django import-export导入具有对象关系的数据

  3. 3

    Django import-export导入具有对象关系的数据

  4. 4

    django-import-export导出用户模型

  5. 5

    使用django-import-export导入FloatField时出错

  6. 6

    扩展django import_export的管理员导入表单

  7. 7

    在django-import-export中处理外键的导入

  8. 8

    Django-import-export始终导入空值

  9. 9

    使用 tablib 和 django-import-export 导入数据,使用 excel 出错,但使用 csv 可以正常工作

  10. 10

    无法导入“ import_export”

  11. 11

    Django-Import-Export:导入期间不会自动递增表ID

  12. 12

    在django-import-export中导入m2m关系

  13. 13

    django-import-export在导入时csv标头触发异常之前的空行

  14. 14

    Django Import-Export: Admin interface "TypeError at /"

  15. 15

    Django Import Export:无法理解文档

  16. 16

    Django Import-Export:管理界面“ TypeError at /”

  17. 17

    Django import-export:仅导出

  18. 18

    如何集成Django Import Export和Wagtail?

  19. 19

    Django Import Export-UNIQUE约束失败

  20. 20

    Django-import-export-如何将用户信息传递给ForeignKeyWidget?

  21. 21

    将django import_export添加到抽象用户模型

  22. 22

    使用django-import-export进行django迁移的外键

  23. 23

    Django:django-import-export,更改文件名

  24. 24

    用django-import-export流式传输响应

  25. 25

    django-import-export ForeignKeyWidget不执行查找

  26. 26

    如何使用django-import-export导出属性值

  27. 27

    无法在Ubuntu 18.04上安装django-import-export

  28. 28

    Django-import-export-从模型的功能导出吗?

  29. 29

    如何使用django-import-export导出属性值

热门标签

归档