Руслан Волшебник24 января 2020 г. 20:52

Как лучше реализовать модели в бд?

database, Django

Здравствуйте. Снова я здесь со своими вопросами)
Как лучше реализовать структуру моделей для файлов?
Хочу хранить файлы в отдельной модели или моделях (это уже как вы мне подскажите).
Ну так вот, какая схема лучше подойдёт?
1. Делать несколько моделей, которые имеют отношения OneToOne с моделью File.
2. Или же одну модель File, где будет поле file_type.

Мне чутьё подсказывает, что 1-ый вариант лучше использовать, если пригодятся какие-то дополнительные поля для каждого типа файла, а во всех остальных 2-ой.

1.

class File(models.Model):
    pass

class AbstractFile(models.Model):
    filename = models.FileField(upload_to='files')
    file = models.OneToOneField(File, on_delete=models.CASCADE, primary_key=True)
    date = models.DateTimeField(auto_now_add=True, null=True, blank=True)

    class Meta:
        abstract = True

    def delete(self, *args, **kwargs):
        self.file.delete()
        return super(self.__class__, self).delete(*args, **kwargs)

class Image(AbstractFile):
    filename = models.ImageField(upload_to='images')

class Document(AbstractFile):
    pass

2.

class File(models.Model):
    DOCUMENT = 0
    IMAGE = 1
    AUDIO = 2
    VIDEO = 3

    FILE_TYPE_CHOICES = (
        (DOCUMENT, 'Документ'),
        (IMAGE, 'Изображение'),
        (AUDIO, 'Видео'),
        (VIDEO, 'Аудио')
    )
    file_type = models.PositiveSmallIntegerField(choices=FILE_TYPE_CHOICES, default=DOCUMENT)
    file = models.FileField(upload_to='files')

И ещё... Как можно сделать, чтобы в 1-ом варианте можно было к File связать только с одним объектом. Допустим два объекта Image я не могу связать через OneToOne с File, это понятно, но можно 1 Image и 1 Document. Как сделать, чтобы можно было связать только с одной либо Image, либо Document...?

Рекомендуем хостинг TIMEWEB
Рекомендуем хостинг TIMEWEB
Стабильный хостинг, на котором располагается социальная сеть EVILEG. Для проектов на Django рекомендуем VDS хостинг.
6

я думаю, правильнее всего начать с того, какую задачу вы пытаетесь решить.
по большому счету imageField отличается от FileField только проверками, что в поле приехала именно картинка

Ну получается тогда, что если мне пока не нужны дополнительные поля для каждого типа файла, то лучше использовать одну модель File.
Я просто думал на всякий случай сделать, может пригодится. Видимо, раз сейчас надобности нет, сделаю одну модель File.

я думаю, если это вдруг отвечает вашим хотелкам, вы можете сделать 2 разные формы для этого, в одной из которых сделать поле ImageField (со всеми вытекающими проверками), а в другой оставить FileField. Но в обоих случаях все складывать в одну модель с типом поля FileField.
Но в любом случае, не зная предметную область, очень сложно дать какой-то вменяемый ответ. Лучше всего всегда начинать с описания, что вы хотите сделать - так будет проще=)

кстати, из интересного - не обязательно указывать OneToOne на модель=)

class File(models.Model):
    pass

class ImageFile(File): # тут будет связь один к одному на модель File
    pass

class AbstractFile(models.Model):
    class Meta:
        abstract = True

class ImageFileInheritance(AbstractFile): # а тут будет уже наследование
    pass

если хотите наследоваться - надо указать в File.Meta.abstract = True

Хорошо. Вот подробности)
Делаю бота в telegram. Telegram каждому файлу загружаемому на их сервера даёт некий file_id. Мне нужно записывать это file_id в бд. И следующий раз не файл отправлять пользователю, а этот file_id.

Сейчас у меня есть модель Image. И есть FileID, которая хранит file_id каждой картинки.
Также у меня есть модель TelegramBot, так как ботов целых 2 штуки. Связи с этим в модель FileID добилось поле bot, которая хранит ссылку на бота. Так сделано потому что у одного файла может быть разный file_id, в зависимости от того в какой бот был загружен файл.

Вот модель FileID для наглядности

class FileID(models.Model):
    name = models.CharField(max_length=100)
    image = models.ForeignKey(Image, on_delete=models.CASCADE, related_name='file_ids')
    bot = models.ForeignKey(TelegramBot, on_delete=models.CASCADE, related_name='file_ids')

    def __str__(self):
        return self.name

