mirror of
				https://github.com/django/django.git
				synced 2025-10-24 22:26:08 +00:00 
			
		
		
		
	Fixed #23609 -- Fixed IntegrityError that prevented altering a NULL column into a NOT NULL one due to existing rows
Thanks to Simon Charette, Loic Bistuer and Tim Graham for the review.
This commit is contained in:
		
				
					committed by
					
						 Loic Bistuer
						Loic Bistuer
					
				
			
			
				
	
			
			
			
						parent
						
							15d350fbce
						
					
				
				
					commit
					f633ba778d
				
			| @@ -44,6 +44,7 @@ class BaseDatabaseSchemaEditor(object): | ||||
|     sql_alter_column_no_default = "ALTER COLUMN %(column)s DROP DEFAULT" | ||||
|     sql_delete_column = "ALTER TABLE %(table)s DROP COLUMN %(column)s CASCADE" | ||||
|     sql_rename_column = "ALTER TABLE %(table)s RENAME COLUMN %(old_column)s TO %(new_column)s" | ||||
|     sql_update_with_default = "UPDATE %(table)s SET %(column)s = %(default)s WHERE %(column)s IS NULL" | ||||
|  | ||||
|     sql_create_check = "ALTER TABLE %(table)s ADD CONSTRAINT %(name)s CHECK (%(check)s)" | ||||
|     sql_delete_check = "ALTER TABLE %(table)s DROP CONSTRAINT %(name)s" | ||||
| @@ -533,12 +534,19 @@ class BaseDatabaseSchemaEditor(object): | ||||
|             }) | ||||
|         # Next, start accumulating actions to do | ||||
|         actions = [] | ||||
|         null_actions = [] | ||||
|         post_actions = [] | ||||
|         # Type change? | ||||
|         if old_type != new_type: | ||||
|             fragment, other_actions = self._alter_column_type_sql(model._meta.db_table, new_field.column, new_type) | ||||
|             actions.append(fragment) | ||||
|             post_actions.extend(other_actions) | ||||
|         # When changing a column NULL constraint to NOT NULL with a given | ||||
|         # default value, we need to perform 4 steps: | ||||
|         #  1. Add a default for new incoming writes | ||||
|         #  2. Update existing NULL rows with new default | ||||
|         #  3. Replace NULL constraint with NOT NULL | ||||
|         #  4. Drop the default again. | ||||
|         # Default change? | ||||
|         old_default = self.effective_default(old_field) | ||||
|         new_default = self.effective_default(new_field) | ||||
| @@ -573,7 +581,7 @@ class BaseDatabaseSchemaEditor(object): | ||||
|         # Nullability change? | ||||
|         if old_field.null != new_field.null: | ||||
|             if new_field.null: | ||||
|                 actions.append(( | ||||
|                 null_actions.append(( | ||||
|                     self.sql_alter_column_null % { | ||||
|                         "column": self.quote_name(new_field.column), | ||||
|                         "type": new_type, | ||||
| @@ -581,14 +589,23 @@ class BaseDatabaseSchemaEditor(object): | ||||
|                     [], | ||||
|                 )) | ||||
|             else: | ||||
|                 actions.append(( | ||||
|                 null_actions.append(( | ||||
|                     self.sql_alter_column_not_null % { | ||||
|                         "column": self.quote_name(new_field.column), | ||||
|                         "type": new_type, | ||||
|                     }, | ||||
|                     [], | ||||
|                 )) | ||||
|         if actions: | ||||
|         # Only if we have a default and there is a change from NULL to NOT NULL | ||||
|         four_way_default_alteration = ( | ||||
|             new_field.has_default() and | ||||
|             (old_field.null and not new_field.null) | ||||
|         ) | ||||
|         if actions or null_actions: | ||||
|             if not four_way_default_alteration: | ||||
|                 # If we don't have to do a 4-way default alteration we can | ||||
|                 # directly run a (NOT) NULL alteration | ||||
|                 actions = actions + null_actions | ||||
|             # Combine actions together if we can (e.g. postgres) | ||||
|             if self.connection.features.supports_combined_alters: | ||||
|                 sql, params = tuple(zip(*actions)) | ||||
| @@ -602,6 +619,26 @@ class BaseDatabaseSchemaEditor(object): | ||||
|                     }, | ||||
|                     params, | ||||
|                 ) | ||||
|             if four_way_default_alteration: | ||||
|                 # Update existing rows with default value | ||||
|                 self.execute( | ||||
|                     self.sql_update_with_default % { | ||||
|                         "table": self.quote_name(model._meta.db_table), | ||||
|                         "column": self.quote_name(new_field.column), | ||||
|                         "default": "%s", | ||||
|                     }, | ||||
|                     [new_default], | ||||
|                 ) | ||||
|                 # Since we didn't run a NOT NULL change before we need to do it | ||||
|                 # now | ||||
|                 for sql, params in null_actions: | ||||
|                     self.execute( | ||||
|                         self.sql_alter_column % { | ||||
|                             "table": self.quote_name(model._meta.db_table), | ||||
|                             "changes": sql, | ||||
|                         }, | ||||
|                         params, | ||||
|                     ) | ||||
|         if post_actions: | ||||
|             for sql, params in post_actions: | ||||
|                 self.execute(sql, params) | ||||
|   | ||||
| @@ -78,7 +78,14 @@ class DatabaseSchemaEditor(BaseDatabaseSchemaEditor): | ||||
|             del body[old_field.name] | ||||
|             del mapping[old_field.column] | ||||
|             body[new_field.name] = new_field | ||||
|             mapping[new_field.column] = self.quote_name(old_field.column) | ||||
|             if old_field.null and not new_field.null: | ||||
|                 case_sql = "coalesce(%(col)s, %(default)s)" % { | ||||
|                     'col': self.quote_name(old_field.column), | ||||
|                     'default': self.quote_value(self.effective_default(new_field)) | ||||
|                 } | ||||
|                 mapping[new_field.column] = case_sql | ||||
|             else: | ||||
|                 mapping[new_field.column] = self.quote_name(old_field.column) | ||||
|             rename_mapping[old_field.name] = new_field.name | ||||
|         # Remove any deleted fields | ||||
|         for field in delete_fields: | ||||
|   | ||||
| @@ -6,8 +6,8 @@ import datetime | ||||
| from itertools import chain | ||||
|  | ||||
| from django.utils import six | ||||
| from django.db import models | ||||
| from django.conf import settings | ||||
| from django.db import models | ||||
| from django.db.migrations import operations | ||||
| from django.db.migrations.migration import Migration | ||||
| from django.db.migrations.questioner import MigrationQuestioner | ||||
| @@ -838,7 +838,6 @@ class MigrationAutodetector(object): | ||||
|         for app_label, model_name, field_name in sorted(self.old_field_keys.intersection(self.new_field_keys)): | ||||
|             # Did the field change? | ||||
|             old_model_name = self.renamed_models.get((app_label, model_name), model_name) | ||||
|             new_model_state = self.to_state.models[app_label, model_name] | ||||
|             old_field_name = self.renamed_fields.get((app_label, model_name, field_name), field_name) | ||||
|             old_field = self.old_apps.get_model(app_label, old_model_name)._meta.get_field_by_name(old_field_name)[0] | ||||
|             new_field = self.new_apps.get_model(app_label, model_name)._meta.get_field_by_name(field_name)[0] | ||||
| @@ -854,12 +853,23 @@ class MigrationAutodetector(object): | ||||
|             old_field_dec = self.deep_deconstruct(old_field) | ||||
|             new_field_dec = self.deep_deconstruct(new_field) | ||||
|             if old_field_dec != new_field_dec: | ||||
|                 preserve_default = True | ||||
|                 if (old_field.null and not new_field.null and not new_field.has_default() and | ||||
|                         not isinstance(new_field, models.ManyToManyField)): | ||||
|                     field = new_field.clone() | ||||
|                     new_default = self.questioner.ask_not_null_alteration(field_name, model_name) | ||||
|                     if new_default is not models.NOT_PROVIDED: | ||||
|                         field.default = new_default | ||||
|                         preserve_default = False | ||||
|                 else: | ||||
|                     field = new_field | ||||
|                 self.add_operation( | ||||
|                     app_label, | ||||
|                     operations.AlterField( | ||||
|                         model_name=model_name, | ||||
|                         name=field_name, | ||||
|                         field=new_model_state.get_field_by_name(field_name), | ||||
|                         field=field, | ||||
|                         preserve_default=preserve_default, | ||||
|                     ) | ||||
|                 ) | ||||
|  | ||||
|   | ||||
| @@ -104,14 +104,20 @@ class AlterField(Operation): | ||||
|     Alters a field's database column (e.g. null, max_length) to the provided new field | ||||
|     """ | ||||
|  | ||||
|     def __init__(self, model_name, name, field): | ||||
|     def __init__(self, model_name, name, field, preserve_default=True): | ||||
|         self.model_name = model_name | ||||
|         self.name = name | ||||
|         self.field = field | ||||
|         self.preserve_default = preserve_default | ||||
|  | ||||
|     def state_forwards(self, app_label, state): | ||||
|         if not self.preserve_default: | ||||
|             field = self.field.clone() | ||||
|             field.default = NOT_PROVIDED | ||||
|         else: | ||||
|             field = self.field | ||||
|         state.models[app_label, self.model_name.lower()].fields = [ | ||||
|             (n, self.field if n == self.name else f) for n, f in state.models[app_label, self.model_name.lower()].fields | ||||
|             (n, field if n == self.name else f) for n, f in state.models[app_label, self.model_name.lower()].fields | ||||
|         ] | ||||
|  | ||||
|     def database_forwards(self, app_label, schema_editor, from_state, to_state): | ||||
| @@ -128,7 +134,11 @@ class AlterField(Operation): | ||||
|                     from_field.rel.to = to_field.rel.to | ||||
|                 elif to_field.rel and isinstance(to_field.rel.to, six.string_types): | ||||
|                     to_field.rel.to = from_field.rel.to | ||||
|             if not self.preserve_default: | ||||
|                 to_field.default = self.field.default | ||||
|             schema_editor.alter_field(from_model, from_field, to_field) | ||||
|             if not self.preserve_default: | ||||
|                 to_field.default = NOT_PROVIDED | ||||
|  | ||||
|     def database_backwards(self, app_label, schema_editor, from_state, to_state): | ||||
|         self.database_forwards(app_label, schema_editor, from_state, to_state) | ||||
|   | ||||
| @@ -1,10 +1,11 @@ | ||||
| from __future__ import unicode_literals | ||||
| from __future__ import print_function, unicode_literals | ||||
|  | ||||
| import importlib | ||||
| import os | ||||
| import sys | ||||
|  | ||||
| from django.apps import apps | ||||
| from django.db.models.fields import NOT_PROVIDED | ||||
| from django.utils import datetime_safe, six, timezone | ||||
| from django.utils.six.moves import input | ||||
|  | ||||
| @@ -55,6 +56,11 @@ class MigrationQuestioner(object): | ||||
|         # None means quit | ||||
|         return None | ||||
|  | ||||
|     def ask_not_null_alteration(self, field_name, model_name): | ||||
|         "Changing a NULL field to NOT NULL" | ||||
|         # None means quit | ||||
|         return None | ||||
|  | ||||
|     def ask_rename(self, model_name, old_name, new_name, field_instance): | ||||
|         "Was this field really renamed?" | ||||
|         return self.defaults.get("ask_rename", False) | ||||
| @@ -92,13 +98,34 @@ class InteractiveMigrationQuestioner(MigrationQuestioner): | ||||
|                 pass | ||||
|             result = input("Please select a valid option: ") | ||||
|  | ||||
|     def _ask_default(self): | ||||
|         print("Please enter the default value now, as valid Python") | ||||
|         print("The datetime and django.utils.timezone modules are available, so you can do e.g. timezone.now()") | ||||
|         while True: | ||||
|             if six.PY3: | ||||
|                 # Six does not correctly abstract over the fact that | ||||
|                 # py3 input returns a unicode string, while py2 raw_input | ||||
|                 # returns a bytestring. | ||||
|                 code = input(">>> ") | ||||
|             else: | ||||
|                 code = input(">>> ").decode(sys.stdin.encoding) | ||||
|             if not code: | ||||
|                 print("Please enter some code, or 'exit' (with no quotes) to exit.") | ||||
|             elif code == "exit": | ||||
|                 sys.exit(1) | ||||
|             else: | ||||
|                 try: | ||||
|                     return eval(code, {}, {"datetime": datetime_safe, "timezone": timezone}) | ||||
|                 except (SyntaxError, NameError) as e: | ||||
|                     print("Invalid input: %s" % e) | ||||
|  | ||||
|     def ask_not_null_addition(self, field_name, model_name): | ||||
|         "Adding a NOT NULL field to a model" | ||||
|         if not self.dry_run: | ||||
|             choice = self._choice_input( | ||||
|                 "You are trying to add a non-nullable field '%s' to %s without a default;\n" % (field_name, model_name) + | ||||
|                 "we can't do that (the database needs something to populate existing rows).\n" + | ||||
|                 "Please select a fix:", | ||||
|                 "You are trying to add a non-nullable field '%s' to %s without a default; " | ||||
|                 "we can't do that (the database needs something to populate existing rows).\n" | ||||
|                 "Please select a fix:" % (field_name, model_name), | ||||
|                 [ | ||||
|                     "Provide a one-off default now (will be set on all existing rows)", | ||||
|                     "Quit, and let me add a default in models.py", | ||||
| @@ -107,26 +134,31 @@ class InteractiveMigrationQuestioner(MigrationQuestioner): | ||||
|             if choice == 2: | ||||
|                 sys.exit(3) | ||||
|             else: | ||||
|                 print("Please enter the default value now, as valid Python") | ||||
|                 print("The datetime and django.utils.timezone modules are " | ||||
|                       "available, so you can do e.g. timezone.now()") | ||||
|                 while True: | ||||
|                     if six.PY3: | ||||
|                         # Six does not correctly abstract over the fact that | ||||
|                         # py3 input returns a unicode string, while py2 raw_input | ||||
|                         # returns a bytestring. | ||||
|                         code = input(">>> ") | ||||
|                     else: | ||||
|                         code = input(">>> ").decode(sys.stdin.encoding) | ||||
|                     if not code: | ||||
|                         print("Please enter some code, or 'exit' (with no quotes) to exit.") | ||||
|                     elif code == "exit": | ||||
|                         sys.exit(1) | ||||
|                     else: | ||||
|                         try: | ||||
|                             return eval(code, {}, {"datetime": datetime_safe, "timezone": timezone}) | ||||
|                         except (SyntaxError, NameError) as e: | ||||
|                             print("Invalid input: %s" % e) | ||||
|                 return self._ask_default() | ||||
|         return None | ||||
|  | ||||
|     def ask_not_null_alteration(self, field_name, model_name): | ||||
|         "Changing a NULL field to NOT NULL" | ||||
|         if not self.dry_run: | ||||
|             choice = self._choice_input( | ||||
|                 "You are trying to change the nullable field '%s' on %s to non-nullable " | ||||
|                 "without a default; we can't do that (the database needs something to " | ||||
|                 "populate existing rows).\n" | ||||
|                 "Please select a fix:" % (field_name, model_name), | ||||
|                 [ | ||||
|                     "Provide a one-off default now (will be set on all existing rows)", | ||||
|                     ("Ignore for now, and let me handle existing rows with NULL myself " | ||||
|                      "(e.g. adding a RunPython or RunSQL operation in the new migration " | ||||
|                      "file before the AlterField operation)"), | ||||
|                     "Quit, and let me add a default in models.py", | ||||
|                 ] | ||||
|             ) | ||||
|             if choice == 2: | ||||
|                 return NOT_PROVIDED | ||||
|             elif choice == 3: | ||||
|                 sys.exit(3) | ||||
|             else: | ||||
|                 return self._ask_default() | ||||
|         return None | ||||
|  | ||||
|     def ask_rename(self, model_name, old_name, new_name, field_instance): | ||||
|   | ||||
| @@ -137,7 +137,7 @@ or if it is temporary and just for this migration (``False``) - usually | ||||
| because the migration is adding a non-nullable field to a table and needs | ||||
| a default value to put into existing rows. It does not effect the behavior | ||||
| of setting defaults in the database directly - Django never sets database | ||||
| defaults, and always applies them in the Django ORM code. | ||||
| defaults and always applies them in the Django ORM code. | ||||
|  | ||||
| RemoveField | ||||
| ----------- | ||||
| @@ -153,16 +153,28 @@ from any data loss, which of course is irreversible). | ||||
| AlterField | ||||
| ---------- | ||||
|  | ||||
| .. class:: AlterField(model_name, name, field) | ||||
| .. class:: AlterField(model_name, name, field, preserve_default=True) | ||||
|  | ||||
| Alters a field's definition, including changes to its type, | ||||
| :attr:`~django.db.models.Field.null`, :attr:`~django.db.models.Field.unique`, | ||||
| :attr:`~django.db.models.Field.db_column` and other field attributes. | ||||
|  | ||||
| The ``preserve_default`` argument indicates whether the field's default | ||||
| value is permanent and should be baked into the project state (``True``), | ||||
| or if it is temporary and just for this migration (``False``) - usually | ||||
| because the migration is altering a nullable field to a non-nullable one and | ||||
| needs a default value to put into existing rows. It does not effect the | ||||
| behavior of setting defaults in the database directly - Django never sets | ||||
| database defaults and always applies them in the Django ORM code. | ||||
|  | ||||
| Note that not all changes are possible on all databases - for example, you | ||||
| cannot change a text-type field like ``models.TextField()`` into a number-type | ||||
| field like ``models.IntegerField()`` on most databases. | ||||
|  | ||||
| .. versionchanged:: 1.7.1 | ||||
|  | ||||
|     The ``preserve_default`` argument was added. | ||||
|  | ||||
| RenameField | ||||
| ----------- | ||||
|  | ||||
|   | ||||
| @@ -106,3 +106,7 @@ Bugfixes | ||||
|  | ||||
| * Made :func:`~django.utils.http.urlsafe_base64_decode` return the proper | ||||
|   type (byte string) on Python 3 (:ticket:`23333`). | ||||
|  | ||||
| * Added a prompt to the migrations questioner when removing the null constraint | ||||
|   from a field to prevent an IntegrityError on existing NULL rows | ||||
|   (:ticket:`23609`). | ||||
|   | ||||
| @@ -26,6 +26,7 @@ class AutodetectorTests(TestCase): | ||||
|  | ||||
|     author_empty = ModelState("testapp", "Author", [("id", models.AutoField(primary_key=True))]) | ||||
|     author_name = ModelState("testapp", "Author", [("id", models.AutoField(primary_key=True)), ("name", models.CharField(max_length=200))]) | ||||
|     author_name_null = ModelState("testapp", "Author", [("id", models.AutoField(primary_key=True)), ("name", models.CharField(max_length=200, null=True))]) | ||||
|     author_name_longer = ModelState("testapp", "Author", [("id", models.AutoField(primary_key=True)), ("name", models.CharField(max_length=400))]) | ||||
|     author_name_renamed = ModelState("testapp", "Author", [("id", models.AutoField(primary_key=True)), ("names", models.CharField(max_length=200))]) | ||||
|     author_name_default = ModelState("testapp", "Author", [("id", models.AutoField(primary_key=True)), ("name", models.CharField(max_length=200, default='Ada Lovelace'))]) | ||||
| @@ -302,6 +303,80 @@ class AutodetectorTests(TestCase): | ||||
|         action = migration.operations[0] | ||||
|         self.assertEqual(action.__class__.__name__, "AlterField") | ||||
|         self.assertEqual(action.name, "name") | ||||
|         self.assertTrue(action.preserve_default) | ||||
|  | ||||
|     def test_alter_field_to_not_null_with_default(self): | ||||
|         "#23609 - Tests autodetection of nullable to non-nullable alterations" | ||||
|         class CustomQuestioner(MigrationQuestioner): | ||||
|             def ask_not_null_alteration(self, field_name, model_name): | ||||
|                 raise Exception("Should not have prompted for not null addition") | ||||
|  | ||||
|         # Make state | ||||
|         before = self.make_project_state([self.author_name_null]) | ||||
|         after = self.make_project_state([self.author_name_default]) | ||||
|         autodetector = MigrationAutodetector(before, after, CustomQuestioner()) | ||||
|         changes = autodetector._detect_changes() | ||||
|         # Right number of migrations? | ||||
|         self.assertEqual(len(changes['testapp']), 1) | ||||
|         # Right number of actions? | ||||
|         migration = changes['testapp'][0] | ||||
|         self.assertEqual(len(migration.operations), 1) | ||||
|         # Right action? | ||||
|         action = migration.operations[0] | ||||
|         self.assertEqual(action.__class__.__name__, "AlterField") | ||||
|         self.assertEqual(action.name, "name") | ||||
|         self.assertTrue(action.preserve_default) | ||||
|         self.assertEqual(action.field.default, 'Ada Lovelace') | ||||
|  | ||||
|     def test_alter_field_to_not_null_without_default(self): | ||||
|         "#23609 - Tests autodetection of nullable to non-nullable alterations" | ||||
|         class CustomQuestioner(MigrationQuestioner): | ||||
|             def ask_not_null_alteration(self, field_name, model_name): | ||||
|                 # Ignore for now, and let me handle existing rows with NULL | ||||
|                 # myself (e.g. adding a RunPython or RunSQL operation in the new | ||||
|                 # migration file before the AlterField operation) | ||||
|                 return models.NOT_PROVIDED | ||||
|  | ||||
|         # Make state | ||||
|         before = self.make_project_state([self.author_name_null]) | ||||
|         after = self.make_project_state([self.author_name]) | ||||
|         autodetector = MigrationAutodetector(before, after, CustomQuestioner()) | ||||
|         changes = autodetector._detect_changes() | ||||
|         # Right number of migrations? | ||||
|         self.assertEqual(len(changes['testapp']), 1) | ||||
|         # Right number of actions? | ||||
|         migration = changes['testapp'][0] | ||||
|         self.assertEqual(len(migration.operations), 1) | ||||
|         # Right action? | ||||
|         action = migration.operations[0] | ||||
|         self.assertEqual(action.__class__.__name__, "AlterField") | ||||
|         self.assertEqual(action.name, "name") | ||||
|         self.assertTrue(action.preserve_default) | ||||
|         self.assertIs(action.field.default, models.NOT_PROVIDED) | ||||
|  | ||||
|     def test_alter_field_to_not_null_oneoff_default(self): | ||||
|         "#23609 - Tests autodetection of nullable to non-nullable alterations" | ||||
|         class CustomQuestioner(MigrationQuestioner): | ||||
|             def ask_not_null_alteration(self, field_name, model_name): | ||||
|                 # Provide a one-off default now (will be set on all existing rows) | ||||
|                 return 'Some Name' | ||||
|  | ||||
|         # Make state | ||||
|         before = self.make_project_state([self.author_name_null]) | ||||
|         after = self.make_project_state([self.author_name]) | ||||
|         autodetector = MigrationAutodetector(before, after, CustomQuestioner()) | ||||
|         changes = autodetector._detect_changes() | ||||
|         # Right number of migrations? | ||||
|         self.assertEqual(len(changes['testapp']), 1) | ||||
|         # Right number of actions? | ||||
|         migration = changes['testapp'][0] | ||||
|         self.assertEqual(len(migration.operations), 1) | ||||
|         # Right action? | ||||
|         action = migration.operations[0] | ||||
|         self.assertEqual(action.__class__.__name__, "AlterField") | ||||
|         self.assertEqual(action.name, "name") | ||||
|         self.assertFalse(action.preserve_default) | ||||
|         self.assertEqual(action.field.default, "Some Name") | ||||
|  | ||||
|     def test_rename_field(self): | ||||
|         "Tests autodetection of renamed fields" | ||||
|   | ||||
| @@ -3,7 +3,8 @@ import unittest | ||||
|  | ||||
| from django.test import TransactionTestCase | ||||
| from django.db import connection, DatabaseError, IntegrityError, OperationalError | ||||
| from django.db.models.fields import IntegerField, TextField, CharField, SlugField, BooleanField, BinaryField | ||||
| from django.db.models.fields import (BinaryField, BooleanField, CharField, IntegerField, | ||||
|     PositiveIntegerField, SlugField, TextField) | ||||
| from django.db.models.fields.related import ManyToManyField, ForeignKey | ||||
| from django.db.transaction import atomic | ||||
| from .models import (Author, AuthorWithM2M, Book, BookWithLongName, | ||||
| @@ -415,6 +416,38 @@ class SchemaTests(TransactionTestCase): | ||||
|         self.assertEqual(columns['name'][0], "TextField") | ||||
|         self.assertEqual(bool(columns['name'][1][6]), False) | ||||
|  | ||||
|     def test_alter_null_to_not_null(self): | ||||
|         """ | ||||
|         #23609 - Tests handling of default values when altering from NULL to NOT NULL. | ||||
|         """ | ||||
|         # Create the table | ||||
|         with connection.schema_editor() as editor: | ||||
|             editor.create_model(Author) | ||||
|         # Ensure the field is right to begin with | ||||
|         columns = self.column_classes(Author) | ||||
|         self.assertTrue(columns['height'][1][6]) | ||||
|         # Create some test data | ||||
|         Author.objects.create(name='Not null author', height=12) | ||||
|         Author.objects.create(name='Null author') | ||||
|         # Verify null value | ||||
|         self.assertEqual(Author.objects.get(name='Not null author').height, 12) | ||||
|         self.assertIsNone(Author.objects.get(name='Null author').height) | ||||
|         # Alter the height field to NOT NULL with default | ||||
|         new_field = PositiveIntegerField(default=42) | ||||
|         new_field.set_attributes_from_name("height") | ||||
|         with connection.schema_editor() as editor: | ||||
|             editor.alter_field( | ||||
|                 Author, | ||||
|                 Author._meta.get_field_by_name("height")[0], | ||||
|                 new_field | ||||
|             ) | ||||
|         # Ensure the field is right afterwards | ||||
|         columns = self.column_classes(Author) | ||||
|         self.assertFalse(columns['height'][1][6]) | ||||
|         # Verify default value | ||||
|         self.assertEqual(Author.objects.get(name='Not null author').height, 12) | ||||
|         self.assertEqual(Author.objects.get(name='Null author').height, 42) | ||||
|  | ||||
|     @unittest.skipUnless(connection.features.supports_foreign_keys, "No FK support") | ||||
|     def test_alter_fk(self): | ||||
|         """ | ||||
|   | ||||
		Reference in New Issue
	
	Block a user