Django вызов 'id' ожидал число, но получил строку - PullRequest
1 голос
/ 27 января 2020

Django ошибки с django -import-export библиотеками.

Я хочу импортировать данные из excel в db через django admin. Я использую для этого django -import-export, но я получил Поле 'id', ожидал число, но получил 'HPI'.

enter image description here

Файл Excel содержит enter image description here

Я нашел ответ, который я должен добавить exclude = ('id',), но это не помогло. Также я занимался миграциями, это тоже не помогло. Как это исправить и иметь возможность импортировать данные из 6 столбцов из Excel в БД через django admin?

models.py


    from django_mysql.models import JSONField, Model
    from django.db import models



    class Category(Model):
        title = models.CharField(max_length=100)

        class Meta:
            ordering = ('-id',)
            verbose_name = 'Category'
            verbose_name_plural = 'Categories'

        def __str__(self):
            return self.title


    class Tag(Model):
        title = models.CharField(max_length=100)

        class Meta:
            ordering = ('-id',)

        def __str__(self):
            return self.title


    class Type(Model):
        title = models.CharField(max_length=100)

        class Meta:
            ordering = ('-id',)
            verbose_name = 'Type'
            verbose_name_plural = 'Types'

        def __str__(self):
            return self.title


    class Macro(Model):
        type = models.ForeignKey(
            Type,
            max_length=100,
            null=True,
            blank=True,
            on_delete=models.SET_NULL)
        tags = models.ManyToManyField(Tag, blank=True)
        category = models.ForeignKey(
            Category, null=True, blank=True, on_delete=models.SET_NULL)
        abbreviation = models.CharField(max_length=100, unique=True)
        title = models.CharField(max_length=100, verbose_name='Title')
        content = models.TextField(max_length=1000, null=True, blank=True)

        class Meta:
            ordering = ('-id',)

        def __str__(self):
            return self.title

admin.py


    from django.contrib import admin

    from import_export import resources
    from import_export.admin import ImportExportModelAdmin

    from .models import Category, Tag, Type, Macro


    class MacroResource(resources.ModelResource):

        class Meta:
            model = Macro
            skip_unchanged = True
            report_skipped = True
            exclude = ('id', )
            export_order = ('type', 'tags', 'category', 'abbreviation', 'title', 'content')


    @admin.register(Macro)
    class MacroAdmin(ImportExportModelAdmin):
        resource_class = MacroResource
        list_display = ('id', 'type', 'tags_list', 'category', 'abbreviation', 'title', 'content')
        search_fields = ('title', 'category__title', 'type__title', 'abbreviation', 'content', )

        def tags_list(self, obj):
            tags = [t for t in obj.tags.all()]
            return ' '.join(str(tags)) if tags else '-'


    @admin.register(Category)
    class CategoryAdmin(admin.ModelAdmin):
        list_display = ('id', 'title')


    @admin.register(Tag)
    class TagAdmin(admin.ModelAdmin):
        list_display = ('id', 'title')

        def __str__(self):
            return self.title


    @admin.register(Type)
    class TypeAdmin(admin.ModelAdmin):
        list_display = ('id', 'title')

Ответы [ 2 ]

1 голос
/ 27 января 2020

Django -import-export ожидает, что первый столбец будет id.

Если это новые объекты, просто оставьте столбец id пустым. В противном случае укажите идентификатор базы данных объекта в этом поле.

Если вы не можете изменить файл или не хотите, и вы всегда будете добавлять новые строк в базе данных (без изменения существующих), вы можете динамически создать поле id в своем классе ресурсов, переопределив метод before_import и заставив get_instance всегда возвращать False.

class MacroResource(resources.ModelResource):

    def before_import(self, dataset, using_transactions, dry_run, **kwargs):
        dataset.insert_col(0, col=["",]*dataset.height, header="id")

    def get_instance(self, instance_loader, row):
        return False

    class Meta:
        model = Macro
        skip_unchanged = True
        report_skipped = True
        export_order = ('type', 'tags', 'category', 'abbreviation', 'title', 'content')
0 голосов
/ 01 февраля 2020

Проблема была с полями ForeignKey и ManyToMany модели базы данных. Так что django -import-export library должна получить виджеты для этих полей.

Подробнее об этом здесь: https://django-import-export.readthedocs.io/en/latest/api_widgets.html#import_export .widgets.ForeignKeyWidget

Решение: admin.py


        class MacroResource(resources.ModelResource):

            type = fields.Field(
                column_name='type',
                attribute='type',
                widget=ForeignKeyWidget(Type, 'title'))

            category = fields.Field(
                column_name='category',
                attribute='category',
                widget=ForeignKeyWidget(Category, 'title'))

            tags = fields.Field(
                column_name='tags',
                attribute='tags',
                widget=ManyToManyWidget(Tag, field='title'))

            class Meta:
                model = Macro
                skip_unchanged = True
                report_skipped = True
                exclude = ('id', )
                import_id_fields = ('title',)

                fields = ('type', 'tags', 'category', 'abbreviation', 'title', 'content')

вместо


        class MacroResource(resources.ModelResource):

            class Meta:
                model = Macro
                skip_unchanged = True
                report_skipped = True
                exclude = ('id', )
                export_order = ('type', 'tags', 'category', 'abbreviation', 'title', 'content')


Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...