Fixed #9039 -- Don't perform unique checks on NULL values, since NULL != NULL in SQL.
git-svn-id: http://code.djangoproject.com/svn/django/trunk@9239 bcc190cf-cafb-0310-a4f2-bffc1f526a37
This commit is contained in:
parent
a14b98e207
commit
49ef21d9ea
|
@ -216,34 +216,31 @@ class BaseModelForm(BaseForm):
|
||||||
def validate_unique(self):
|
def validate_unique(self):
|
||||||
from django.db.models.fields import FieldDoesNotExist
|
from django.db.models.fields import FieldDoesNotExist
|
||||||
|
|
||||||
# Gather a list of checks to perform. Since this is a ModelForm, some
|
# Gather a list of checks to perform. We only perform unique checks
|
||||||
# fields may have been excluded; we can't perform a unique check on a
|
# for fields present and not None in cleaned_data. Since this is a
|
||||||
# form that is missing fields involved in that check.
|
# ModelForm, some fields may have been excluded; we can't perform a unique
|
||||||
|
# check on a form that is missing fields involved in that check. It also does
|
||||||
|
# not make sense to check data that didn't validate, and since NULL does not
|
||||||
|
# equal NULL in SQL we should not do any unique checking for NULL values.
|
||||||
unique_checks = []
|
unique_checks = []
|
||||||
for check in self.instance._meta.unique_together[:]:
|
for check in self.instance._meta.unique_together[:]:
|
||||||
fields_on_form = [field for field in check if field in self.fields]
|
fields_on_form = [field for field in check if field in self.cleaned_data and not self.cleaned_data[field] is None]
|
||||||
if len(fields_on_form) == len(check):
|
if len(fields_on_form) == len(check):
|
||||||
unique_checks.append(check)
|
unique_checks.append(check)
|
||||||
|
|
||||||
form_errors = []
|
form_errors = []
|
||||||
|
|
||||||
# Gather a list of checks for fields declared as unique and add them to
|
# Gather a list of checks for fields declared as unique and add them to
|
||||||
# the list of checks. Again, skip fields not on the form.
|
# the list of checks. Again, skip empty fields and any that did not validate.
|
||||||
for name, field in self.fields.items():
|
for name, field in self.fields.items():
|
||||||
try:
|
try:
|
||||||
f = self.instance._meta.get_field_by_name(name)[0]
|
f = self.instance._meta.get_field_by_name(name)[0]
|
||||||
except FieldDoesNotExist:
|
except FieldDoesNotExist:
|
||||||
# This is an extra field that's not on the ModelForm, ignore it
|
# This is an extra field that's not on the ModelForm, ignore it
|
||||||
continue
|
continue
|
||||||
# MySQL can't handle ... WHERE pk IS NULL, so make sure we
|
if f.unique and name in self.cleaned_data and not self.cleaned_data[name] is None:
|
||||||
# don't generate queries of that form.
|
|
||||||
is_null_pk = f.primary_key and self.cleaned_data[name] is None
|
|
||||||
if name in self.cleaned_data and f.unique and not is_null_pk:
|
|
||||||
unique_checks.append((name,))
|
unique_checks.append((name,))
|
||||||
|
|
||||||
# Don't run unique checks on fields that already have an error.
|
|
||||||
unique_checks = [check for check in unique_checks if not [x in self._errors for x in check if x in self._errors]]
|
|
||||||
|
|
||||||
bad_fields = set()
|
bad_fields = set()
|
||||||
for unique_check in unique_checks:
|
for unique_check in unique_checks:
|
||||||
# Try to look up an existing object with the same values as this
|
# Try to look up an existing object with the same values as this
|
||||||
|
|
|
@ -145,7 +145,15 @@ class Inventory(models.Model):
|
||||||
|
|
||||||
def __unicode__(self):
|
def __unicode__(self):
|
||||||
return self.name
|
return self.name
|
||||||
|
|
||||||
|
class Book(models.Model):
|
||||||
|
title = models.CharField(max_length=40)
|
||||||
|
author = models.ForeignKey(Writer, blank=True, null=True)
|
||||||
|
special_id = models.IntegerField(blank=True, null=True, unique=True)
|
||||||
|
|
||||||
|
class Meta:
|
||||||
|
unique_together = ('title', 'author')
|
||||||
|
|
||||||
__test__ = {'API_TESTS': """
|
__test__ = {'API_TESTS': """
|
||||||
>>> from django import forms
|
>>> from django import forms
|
||||||
>>> from django.forms.models import ModelForm, model_to_dict
|
>>> from django.forms.models import ModelForm, model_to_dict
|
||||||
|
@ -1201,6 +1209,32 @@ False
|
||||||
>>> form.is_valid()
|
>>> form.is_valid()
|
||||||
True
|
True
|
||||||
|
|
||||||
|
# Unique & unique together with null values
|
||||||
|
>>> class BookForm(ModelForm):
|
||||||
|
... class Meta:
|
||||||
|
... model = Book
|
||||||
|
>>> w = Writer.objects.get(name='Mike Royko')
|
||||||
|
>>> form = BookForm({'title': 'I May Be Wrong But I Doubt It', 'author' : w.pk})
|
||||||
|
>>> form.is_valid()
|
||||||
|
True
|
||||||
|
>>> form.save()
|
||||||
|
<Book: Book object>
|
||||||
|
>>> form = BookForm({'title': 'I May Be Wrong But I Doubt It', 'author' : w.pk})
|
||||||
|
>>> form.is_valid()
|
||||||
|
False
|
||||||
|
>>> form._errors
|
||||||
|
{'__all__': [u'Book with this Title and Author already exists.']}
|
||||||
|
>>> form = BookForm({'title': 'I May Be Wrong But I Doubt It'})
|
||||||
|
>>> form.is_valid()
|
||||||
|
True
|
||||||
|
>>> form.save()
|
||||||
|
<Book: Book object>
|
||||||
|
>>> form = BookForm({'title': 'I May Be Wrong But I Doubt It'})
|
||||||
|
>>> form.is_valid()
|
||||||
|
True
|
||||||
|
>>> form.save()
|
||||||
|
<Book: Book object>
|
||||||
|
|
||||||
# Choices on CharField and IntegerField
|
# Choices on CharField and IntegerField
|
||||||
>>> class ArticleForm(ModelForm):
|
>>> class ArticleForm(ModelForm):
|
||||||
... class Meta:
|
... class Meta:
|
||||||
|
|
Loading…
Reference in New Issue