mirror of https://github.com/django/django.git
1535 lines
52 KiB
Python
1535 lines
52 KiB
Python
import builtins
|
|
import getpass
|
|
import os
|
|
import sys
|
|
from datetime import date
|
|
from io import StringIO
|
|
from unittest import mock
|
|
|
|
from django.apps import apps
|
|
from django.contrib.auth import get_permission_codename, management
|
|
from django.contrib.auth.management import create_permissions, get_default_username
|
|
from django.contrib.auth.management.commands import changepassword, createsuperuser
|
|
from django.contrib.auth.models import Group, Permission, User
|
|
from django.contrib.contenttypes.models import ContentType
|
|
from django.core.management import call_command
|
|
from django.core.management.base import CommandError
|
|
from django.db import migrations
|
|
from django.test import TestCase, override_settings
|
|
from django.utils.translation import gettext_lazy as _
|
|
|
|
from .models import (
|
|
CustomUser,
|
|
CustomUserNonUniqueUsername,
|
|
CustomUserWithFK,
|
|
CustomUserWithM2M,
|
|
CustomUserWithUniqueConstraint,
|
|
Email,
|
|
Organization,
|
|
UserProxy,
|
|
)
|
|
|
|
MOCK_INPUT_KEY_TO_PROMPTS = {
|
|
# @mock_inputs dict key: [expected prompt messages],
|
|
"bypass": ["Bypass password validation and create user anyway? [y/N]: "],
|
|
"email": ["Email address: "],
|
|
"date_of_birth": ["Date of birth: "],
|
|
"first_name": ["First name: "],
|
|
"username": [
|
|
"Username: ",
|
|
lambda: "Username (leave blank to use '%s'): " % get_default_username(),
|
|
],
|
|
}
|
|
|
|
|
|
def mock_inputs(inputs):
|
|
"""
|
|
Decorator to temporarily replace input/getpass to allow interactive
|
|
createsuperuser.
|
|
"""
|
|
|
|
def inner(test_func):
|
|
def wrapper(*args):
|
|
class mock_getpass:
|
|
@staticmethod
|
|
def getpass(prompt=b"Password: ", stream=None):
|
|
if callable(inputs["password"]):
|
|
return inputs["password"]()
|
|
return inputs["password"]
|
|
|
|
def mock_input(prompt):
|
|
assert "__proxy__" not in prompt
|
|
response = None
|
|
for key, val in inputs.items():
|
|
if val == "KeyboardInterrupt":
|
|
raise KeyboardInterrupt
|
|
# get() fallback because sometimes 'key' is the actual
|
|
# prompt rather than a shortcut name.
|
|
prompt_msgs = MOCK_INPUT_KEY_TO_PROMPTS.get(key, key)
|
|
if isinstance(prompt_msgs, list):
|
|
prompt_msgs = [
|
|
msg() if callable(msg) else msg for msg in prompt_msgs
|
|
]
|
|
if prompt in prompt_msgs:
|
|
if callable(val):
|
|
response = val()
|
|
else:
|
|
response = val
|
|
break
|
|
if response is None:
|
|
raise ValueError("Mock input for %r not found." % prompt)
|
|
return response
|
|
|
|
old_getpass = createsuperuser.getpass
|
|
old_input = builtins.input
|
|
createsuperuser.getpass = mock_getpass
|
|
builtins.input = mock_input
|
|
try:
|
|
test_func(*args)
|
|
finally:
|
|
createsuperuser.getpass = old_getpass
|
|
builtins.input = old_input
|
|
|
|
return wrapper
|
|
|
|
return inner
|
|
|
|
|
|
class MockTTY:
|
|
"""
|
|
A fake stdin object that pretends to be a TTY to be used in conjunction
|
|
with mock_inputs.
|
|
"""
|
|
|
|
def isatty(self):
|
|
return True
|
|
|
|
|
|
class MockInputTests(TestCase):
|
|
@mock_inputs({"username": "alice"})
|
|
def test_input_not_found(self):
|
|
with self.assertRaisesMessage(
|
|
ValueError, "Mock input for 'Email address: ' not found."
|
|
):
|
|
call_command("createsuperuser", stdin=MockTTY())
|
|
|
|
|
|
class GetDefaultUsernameTestCase(TestCase):
|
|
databases = {"default", "other"}
|
|
|
|
def setUp(self):
|
|
self.old_get_system_username = management.get_system_username
|
|
|
|
def tearDown(self):
|
|
management.get_system_username = self.old_get_system_username
|
|
|
|
def test_actual_implementation(self):
|
|
self.assertIsInstance(management.get_system_username(), str)
|
|
|
|
def test_simple(self):
|
|
management.get_system_username = lambda: "joe"
|
|
self.assertEqual(management.get_default_username(), "joe")
|
|
|
|
def test_existing(self):
|
|
User.objects.create(username="joe")
|
|
management.get_system_username = lambda: "joe"
|
|
self.assertEqual(management.get_default_username(), "")
|
|
self.assertEqual(management.get_default_username(check_db=False), "joe")
|
|
|
|
def test_i18n(self):
|
|
# 'Julia' with accented 'u':
|
|
management.get_system_username = lambda: "J\xfalia"
|
|
self.assertEqual(management.get_default_username(), "julia")
|
|
|
|
def test_with_database(self):
|
|
User.objects.create(username="joe")
|
|
management.get_system_username = lambda: "joe"
|
|
self.assertEqual(management.get_default_username(), "")
|
|
self.assertEqual(management.get_default_username(database="other"), "joe")
|
|
|
|
User.objects.using("other").create(username="joe")
|
|
self.assertEqual(management.get_default_username(database="other"), "")
|
|
|
|
|
|
@override_settings(
|
|
AUTH_PASSWORD_VALIDATORS=[
|
|
{"NAME": "django.contrib.auth.password_validation.NumericPasswordValidator"},
|
|
]
|
|
)
|
|
class ChangepasswordManagementCommandTestCase(TestCase):
|
|
@classmethod
|
|
def setUpTestData(cls):
|
|
cls.user = User.objects.create_user(username="joe", password="qwerty")
|
|
|
|
def setUp(self):
|
|
self.stdout = StringIO()
|
|
self.addCleanup(self.stdout.close)
|
|
self.stderr = StringIO()
|
|
self.addCleanup(self.stderr.close)
|
|
|
|
@mock.patch.object(getpass, "getpass", return_value="password")
|
|
def test_get_pass(self, mock_get_pass):
|
|
call_command("changepassword", username="joe", stdout=self.stdout)
|
|
self.assertIs(User.objects.get(username="joe").check_password("password"), True)
|
|
|
|
@mock.patch.object(getpass, "getpass", return_value="")
|
|
def test_get_pass_no_input(self, mock_get_pass):
|
|
with self.assertRaisesMessage(CommandError, "aborted"):
|
|
call_command("changepassword", username="joe", stdout=self.stdout)
|
|
|
|
@mock.patch.object(changepassword.Command, "_get_pass", return_value="new_password")
|
|
def test_system_username(self, mock_get_pass):
|
|
"""The system username is used if --username isn't provided."""
|
|
username = getpass.getuser()
|
|
User.objects.create_user(username=username, password="qwerty")
|
|
call_command("changepassword", stdout=self.stdout)
|
|
self.assertIs(
|
|
User.objects.get(username=username).check_password("new_password"), True
|
|
)
|
|
|
|
def test_nonexistent_username(self):
|
|
with self.assertRaisesMessage(CommandError, "user 'test' does not exist"):
|
|
call_command("changepassword", username="test", stdout=self.stdout)
|
|
|
|
@mock.patch.object(changepassword.Command, "_get_pass", return_value="not qwerty")
|
|
def test_that_changepassword_command_changes_joes_password(self, mock_get_pass):
|
|
"Executing the changepassword management command should change joe's password"
|
|
self.assertTrue(self.user.check_password("qwerty"))
|
|
|
|
call_command("changepassword", username="joe", stdout=self.stdout)
|
|
command_output = self.stdout.getvalue().strip()
|
|
|
|
self.assertEqual(
|
|
command_output,
|
|
"Changing password for user 'joe'\n"
|
|
"Password changed successfully for user 'joe'",
|
|
)
|
|
self.assertTrue(User.objects.get(username="joe").check_password("not qwerty"))
|
|
|
|
@mock.patch.object(
|
|
changepassword.Command, "_get_pass", side_effect=lambda *args: str(args)
|
|
)
|
|
def test_that_max_tries_exits_1(self, mock_get_pass):
|
|
"""
|
|
A CommandError should be thrown by handle() if the user enters in
|
|
mismatched passwords three times.
|
|
"""
|
|
msg = "Aborting password change for user 'joe' after 3 attempts"
|
|
with self.assertRaisesMessage(CommandError, msg):
|
|
call_command(
|
|
"changepassword", username="joe", stdout=self.stdout, stderr=self.stderr
|
|
)
|
|
|
|
@mock.patch.object(changepassword.Command, "_get_pass", return_value="1234567890")
|
|
def test_password_validation(self, mock_get_pass):
|
|
"""
|
|
A CommandError should be raised if the user enters in passwords which
|
|
fail validation three times.
|
|
"""
|
|
abort_msg = "Aborting password change for user 'joe' after 3 attempts"
|
|
with self.assertRaisesMessage(CommandError, abort_msg):
|
|
call_command(
|
|
"changepassword", username="joe", stdout=self.stdout, stderr=self.stderr
|
|
)
|
|
self.assertIn("This password is entirely numeric.", self.stderr.getvalue())
|
|
|
|
@mock.patch.object(changepassword.Command, "_get_pass", return_value="not qwerty")
|
|
def test_that_changepassword_command_works_with_nonascii_output(
|
|
self, mock_get_pass
|
|
):
|
|
"""
|
|
#21627 -- Executing the changepassword management command should allow
|
|
non-ASCII characters from the User object representation.
|
|
"""
|
|
# 'Julia' with accented 'u':
|
|
User.objects.create_user(username="J\xfalia", password="qwerty")
|
|
call_command("changepassword", username="J\xfalia", stdout=self.stdout)
|
|
|
|
|
|
class MultiDBChangepasswordManagementCommandTestCase(TestCase):
|
|
databases = {"default", "other"}
|
|
|
|
@mock.patch.object(changepassword.Command, "_get_pass", return_value="not qwerty")
|
|
def test_that_changepassword_command_with_database_option_uses_given_db(
|
|
self, mock_get_pass
|
|
):
|
|
"""
|
|
changepassword --database should operate on the specified DB.
|
|
"""
|
|
user = User.objects.db_manager("other").create_user(
|
|
username="joe", password="qwerty"
|
|
)
|
|
self.assertTrue(user.check_password("qwerty"))
|
|
|
|
out = StringIO()
|
|
call_command("changepassword", username="joe", database="other", stdout=out)
|
|
command_output = out.getvalue().strip()
|
|
|
|
self.assertEqual(
|
|
command_output,
|
|
"Changing password for user 'joe'\n"
|
|
"Password changed successfully for user 'joe'",
|
|
)
|
|
self.assertTrue(
|
|
User.objects.using("other").get(username="joe").check_password("not qwerty")
|
|
)
|
|
|
|
|
|
@override_settings(
|
|
SILENCED_SYSTEM_CHECKS=["fields.W342"], # ForeignKey(unique=True)
|
|
AUTH_PASSWORD_VALIDATORS=[
|
|
{"NAME": "django.contrib.auth.password_validation.NumericPasswordValidator"}
|
|
],
|
|
)
|
|
class CreatesuperuserManagementCommandTestCase(TestCase):
|
|
def test_no_email_argument(self):
|
|
new_io = StringIO()
|
|
with self.assertRaisesMessage(
|
|
CommandError, "You must use --email with --noinput."
|
|
):
|
|
call_command(
|
|
"createsuperuser", interactive=False, username="joe", stdout=new_io
|
|
)
|
|
|
|
def test_basic_usage(self):
|
|
"Check the operation of the createsuperuser management command"
|
|
# We can use the management command to create a superuser
|
|
new_io = StringIO()
|
|
call_command(
|
|
"createsuperuser",
|
|
interactive=False,
|
|
username="joe",
|
|
email="joe@somewhere.org",
|
|
stdout=new_io,
|
|
)
|
|
command_output = new_io.getvalue().strip()
|
|
self.assertEqual(command_output, "Superuser created successfully.")
|
|
u = User.objects.get(username="joe")
|
|
self.assertEqual(u.email, "joe@somewhere.org")
|
|
|
|
# created password should be unusable
|
|
self.assertFalse(u.has_usable_password())
|
|
|
|
def test_validate_username(self):
|
|
msg = (
|
|
"Enter a valid username. This value may contain only letters, numbers, "
|
|
"and @/./+/-/_ characters."
|
|
)
|
|
with self.assertRaisesMessage(CommandError, msg):
|
|
call_command(
|
|
"createsuperuser",
|
|
interactive=False,
|
|
username="🤠",
|
|
email="joe@somewhere.org",
|
|
)
|
|
|
|
def test_non_ascii_verbose_name(self):
|
|
@mock_inputs(
|
|
{
|
|
"password": "nopasswd",
|
|
"Uživatel (leave blank to use '%s'): "
|
|
% get_default_username(): "foo", # username (cz)
|
|
"email": "nolocale@somewhere.org",
|
|
}
|
|
)
|
|
def test(self):
|
|
username_field = User._meta.get_field("username")
|
|
old_verbose_name = username_field.verbose_name
|
|
username_field.verbose_name = _("u\u017eivatel")
|
|
new_io = StringIO()
|
|
try:
|
|
call_command(
|
|
"createsuperuser",
|
|
interactive=True,
|
|
stdout=new_io,
|
|
stdin=MockTTY(),
|
|
)
|
|
finally:
|
|
username_field.verbose_name = old_verbose_name
|
|
|
|
command_output = new_io.getvalue().strip()
|
|
self.assertEqual(command_output, "Superuser created successfully.")
|
|
|
|
test(self)
|
|
|
|
def test_verbosity_zero(self):
|
|
# We can suppress output on the management command
|
|
new_io = StringIO()
|
|
call_command(
|
|
"createsuperuser",
|
|
interactive=False,
|
|
username="joe2",
|
|
email="joe2@somewhere.org",
|
|
verbosity=0,
|
|
stdout=new_io,
|
|
)
|
|
command_output = new_io.getvalue().strip()
|
|
self.assertEqual(command_output, "")
|
|
u = User.objects.get(username="joe2")
|
|
self.assertEqual(u.email, "joe2@somewhere.org")
|
|
self.assertFalse(u.has_usable_password())
|
|
|
|
def test_email_in_username(self):
|
|
call_command(
|
|
"createsuperuser",
|
|
interactive=False,
|
|
username="joe+admin@somewhere.org",
|
|
email="joe@somewhere.org",
|
|
verbosity=0,
|
|
)
|
|
u = User._default_manager.get(username="joe+admin@somewhere.org")
|
|
self.assertEqual(u.email, "joe@somewhere.org")
|
|
self.assertFalse(u.has_usable_password())
|
|
|
|
@override_settings(AUTH_USER_MODEL="auth_tests.CustomUser")
|
|
def test_swappable_user(self):
|
|
"A superuser can be created when a custom user model is in use"
|
|
# We can use the management command to create a superuser
|
|
# We skip validation because the temporary substitution of the
|
|
# swappable User model messes with validation.
|
|
new_io = StringIO()
|
|
call_command(
|
|
"createsuperuser",
|
|
interactive=False,
|
|
email="joe@somewhere.org",
|
|
date_of_birth="1976-04-01",
|
|
first_name="Joe",
|
|
stdout=new_io,
|
|
)
|
|
command_output = new_io.getvalue().strip()
|
|
self.assertEqual(command_output, "Superuser created successfully.")
|
|
u = CustomUser._default_manager.get(email="joe@somewhere.org")
|
|
self.assertEqual(u.date_of_birth, date(1976, 4, 1))
|
|
|
|
# created password should be unusable
|
|
self.assertFalse(u.has_usable_password())
|
|
|
|
@override_settings(AUTH_USER_MODEL="auth_tests.CustomUser")
|
|
def test_swappable_user_missing_required_field(self):
|
|
"A Custom superuser won't be created when a required field isn't provided"
|
|
# We can use the management command to create a superuser
|
|
# We skip validation because the temporary substitution of the
|
|
# swappable User model messes with validation.
|
|
new_io = StringIO()
|
|
with self.assertRaisesMessage(
|
|
CommandError, "You must use --email with --noinput."
|
|
):
|
|
call_command(
|
|
"createsuperuser",
|
|
interactive=False,
|
|
stdout=new_io,
|
|
stderr=new_io,
|
|
)
|
|
|
|
self.assertEqual(CustomUser._default_manager.count(), 0)
|
|
|
|
@override_settings(
|
|
AUTH_USER_MODEL="auth_tests.CustomUserNonUniqueUsername",
|
|
AUTHENTICATION_BACKENDS=["my.custom.backend"],
|
|
)
|
|
def test_swappable_user_username_non_unique(self):
|
|
@mock_inputs(
|
|
{
|
|
"username": "joe",
|
|
"password": "nopasswd",
|
|
}
|
|
)
|
|
def createsuperuser():
|
|
new_io = StringIO()
|
|
call_command(
|
|
"createsuperuser",
|
|
interactive=True,
|
|
email="joe@somewhere.org",
|
|
stdout=new_io,
|
|
stdin=MockTTY(),
|
|
)
|
|
command_output = new_io.getvalue().strip()
|
|
self.assertEqual(command_output, "Superuser created successfully.")
|
|
|
|
for i in range(2):
|
|
createsuperuser()
|
|
|
|
users = CustomUserNonUniqueUsername.objects.filter(username="joe")
|
|
self.assertEqual(users.count(), 2)
|
|
|
|
def test_skip_if_not_in_TTY(self):
|
|
"""
|
|
If the command is not called from a TTY, it should be skipped and a
|
|
message should be displayed (#7423).
|
|
"""
|
|
|
|
class FakeStdin:
|
|
"""A fake stdin object that has isatty() return False."""
|
|
|
|
def isatty(self):
|
|
return False
|
|
|
|
out = StringIO()
|
|
call_command(
|
|
"createsuperuser",
|
|
stdin=FakeStdin(),
|
|
stdout=out,
|
|
interactive=True,
|
|
)
|
|
|
|
self.assertEqual(User._default_manager.count(), 0)
|
|
self.assertIn("Superuser creation skipped", out.getvalue())
|
|
|
|
def test_passing_stdin(self):
|
|
"""
|
|
You can pass a stdin object as an option and it should be
|
|
available on self.stdin.
|
|
If no such option is passed, it defaults to sys.stdin.
|
|
"""
|
|
sentinel = object()
|
|
command = createsuperuser.Command()
|
|
call_command(
|
|
command,
|
|
stdin=sentinel,
|
|
interactive=False,
|
|
verbosity=0,
|
|
username="janet",
|
|
email="janet@example.com",
|
|
)
|
|
self.assertIs(command.stdin, sentinel)
|
|
|
|
command = createsuperuser.Command()
|
|
call_command(
|
|
command,
|
|
interactive=False,
|
|
verbosity=0,
|
|
username="joe",
|
|
email="joe@example.com",
|
|
)
|
|
self.assertIs(command.stdin, sys.stdin)
|
|
|
|
@override_settings(AUTH_USER_MODEL="auth_tests.CustomUserWithFK")
|
|
def test_fields_with_fk(self):
|
|
new_io = StringIO()
|
|
group = Group.objects.create(name="mygroup")
|
|
email = Email.objects.create(email="mymail@gmail.com")
|
|
call_command(
|
|
"createsuperuser",
|
|
interactive=False,
|
|
username=email.pk,
|
|
email=email.email,
|
|
group=group.pk,
|
|
stdout=new_io,
|
|
)
|
|
command_output = new_io.getvalue().strip()
|
|
self.assertEqual(command_output, "Superuser created successfully.")
|
|
u = CustomUserWithFK._default_manager.get(email=email)
|
|
self.assertEqual(u.username, email)
|
|
self.assertEqual(u.group, group)
|
|
|
|
non_existent_email = "mymail2@gmail.com"
|
|
msg = "email instance with email %r does not exist." % non_existent_email
|
|
with self.assertRaisesMessage(CommandError, msg):
|
|
call_command(
|
|
"createsuperuser",
|
|
interactive=False,
|
|
username=email.pk,
|
|
email=non_existent_email,
|
|
stdout=new_io,
|
|
)
|
|
|
|
@override_settings(AUTH_USER_MODEL="auth_tests.CustomUserWithFK")
|
|
def test_fields_with_fk_interactive(self):
|
|
new_io = StringIO()
|
|
group = Group.objects.create(name="mygroup")
|
|
email = Email.objects.create(email="mymail@gmail.com")
|
|
|
|
@mock_inputs(
|
|
{
|
|
"password": "nopasswd",
|
|
"Username (Email.id): ": email.pk,
|
|
"Email (Email.email): ": email.email,
|
|
"Group (Group.id): ": group.pk,
|
|
}
|
|
)
|
|
def test(self):
|
|
call_command(
|
|
"createsuperuser",
|
|
interactive=True,
|
|
stdout=new_io,
|
|
stdin=MockTTY(),
|
|
)
|
|
|
|
command_output = new_io.getvalue().strip()
|
|
self.assertEqual(command_output, "Superuser created successfully.")
|
|
u = CustomUserWithFK._default_manager.get(email=email)
|
|
self.assertEqual(u.username, email)
|
|
self.assertEqual(u.group, group)
|
|
|
|
test(self)
|
|
|
|
@override_settings(AUTH_USER_MODEL="auth_tests.CustomUserWithFK")
|
|
def test_fields_with_fk_via_option_interactive(self):
|
|
new_io = StringIO()
|
|
group = Group.objects.create(name="mygroup")
|
|
email = Email.objects.create(email="mymail@gmail.com")
|
|
|
|
@mock_inputs({"password": "nopasswd"})
|
|
def test(self):
|
|
call_command(
|
|
"createsuperuser",
|
|
interactive=True,
|
|
username=email.pk,
|
|
email=email.email,
|
|
group=group.pk,
|
|
stdout=new_io,
|
|
stdin=MockTTY(),
|
|
)
|
|
|
|
command_output = new_io.getvalue().strip()
|
|
self.assertEqual(command_output, "Superuser created successfully.")
|
|
u = CustomUserWithFK._default_manager.get(email=email)
|
|
self.assertEqual(u.username, email)
|
|
self.assertEqual(u.group, group)
|
|
|
|
test(self)
|
|
|
|
@override_settings(AUTH_USER_MODEL="auth_tests.CustomUserWithFK")
|
|
def test_validate_fk(self):
|
|
email = Email.objects.create(email="mymail@gmail.com")
|
|
Group.objects.all().delete()
|
|
nonexistent_group_id = 1
|
|
msg = f"group instance with id {nonexistent_group_id} does not exist."
|
|
|
|
with self.assertRaisesMessage(CommandError, msg):
|
|
call_command(
|
|
"createsuperuser",
|
|
interactive=False,
|
|
username=email.pk,
|
|
email=email.email,
|
|
group=nonexistent_group_id,
|
|
verbosity=0,
|
|
)
|
|
|
|
@override_settings(AUTH_USER_MODEL="auth_tests.CustomUserWithFK")
|
|
def test_validate_fk_environment_variable(self):
|
|
email = Email.objects.create(email="mymail@gmail.com")
|
|
Group.objects.all().delete()
|
|
nonexistent_group_id = 1
|
|
msg = f"group instance with id {nonexistent_group_id} does not exist."
|
|
|
|
with mock.patch.dict(
|
|
os.environ,
|
|
{"DJANGO_SUPERUSER_GROUP": str(nonexistent_group_id)},
|
|
):
|
|
with self.assertRaisesMessage(CommandError, msg):
|
|
call_command(
|
|
"createsuperuser",
|
|
interactive=False,
|
|
username=email.pk,
|
|
email=email.email,
|
|
verbosity=0,
|
|
)
|
|
|
|
@override_settings(AUTH_USER_MODEL="auth_tests.CustomUserWithFK")
|
|
def test_validate_fk_via_option_interactive(self):
|
|
email = Email.objects.create(email="mymail@gmail.com")
|
|
Group.objects.all().delete()
|
|
nonexistent_group_id = 1
|
|
msg = f"group instance with id {nonexistent_group_id} does not exist."
|
|
|
|
@mock_inputs(
|
|
{
|
|
"password": "nopasswd",
|
|
"Username (Email.id): ": email.pk,
|
|
"Email (Email.email): ": email.email,
|
|
}
|
|
)
|
|
def test(self):
|
|
with self.assertRaisesMessage(CommandError, msg):
|
|
call_command(
|
|
"createsuperuser",
|
|
group=nonexistent_group_id,
|
|
stdin=MockTTY(),
|
|
verbosity=0,
|
|
)
|
|
|
|
test(self)
|
|
|
|
@override_settings(AUTH_USER_MODEL="auth_tests.CustomUserWithM2m")
|
|
def test_fields_with_m2m(self):
|
|
new_io = StringIO()
|
|
org_id_1 = Organization.objects.create(name="Organization 1").pk
|
|
org_id_2 = Organization.objects.create(name="Organization 2").pk
|
|
call_command(
|
|
"createsuperuser",
|
|
interactive=False,
|
|
username="joe",
|
|
orgs=[org_id_1, org_id_2],
|
|
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)
|
|
|
|
@override_settings(AUTH_USER_MODEL="auth_tests.CustomUserWithM2M")
|
|
def test_fields_with_m2m_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
|
|
|
|
@mock_inputs(
|
|
{
|
|
"password": "nopasswd",
|
|
"Username: ": "joe",
|
|
"Orgs (Organization.id): ": "%s, %s" % (org_id_1, org_id_2),
|
|
}
|
|
)
|
|
def test(self):
|
|
call_command(
|
|
"createsuperuser",
|
|
interactive=True,
|
|
stdout=new_io,
|
|
stdin=MockTTY(),
|
|
)
|
|
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)
|
|
|
|
test(self)
|
|
|
|
@override_settings(AUTH_USER_MODEL="auth_tests.CustomUserWithM2M")
|
|
def test_fields_with_m2m_interactive_blank(self):
|
|
new_io = StringIO()
|
|
org_id = Organization.objects.create(name="Organization").pk
|
|
entered_orgs = [str(org_id), " "]
|
|
|
|
def return_orgs():
|
|
return entered_orgs.pop()
|
|
|
|
@mock_inputs(
|
|
{
|
|
"password": "nopasswd",
|
|
"Username: ": "joe",
|
|
"Orgs (Organization.id): ": return_orgs,
|
|
}
|
|
)
|
|
def test(self):
|
|
call_command(
|
|
"createsuperuser",
|
|
interactive=True,
|
|
stdout=new_io,
|
|
stderr=new_io,
|
|
stdin=MockTTY(),
|
|
)
|
|
self.assertEqual(
|
|
new_io.getvalue().strip(),
|
|
"Error: This field cannot be blank.\n"
|
|
"Superuser created successfully.",
|
|
)
|
|
|
|
test(self)
|
|
|
|
@override_settings(AUTH_USER_MODEL="auth_tests.CustomUserWithM2MThrough")
|
|
def test_fields_with_m2m_and_through(self):
|
|
msg = (
|
|
"Required field 'orgs' specifies a many-to-many relation through "
|
|
"model, which is not supported."
|
|
)
|
|
with self.assertRaisesMessage(CommandError, msg):
|
|
call_command("createsuperuser")
|
|
|
|
def test_default_username(self):
|
|
"""createsuperuser uses a default username when one isn't provided."""
|
|
# Get the default username before creating a user.
|
|
default_username = get_default_username()
|
|
new_io = StringIO()
|
|
entered_passwords = ["password", "password"]
|
|
|
|
def return_passwords():
|
|
return entered_passwords.pop(0)
|
|
|
|
@mock_inputs({"password": return_passwords, "username": "", "email": ""})
|
|
def test(self):
|
|
call_command(
|
|
"createsuperuser",
|
|
interactive=True,
|
|
stdin=MockTTY(),
|
|
stdout=new_io,
|
|
stderr=new_io,
|
|
)
|
|
self.assertEqual(
|
|
new_io.getvalue().strip(), "Superuser created successfully."
|
|
)
|
|
self.assertTrue(User.objects.filter(username=default_username).exists())
|
|
|
|
test(self)
|
|
|
|
def test_password_validation(self):
|
|
"""
|
|
Creation should fail if the password fails validation.
|
|
"""
|
|
new_io = StringIO()
|
|
entered_passwords = ["1234567890", "1234567890", "password", "password"]
|
|
|
|
def bad_then_good_password():
|
|
return entered_passwords.pop(0)
|
|
|
|
@mock_inputs(
|
|
{
|
|
"password": bad_then_good_password,
|
|
"username": "joe1234567890",
|
|
"email": "",
|
|
"bypass": "n",
|
|
}
|
|
)
|
|
def test(self):
|
|
call_command(
|
|
"createsuperuser",
|
|
interactive=True,
|
|
stdin=MockTTY(),
|
|
stdout=new_io,
|
|
stderr=new_io,
|
|
)
|
|
self.assertEqual(
|
|
new_io.getvalue().strip(),
|
|
"This password is entirely numeric.\n"
|
|
"Superuser created successfully.",
|
|
)
|
|
|
|
test(self)
|
|
|
|
@override_settings(
|
|
AUTH_PASSWORD_VALIDATORS=[
|
|
{
|
|
"NAME": (
|
|
"django.contrib.auth.password_validation."
|
|
"UserAttributeSimilarityValidator"
|
|
)
|
|
},
|
|
]
|
|
)
|
|
def test_validate_password_against_username(self):
|
|
new_io = StringIO()
|
|
username = "supremelycomplex"
|
|
entered_passwords = [
|
|
username,
|
|
username,
|
|
"superduperunguessablepassword",
|
|
"superduperunguessablepassword",
|
|
]
|
|
|
|
def bad_then_good_password():
|
|
return entered_passwords.pop(0)
|
|
|
|
@mock_inputs(
|
|
{
|
|
"password": bad_then_good_password,
|
|
"username": username,
|
|
"email": "",
|
|
"bypass": "n",
|
|
}
|
|
)
|
|
def test(self):
|
|
call_command(
|
|
"createsuperuser",
|
|
interactive=True,
|
|
stdin=MockTTY(),
|
|
stdout=new_io,
|
|
stderr=new_io,
|
|
)
|
|
self.assertEqual(
|
|
new_io.getvalue().strip(),
|
|
"The password is too similar to the username.\n"
|
|
"Superuser created successfully.",
|
|
)
|
|
|
|
test(self)
|
|
|
|
@override_settings(
|
|
AUTH_USER_MODEL="auth_tests.CustomUser",
|
|
AUTH_PASSWORD_VALIDATORS=[
|
|
{
|
|
"NAME": (
|
|
"django.contrib.auth.password_validation."
|
|
"UserAttributeSimilarityValidator"
|
|
)
|
|
},
|
|
],
|
|
)
|
|
def test_validate_password_against_required_fields(self):
|
|
new_io = StringIO()
|
|
first_name = "josephine"
|
|
entered_passwords = [
|
|
first_name,
|
|
first_name,
|
|
"superduperunguessablepassword",
|
|
"superduperunguessablepassword",
|
|
]
|
|
|
|
def bad_then_good_password():
|
|
return entered_passwords.pop(0)
|
|
|
|
@mock_inputs(
|
|
{
|
|
"password": bad_then_good_password,
|
|
"username": "whatever",
|
|
"first_name": first_name,
|
|
"date_of_birth": "1970-01-01",
|
|
"email": "joey@example.com",
|
|
"bypass": "n",
|
|
}
|
|
)
|
|
def test(self):
|
|
call_command(
|
|
"createsuperuser",
|
|
interactive=True,
|
|
stdin=MockTTY(),
|
|
stdout=new_io,
|
|
stderr=new_io,
|
|
)
|
|
self.assertEqual(
|
|
new_io.getvalue().strip(),
|
|
"The password is too similar to the first name.\n"
|
|
"Superuser created successfully.",
|
|
)
|
|
|
|
test(self)
|
|
|
|
@override_settings(
|
|
AUTH_USER_MODEL="auth_tests.CustomUser",
|
|
AUTH_PASSWORD_VALIDATORS=[
|
|
{
|
|
"NAME": (
|
|
"django.contrib.auth.password_validation."
|
|
"UserAttributeSimilarityValidator"
|
|
)
|
|
},
|
|
],
|
|
)
|
|
def test_validate_password_against_required_fields_via_option(self):
|
|
new_io = StringIO()
|
|
first_name = "josephine"
|
|
entered_passwords = [
|
|
first_name,
|
|
first_name,
|
|
"superduperunguessablepassword",
|
|
"superduperunguessablepassword",
|
|
]
|
|
|
|
def bad_then_good_password():
|
|
return entered_passwords.pop(0)
|
|
|
|
@mock_inputs(
|
|
{
|
|
"password": bad_then_good_password,
|
|
"bypass": "n",
|
|
}
|
|
)
|
|
def test(self):
|
|
call_command(
|
|
"createsuperuser",
|
|
interactive=True,
|
|
first_name=first_name,
|
|
date_of_birth="1970-01-01",
|
|
email="joey@example.com",
|
|
stdin=MockTTY(),
|
|
stdout=new_io,
|
|
stderr=new_io,
|
|
)
|
|
self.assertEqual(
|
|
new_io.getvalue().strip(),
|
|
"The password is too similar to the first name.\n"
|
|
"Superuser created successfully.",
|
|
)
|
|
|
|
test(self)
|
|
|
|
def test_blank_username(self):
|
|
"""Creation fails if --username is blank."""
|
|
new_io = StringIO()
|
|
with self.assertRaisesMessage(CommandError, "Username cannot be blank."):
|
|
call_command(
|
|
"createsuperuser",
|
|
username="",
|
|
stdin=MockTTY(),
|
|
stdout=new_io,
|
|
stderr=new_io,
|
|
)
|
|
|
|
def test_blank_username_non_interactive(self):
|
|
new_io = StringIO()
|
|
with self.assertRaisesMessage(CommandError, "Username cannot be blank."):
|
|
call_command(
|
|
"createsuperuser",
|
|
username="",
|
|
interactive=False,
|
|
stdin=MockTTY(),
|
|
stdout=new_io,
|
|
stderr=new_io,
|
|
)
|
|
|
|
def test_blank_email_allowed_non_interactive(self):
|
|
new_io = StringIO()
|
|
|
|
call_command(
|
|
"createsuperuser",
|
|
email="",
|
|
username="joe",
|
|
interactive=False,
|
|
stdout=new_io,
|
|
stderr=new_io,
|
|
)
|
|
self.assertEqual(new_io.getvalue().strip(), "Superuser created successfully.")
|
|
u = User.objects.get(username="joe")
|
|
self.assertEqual(u.email, "")
|
|
|
|
@mock.patch.dict(os.environ, {"DJANGO_SUPERUSER_EMAIL": ""})
|
|
def test_blank_email_allowed_non_interactive_environment_variable(self):
|
|
new_io = StringIO()
|
|
|
|
call_command(
|
|
"createsuperuser",
|
|
username="joe",
|
|
interactive=False,
|
|
stdout=new_io,
|
|
stderr=new_io,
|
|
)
|
|
self.assertEqual(new_io.getvalue().strip(), "Superuser created successfully.")
|
|
u = User.objects.get(username="joe")
|
|
self.assertEqual(u.email, "")
|
|
|
|
def test_password_validation_bypass(self):
|
|
"""
|
|
Password validation can be bypassed by entering 'y' at the prompt.
|
|
"""
|
|
new_io = StringIO()
|
|
|
|
@mock_inputs(
|
|
{
|
|
"password": "1234567890",
|
|
"username": "joe1234567890",
|
|
"email": "",
|
|
"bypass": "y",
|
|
}
|
|
)
|
|
def test(self):
|
|
call_command(
|
|
"createsuperuser",
|
|
interactive=True,
|
|
stdin=MockTTY(),
|
|
stdout=new_io,
|
|
stderr=new_io,
|
|
)
|
|
self.assertEqual(
|
|
new_io.getvalue().strip(),
|
|
"This password is entirely numeric.\n"
|
|
"Superuser created successfully.",
|
|
)
|
|
|
|
test(self)
|
|
|
|
def test_invalid_username(self):
|
|
"""Creation fails if the username fails validation."""
|
|
user_field = User._meta.get_field(User.USERNAME_FIELD)
|
|
new_io = StringIO()
|
|
entered_passwords = ["password", "password"]
|
|
# Enter an invalid (too long) username first and then a valid one.
|
|
invalid_username = ("x" * user_field.max_length) + "y"
|
|
entered_usernames = [invalid_username, "janet"]
|
|
|
|
def return_passwords():
|
|
return entered_passwords.pop(0)
|
|
|
|
def return_usernames():
|
|
return entered_usernames.pop(0)
|
|
|
|
@mock_inputs(
|
|
{"password": return_passwords, "username": return_usernames, "email": ""}
|
|
)
|
|
def test(self):
|
|
call_command(
|
|
"createsuperuser",
|
|
interactive=True,
|
|
stdin=MockTTY(),
|
|
stdout=new_io,
|
|
stderr=new_io,
|
|
)
|
|
self.assertEqual(
|
|
new_io.getvalue().strip(),
|
|
"Error: Ensure this value has at most %s characters (it has %s).\n"
|
|
"Superuser created successfully."
|
|
% (user_field.max_length, len(invalid_username)),
|
|
)
|
|
|
|
test(self)
|
|
|
|
@mock_inputs({"username": "KeyboardInterrupt"})
|
|
def test_keyboard_interrupt(self):
|
|
new_io = StringIO()
|
|
with self.assertRaises(SystemExit):
|
|
call_command(
|
|
"createsuperuser",
|
|
interactive=True,
|
|
stdin=MockTTY(),
|
|
stdout=new_io,
|
|
stderr=new_io,
|
|
)
|
|
self.assertEqual(new_io.getvalue(), "\nOperation cancelled.\n")
|
|
|
|
def test_existing_username(self):
|
|
"""Creation fails if the username already exists."""
|
|
user = User.objects.create(username="janet")
|
|
new_io = StringIO()
|
|
entered_passwords = ["password", "password"]
|
|
# Enter the existing username first and then a new one.
|
|
entered_usernames = [user.username, "joe"]
|
|
|
|
def return_passwords():
|
|
return entered_passwords.pop(0)
|
|
|
|
def return_usernames():
|
|
return entered_usernames.pop(0)
|
|
|
|
@mock_inputs(
|
|
{"password": return_passwords, "username": return_usernames, "email": ""}
|
|
)
|
|
def test(self):
|
|
call_command(
|
|
"createsuperuser",
|
|
interactive=True,
|
|
stdin=MockTTY(),
|
|
stdout=new_io,
|
|
stderr=new_io,
|
|
)
|
|
self.assertEqual(
|
|
new_io.getvalue().strip(),
|
|
"Error: That username is already taken.\n"
|
|
"Superuser created successfully.",
|
|
)
|
|
|
|
test(self)
|
|
|
|
@override_settings(AUTH_USER_MODEL="auth_tests.CustomUserWithUniqueConstraint")
|
|
def test_existing_username_meta_unique_constraint(self):
|
|
"""
|
|
Creation fails if the username already exists and a custom user model
|
|
has UniqueConstraint.
|
|
"""
|
|
user = CustomUserWithUniqueConstraint.objects.create(username="janet")
|
|
new_io = StringIO()
|
|
entered_passwords = ["password", "password"]
|
|
# Enter the existing username first and then a new one.
|
|
entered_usernames = [user.username, "joe"]
|
|
|
|
def return_passwords():
|
|
return entered_passwords.pop(0)
|
|
|
|
def return_usernames():
|
|
return entered_usernames.pop(0)
|
|
|
|
@mock_inputs({"password": return_passwords, "username": return_usernames})
|
|
def test(self):
|
|
call_command(
|
|
"createsuperuser",
|
|
interactive=True,
|
|
stdin=MockTTY(),
|
|
stdout=new_io,
|
|
stderr=new_io,
|
|
)
|
|
self.assertEqual(
|
|
new_io.getvalue().strip(),
|
|
"Error: That username is already taken.\n"
|
|
"Superuser created successfully.",
|
|
)
|
|
|
|
test(self)
|
|
|
|
def test_existing_username_non_interactive(self):
|
|
"""Creation fails if the username already exists."""
|
|
User.objects.create(username="janet")
|
|
new_io = StringIO()
|
|
with self.assertRaisesMessage(
|
|
CommandError, "Error: That username is already taken."
|
|
):
|
|
call_command(
|
|
"createsuperuser",
|
|
username="janet",
|
|
email="",
|
|
interactive=False,
|
|
stdout=new_io,
|
|
)
|
|
|
|
def test_existing_username_provided_via_option_and_interactive(self):
|
|
"""call_command() gets username='janet' and interactive=True."""
|
|
new_io = StringIO()
|
|
entered_passwords = ["password", "password"]
|
|
User.objects.create(username="janet")
|
|
|
|
def return_passwords():
|
|
return entered_passwords.pop(0)
|
|
|
|
@mock_inputs(
|
|
{
|
|
"password": return_passwords,
|
|
"username": "janet1",
|
|
"email": "test@test.com",
|
|
}
|
|
)
|
|
def test(self):
|
|
call_command(
|
|
"createsuperuser",
|
|
username="janet",
|
|
interactive=True,
|
|
stdin=MockTTY(),
|
|
stdout=new_io,
|
|
stderr=new_io,
|
|
)
|
|
msg = (
|
|
"Error: That username is already taken.\n"
|
|
"Superuser created successfully."
|
|
)
|
|
self.assertEqual(new_io.getvalue().strip(), msg)
|
|
|
|
test(self)
|
|
|
|
def test_validation_mismatched_passwords(self):
|
|
"""
|
|
Creation should fail if the user enters mismatched passwords.
|
|
"""
|
|
new_io = StringIO()
|
|
|
|
# The first two passwords do not match, but the second two do match and
|
|
# are valid.
|
|
entered_passwords = ["password", "not password", "password2", "password2"]
|
|
|
|
def mismatched_passwords_then_matched():
|
|
return entered_passwords.pop(0)
|
|
|
|
@mock_inputs(
|
|
{
|
|
"password": mismatched_passwords_then_matched,
|
|
"username": "joe1234567890",
|
|
"email": "",
|
|
}
|
|
)
|
|
def test(self):
|
|
call_command(
|
|
"createsuperuser",
|
|
interactive=True,
|
|
stdin=MockTTY(),
|
|
stdout=new_io,
|
|
stderr=new_io,
|
|
)
|
|
self.assertEqual(
|
|
new_io.getvalue().strip(),
|
|
"Error: Your passwords didn't match.\n"
|
|
"Superuser created successfully.",
|
|
)
|
|
|
|
test(self)
|
|
|
|
def test_validation_blank_password_entered(self):
|
|
"""
|
|
Creation should fail if the user enters blank passwords.
|
|
"""
|
|
new_io = StringIO()
|
|
|
|
# The first two passwords are empty strings, but the second two are
|
|
# valid.
|
|
entered_passwords = ["", "", "password2", "password2"]
|
|
|
|
def blank_passwords_then_valid():
|
|
return entered_passwords.pop(0)
|
|
|
|
@mock_inputs(
|
|
{
|
|
"password": blank_passwords_then_valid,
|
|
"username": "joe1234567890",
|
|
"email": "",
|
|
}
|
|
)
|
|
def test(self):
|
|
call_command(
|
|
"createsuperuser",
|
|
interactive=True,
|
|
stdin=MockTTY(),
|
|
stdout=new_io,
|
|
stderr=new_io,
|
|
)
|
|
self.assertEqual(
|
|
new_io.getvalue().strip(),
|
|
"Error: Blank passwords aren't allowed.\n"
|
|
"Superuser created successfully.",
|
|
)
|
|
|
|
test(self)
|
|
|
|
@override_settings(AUTH_USER_MODEL="auth_tests.NoPasswordUser")
|
|
def test_usermodel_without_password(self):
|
|
new_io = StringIO()
|
|
call_command(
|
|
"createsuperuser",
|
|
interactive=False,
|
|
stdin=MockTTY(),
|
|
stdout=new_io,
|
|
stderr=new_io,
|
|
username="username",
|
|
)
|
|
self.assertEqual(new_io.getvalue().strip(), "Superuser created successfully.")
|
|
|
|
@override_settings(AUTH_USER_MODEL="auth_tests.NoPasswordUser")
|
|
def test_usermodel_without_password_interactive(self):
|
|
new_io = StringIO()
|
|
|
|
@mock_inputs({"username": "username"})
|
|
def test(self):
|
|
call_command(
|
|
"createsuperuser",
|
|
interactive=True,
|
|
stdin=MockTTY(),
|
|
stdout=new_io,
|
|
stderr=new_io,
|
|
)
|
|
self.assertEqual(
|
|
new_io.getvalue().strip(), "Superuser created successfully."
|
|
)
|
|
|
|
test(self)
|
|
|
|
@mock.patch.dict(
|
|
os.environ,
|
|
{
|
|
"DJANGO_SUPERUSER_PASSWORD": "test_password",
|
|
"DJANGO_SUPERUSER_USERNAME": "test_superuser",
|
|
"DJANGO_SUPERUSER_EMAIL": "joe@somewhere.org",
|
|
"DJANGO_SUPERUSER_FIRST_NAME": "ignored_first_name",
|
|
},
|
|
)
|
|
def test_environment_variable_non_interactive(self):
|
|
call_command("createsuperuser", interactive=False, verbosity=0)
|
|
user = User.objects.get(username="test_superuser")
|
|
self.assertEqual(user.email, "joe@somewhere.org")
|
|
self.assertTrue(user.check_password("test_password"))
|
|
# 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",
|
|
},
|
|
)
|
|
def test_ignore_environment_variable_non_interactive(self):
|
|
# Environment variables are ignored in non-interactive mode, if
|
|
# provided by a command line arguments.
|
|
call_command(
|
|
"createsuperuser",
|
|
interactive=False,
|
|
username="cmd_superuser",
|
|
email="cmd@somewhere.org",
|
|
verbosity=0,
|
|
)
|
|
user = User.objects.get(username="cmd_superuser")
|
|
self.assertEqual(user.email, "cmd@somewhere.org")
|
|
self.assertFalse(user.has_usable_password())
|
|
|
|
@mock.patch.dict(
|
|
os.environ,
|
|
{
|
|
"DJANGO_SUPERUSER_PASSWORD": "test_password",
|
|
"DJANGO_SUPERUSER_USERNAME": "test_superuser",
|
|
"DJANGO_SUPERUSER_EMAIL": "joe@somewhere.org",
|
|
},
|
|
)
|
|
def test_ignore_environment_variable_interactive(self):
|
|
# Environment variables are ignored in interactive mode.
|
|
@mock_inputs({"password": "cmd_password"})
|
|
def test(self):
|
|
call_command(
|
|
"createsuperuser",
|
|
interactive=True,
|
|
username="cmd_superuser",
|
|
email="cmd@somewhere.org",
|
|
stdin=MockTTY(),
|
|
verbosity=0,
|
|
)
|
|
user = User.objects.get(username="cmd_superuser")
|
|
self.assertEqual(user.email, "cmd@somewhere.org")
|
|
self.assertTrue(user.check_password("cmd_password"))
|
|
|
|
test(self)
|
|
|
|
|
|
class MultiDBCreatesuperuserTestCase(TestCase):
|
|
databases = {"default", "other"}
|
|
|
|
def test_createsuperuser_command_with_database_option(self):
|
|
"""
|
|
createsuperuser --database should operate on the specified DB.
|
|
"""
|
|
new_io = StringIO()
|
|
call_command(
|
|
"createsuperuser",
|
|
interactive=False,
|
|
username="joe",
|
|
email="joe@somewhere.org",
|
|
database="other",
|
|
stdout=new_io,
|
|
)
|
|
command_output = new_io.getvalue().strip()
|
|
self.assertEqual(command_output, "Superuser created successfully.")
|
|
user = User.objects.using("other").get(username="joe")
|
|
self.assertEqual(user.email, "joe@somewhere.org")
|
|
|
|
def test_createsuperuser_command_suggested_username_with_database_option(self):
|
|
default_username = get_default_username(database="other")
|
|
qs = User.objects.using("other")
|
|
|
|
@mock_inputs({"password": "nopasswd", "username": "", "email": ""})
|
|
def test_other_create_with_suggested_username(self):
|
|
call_command(
|
|
"createsuperuser",
|
|
interactive=True,
|
|
stdin=MockTTY(),
|
|
verbosity=0,
|
|
database="other",
|
|
)
|
|
self.assertIs(qs.filter(username=default_username).exists(), True)
|
|
|
|
test_other_create_with_suggested_username(self)
|
|
|
|
@mock_inputs({"password": "nopasswd", "Username: ": "other", "email": ""})
|
|
def test_other_no_suggestion(self):
|
|
call_command(
|
|
"createsuperuser",
|
|
interactive=True,
|
|
stdin=MockTTY(),
|
|
verbosity=0,
|
|
database="other",
|
|
)
|
|
self.assertIs(qs.filter(username="other").exists(), True)
|
|
|
|
test_other_no_suggestion(self)
|
|
|
|
|
|
class CreatePermissionsTests(TestCase):
|
|
def setUp(self):
|
|
self._original_permissions = Permission._meta.permissions[:]
|
|
self._original_default_permissions = Permission._meta.default_permissions
|
|
self.app_config = apps.get_app_config("auth")
|
|
|
|
def tearDown(self):
|
|
Permission._meta.permissions = self._original_permissions
|
|
Permission._meta.default_permissions = self._original_default_permissions
|
|
ContentType.objects.clear_cache()
|
|
|
|
def test_default_permissions(self):
|
|
permission_content_type = ContentType.objects.get_by_natural_key(
|
|
"auth", "permission"
|
|
)
|
|
Permission._meta.permissions = [
|
|
("my_custom_permission", "Some permission"),
|
|
]
|
|
create_permissions(self.app_config, verbosity=0)
|
|
|
|
# view/add/change/delete permission by default + custom permission
|
|
self.assertEqual(
|
|
Permission.objects.filter(
|
|
content_type=permission_content_type,
|
|
).count(),
|
|
5,
|
|
)
|
|
|
|
Permission.objects.filter(content_type=permission_content_type).delete()
|
|
Permission._meta.default_permissions = []
|
|
create_permissions(self.app_config, verbosity=0)
|
|
|
|
# custom permission only since default permissions is empty
|
|
self.assertEqual(
|
|
Permission.objects.filter(
|
|
content_type=permission_content_type,
|
|
).count(),
|
|
1,
|
|
)
|
|
|
|
def test_unavailable_models(self):
|
|
"""
|
|
#24075 - Permissions shouldn't be created or deleted if the ContentType
|
|
or Permission models aren't available.
|
|
"""
|
|
state = migrations.state.ProjectState()
|
|
# Unavailable contenttypes.ContentType
|
|
with self.assertNumQueries(0):
|
|
create_permissions(self.app_config, verbosity=0, apps=state.apps)
|
|
# Unavailable auth.Permission
|
|
state = migrations.state.ProjectState(real_apps={"contenttypes"})
|
|
with self.assertNumQueries(0):
|
|
create_permissions(self.app_config, verbosity=0, apps=state.apps)
|
|
|
|
def test_create_permissions_checks_contenttypes_created(self):
|
|
"""
|
|
`post_migrate` handler ordering isn't guaranteed. Simulate a case
|
|
where create_permissions() is called before create_contenttypes().
|
|
"""
|
|
# Warm the manager cache.
|
|
ContentType.objects.get_for_model(Group)
|
|
# Apply a deletion as if e.g. a database 'flush' had been executed.
|
|
ContentType.objects.filter(app_label="auth", model="group").delete()
|
|
# This fails with a foreign key constraint without the fix.
|
|
create_permissions(apps.get_app_config("auth"), interactive=False, verbosity=0)
|
|
|
|
def test_permission_with_proxy_content_type_created(self):
|
|
"""
|
|
A proxy model's permissions use its own content type rather than the
|
|
content type of the concrete model.
|
|
"""
|
|
opts = UserProxy._meta
|
|
codename = get_permission_codename("add", opts)
|
|
self.assertTrue(
|
|
Permission.objects.filter(
|
|
content_type__model=opts.model_name,
|
|
content_type__app_label=opts.app_label,
|
|
codename=codename,
|
|
).exists()
|
|
)
|
|
|
|
|
|
class DefaultDBRouter:
|
|
"""Route all writes to default."""
|
|
|
|
def db_for_write(self, model, **hints):
|
|
return "default"
|
|
|
|
|
|
@override_settings(DATABASE_ROUTERS=[DefaultDBRouter()])
|
|
class CreatePermissionsMultipleDatabasesTests(TestCase):
|
|
databases = {"default", "other"}
|
|
|
|
def test_set_permissions_fk_to_using_parameter(self):
|
|
Permission.objects.using("other").delete()
|
|
with self.assertNumQueries(6, using="other") as captured_queries:
|
|
create_permissions(apps.get_app_config("auth"), verbosity=0, using="other")
|
|
self.assertIn("INSERT INTO", captured_queries[-1]["sql"].upper())
|
|
self.assertGreater(Permission.objects.using("other").count(), 0)
|