upload_to 动态生成的 url 到 callable

Emi*_*nte 2 python django models

我看过很多关于这个问题的帖子,但没有真正理解如何解决它。

我有这个模型:

class Project(TimeStampedModel):
    name = models.TextField(max_length=100, default='no name')
    logo = models.ImageField()
Run Code Online (Sandbox Code Playgroud)

我想按照此模板将我的图像保存到媒体根目录:

<name>/logo/<filename>
Run Code Online (Sandbox Code Playgroud)

乍一看,我想做:

logo = models.ImageField(upload_to="{}/logo/".format(name))

但它引发了这个错误: AttributeError: 'TextField' object has no attribute 'model'

使用 callable 会很好,但部分是:

def upload_to_project(self, filename):
    url = ("%s/%s") % (self.name, filename)
    return url
Run Code Online (Sandbox Code Playgroud)

并使用:

logo = models.ImageField(upload_to=upload_to_project)
Run Code Online (Sandbox Code Playgroud)

至少我有: <name>/<filename>

但是在这种情况下如何传递参数呢?我想重用我的功能在其他子文件夹中上传,不仅logo是:

<name>/logo/<filename>
<name>/history/<filename>
<name>/whatever/<filename>
Run Code Online (Sandbox Code Playgroud)

知道我能做什么吗?

bru*_*ers 6

看起来(重新阅读您的帖子并不是 100% 清楚)您想要的是部分应用程序。好消息,它是 Python 标准库的一部分:

import os
from functools import partial

def generic_upload_to(instance, filename, folder):
    return os.path.join(instance.name, folder, filename)


class Project(TimeStampedModel):
    name = models.TextField(max_length=100, default='no name')
    logo = models.ImageField(
        upload_to=partial(generic_upload_to, folder="logo")
        )
Run Code Online (Sandbox Code Playgroud)

请注意,此实现假定instance有一个name属性...如果您想用作第一部分的实例属性也必须是可配置的,您可以将您的重写upload_to为:

def generic_upload_to(instance, filename, folder, attrname):
    return os.path.join(getattr(instance, attrname), folder, filename)
Run Code Online (Sandbox Code Playgroud)

然后将其用作

class Project(TimeStampedModel):
    name = models.TextField(max_length=100, default='no name')
    logo = models.ImageField(
        upload_to=partial(generic_upload_to, attrname="name", folder="logo")
        )
Run Code Online (Sandbox Code Playgroud)

如果您的模型中有多个FileFieldImageField并且不想重复该attrname部分:

class Something(TimeStampedModel):
    my_upload_to = partial(generic_upload_to, attrname="label")

    label = models.CharField(max_length=100, default='no label')
    logo = models.ImageField(
        upload_to=partial(my_upload_to, folder="logo")
        )
    attachment = models.FileField(
        upload_to=partial(my_upload_to, folder="attachment")
        )
Run Code Online (Sandbox Code Playgroud)