mirror of
				https://github.com/django/django.git
				synced 2025-10-25 22:56:12 +00:00 
			
		
		
		
	Prevent Oracle from changing field.null to True
Fixed #17957 -- when using Oracle and character fields, the fields were set null = True to ease the handling of empty strings. This caused problems when using multiple databases from different vendors, or when the character field happened to be also a primary key. The handling was changed so that NOT NULL is not emitted on Oracle even if field.null = False, and field.null is not touched otherwise. Thanks to bhuztez for the report, ramiro for triaging & comments, ikelly for the patch and alex for reviewing.
This commit is contained in:
		| @@ -50,7 +50,13 @@ class BaseDatabaseCreation(object): | |||||||
|             # Make the definition (e.g. 'foo VARCHAR(30)') for this field. |             # Make the definition (e.g. 'foo VARCHAR(30)') for this field. | ||||||
|             field_output = [style.SQL_FIELD(qn(f.column)), |             field_output = [style.SQL_FIELD(qn(f.column)), | ||||||
|                 style.SQL_COLTYPE(col_type)] |                 style.SQL_COLTYPE(col_type)] | ||||||
|             if not f.null: |             # Oracle treats the empty string ('') as null, so coerce the null | ||||||
|  |             # option whenever '' is a possible value. | ||||||
|  |             null = f.null | ||||||
|  |             if (f.empty_strings_allowed and not f.primary_key and | ||||||
|  |                     self.connection.features.interprets_empty_strings_as_nulls): | ||||||
|  |                 null = True | ||||||
|  |             if not null: | ||||||
|                 field_output.append(style.SQL_KEYWORD('NOT NULL')) |                 field_output.append(style.SQL_KEYWORD('NOT NULL')) | ||||||
|             if f.primary_key: |             if f.primary_key: | ||||||
|                 field_output.append(style.SQL_KEYWORD('PRIMARY KEY')) |                 field_output.append(style.SQL_KEYWORD('PRIMARY KEY')) | ||||||
|   | |||||||
| @@ -85,11 +85,6 @@ class Field(object): | |||||||
|         self.primary_key = primary_key |         self.primary_key = primary_key | ||||||
|         self.max_length, self._unique = max_length, unique |         self.max_length, self._unique = max_length, unique | ||||||
|         self.blank, self.null = blank, null |         self.blank, self.null = blank, null | ||||||
|         # Oracle treats the empty string ('') as null, so coerce the null |  | ||||||
|         # option whenever '' is a possible value. |  | ||||||
|         if (self.empty_strings_allowed and |  | ||||||
|             connection.features.interprets_empty_strings_as_nulls): |  | ||||||
|             self.null = True |  | ||||||
|         self.rel = rel |         self.rel = rel | ||||||
|         self.default = default |         self.default = default | ||||||
|         self.editable = editable |         self.editable = editable | ||||||
|   | |||||||
| @@ -1193,7 +1193,7 @@ class Query(object): | |||||||
|                             entry.negate() |                             entry.negate() | ||||||
|                             self.where.add(entry, AND) |                             self.where.add(entry, AND) | ||||||
|                             break |                             break | ||||||
|                 if field.null: |                 if self.is_nullable(field): | ||||||
|                     # In SQL NULL = anyvalue returns unknown, and NOT unknown |                     # In SQL NULL = anyvalue returns unknown, and NOT unknown | ||||||
|                     # is still unknown. However, in Python None = anyvalue is False |                     # is still unknown. However, in Python None = anyvalue is False | ||||||
|                     # (and not False is True...), and we want to return this Python's |                     # (and not False is True...), and we want to return this Python's | ||||||
| @@ -1396,7 +1396,8 @@ class Query(object): | |||||||
|                                 opts, target) |                                 opts, target) | ||||||
|  |  | ||||||
|                     alias = self.join((alias, table, from_col, to_col), |                     alias = self.join((alias, table, from_col, to_col), | ||||||
|                             exclusions=exclusions, nullable=field.null) |                                       exclusions=exclusions, | ||||||
|  |                                       nullable=self.is_nullable(field)) | ||||||
|                     joins.append(alias) |                     joins.append(alias) | ||||||
|                 else: |                 else: | ||||||
|                     # Non-relation fields. |                     # Non-relation fields. | ||||||
| @@ -1946,6 +1947,24 @@ class Query(object): | |||||||
|         self.select = [(select_alias, select_col)] |         self.select = [(select_alias, select_col)] | ||||||
|         self.remove_inherited_models() |         self.remove_inherited_models() | ||||||
|  |  | ||||||
|  |     def is_nullable(self, field): | ||||||
|  |         """ | ||||||
|  |         A helper to check if the given field should be treated as nullable. | ||||||
|  |  | ||||||
|  |         Some backends treat '' as null and Django treats such fields as | ||||||
|  |         nullable for those backends. In such situations field.null can be | ||||||
|  |         False even if we should treat the field as nullable. | ||||||
|  |         """ | ||||||
|  |         # We need to use DEFAULT_DB_ALIAS here, as QuerySet does not have | ||||||
|  |         # (nor should it have) knowledge of which connection is going to be | ||||||
|  |         # used. The proper fix would be to defer all decisions where | ||||||
|  |         # is_nullable() is needed to the compiler stage, but that is not easy | ||||||
|  |         # to do currently. | ||||||
|  |         if ((connections[DEFAULT_DB_ALIAS].features.interprets_empty_strings_as_nulls) | ||||||
|  |             and field.empty_strings_allowed): | ||||||
|  |             return True | ||||||
|  |         else: | ||||||
|  |             return field.null | ||||||
|  |  | ||||||
| def get_order_dir(field, default='ASC'): | def get_order_dir(field, default='ASC'): | ||||||
|     """ |     """ | ||||||
|   | |||||||
| @@ -685,11 +685,11 @@ NULL and empty strings | |||||||
|  |  | ||||||
| Django generally prefers to use the empty string ('') rather than | Django generally prefers to use the empty string ('') rather than | ||||||
| NULL, but Oracle treats both identically. To get around this, the | NULL, but Oracle treats both identically. To get around this, the | ||||||
| Oracle backend coerces the ``null=True`` option on fields that have | Oracle backend ignores an explicit ``null`` option on fields that | ||||||
| the empty string as a possible value. When fetching from the database, | have the empty string as a possible value and generates DDL as if | ||||||
| it is assumed that a NULL value in one of these fields really means | ``null=True``. When fetching from the database, it is assumed that | ||||||
| the empty string, and the data is silently converted to reflect this | a ``NULL`` value in one of these fields really means the empty | ||||||
| assumption. | string, and the data is silently converted to reflect this assumption. | ||||||
|  |  | ||||||
| ``TextField`` limitations | ``TextField`` limitations | ||||||
| ------------------------- | ------------------------- | ||||||
|   | |||||||
| @@ -55,9 +55,8 @@ string, not ``NULL``. | |||||||
|  |  | ||||||
| .. note:: | .. note:: | ||||||
|  |  | ||||||
|     When using the Oracle database backend, the ``null=True`` option will be |     When using the Oracle database backend, the value ``NULL`` will be stored to | ||||||
|     coerced for string-based fields that have the empty string as a possible |     denote the empty string regardless of this attribute. | ||||||
|     value, and the value ``NULL`` will be stored to denote the empty string. |  | ||||||
|  |  | ||||||
| If you want to accept :attr:`~Field.null` values with :class:`BooleanField`, | If you want to accept :attr:`~Field.null` values with :class:`BooleanField`, | ||||||
| use :class:`NullBooleanField` instead. | use :class:`NullBooleanField` instead. | ||||||
|   | |||||||
| @@ -1930,3 +1930,25 @@ class NullInExcludeTest(TestCase): | |||||||
|         self.assertQuerysetEqual( |         self.assertQuerysetEqual( | ||||||
|             NullableName.objects.exclude(name__in=[None]), |             NullableName.objects.exclude(name__in=[None]), | ||||||
|             ['i1'], attrgetter('name')) |             ['i1'], attrgetter('name')) | ||||||
|  |  | ||||||
|  | class EmptyStringsAsNullTest(TestCase): | ||||||
|  |     """ | ||||||
|  |     Test that filtering on non-null character fields works as expected. | ||||||
|  |     The reason for these tests is that Oracle treats '' as NULL, and this | ||||||
|  |     can cause problems in query construction. Refs #17957. | ||||||
|  |     """ | ||||||
|  |  | ||||||
|  |     def setUp(self): | ||||||
|  |         self.nc = NamedCategory.objects.create(name='') | ||||||
|  |  | ||||||
|  |     def test_direct_exclude(self): | ||||||
|  |         self.assertQuerysetEqual( | ||||||
|  |             NamedCategory.objects.exclude(name__in=['nonexisting']), | ||||||
|  |             [self.nc.pk], attrgetter('pk') | ||||||
|  |         ) | ||||||
|  |  | ||||||
|  |     def test_joined_exclude(self): | ||||||
|  |         self.assertQuerysetEqual( | ||||||
|  |             DumbCategory.objects.exclude(namedcategory__name__in=['nonexisting']), | ||||||
|  |             [self.nc.pk], attrgetter('pk') | ||||||
|  |         ) | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user