24 января 2020 г. 9: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 со всеми этими полями, и в зависимости от действия пользователя, например, "добавить документ", "добавить картинку" и т.д. класть все это в разные поля?

Комментарии

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

Здравствуйте, уважаемые пользователи EVILEG !!!

Если сайт вам помог, то поддержите разработку сайта финансово, пожалуйста.

Вы можете сделать это следующими способами:

Спасибо, Евгений Легоцкой

p
17 февраля 2020 г. 14:41
pstMem

C++ - Тест 003. Условия и циклы

  • Результат:85баллов,
  • Очки рейтинга6
z
17 февраля 2020 г. 6:02
zet

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

  • Результат:80баллов,
  • Очки рейтинга4
z
17 февраля 2020 г. 5:49
zet

C++ - Тест 001. Первая программа и типы данных

  • Результат:80баллов,
  • Очки рейтинга4
Последние комментарии
17 февраля 2020 г. 3:22
Евгений Легоцкой

Добрый день. Это кастомный тег, помещается в файл, который находится в каталоге templatetags myapp/ templatetags/ myapp.py
B
16 февраля 2020 г. 13:36
BahaMeirman

Добрый вечер! Монжно по подробней о теге get_companion? ссылка не работает.
16 февраля 2020 г. 9:35
Евгений Легоцкой

Добрый день. На GitHub исходники, можете посмотреть в официальном репозитории
B
16 февраля 2020 г. 9:29
BahaMeirman

Здравстсвуйте Евгений, непонятно мне где эти исходники найти?
Сейчас обсуждают на форуме
19 февраля 2020 г. 11:58
Юрий

Спасибо, все понятно.
19 февраля 2020 г. 8:55
Михаиллл

Можно через сервер сделать
V
19 февраля 2020 г. 7:09
Vitali

Да, прямо сходу не заработало, а сейчас просто некогда разбираться, да и я уже решил не использовать в этом приложении WebEngine. Ошибка: WebEngineContext used before QtWebEngine::initialize(…
19 февраля 2020 г. 7:01
BlinCT

Просто реально не вижу тут каких то проблем в модели, вот вообще ничего. Но она все равно не отображается, то есть ладно бы если данные бы не появлялись а сама таблица была бы. Так и таблиц…
19 февраля 2020 г. 6:54
Михаиллл

понятно, думал что дебаг будет стандартно работать. спасибо за помощь!
EVILEG
О нас
Услуги
© EVILEG 2015-2019
Рекомендует хостинг TIMEWEB