mirror of
				https://github.com/django/django.git
				synced 2025-10-25 06:36:07 +00:00 
			
		
		
		
	[1.0.X] Fixed #9039 -- Don't perform unique checks on NULL values, since NULL != NULL in SQL.
Backport of [9239] from trunk. git-svn-id: http://code.djangoproject.com/svn/django/branches/releases/1.0.X@9240 bcc190cf-cafb-0310-a4f2-bffc1f526a37
This commit is contained in:
		| @@ -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: | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user