2008-06-08 13:31:16 +08:00
|
|
|
"""
|
|
|
|
Management utility to create superusers.
|
|
|
|
"""
|
|
|
|
|
|
|
|
import getpass
|
|
|
|
import os
|
|
|
|
import re
|
|
|
|
import sys
|
|
|
|
from optparse import make_option
|
2008-07-22 11:24:09 +08:00
|
|
|
from django.contrib.auth.models import User
|
Removed oldforms, validators, and related code:
* Removed `Manipulator`, `AutomaticManipulator`, and related classes.
* Removed oldforms specific bits from model fields:
* Removed `validator_list` and `core` arguments from constructors.
* Removed the methods:
* `get_manipulator_field_names`
* `get_manipulator_field_objs`
* `get_manipulator_fields`
* `get_manipulator_new_data`
* `prepare_field_objs_and_params`
* `get_follow`
* Renamed `flatten_data` method to `value_to_string` for better alignment with its use by the serialization framework, which was the only remaining code using `flatten_data`.
* Removed oldforms methods from `django.db.models.Options` class: `get_followed_related_objects`, `get_data_holders`, `get_follow`, and `has_field_type`.
* Removed oldforms-admin specific options from `django.db.models.fields.related` classes: `num_in_admin`, `min_num_in_admin`, `max_num_in_admin`, `num_extra_on_change`, and `edit_inline`.
* Serialization framework
* `Serializer.get_string_value` now calls the model fields' renamed `value_to_string` methods.
* Removed a special-casing of `models.DateTimeField` in `core.serializers.base.Serializer.get_string_value` that's handled by `django.db.models.fields.DateTimeField.value_to_string`.
* Removed `django.core.validators`:
* Moved `ValidationError` exception to `django.core.exceptions`.
* For the couple places that were using validators, brought over the necessary code to maintain the same functionality.
* Introduced a SlugField form field for validation and to compliment the SlugField model field (refs #8040).
* Removed an oldforms-style model creation hack (refs #2160).
git-svn-id: http://code.djangoproject.com/svn/django/trunk@8616 bcc190cf-cafb-0310-a4f2-bffc1f526a37
2008-08-27 15:19:44 +08:00
|
|
|
from django.core import exceptions
|
2008-06-08 13:31:16 +08:00
|
|
|
from django.core.management.base import BaseCommand, CommandError
|
Removed oldforms, validators, and related code:
* Removed `Manipulator`, `AutomaticManipulator`, and related classes.
* Removed oldforms specific bits from model fields:
* Removed `validator_list` and `core` arguments from constructors.
* Removed the methods:
* `get_manipulator_field_names`
* `get_manipulator_field_objs`
* `get_manipulator_fields`
* `get_manipulator_new_data`
* `prepare_field_objs_and_params`
* `get_follow`
* Renamed `flatten_data` method to `value_to_string` for better alignment with its use by the serialization framework, which was the only remaining code using `flatten_data`.
* Removed oldforms methods from `django.db.models.Options` class: `get_followed_related_objects`, `get_data_holders`, `get_follow`, and `has_field_type`.
* Removed oldforms-admin specific options from `django.db.models.fields.related` classes: `num_in_admin`, `min_num_in_admin`, `max_num_in_admin`, `num_extra_on_change`, and `edit_inline`.
* Serialization framework
* `Serializer.get_string_value` now calls the model fields' renamed `value_to_string` methods.
* Removed a special-casing of `models.DateTimeField` in `core.serializers.base.Serializer.get_string_value` that's handled by `django.db.models.fields.DateTimeField.value_to_string`.
* Removed `django.core.validators`:
* Moved `ValidationError` exception to `django.core.exceptions`.
* For the couple places that were using validators, brought over the necessary code to maintain the same functionality.
* Introduced a SlugField form field for validation and to compliment the SlugField model field (refs #8040).
* Removed an oldforms-style model creation hack (refs #2160).
git-svn-id: http://code.djangoproject.com/svn/django/trunk@8616 bcc190cf-cafb-0310-a4f2-bffc1f526a37
2008-08-27 15:19:44 +08:00
|
|
|
from django.utils.translation import ugettext as _
|
2008-06-08 13:31:16 +08:00
|
|
|
|
2010-05-21 22:08:49 +08:00
|
|
|
RE_VALID_USERNAME = re.compile('[\w.@+-]+$')
|
|
|
|
|
Removed oldforms, validators, and related code:
* Removed `Manipulator`, `AutomaticManipulator`, and related classes.
* Removed oldforms specific bits from model fields:
* Removed `validator_list` and `core` arguments from constructors.
* Removed the methods:
* `get_manipulator_field_names`
* `get_manipulator_field_objs`
* `get_manipulator_fields`
* `get_manipulator_new_data`
* `prepare_field_objs_and_params`
* `get_follow`
* Renamed `flatten_data` method to `value_to_string` for better alignment with its use by the serialization framework, which was the only remaining code using `flatten_data`.
* Removed oldforms methods from `django.db.models.Options` class: `get_followed_related_objects`, `get_data_holders`, `get_follow`, and `has_field_type`.
* Removed oldforms-admin specific options from `django.db.models.fields.related` classes: `num_in_admin`, `min_num_in_admin`, `max_num_in_admin`, `num_extra_on_change`, and `edit_inline`.
* Serialization framework
* `Serializer.get_string_value` now calls the model fields' renamed `value_to_string` methods.
* Removed a special-casing of `models.DateTimeField` in `core.serializers.base.Serializer.get_string_value` that's handled by `django.db.models.fields.DateTimeField.value_to_string`.
* Removed `django.core.validators`:
* Moved `ValidationError` exception to `django.core.exceptions`.
* For the couple places that were using validators, brought over the necessary code to maintain the same functionality.
* Introduced a SlugField form field for validation and to compliment the SlugField model field (refs #8040).
* Removed an oldforms-style model creation hack (refs #2160).
git-svn-id: http://code.djangoproject.com/svn/django/trunk@8616 bcc190cf-cafb-0310-a4f2-bffc1f526a37
2008-08-27 15:19:44 +08:00
|
|
|
EMAIL_RE = re.compile(
|
|
|
|
r"(^[-!#$%&'*+/=?^_`{}|~0-9A-Z]+(\.[-!#$%&'*+/=?^_`{}|~0-9A-Z]+)*" # dot-atom
|
|
|
|
r'|^"([\001-\010\013\014\016-\037!#-\[\]-\177]|\\[\001-\011\013\014\016-\177])*"' # quoted-string
|
|
|
|
r')@(?:[A-Z0-9-]+\.)+[A-Z]{2,6}$', re.IGNORECASE) # domain
|
|
|
|
|
|
|
|
def is_valid_email(value):
|
|
|
|
if not EMAIL_RE.search(value):
|
|
|
|
raise exceptions.ValidationError(_('Enter a valid e-mail address.'))
|
2008-06-08 13:31:16 +08:00
|
|
|
|
|
|
|
class Command(BaseCommand):
|
|
|
|
option_list = BaseCommand.option_list + (
|
|
|
|
make_option('--username', dest='username', default=None,
|
|
|
|
help='Specifies the username for the superuser.'),
|
|
|
|
make_option('--email', dest='email', default=None,
|
|
|
|
help='Specifies the email address for the superuser.'),
|
|
|
|
make_option('--noinput', action='store_false', dest='interactive', default=True,
|
|
|
|
help='Tells Django to NOT prompt the user for input of any kind. ' \
|
|
|
|
'You must use --username and --email with --noinput, and ' \
|
|
|
|
'superusers created with --noinput will not be able to log in ' \
|
|
|
|
'until they\'re given a valid password.'),
|
|
|
|
)
|
|
|
|
help = 'Used to create a superuser.'
|
|
|
|
|
|
|
|
def handle(self, *args, **options):
|
|
|
|
username = options.get('username', None)
|
|
|
|
email = options.get('email', None)
|
|
|
|
interactive = options.get('interactive')
|
2010-10-09 11:34:08 +08:00
|
|
|
verbosity = int(options.get('verbosity', 1))
|
|
|
|
|
2008-06-08 13:31:16 +08:00
|
|
|
# Do quick and dirty validation if --noinput
|
|
|
|
if not interactive:
|
|
|
|
if not username or not email:
|
|
|
|
raise CommandError("You must use --username and --email with --noinput.")
|
|
|
|
if not RE_VALID_USERNAME.match(username):
|
|
|
|
raise CommandError("Invalid username. Use only letters, digits, and underscores")
|
|
|
|
try:
|
Removed oldforms, validators, and related code:
* Removed `Manipulator`, `AutomaticManipulator`, and related classes.
* Removed oldforms specific bits from model fields:
* Removed `validator_list` and `core` arguments from constructors.
* Removed the methods:
* `get_manipulator_field_names`
* `get_manipulator_field_objs`
* `get_manipulator_fields`
* `get_manipulator_new_data`
* `prepare_field_objs_and_params`
* `get_follow`
* Renamed `flatten_data` method to `value_to_string` for better alignment with its use by the serialization framework, which was the only remaining code using `flatten_data`.
* Removed oldforms methods from `django.db.models.Options` class: `get_followed_related_objects`, `get_data_holders`, `get_follow`, and `has_field_type`.
* Removed oldforms-admin specific options from `django.db.models.fields.related` classes: `num_in_admin`, `min_num_in_admin`, `max_num_in_admin`, `num_extra_on_change`, and `edit_inline`.
* Serialization framework
* `Serializer.get_string_value` now calls the model fields' renamed `value_to_string` methods.
* Removed a special-casing of `models.DateTimeField` in `core.serializers.base.Serializer.get_string_value` that's handled by `django.db.models.fields.DateTimeField.value_to_string`.
* Removed `django.core.validators`:
* Moved `ValidationError` exception to `django.core.exceptions`.
* For the couple places that were using validators, brought over the necessary code to maintain the same functionality.
* Introduced a SlugField form field for validation and to compliment the SlugField model field (refs #8040).
* Removed an oldforms-style model creation hack (refs #2160).
git-svn-id: http://code.djangoproject.com/svn/django/trunk@8616 bcc190cf-cafb-0310-a4f2-bffc1f526a37
2008-08-27 15:19:44 +08:00
|
|
|
is_valid_email(email)
|
|
|
|
except exceptions.ValidationError:
|
2008-06-08 13:31:16 +08:00
|
|
|
raise CommandError("Invalid email address.")
|
2008-06-09 02:18:01 +08:00
|
|
|
|
|
|
|
password = ''
|
2008-06-08 13:31:16 +08:00
|
|
|
|
|
|
|
# Try to determine the current system user's username to use as a default.
|
|
|
|
try:
|
|
|
|
import pwd
|
|
|
|
default_username = pwd.getpwuid(os.getuid())[0].replace(' ', '').lower()
|
2008-10-06 12:54:14 +08:00
|
|
|
except (ImportError, KeyError):
|
|
|
|
# KeyError will be raised by getpwuid() if there is no
|
|
|
|
# corresponding entry in the /etc/passwd file (a very restricted
|
|
|
|
# chroot environment, for example).
|
|
|
|
default_username = ''
|
2008-06-08 13:31:16 +08:00
|
|
|
|
|
|
|
# Determine whether the default username is taken, so we don't display
|
|
|
|
# it as an option.
|
|
|
|
if default_username:
|
|
|
|
try:
|
|
|
|
User.objects.get(username=default_username)
|
|
|
|
except User.DoesNotExist:
|
|
|
|
pass
|
|
|
|
else:
|
|
|
|
default_username = ''
|
|
|
|
|
|
|
|
# Prompt for username/email/password. Enclose this whole thing in a
|
|
|
|
# try/except to trap for a keyboard interrupt and exit gracefully.
|
|
|
|
if interactive:
|
|
|
|
try:
|
2010-10-09 11:34:08 +08:00
|
|
|
|
2008-06-08 13:31:16 +08:00
|
|
|
# Get a username
|
|
|
|
while 1:
|
|
|
|
if not username:
|
|
|
|
input_msg = 'Username'
|
|
|
|
if default_username:
|
|
|
|
input_msg += ' (Leave blank to use %r)' % default_username
|
|
|
|
username = raw_input(input_msg + ': ')
|
|
|
|
if default_username and username == '':
|
|
|
|
username = default_username
|
|
|
|
if not RE_VALID_USERNAME.match(username):
|
|
|
|
sys.stderr.write("Error: That username is invalid. Use only letters, digits and underscores.\n")
|
|
|
|
username = None
|
|
|
|
continue
|
|
|
|
try:
|
|
|
|
User.objects.get(username=username)
|
|
|
|
except User.DoesNotExist:
|
|
|
|
break
|
|
|
|
else:
|
|
|
|
sys.stderr.write("Error: That username is already taken.\n")
|
|
|
|
username = None
|
2010-10-09 11:34:08 +08:00
|
|
|
|
2008-06-08 13:31:16 +08:00
|
|
|
# Get an email
|
|
|
|
while 1:
|
|
|
|
if not email:
|
|
|
|
email = raw_input('E-mail address: ')
|
|
|
|
try:
|
Removed oldforms, validators, and related code:
* Removed `Manipulator`, `AutomaticManipulator`, and related classes.
* Removed oldforms specific bits from model fields:
* Removed `validator_list` and `core` arguments from constructors.
* Removed the methods:
* `get_manipulator_field_names`
* `get_manipulator_field_objs`
* `get_manipulator_fields`
* `get_manipulator_new_data`
* `prepare_field_objs_and_params`
* `get_follow`
* Renamed `flatten_data` method to `value_to_string` for better alignment with its use by the serialization framework, which was the only remaining code using `flatten_data`.
* Removed oldforms methods from `django.db.models.Options` class: `get_followed_related_objects`, `get_data_holders`, `get_follow`, and `has_field_type`.
* Removed oldforms-admin specific options from `django.db.models.fields.related` classes: `num_in_admin`, `min_num_in_admin`, `max_num_in_admin`, `num_extra_on_change`, and `edit_inline`.
* Serialization framework
* `Serializer.get_string_value` now calls the model fields' renamed `value_to_string` methods.
* Removed a special-casing of `models.DateTimeField` in `core.serializers.base.Serializer.get_string_value` that's handled by `django.db.models.fields.DateTimeField.value_to_string`.
* Removed `django.core.validators`:
* Moved `ValidationError` exception to `django.core.exceptions`.
* For the couple places that were using validators, brought over the necessary code to maintain the same functionality.
* Introduced a SlugField form field for validation and to compliment the SlugField model field (refs #8040).
* Removed an oldforms-style model creation hack (refs #2160).
git-svn-id: http://code.djangoproject.com/svn/django/trunk@8616 bcc190cf-cafb-0310-a4f2-bffc1f526a37
2008-08-27 15:19:44 +08:00
|
|
|
is_valid_email(email)
|
|
|
|
except exceptions.ValidationError:
|
2008-06-08 13:31:16 +08:00
|
|
|
sys.stderr.write("Error: That e-mail address is invalid.\n")
|
|
|
|
email = None
|
|
|
|
else:
|
|
|
|
break
|
2010-10-09 11:34:08 +08:00
|
|
|
|
2008-06-08 13:31:16 +08:00
|
|
|
# Get a password
|
|
|
|
while 1:
|
|
|
|
if not password:
|
|
|
|
password = getpass.getpass()
|
|
|
|
password2 = getpass.getpass('Password (again): ')
|
|
|
|
if password != password2:
|
|
|
|
sys.stderr.write("Error: Your passwords didn't match.\n")
|
|
|
|
password = None
|
|
|
|
continue
|
|
|
|
if password.strip() == '':
|
|
|
|
sys.stderr.write("Error: Blank passwords aren't allowed.\n")
|
|
|
|
password = None
|
|
|
|
continue
|
|
|
|
break
|
|
|
|
except KeyboardInterrupt:
|
|
|
|
sys.stderr.write("\nOperation cancelled.\n")
|
|
|
|
sys.exit(1)
|
2010-10-09 11:34:08 +08:00
|
|
|
|
2008-06-08 13:31:16 +08:00
|
|
|
User.objects.create_superuser(username, email, password)
|
2010-10-09 11:34:08 +08:00
|
|
|
if verbosity >= 1:
|
|
|
|
self.stdout.write("Superuser created successfully.\n")
|
|
|
|
|