Django 1.8 makemigrations генерирует дублированную миграцию каждый раз из-за валидаторов - PullRequest
0 голосов
/ 21 мая 2018

У меня есть модель, у которой есть поле со следующим валидатором:

validators=[
        FileValidator(
            allowed_extensions=ALLOWED_PHOTO_EXT,
            allowed_mimetypes=ALLOWED_PHOTO_MIME_TYPES,
            max_size=ALLOWED_PHOTO_MAX_SIZE,
        )
    ],

Здесь идет сам валидатор

@deconstructible
class FileValidator(object):
    """
    Validator for files, checking the size, extension and mimetype.
    Initialization parameters:
        allowed_extensions: iterable with allowed file extensions
            ie. ('txt', 'doc')
        allowd_mimetypes: iterable with allowed mimetypes
            ie. ('image/png', )
        min_size: minimum number of bytes allowed
            ie. 100
        max_size: maximum number of bytes allowed
            ie. 24*1024*1024 for 24 MB
    Usage example::
        MyModel(models.Model):
            myfile = FileField(validators=FileValidator(max_size=24*1024*1024), ...)

    See https://gist.github.com/jrosebr1/2140738 (improved)
    """

    extension_message = _("Extension '%(extension)s' not allowed. Allowed extensions are: '%(allowed_extensions)s.'")
    mime_message = _("MIME type '%(mimetype)s' is not valid. Allowed types are: %(allowed_mimetypes)s.")
    min_size_message = _('The current file %(size)s, which is too small. The minumum file size is %(allowed_size)s.')
    max_size_message = _('The current file %(size)s, which is too large. The maximum file size is %(allowed_size)s.')

    def __init__(self, *args, **kwargs):
        self.allowed_extensions = kwargs.pop('allowed_extensions', None)
        self.allowed_mimetypes = kwargs.pop('allowed_mimetypes', None)
        self.min_size = kwargs.pop('min_size', 0)
        self.max_size = kwargs.pop('max_size', DEFAULT_FILE_MAX_SIZE)

    def __call__(self, value):
        """
        Check the extension, content type and file size.
        """

        # Check the extension
        ext = splitext(value.name)[1][1:].lower()
        if self.allowed_extensions and not ext in self.allowed_extensions:
            message = self.extension_message % {
                'extension': ext,
                'allowed_extensions': ', '.join(self.allowed_extensions)
            }

            raise ValidationError(message)

        # Check the content type
        # mimetype = mimetypes.guess_type(value.name)[0] # XXX Alternative guessing way, unsure
        mimetype = magic.from_buffer(value.read(1024), mime=True)
        if self.allowed_mimetypes and not mimetype in self.allowed_mimetypes:
            message = self.mime_message % {
                'mimetype': mimetype,
                'allowed_mimetypes': ', '.join(self.allowed_mimetypes)
            }

            raise ValidationError(message)

        # Check the file size
        filesize = len(value)
        if self.max_size and filesize > self.max_size:
            message = self.max_size_message % {
                'size': filesizeformat(filesize),
                'allowed_size': filesizeformat(self.max_size)
            }

            raise ValidationError(message)

        elif filesize < self.min_size:
            message = self.min_size_message % {
                'size': filesizeformat(filesize),
                'allowed_size': filesizeformat(self.min_size)
            }

            raise ValidationError(message)

Каждый раз, когда я запускаю ./manage.py makemigrations app1

0003_auto_20180521_0325.py:
    - Alter field xxx on yyy

Он каждый раз генерирует новую миграцию, которая всегда делает одно и то же (поле alter)

Если я закомментирую свой валидатор, это поведение прекращается, и makemigrations отображает сообщение «Обнаружено отсутствие изменений».

Что не так?Как я могу избежать этого поведения?


Редактировать: После ответа @ benjamin вот полный код, включая исправление.(__eq__ функция)

# -*- coding: utf-8 -*-

import magic

from os.path import splitext
from django.core.exceptions import ValidationError
from django.utils.deconstruct import deconstructible
from django.utils.translation import gettext_lazy as _
from django.template.defaultfilters import filesizeformat

DEFAULT_FILE_MAX_SIZE = 10 * 1024 * 1024  # 10Mo

# Photo, like profile picture
# Only allow web-friendly extensions/mime types, since they'll be displayed on web pages
# TODO Size is huge, should be optimised
ALLOWED_PHOTO_EXT = [
    'jpg',
    'jpeg',
    'png',
]

ALLOWED_PHOTO_MIME_TYPES = [
    'image/jpeg',
    'image/pjpeg',
    'image/png',
    'image/x-png',
]

ALLOWED_PHOTO_MAX_SIZE = 10 * 1024 * 1024  # 10Mo

# Any document
# Allow a wide range of extensions and mime types
# TODO Size is huge, should be optimised
ALLOWED_DOCUMENT_EXT = [
    'jpg',
    'jpeg',
    'png',
    'tif',
    'bmp',
    'pdf',
    'doc',
    'dot',
    'docx',
    'dotx',
    'xls',
    'xlt',
    'xla',
    'xlsx',
    'xltx',
    'pptx',
    'potx',
    'ppsx',
]

