Руслан Волшебник
Руслан ВолшебникJan. 24, 2020, 9:52 a.m.

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

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...?

We recommend hosting TIMEWEB
We recommend hosting TIMEWEB
Stable hosting, on which the social network EVILEG is located. For projects on Django we recommend VDS hosting.

Do you like it? Share on social networks!

6
Илья Чичак
  • Jan. 24, 2020, 10:32 a.m.

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

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

      Илья Чичак
      • Jan. 24, 2020, 10:48 a.m.
      • (edited)

      я думаю, если это вдруг отвечает вашим хотелкам, вы можете сделать 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

        Руслан Волшебник
        • Jan. 24, 2020, 11:34 a.m.
        • (edited)

        Хорошо. Вот подробности)
        Делаю бота в 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)
          

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

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

            Илья Чичак
            • Jan. 27, 2020, 3:17 a.m.

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

              Comments

              Only authorized users can post comments.
              Please, Log in or Sign up
              d
              • dsfs
              • April 26, 2024, 1:56 a.m.

              C ++ - Test 004. Pointers, Arrays and Loops

              • Result:80points,
              • Rating points4
              d
              • dsfs
              • April 26, 2024, 1:45 a.m.

              C++ - Test 002. Constants

              • Result:50points,
              • Rating points-4
              d
              • dsfs
              • April 26, 2024, 1:35 a.m.

              C++ - Test 001. The first program and data types

              • Result:73points,
              • Rating points1
              Last comments
              k
              kmssrFeb. 8, 2024, 3:43 p.m.
              Qt Linux - Lesson 001. Autorun Qt application under Linux как сделать автозапуск для флэтпака, который не даёт создавать файлы в ~/.config - вот это вопрос ))
              Qt WinAPI - Lesson 007. Working with ICMP Ping in Qt Без строки #include <QRegularExpressionValidator> в заголовочном файле не работает валидатор.
              EVA
              EVADec. 25, 2023, 7:30 a.m.
              Boost - static linking in CMake project under Windows Ошибка LNK1104 часто возникает, когда компоновщик не может найти или открыть файл библиотеки. В вашем случае, это файл libboost_locale-vc142-mt-gd-x64-1_74.lib из библиотеки Boost для C+…
              J
              JonnyJoDec. 25, 2023, 5:38 a.m.
              Boost - static linking in CMake project under Windows Сделал всё по-как у вас, но выдаёт ошибку [build] LINK : fatal error LNK1104: не удается открыть файл "libboost_locale-vc142-mt-gd-x64-1_74.lib" Хоть убей, не могу понять в чём дел…
              G
              GvozdikDec. 18, 2023, 6:01 p.m.
              Qt/C++ - Lesson 056. Connecting the Boost library in Qt for MinGW and MSVC compilers Для решения твой проблемы добавь в файл .pro строчку "LIBS += -lws2_32" она решит проблему , лично мне помогло.
              Now discuss on the forum
              IscanderChe
              IscanderCheApril 30, 2024, 1:22 a.m.
              Во Flask рендер шаблона не передаётся в браузер Доброе утро! Имеется вот такой шаблон: <!doctype html><html> <head> <title>{{ title }}</title> <link rel="stylesheet" href="{{ url_…
              G
              GarApril 22, 2024, 2:46 a.m.
              Clipboard Как скопировать окно целиком в clipb?
              DA
              Dr Gangil AcademicsApril 20, 2024, 4:45 a.m.
              Unlock Your Aesthetic Potential: Explore MSC in Facial Aesthetics and Cosmetology in India Embark on a transformative journey with an msc in facial aesthetics and cosmetology in india . Delve into the intricate world of beauty and rejuvenation, guided by expert faculty and …
              a
              a_vlasovApril 14, 2024, 3:41 a.m.
              Мобильное приложение на C++Qt и бэкенд к нему на Django Rest Framework Евгений, добрый день! Такой вопрос. Верно ли следующее утверждение: Любое Android-приложение, написанное на Java/Kotlin чисто теоретически (пусть и с большими трудностями) можно написать и на C+…
              Павел Дорофеев
              Павел ДорофеевApril 13, 2024, 11:35 p.m.
              QTableWidget с 2 заголовками Вот тут есть кастомный QTableView с многорядностью проект поддерживается, обращайтесь

              Follow us in social networks