ユーザー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を処理する独自の関数を定義できますが、その関数にはユーザーが誰なのかわからないため、少し迷っています。
助けてくれてありがとう!
おそらく ドキュメント を読んだことがあるので、ここでわかりやすくするための簡単な例を示します。
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 callableでそれをオーバーライドすることもできます。
これは本当に役に立ちました。もう少し簡潔にするために、私の場合はラムダを使用することにしました。
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)