Руслан Волшебник
Руслан Волшебник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
              AD

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

              • Result:50points,
              • Rating points-4
              m

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

              • Result:80points,
              • Rating points4
              m

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

              • Result:20points,
              • Rating points-10
              Last comments
              i
              innorwallNov. 15, 2024, 5:26 a.m.
              Qt/C++ - Lesson 031. QCustomPlot – The build of charts with time buy generic priligy We can just chat, and we will not lose too much time anyway
              i
              innorwallNov. 15, 2024, 3:03 a.m.
              Qt/C++ - Lesson 060. Configuring the appearance of the application in runtime I didnt have an issue work colors priligy dapoxetine 60mg revia cost uk August 3, 2022 Reply
              i
              innorwallNov. 14, 2024, 8:07 p.m.
              Circuit switching and packet data transmission networks Angioedema 1 priligy dapoxetine
              i
              innorwallNov. 14, 2024, 7:42 p.m.
              How to Copy Files in Linux If only females relatives with DZ offspring were considered these percentages were 23 order priligy online uk
              i
              innorwallNov. 14, 2024, 5:09 p.m.
              Qt/C++ - Tutorial 068. Hello World using the CMAKE build system in CLion ditropan pristiq dosing With the Yankees leading, 4 3, Rivera jogged in from the bullpen to a standing ovation as he prepared for his final appearance in Chicago buy priligy pakistan
              Now discuss on the forum
              i
              innorwallNov. 14, 2024, 11:39 a.m.
              добавить qlineseries в функции priligy amazon canada 93 GREB1 protein GREB1 AB011147 6
              i
              innorwallNov. 11, 2024, 6:55 p.m.
              Всё ещё разбираюсь с кешем. priligy walgreens levitra dulcolax carbs The third ring was found to be made up of ultra relativistic electrons, which are also present in both the outer and inner rings
              9
              9AnonimOct. 25, 2024, 4:10 p.m.
              Машина тьюринга // Начальное состояние 0 0, ,<,1 // Переход в состояние 1 при пустом символе 0,0,>,0 // Остаемся в состоянии 0, двигаясь вправо при встрече 0 0,1,>…

              Follow us in social networks