Перезапись изображения с тем же именем - Джанго - PullRequest
0 голосов
/ 10 апреля 2019

Благодаря моему проекту пользователи загружают фотографию профиля. Я сохраняю изображение профиля как userID.jpg. Если они загружают новое изображение профиля, я хочу перезаписать старое изображение профиля, чтобы не тратить место на диске. Просматривая ранее задаваемые вопросы по stackoverflow, я переопределил OverwriteStorage:

class OverwriteStorage(FileSystemStorage):
    def get_available_name(self, name, max_length=None):
        if self.exists(name):
             os.remove(os.path.join(settings.MEDIA_ROOT, name))
        return name

Когда я загружаю изображение профиля, я вижу в каталоге на моем компьютере, что изображение было успешно перезаписано. Изображение сохраняется с путем "media / profile / userID.jpg". Тем не менее, когда я показываю изображение на моем сайте, это все еще старая фотография. Через сайт Django, когда я открываю путь, я вижу старую картинку, и когда я пытаюсь изменить ее через администратора, я получаю следующую ошибку:

[WinError 32] The process cannot access the file because it is being used by another process: '\media\\profile\\userID.jpg'

Полагаю, я неправильно перезаписываю файл, а другой экземпляр все еще открыт, и для его решения мне нужно правильно закрыть изображение перед перезаписью. Я пытался сделать это, но безуспешно.

Ответы [ 2 ]

0 голосов
/ 10 апреля 2019

Я сделал нечто подобное, но использовал сигналы для обновления и удаления изображений.

Firstable, я определил имя изображения в helpers.py

from django.conf import settings
from datetime import datetime

def upload_to_image_post(self, filename):
    """
    Stores the image in a specific path regards to date 
    and changes the name of the image with for the name of the post
    """    
    ext = filename.split('.')[-1]
    current_date = datetime.now()

    return '%s/posts/main/{year}/{month}/{day}/%s'.format(
        year=current_date.strftime('%Y'), month=current_date.strftime('%m'), 
        day=current_date.strftime('%d')) % (settings.MEDIA_ROOT, filename)

ТАК, я вызвал def в моей модели, в частности в поле изображения

from django.db import models
from django.utils.text import slugify
from .helpers import upload_to_image_post

class Post(models.Model):
    """
    Store a simple Post entry. 
    """
    title = models.CharField('Title', max_length=200, help_text='Title of the post')
    body = models.TextField('Body', help_text='Enter the description of the post')   
    slug = models.SlugField('Slug', max_length=200, db_index=True, unique=True, help_text='Title in format of URL')        
    image_post = models.ImageField('Image', max_length=80, blank=True, upload_to=upload_to_image_post, help_text='Main image of the post')

    class Meta:
        verbose_name = 'Post'
        verbose_name_plural = 'Posts'

Наконец, я определил сигналы для обновления или удаления изображения до того, как в модели произойдут действия (обновить или удалить).

import os
from django.db import models
from django.dispatch import receiver
from django.db.models.signals import pre_delete, pre_save
from .models import Post

@receiver(pre_delete, sender=Post)
def post_delete(sender, instance, **kwargs):
    """
    Deleting the specific image of a Post after delete it
    """
    if instance.image_post:
        if os.path.isfile(instance.image_post.path):
            os.remove(instance.image_post.path)

@receiver(pre_save, sender=Post)
def post_update(sender, instance, **kwargs):
    """
    Replacing the specific image of a Post after update
    """
    if not instance.pk:
        return False

    if sender.objects.get(pk=instance.pk).image_post:
        old_image = sender.objects.get(pk=instance.pk).image_post
        new_image = instance.image_post
        if not old_image == new_image:
            if os.path.isfile(old_image.path):
                os.remove(old_image.path)
    else:
        return False

Надеюсь, это помогло вам.

0 голосов
/ 10 апреля 2019

Переименуйте старый в userID-old.jpg, а затем сохраните userID.jpg.Скоро никто не заметит, что это происходит.

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...