How to save a file field of a new django model object to the path with ID?

3.1k Views Asked by At

In Django I use the model

class Specialist(models.Model):
    ...
    photo = models.ImageField(_('photo'), upload_to='spec_foto')
    ...

After I create and save a new object, I have the "photo" field at .../spec_photo/filename.jpg But I'd like to move the file to .../spec_photo/ID/photo.jpg, where ID belongs to the Specialist object. In order to do that, I override Model.save method

def save(self):
    # Saving a new object and getting ID
    super(Specialist, self).save()
    # After getting ID, move photo file to the right destination
    # ????
    # Saving the object with the right file destination
    super(Specialist, self).save()

The question is, what should I do to move file? (??? in the code). Or maybe is there a simpler way?

2

There are 2 best solutions below

0
On

Instead of setting the 'upload_to' as a string, you can set it as a callable (ie a function) which will return the path you require (https://docs.djangoproject.com/en/dev/ref/models/fields/#django.db.models.FileField.upload_to):

class Specialist(models.Model):
    ...
    photo = models.ImageField(_('photo'), upload_to=get_file_path)
    ...

def get_file_path(instance, filename):
    if not instance.id:
        instance.save()

    return 'spec_photo/%s/photo.jpg' % instance.id
1
On

this is my code.
in models.py

class ChatUser(models.Model):
"""User Model"""
username = models.CharField(max_length=64)
password = models.CharField(max_length=64)
GENDER_CHOICES = (
    (0, u'女'),
    (1, u'男')
)
sex = models.IntegerField(default=0, choices=GENDER_CHOICES)
description = models.CharField(max_length = 256, blank=True, default=None)
headphoto = models.ImageField(upload_to='photos/users/' , blank=True, default=None)

class Meta:
    db_table    = 'user'

def __unicode__(self):
    return "<ChatUser {'%s'}>" % self.username

in forms.py

class UserForm(ModelForm):

    class Meta:
        model = ChatUser
        fields = ( 'username' , 'password' , 'sex' , 'description', 'headphoto')

and views.py

@csrf_exempt
def index(request):
    c = {}
    if request.method == 'POST':
        form = UserForm(request.POST , request.FILES)
        if form.is_valid():
            username = form.cleaned_data['username']
            password = form.cleaned_data['password']

            user = ChatUser.objects.filter( username = username, password = password )
            if user:
                print 'userhead ' , user[0].headphoto
                path = settings.WEB_BASE_PATH + '/' + str(user[0].headphoto)
                print path
                import os
                try:
                    os.remove( path )
                except:
                    pass
                import Image
                uploaded = request.FILES['headphoto']
                from django.core.files.base import ContentFile
                file_content = ContentFile(uploaded.read())
                user[0].headphoto.save( str(user[0].headphoto), file_content )
            else:
                form.save()
            return HttpResponseRedirect('thanks.html')
        else:
            print 'error ' , form.errors
    else:
        form = UserForm(initial={'username':'watsy', 'password':'123123'})
    c['form'] = form
    return render_to_response('index.html', c)