ALLOWED_DOCUMENT_MIME_TYPES = [
    'image/jpeg',
    'image/pjpeg',
    'image/png',
    'image/x-png',
    'image/tiff',
    'image/bmp',
    'application/pdf',
    'application/vnd.ms-excel',
    'application/vnd.openxmlformats-officedocument.spreadsheetml.sheet',
    'application/vnd.ms-powerpoint',
    'application/vnd.openxmlformats-officedocument.presentationml.presentation',
    'application/msword',
    'application/vnd.openxmlformats-officedocument.wordprocessingml.document',
    'application/zip',  # XXX PPTX can be detected as ZIP for some reasons
]

ALLOWED_DOCUMENT_MAX_SIZE = 10 * 1024 * 1024  # 10Mo

# Any document sent to SMoney, which have their own rules and limitations
# Allow a wide range of extensions and mime types
# TODO Size is huge, should be optimised
ALLOWED_DOCUMENT_SMONEY_EXT = [
    'jpg',
    'jpeg',
    'png',
    'tif',
    'bmp',
    'pdf',
]

ALLOWED_DOCUMENT_SMONEY_MIME_TYPES = [
    'image/jpeg',
    'image/pjpeg',
    'image/png',
    'image/x-png',
    'image/tiff',
    'image/bmp',
    'application/pdf',
]

ALLOWED_DOCUMENT_SMONEY_MAX_SIZE = 3 * 1024 * 1024  # 3Mo


@deconstructible
class FileValidator(object):
    """
    Validator for files, checking the size, extension and mimetype.
    Initialization parameters:
        allowed_extensions: iterable with allowed file extensions
            ie. ('txt', 'doc')
        allowd_mimetypes: iterable with allowed mimetypes
            ie. ('image/png', )
        min_size: minimum number of bytes allowed
            ie. 100
        max_size: maximum number of bytes allowed
            ie. 24*1024*1024 for 24 MB
    Usage example::
        MyModel(models.Model):
            myfile = FileField(validators=FileValidator(max_size=24*1024*1024), ...)

    See https://gist.github.com/jrosebr1/2140738 (improved)
    """

    extension_message = _("Extension '%(extension)s' not allowed. Allowed extensions are: '%(allowed_extensions)s.'")
    mime_message = _("MIME type '%(mimetype)s' is not valid. Allowed types are: %(allowed_mimetypes)s.")
    min_size_message = _('The current file %(size)s, which is too small. The minumum file size is %(allowed_size)s.')
    max_size_message = _('The current file %(size)s, which is too large. The maximum file size is %(allowed_size)s.')

    def __init__(self, *args, **kwargs):
        self.allowed_extensions = kwargs.pop('allowed_extensions', None)
        self.allowed_mimetypes = kwargs.pop('allowed_mimetypes', None)
        self.min_size = kwargs.pop('min_size', 0)
        self.max_size = kwargs.pop('max_size', DEFAULT_FILE_MAX_SIZE)

    def __call__(self, value):
        """
        Check the extension, content type and file size.
        """

        # Check the extension
        ext = splitext(value.name)[1][1:].lower()
        if self.allowed_extensions and not ext in self.allowed_extensions:
            message = self.extension_message % {
                'extension': ext,
                'allowed_extensions': ', '.join(self.allowed_extensions)
            }

            raise ValidationError(message)

        # Check the content type
        # mimetype = mimetypes.guess_type(value.name)[0] # XXX Alternative guessing way, unsure
        mimetype = magic.from_buffer(value.read(1024), mime=True)
        if self.allowed_mimetypes and not mimetype in self.allowed_mimetypes:
            message = self.mime_message % {
                'mimetype': mimetype,
                'allowed_mimetypes': ', '.join(self.allowed_mimetypes)
            }

            raise ValidationError(message)

        # Check the file size
        filesize = len(value)
        if self.max_size and filesize > self.max_size:
            message = self.max_size_message % {
                'size': filesizeformat(filesize),
                'allowed_size': filesizeformat(self.max_size)
            }

            raise ValidationError(message)

        elif filesize < self.min_size:
            message = self.min_size_message % {
                'size': filesizeformat(filesize),
                'allowed_size': filesizeformat(self.min_size)
            }

            raise ValidationError(message)

    def __eq__(self, other):
        return (
                isinstance(other, self.__class__) and
                self.allowed_extensions == other.allowed_extensions and
                self.allowed_mimetypes == other.allowed_mimetypes and
                self.min_size == self.min_size and
                self.max_size == self.max_size
        )

Ответы [ 2 ]

0 голосов
/ 21 мая 2018

Я хочу догадаться, что это из-за отсутствия метода __eq__ в классе валидатора.Что-то в сравнениях по умолчанию выглядит «не равным» в двух экземплярах, инициализированных одинаково на основе его __dict__.Попробуйте добавить явное:

def __eq__(self, other):
    return (isinstance(other, self.__class__) and 
            self.allowed_extensions == other.allowed_extensions and
            self.allowed_mimetypes == other.allowed_mimetypes and
            self.min_size == self.min_size and
            self.max_size == self.max_size
    )
0 голосов
/ 21 мая 2018
/app_name/migrations

удалить все .py файлы, кроме __ init __. Py

python manage.py makemigrations
python manage.py migrate

Если это не сработает, вы можете попробовать что-то, что я уже сделал

Внутри вида я проверяю, является ли изображение действительным, как это

def check_image(request):

    if len(request.FILES) != 0:

        image = request.FILES['image'] # <input type="file" name="image" ....>
        ext = os.path.splitext(image.name)[1]

        if ext in ['.jpg', '.jpeg', '.png', '.gif']:
            # file is image
            x=0
        else:
            # file is not image
            y=0

    return HttpResponseRedirect('/')
...