Warning: file_get_contents(/data/phpspider/zhask/data//catemap/2/python/334.json): failed to open stream: No such file or directory in /data/phpspider/zhask/libs/function.php on line 167

Warning: Invalid argument supplied for foreach() in /data/phpspider/zhask/libs/tag.function.php on line 1116

Notice: Undefined index: in /data/phpspider/zhask/libs/function.php on line 180

Warning: array_chunk() expects parameter 1 to be array, null given in /data/phpspider/zhask/libs/function.php on line 181
Python Django文件字段,在运行时确定上载到_Python_Django_Django Models - Fatal编程技术网

Python Django文件字段,在运行时确定上载到

Python Django文件字段,在运行时确定上载到,python,django,django-models,Python,Django,Django Models,我正在尝试设置我的上传,这样如果用户joe上传一个文件,它就会转到MEDIA\u ROOT/joe,而不是让每个人的文件都转到MEDIA\u ROOT。问题是我不知道如何在模型中定义它。以下是它当前的外观: class Content(models.Model): name = models.CharField(max_length=200) user = models.ForeignKey(User) file = models.FileField(upload_to=

我正在尝试设置我的上传,这样如果用户joe上传一个文件,它就会转到MEDIA\u ROOT/joe,而不是让每个人的文件都转到MEDIA\u ROOT。问题是我不知道如何在模型中定义它。以下是它当前的外观:

class Content(models.Model):
    name = models.CharField(max_length=200)
    user = models.ForeignKey(User)
    file = models.FileField(upload_to='.')
因此,我想要的不是作为上传对象的“.”,而是将其作为用户名

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

谢谢你的帮助

你可能已经读过了,所以这里有一个简单的例子来说明这一点:

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)
如您所见,您甚至不需要使用给定的文件名-如果您愿意,您可以在上载中覆盖该文件名以使其也可调用。

您可能已经阅读过,因此这里有一个简单的示例来说明这一点:

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)

正如您所看到的,您甚至不需要使用给定的文件名-如果您愿意,您可以在上载中覆盖该文件名以使其也可调用。

这真的很有帮助。为了简洁起见,我决定在我的案例中使用lambda:

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

这真的很有帮助。为了简洁起见,我决定在我的案例中使用lambda:

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

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

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


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

关于使用“实例”对象pk值的注意事项。根据文件:

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


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

如果您在迁移方面有问题,您可能应该使用
@deconstructible
decorator

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)

如果您在迁移过程中遇到问题,您可能应该使用
@deconstructible
decorator

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)

如果您有一个用户实例,那么可以快速设置生成

/-/filename random.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)


例如:
/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版本]

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

/-/filename random.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)


例如:
/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的修改版本]

是的,它可能确实属于文档-这是IRC上的一个合理常见问题解答。这对ModelForm有用吗?我可以看到该实例具有类模型的所有属性,但没有值(只有字段名的str)。在模板中,用户是隐藏的。我可能需要提交一个问题,我已经在谷歌上搜索了几个小时。奇怪的是,在基本相同的设置中,这对我来说是失败的。instance.user上没有属性。您可能希望使用
os.path.join
而不是
“/”。join
以确保它也能在非Unix系统上工作。它们可能很少见,但这是很好的做法;)您好,我尝试了相同的代码,将它们放在models.py中,但get error Content对象没有“user”属性。是的,它可能属于文档-这是IRC上的常见问题解答。这对ModelForm有效吗?我可以看到该实例具有类模型的所有属性,但没有值(只有字段名的str)。在模板中,用户是隐藏的。我可能需要提交一个问题,我已经在谷歌上搜索了几个小时。奇怪的是,在基本相同的设置中,这对我来说是失败的。instance.user上没有属性。您可能希望使用
os.path.join
而不是
“/”。join
以确保它也能在非Unix系统上工作。它们可能很少见,但这是很好的做法;)您好,我尝试了相同的代码,将它们放在models.py中,但get error Content对象没有“user”属性。在Django 1.7中使用迁移时,这对我不起作用。最终创建了一个函数,迁移耗时。即使您无法让lambda使用str(instance.pk)工作,如果您在不希望文件被覆盖时遇到问题,使用str(instance.pk)也是一个好主意。instance在保存之前没有
pk
。它只适用于更新,不适用于创建(插入)。lambda在
迁移
操作中不起作用,因为它无法根据使用迁移的Django 1.7中这不适用于我。最终创建了一个函数,迁移耗时。即使您无法让lambda使用str(instance.pk)工作,如果您在不希望文件被覆盖时遇到问题,使用str(instance.pk)也是一个好主意。instance在保存之前没有
pk
。它只适用于更新,不适用于创建(插入)。lambda在
迁移
操作中不起作用,因为它不能根据“我没有得到任何”作为值进行序列化。我想不出怎么修理它。你能详细解释一下吗?我的价值是零。我想不出怎么修理它。你能详细解释一下吗。