在运行时确定带有upload_to的Django FileField


130

我正在尝试设置我的上传文件,以便如果用户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,但是该函数也不知道谁将成为谁,所以我有点迷失了。

谢谢您的帮助!

Answers:


256

您可能已经阅读了文档,所以这里有一个简单的示例可以使之有意义:

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上是一个合理的常见问题解答
SmileyChris

2
这对ModelForm有用吗?我可以看到实例具有类模型的所有属性,但是没有值(只是字段名称的str)。在模板中,用户是隐藏的。我可能不得不提一个问题,这个问题我已经搜索了好几个小时。
mgag

3
奇怪的是,这基本上在相同的设置中对我来说是失败的。instance.user上没有任何属性。
鲍勃·斯普林

11
您可能要使用os.path.join而不是'/'.join确保它也可以在非Unix系统上使用。它们可能很少见,但这是一个好习惯;)
Xudonax 2014年

2
嗨,我尝试了相同的代码,将它们放在models.py中,但出现错误Content对象没有属性'user'。
哈利

12

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

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

4
在使用迁移的Django 1.7中,这对我不起作用。最终创建了一个函数,然后进行了迁移。
aboutaaron

即使您不希望使用lambda来使用str(instance.pk)也是一个好主意,如果您在文件覆盖方面遇到问题时不愿使用它们。
约瑟夫·达蒂洛

2
实例没有pk保存前。它仅适用于更新,不适用于创建(插入)。
Mohammad Jafar Mashhadi

lambda在migrations操作中不起作用,因为它无法根据文档
Ebrahim Karimi

4

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

在大多数情况下,此对象尚未保存到数据库,因此,如果使用默认的AutoField,则它的主键字段可能尚未具有值。

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


我得到了None作为价值。我不知道如何解决它。您能否详细解释一下。
阿曼(Aman)Deep

1

如果您在迁移时遇到问题,则可能应该使用@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)
By using our site, you acknowledge that you have read and understand our Cookie Policy and Privacy Policy.
Licensed under cc by-sa 3.0 with attribution required.