web-dev-qa-db-ja.com

Django実行時にupload_toが決定された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を処理する独自の関数を定義できますが、その関数にはユーザーが誰なのかわからないため、少し迷っています。

助けてくれてありがとう!

126
Teebes

おそらく ドキュメント を読んだことがあるので、ここでわかりやすくするための簡単な例を示します。

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でそれをオーバーライドすることもできます。

248
SmileyChris

これは本当に役に立ちました。もう少し簡潔にするために、私の場合はラムダを使用することにしました。

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

「インスタンス」オブジェクトのpk値の使用に関する注意。ドキュメントによると:

ほとんどの場合、このオブジェクトはまだデータベースに保存されていないため、デフォルトのAutoFieldを使用する場合、主キーフィールドの値がまだない場合があります。

したがって、pkを使用する有効性は、特定のモデルがどのように定義されているかによって異なります。

4
Max Dudzinski

移行に問題がある場合は、おそらく@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)
0
michal-michalak