[python] 런타임시 upload_to가있는 Django FileField

사용자 joe가 파일을 업로드하면 모든 사람의 파일이 MEDIA_ROOT로 이동하는 대신 MEDIA_ROOT / joe로 이동하도록 업로드를 설정하려고합니다. 문제는 모델에서 이것을 정의하는 방법을 모른다는 것입니다. 현재 모습은 다음과 같습니다.

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를 처리하기 위해 자신의 함수를 정의 할 수 있지만 그 함수는 사용자가 누구인지 알지 못하므로 조금 잃어 버렸습니다.

도와 주셔서 감사합니다!



답변

아마도 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 호출 가능 파일에서도 파일 이름을 무시할 수 있습니다.


답변

정말 도움이되었습니다. 좀 더 간결하게하기 위해 필자의 경우 람다를 사용하기로 결정했습니다.

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


답변

‘인스턴스’객체의 pk 값 사용에 대한 참고 사항입니다. 설명서에 따르면 :

대부분의 경우이 개체는 아직 데이터베이스에 저장되지 않았으므로 기본 AutoField를 사용하는 경우 기본 키 필드 값이 아직 없을 수 있습니다.

따라서 pk 사용의 유효성은 특정 모델이 정의 된 방법에 따라 다릅니다.


답변

마이그레이션에 문제가있는 경우 @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)


답변