В поле name хранится тот самый file_id.

Всё это время я работал только с картинками, но теперь нужно добавить и другие файлы.
Поэтому решил сделать отдельную модель File, ссылку на которую добавлю в FileID вместо Image.

Ещё появилась мысля сделать, отдельно модели Image, Document, Video, Audio. А в FileID использовать GenericForeignKey.
Или сделать так:

class File(models.Model):
    image = ForeignKey(Image, null=True, blank=True)
    video = ForeignKey(Video, null=True, blank=True)
    audio = ForeignKey(Audio, null=True, blank=True)
    document = ForeignKey(Document, null=True, blank=True)

А при сохранении разрешить только один тип файла.

Я уже, честно говоря, запутался.

а почему бы не сделать одну модель, например Attachement со всеми этими полями, и в зависимости от действия пользователя, например, "добавить документ", "добавить картинку" и т.д. класть все это в разные поля?

Комментарии

Только авторизованные пользователи могут публиковать комментарии.
Пожалуйста, авторизуйтесь или зарегистрируйтесь
Timeweb

Позвольте мне порекомендовать вам отличный хостинг, на котором расположен EVILEG.

В течение многих лет Timeweb доказывает свою стабильность.

Для проектов на Django рекомендую VDS хостинг

Посмотреть Хостинг
PH

C++ - Тест 004. Указатели, Массивы и Циклы

  • Результат:60баллов,
  • Очки рейтинга-1
СН

C++ - Тест 004. Указатели, Массивы и Циклы

  • Результат:10баллов,
  • Очки рейтинга-10
k
  • knobu
  • 23 сентября 2020 г. 12:34

C++ - Тест 006. Перечисления

  • Результат:60баллов,
  • Очки рейтинга-1
Последние комментарии
ДИ

Qt/C++ - Урок 015. QTableWidget или Как сделать таблицу с чекбоксами

Кто-нибудь знает, как сделать так, чтобы в QTableWidget состоящей из чекбоксов в строке таблицы можно было выбрать только один checkbox ?

Qt/C++ - Урок 006. QSqlQueryModel - Таблицы в Qt с помощью SQL-запросов

QSqlTableModel выполняет ряд стандартных операций для одной таблицы из базы данных. Поэтому там и реализован функционал по удалению и редактированию. QSqlQueryModel позволяет выполнить запр…
VB

Qt/C++ - Урок 006. QSqlQueryModel - Таблицы в Qt с помощью SQL-запросов

Добрый день. Хотел спросить вот что. Создал проект на основе QAbstractTableModel. В MainWindow cоответственно создал модель и связал с представлением. Поиск веду по списку элементов модели,…

QCheckBox в качестве делегата QTableView

До тех пор, пока у вас проект содержит только одну таблицу, или несколько то может быть. Когда их будет 1000 и чекбоксы в разных колонках, то без делегатов и переопределения возвращаемых ре…
D
  • Damir
  • 21 сентября 2020 г. 1:34

QCheckBox в качестве делегата QTableView

bool Node::setData(const QModelIndex& index, const QVariant& value, int role){ switch (index.column()) { case 0: switch (role) { case Qt::CheckStateRole:// <- т…
Сейчас обсуждают на форуме

Как в Qt в qmenu добавить scrollarea

Вот это наследованный класс меню. Но посути это обычное меню. #pragma once#include <QtWidgets>class TransMenu : public QMenu { Q_OBJECTpublic: TransMenu(QWidget* parent = …
  • Nomad
  • 1 октября 2020 г. 15:22

MyForm(forms.Form): - непонятка

понятно спасибо
ДИ

Как в QTableWidget, состоящей из чекбоксов, в строке таблицы можно было выбрать только один checkbox ?

Сделал таблицу состоящую из чекбоксов по уроку https://evileg.com/ru/post/78/ Кто-нибудь знает, как сделать так, чтобы в QTableWidget состоящей из чекбоксов в строке таблицы можно …
U

как скрыть елемент с копии виджета

Удалить пост нельзя... Поэтому удачки, Катту)
H

Тесты падают при сборке под MinGW.

Всем привет! При сборке под MinGW некоторые тесты при старте выдают FATAL "не удалось запустить тест проекта" и подвешивают Qt Creator - приходится его перезапускать. При сборке под MSVC та…
О нас
Услуги
© EVILEG 2015-2020
Рекомендует хостинг TIMEWEB