【问题标题】:Django FileField with upload_to determined at runtimeDjango FileField with upload_to 在运行时确定
【发布时间】:2010-11-14 12:16:03
【问题描述】:

我正在尝试设置我的上传,以便如果用户 joe 上传一个文件,它会转到 MEDIA_ROOT/joe,而不是让每个人的文件都转到 MEDIA_ROOT。问题是我不知道如何在模型中定义它。以下是它目前的样子:

class Content(models.Model):
    name = models.CharField(max_length=200)
    user = models.ForeignKey(User)
    file = models.FileField(upload_to='.')

所以我想要的是而不是'。作为upload_to,让它成为用户名。

我知道,从 Django 1.0 开始,您可以定义自己的函数来处理 upload_to,但该函数也不知道用户是谁,所以我有点迷茫。

感谢您的帮助!

【问题讨论】:

    标签: python django django-models


    【解决方案1】:

    如果您有用户实例,请快速设置以生成

    <model-slug>/<username>-<first_name>-<last_name>/filename-random.png

    例如: /medias/content/ft0004-john-doe/filename-lkl9237.png

    
    def upload_directory_name(instance, filename):
    
        user = getattr(instance, 'user', None)
        if user:
            name = f"{user.username}-{user.get_full_name().replace(' ', '-')}"
        else:
            name=str(instance)
        model_name = instance._meta.verbose_name.replace(' ', '-')
        return str(os.path.pathsep).join([model_name, name, filename])
    
    
    class Content(models.Model):
        name = models.CharField(max_length=200)
        user = models.ForeignKey(User)
        file = models.FileField(upload_to=upload_directory_name)
    
    
    

    [@SmileyChris 的修改版]

    【讨论】:

      【解决方案2】:

      如果您在迁移时遇到问题,您可能应该使用 @deconstructible 装饰器。

      import datetime
      import os
      import unicodedata
      
      from django.core.files.storage import default_storage
      from django.utils.deconstruct import deconstructible
      from django.utils.encoding import force_text, force_str
      
      
      @deconstructible
      class UploadToPath(object):
          def __init__(self, upload_to):
              self.upload_to = upload_to
      
          def __call__(self, instance, filename):
              return self.generate_filename(filename)
      
          def get_directory_name(self):
              return os.path.normpath(force_text(datetime.datetime.now().strftime(force_str(self.upload_to))))
      
          def get_filename(self, filename):
              filename = default_storage.get_valid_name(os.path.basename(filename))
              filename = force_text(filename)
              filename = unicodedata.normalize('NFKD', filename).encode('ascii', 'ignore').decode('ascii')
              return os.path.normpath(filename)
      
          def generate_filename(self, filename):
              return os.path.join(self.get_directory_name(), self.get_filename(filename))
      

      用法:

      class MyModel(models.Model):
          file = models.FileField(upload_to=UploadToPath('files/%Y/%m/%d'), max_length=255)
      

      【讨论】:

        【解决方案3】:

        关于使用“实例”对象的 pk 值的说明。根据文档:

        在大多数情况下,这个对象还没有保存到数据库中,所以如果它使用默认的 AutoField,它可能还没有主键字段的值。

        因此,使用 pk 的有效性取决于您的特定模型是如何定义的。

        【讨论】:

        • 我得到 None 作为值。我不知道如何解决它。能不能详细解释一下。
        【解决方案4】:

        这真的很有帮助。为了更简洁起见,决定在我的情况下使用 lambda:

        file = models.FileField(
            upload_to=lambda instance, filename: '/'.join(['mymodel', str(instance.pk), filename]),
        )
        

        【讨论】:

        • 这对我在 Django 1.7 中使用迁移不起作用。最终创建了一个函数并进行了迁移。
        • 即使您无法使用 str(instance.pk) 让 lambda 工作,如果您在不希望文件覆盖时遇到问题,也是一个好主意。
        • 实例在保存前没有pk。它仅适用于更新而不是创建(插入)。
        • lambda 在migrations 操作中不起作用,因为它不能根据docs 进行序列化
        【解决方案5】:

        您可能已经阅读过the documentation,所以这里有一个简单的例子来说明它的含义:

        def content_file_name(instance, filename):
            return '/'.join(['content', instance.user.username, filename])
        
        class Content(models.Model):
            name = models.CharField(max_length=200)
            user = models.ForeignKey(User)
            file = models.FileField(upload_to=content_file_name)
        

        如您所见,您甚至不需要使用给定的文件名 - 如果您愿意,也可以在您的 upload_to 可调用文件中覆盖它。

        【讨论】:

        • 是的,它可能确实属于文档 - 这是 IRC 上一个合理的常见问题解答
        • 这适用于 ModelForm 吗?我可以看到该实例具有类模型的所有属性,但没有值(只是字段名称的 str)。在模板中,用户是隐藏的。我可能需要提交一个问题,我已经在谷歌上搜索了几个小时。
        • 奇怪的是,在基本相同的设置中,这对我来说是失败的。 instance.user 上没有任何属性。
        • 您可能希望使用os.path.join 而不是'/'.join 以确保它也适用于非Unix 系统。它们可能很少见,但这是一种很好的做法;)
        • 嗨,我尝试了相同的代码,将它们放在models.py中,但得到错误内容对象没有属性'user'。
        猜你喜欢
        • 2018-11-08
        • 1970-01-01
        • 2020-06-29
        • 1970-01-01
        • 1970-01-01
        • 1970-01-01
        • 2010-12-04
        • 2013-07-06
        相关资源
        最近更新 更多