From f2a59630f48f0daac46be62e234942dcdc1f366a Mon Sep 17 00:00:00 2001 From: Christophe Henry Date: Thu, 7 Oct 2021 12:00:47 +0200 Subject: [PATCH] [4.0.x] Fixed #33151 -- Fixed createsuperuser crash for many-to-many required fields in non-interactive mode. Backport of df2d2bc95c451c6366fd522a5a1e6ed84f459f31 from main --- .../management/commands/createsuperuser.py | 4 ++++ tests/auth_tests/test_management.py | 19 +++++++++++++++++++ 2 files changed, 23 insertions(+) diff --git a/django/contrib/auth/management/commands/createsuperuser.py b/django/contrib/auth/management/commands/createsuperuser.py index 3eb8abff066..61b4061252f 100644 --- a/django/contrib/auth/management/commands/createsuperuser.py +++ b/django/contrib/auth/management/commands/createsuperuser.py @@ -185,6 +185,10 @@ class Command(BaseCommand): raise CommandError('You must use --%s with --noinput.' % field_name) field = self.UserModel._meta.get_field(field_name) user_data[field_name] = field.clean(value, None) + if field.many_to_many and isinstance(user_data[field_name], str): + user_data[field_name] = [ + pk.strip() for pk in user_data[field_name].split(',') + ] self.UserModel._default_manager.db_manager(database).create_superuser(**user_data) if options['verbosity'] >= 1: diff --git a/tests/auth_tests/test_management.py b/tests/auth_tests/test_management.py index 61807732110..8a4e23952c3 100644 --- a/tests/auth_tests/test_management.py +++ b/tests/auth_tests/test_management.py @@ -994,6 +994,25 @@ class CreatesuperuserManagementCommandTestCase(TestCase): # Environment variables are ignored for non-required fields. self.assertEqual(user.first_name, '') + @override_settings(AUTH_USER_MODEL='auth_tests.CustomUserWithM2m') + def test_environment_variable_m2m_non_interactive(self): + new_io = StringIO() + org_id_1 = Organization.objects.create(name='Organization 1').pk + org_id_2 = Organization.objects.create(name='Organization 2').pk + with mock.patch.dict(os.environ, { + 'DJANGO_SUPERUSER_ORGS': f'{org_id_1},{org_id_2}', + }): + call_command( + 'createsuperuser', + interactive=False, + username='joe', + stdout=new_io, + ) + command_output = new_io.getvalue().strip() + self.assertEqual(command_output, 'Superuser created successfully.') + user = CustomUserWithM2M._default_manager.get(username='joe') + self.assertEqual(user.orgs.count(), 2) + @mock.patch.dict(os.environ, { 'DJANGO_SUPERUSER_USERNAME': 'test_superuser', 'DJANGO_SUPERUSER_EMAIL': 'joe@somewhere.org',