mirror of
				https://github.com/django/django.git
				synced 2025-10-24 22:26:08 +00:00 
			
		
		
		
	Fixed #23405 -- Fixed makemigrations prompt when adding Text/CharField.
A default is no longer required.
This commit is contained in:
		
				
					committed by
					
						 Tim Graham
						Tim Graham
					
				
			
			
				
	
			
			
			
						parent
						
							d39461eb46
						
					
				
				
					commit
					d8f3b86a76
				
			| @@ -797,8 +797,11 @@ class MigrationAutodetector(object): | |||||||
|                         None, |                         None, | ||||||
|                         True |                         True | ||||||
|                     )) |                     )) | ||||||
|             # You can't just add NOT NULL fields with no default |             # You can't just add NOT NULL fields with no default or fields | ||||||
|             if not field.null and not field.has_default() and not isinstance(field, models.ManyToManyField): |             # which don't allow empty strings as default. | ||||||
|  |             if (not field.null and not field.has_default() and | ||||||
|  |                     not isinstance(field, models.ManyToManyField) and | ||||||
|  |                     not (field.blank and field.empty_strings_allowed)): | ||||||
|                 field = field.clone() |                 field = field.clone() | ||||||
|                 field.default = self.questioner.ask_not_null_addition(field_name, model_name) |                 field.default = self.questioner.ask_not_null_addition(field_name, model_name) | ||||||
|                 self.add_operation( |                 self.add_operation( | ||||||
|   | |||||||
| @@ -140,3 +140,6 @@ Bugfixes | |||||||
|   a user specified ``default``. For example, a ``CharField`` with ``blank=True`` |   a user specified ``default``. For example, a ``CharField`` with ``blank=True`` | ||||||
|   didn't set existing rows to an empty string which resulted in a crash when |   didn't set existing rows to an empty string which resulted in a crash when | ||||||
|   adding the ``NOT NULL`` constraint (:ticket:`23987`). |   adding the ``NOT NULL`` constraint (:ticket:`23987`). | ||||||
|  |  | ||||||
|  | * ``makemigrations`` no longer prompts for a default value when adding | ||||||
|  |   ``TextField()`` or ``CharField()`` without a ``default`` (:ticket:`23405`). | ||||||
|   | |||||||
| @@ -1,6 +1,6 @@ | |||||||
| # -*- coding: utf-8 -*- | # -*- coding: utf-8 -*- | ||||||
| from django.conf import settings | from django.conf import settings | ||||||
| from django.test import TestCase, override_settings | from django.test import TestCase, mock, override_settings | ||||||
| from django.db.migrations.autodetector import MigrationAutodetector | from django.db.migrations.autodetector import MigrationAutodetector | ||||||
| from django.db.migrations.questioner import MigrationQuestioner | from django.db.migrations.questioner import MigrationQuestioner | ||||||
| from django.db.migrations.state import ProjectState, ModelState | from django.db.migrations.state import ProjectState, ModelState | ||||||
| @@ -64,6 +64,16 @@ class AutodetectorTests(TestCase): | |||||||
|         ("name", models.CharField(max_length=200, default=models.IntegerField())), |         ("name", models.CharField(max_length=200, default=models.IntegerField())), | ||||||
|     ]) |     ]) | ||||||
|     author_custom_pk = ModelState("testapp", "Author", [("pk_field", models.IntegerField(primary_key=True))]) |     author_custom_pk = ModelState("testapp", "Author", [("pk_field", models.IntegerField(primary_key=True))]) | ||||||
|  |     author_with_biography_non_blank = ModelState("testapp", "Author", [ | ||||||
|  |         ("id", models.AutoField(primary_key=True)), | ||||||
|  |         ("name", models.CharField()), | ||||||
|  |         ("biography", models.TextField()), | ||||||
|  |     ]) | ||||||
|  |     author_with_biography_blank = ModelState("testapp", "Author", [ | ||||||
|  |         ("id", models.AutoField(primary_key=True)), | ||||||
|  |         ("name", models.CharField(blank=True)), | ||||||
|  |         ("biography", models.TextField(blank=True)), | ||||||
|  |     ]) | ||||||
|     author_with_book = ModelState("testapp", "Author", [ |     author_with_book = ModelState("testapp", "Author", [ | ||||||
|         ("id", models.AutoField(primary_key=True)), |         ("id", models.AutoField(primary_key=True)), | ||||||
|         ("name", models.CharField(max_length=200)), |         ("name", models.CharField(max_length=200)), | ||||||
| @@ -1711,3 +1721,37 @@ class AutodetectorTests(TestCase): | |||||||
|         self.assertNumberMigrations(changes, 'a', 1) |         self.assertNumberMigrations(changes, 'a', 1) | ||||||
|         self.assertOperationTypes(changes, 'a', 0, ["CreateModel"]) |         self.assertOperationTypes(changes, 'a', 0, ["CreateModel"]) | ||||||
|         self.assertMigrationDependencies(changes, 'a', 0, []) |         self.assertMigrationDependencies(changes, 'a', 0, []) | ||||||
|  |  | ||||||
|  |     def test_add_blank_textfield_and_charfield(self): | ||||||
|  |         """ | ||||||
|  |         #23405 - Adding a NOT NULL and blank `CharField` or `TextField` | ||||||
|  |         without default should not prompt for a default. | ||||||
|  |         """ | ||||||
|  |         class CustomQuestioner(MigrationQuestioner): | ||||||
|  |             def ask_not_null_addition(self, field_name, model_name): | ||||||
|  |                 raise Exception("Should not have prompted for not null addition") | ||||||
|  |  | ||||||
|  |         before = self.make_project_state([self.author_empty]) | ||||||
|  |         after = self.make_project_state([self.author_with_biography_blank]) | ||||||
|  |         autodetector = MigrationAutodetector(before, after, CustomQuestioner()) | ||||||
|  |         changes = autodetector._detect_changes() | ||||||
|  |         self.assertNumberMigrations(changes, 'testapp', 1) | ||||||
|  |         self.assertOperationTypes(changes, 'testapp', 0, ["AddField", "AddField"]) | ||||||
|  |         self.assertOperationAttributes(changes, 'testapp', 0, 0) | ||||||
|  |  | ||||||
|  |     @mock.patch('django.db.migrations.questioner.MigrationQuestioner.ask_not_null_addition') | ||||||
|  |     def test_add_non_blank_textfield_and_charfield(self, mocked_ask_method): | ||||||
|  |         """ | ||||||
|  |         #23405 - Adding a NOT NULL and non-blank `CharField` or `TextField` | ||||||
|  |         without default should prompt for a default. | ||||||
|  |         """ | ||||||
|  |         before = self.make_project_state([self.author_empty]) | ||||||
|  |         after = self.make_project_state([self.author_with_biography_non_blank]) | ||||||
|  |         autodetector = MigrationAutodetector(before, after, MigrationQuestioner()) | ||||||
|  |         changes = autodetector._detect_changes() | ||||||
|  |         # need to check for questioner call | ||||||
|  |         self.assertTrue(mocked_ask_method.called) | ||||||
|  |         self.assertEqual(mocked_ask_method.call_count, 2) | ||||||
|  |         self.assertNumberMigrations(changes, 'testapp', 1) | ||||||
|  |         self.assertOperationTypes(changes, 'testapp', 0, ["AddField", "AddField"]) | ||||||
|  |         self.assertOperationAttributes(changes, 'testapp', 0, 0) | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user