Increased test coverage of django.db.migrations.questioner.

This commit is contained in:
Jacob Walls 2021-11-07 09:34:16 -05:00 committed by Mariusz Felisiak
parent 448145092c
commit 1833a9eee9
2 changed files with 101 additions and 6 deletions

View File

@ -1458,6 +1458,12 @@ class MakeMigrationsTests(MigrationTestBase):
" 3) Quit and manually define a default value in models.py." " 3) Quit and manually define a default value in models.py."
) )
with self.temporary_migration_module(module='migrations.test_migrations'): with self.temporary_migration_module(module='migrations.test_migrations'):
# No message appears if --dry-run.
with captured_stdout() as out:
call_command(
'makemigrations', 'migrations', interactive=True, dry_run=True,
)
self.assertNotIn(input_msg, out.getvalue())
# 3 - quit. # 3 - quit.
with mock.patch('builtins.input', return_value='3'): with mock.patch('builtins.input', return_value='3'):
with captured_stdout() as out, self.assertRaises(SystemExit): with captured_stdout() as out, self.assertRaises(SystemExit):
@ -1512,6 +1518,39 @@ class MakeMigrationsTests(MigrationTestBase):
self.assertIn("Remove field silly_field from sillymodel", out.getvalue()) self.assertIn("Remove field silly_field from sillymodel", out.getvalue())
self.assertIn("Add field silly_rename to sillymodel", out.getvalue()) self.assertIn("Add field silly_rename to sillymodel", out.getvalue())
@mock.patch('builtins.input', return_value='Y')
def test_makemigrations_model_rename_interactive(self, mock_input):
class RenamedModel(models.Model):
silly_field = models.BooleanField(default=False)
class Meta:
app_label = 'migrations'
with self.temporary_migration_module(
module='migrations.test_migrations_no_default',
):
with captured_stdout() as out:
call_command('makemigrations', 'migrations', interactive=True)
self.assertIn('Rename model SillyModel to RenamedModel', out.getvalue())
@mock.patch('builtins.input', return_value='Y')
def test_makemigrations_field_rename_interactive(self, mock_input):
class SillyModel(models.Model):
silly_rename = models.BooleanField(default=False)
class Meta:
app_label = 'migrations'
with self.temporary_migration_module(
module='migrations.test_migrations_no_default',
):
with captured_stdout() as out:
call_command('makemigrations', 'migrations', interactive=True)
self.assertIn(
'Rename field silly_field on sillymodel to silly_rename',
out.getvalue(),
)
def test_makemigrations_handle_merge(self): def test_makemigrations_handle_merge(self):
""" """
makemigrations properly merges the conflicting migrations with --noinput. makemigrations properly merges the conflicting migrations with --noinput.
@ -1581,6 +1620,7 @@ class MakeMigrationsTests(MigrationTestBase):
class SillyModel(models.Model): class SillyModel(models.Model):
silly_field = models.BooleanField(default=False) silly_field = models.BooleanField(default=False)
silly_date = models.DateField() # Added field without a default silly_date = models.DateField() # Added field without a default
silly_auto_now = models.DateTimeField(auto_now_add=True)
class Meta: class Meta:
app_label = "migrations" app_label = "migrations"

View File

@ -18,12 +18,9 @@ class QuestionerTests(SimpleTestCase):
questioner = MigrationQuestioner() questioner = MigrationQuestioner()
self.assertIs(False, questioner.ask_initial('migrations')) self.assertIs(False, questioner.ask_initial('migrations'))
@mock.patch('builtins.input', return_value='datetime.timedelta(days=1)') def test_ask_not_null_alteration(self):
def test_timedelta_default(self, mock): questioner = MigrationQuestioner()
questioner = InteractiveMigrationQuestioner() self.assertIsNone(questioner.ask_not_null_alteration('field_name', 'model_name'))
with captured_stdout():
value = questioner._ask_default()
self.assertEqual(value, datetime.timedelta(days=1))
@mock.patch('builtins.input', return_value='2') @mock.patch('builtins.input', return_value='2')
def test_ask_not_null_alteration_not_provided(self, mock): def test_ask_not_null_alteration_not_provided(self, mock):
@ -31,3 +28,61 @@ class QuestionerTests(SimpleTestCase):
with captured_stdout(): with captured_stdout():
question = questioner.ask_not_null_alteration('field_name', 'model_name') question = questioner.ask_not_null_alteration('field_name', 'model_name')
self.assertEqual(question, NOT_PROVIDED) self.assertEqual(question, NOT_PROVIDED)
class QuestionerHelperMethodsTests(SimpleTestCase):
questioner = InteractiveMigrationQuestioner()
@mock.patch('builtins.input', return_value='datetime.timedelta(days=1)')
def test_questioner_default_timedelta(self, mock_input):
questioner = InteractiveMigrationQuestioner()
with captured_stdout():
value = questioner._ask_default()
self.assertEqual(value, datetime.timedelta(days=1))
@mock.patch('builtins.input', return_value='')
def test_questioner_default_no_user_entry(self, mock_input):
with captured_stdout():
value = self.questioner._ask_default(default='datetime.timedelta(days=1)')
self.assertEqual(value, datetime.timedelta(days=1))
@mock.patch('builtins.input', side_effect=['', 'exit'])
def test_questioner_no_default_no_user_entry(self, mock_input):
with captured_stdout() as stdout, self.assertRaises(SystemExit):
self.questioner._ask_default()
self.assertIn(
"Please enter some code, or 'exit' (without quotes) to exit.",
stdout.getvalue(),
)
@mock.patch('builtins.input', side_effect=['bad code', 'exit'])
def test_questioner_no_default_bad_user_entry_code(self, mock_input):
with captured_stdout() as stdout, self.assertRaises(SystemExit):
self.questioner._ask_default()
self.assertIn('Invalid input: unexpected EOF while parsing', stdout.getvalue())
@mock.patch('builtins.input', side_effect=['', 'n'])
def test_questioner_no_default_no_user_entry_boolean(self, mock_input):
with captured_stdout():
value = self.questioner._boolean_input('Proceed?')
self.assertIs(value, False)
@mock.patch('builtins.input', return_value='')
def test_questioner_default_no_user_entry_boolean(self, mock_input):
with captured_stdout():
value = self.questioner._boolean_input('Proceed?', default=True)
self.assertIs(value, True)
@mock.patch('builtins.input', side_effect=[10, 'garbage', 1])
def test_questioner_bad_user_choice(self, mock_input):
question = 'Make a choice:'
with captured_stdout() as stdout:
value = self.questioner._choice_input(question, choices='abc')
expected_msg = (
f'{question}\n'
f' 1) a\n'
f' 2) b\n'
f' 3) c\n'
)
self.assertIn(expected_msg, stdout.getvalue())
self.assertEqual(value, 1)