2015-08-21 02:10:41 +02:00
|
|
|
import argparse
|
2017-11-16 00:43:27 +01:00
|
|
|
import sys
|
|
|
|
from typing import Any
|
2012-12-06 23:26:47 +01:00
|
|
|
|
2017-11-16 00:43:27 +01:00
|
|
|
from django.core import validators
|
2012-12-06 23:26:47 +01:00
|
|
|
from django.core.exceptions import ValidationError
|
2017-11-16 00:43:27 +01:00
|
|
|
from django.core.management.base import CommandError
|
2012-12-06 23:26:47 +01:00
|
|
|
from django.db.utils import IntegrityError
|
|
|
|
|
2018-12-17 07:22:52 +01:00
|
|
|
from zerver.lib.actions import do_create_user
|
2013-07-29 23:03:31 +02:00
|
|
|
from zerver.lib.initial_password import initial_password
|
2017-08-07 17:15:11 +02:00
|
|
|
from zerver.lib.management import ZulipBaseCommand
|
2017-11-16 00:43:27 +01:00
|
|
|
from zerver.models import email_to_username
|
2012-12-06 23:26:47 +01:00
|
|
|
|
2020-01-14 21:59:46 +01:00
|
|
|
|
2017-08-07 17:15:11 +02:00
|
|
|
class Command(ZulipBaseCommand):
|
2013-01-08 23:05:42 +01:00
|
|
|
help = """Create the specified user with a default initial password.
|
|
|
|
|
2018-12-17 07:23:35 +01:00
|
|
|
Set tos_version=None, so that the user needs to do a ToS flow on login.
|
2015-08-21 02:10:41 +02:00
|
|
|
|
|
|
|
Omit both <email> and <full name> for interactive user creation.
|
2013-01-08 23:05:42 +01:00
|
|
|
"""
|
|
|
|
|
2017-10-26 11:35:57 +02:00
|
|
|
def add_arguments(self, parser: argparse.ArgumentParser) -> None:
|
2015-08-21 02:10:41 +02:00
|
|
|
parser.add_argument('--this-user-has-accepted-the-tos',
|
|
|
|
dest='tos',
|
|
|
|
action="store_true",
|
|
|
|
default=False,
|
|
|
|
help='Acknowledgement that the user has already accepted the ToS.')
|
2017-07-08 07:25:17 +02:00
|
|
|
parser.add_argument('--password',
|
|
|
|
dest='password',
|
|
|
|
type=str,
|
|
|
|
default='',
|
2017-07-18 05:50:54 +02:00
|
|
|
help='password of new user. For development only.'
|
|
|
|
'Note that we recommend against setting '
|
2017-07-08 23:12:34 +02:00
|
|
|
'passwords this way, since they can be snooped by any user account '
|
2017-07-18 05:50:54 +02:00
|
|
|
'on the server via `ps -ef` or by any superuser with'
|
|
|
|
'read access to the user\'s bash history.')
|
|
|
|
parser.add_argument('--password-file',
|
|
|
|
dest='password_file',
|
|
|
|
type=str,
|
|
|
|
default='',
|
|
|
|
help='The file containing the password of the new user.')
|
2015-08-21 02:10:41 +02:00
|
|
|
parser.add_argument('email', metavar='<email>', type=str, nargs='?', default=argparse.SUPPRESS,
|
|
|
|
help='email address of new user')
|
2017-11-09 11:45:56 +01:00
|
|
|
parser.add_argument('full_name', metavar='<full name>', type=str, nargs='?',
|
|
|
|
default=argparse.SUPPRESS,
|
2015-08-21 02:10:41 +02:00
|
|
|
help='full name of new user')
|
2017-08-07 17:15:11 +02:00
|
|
|
self.add_realm_args(parser, True, "The name of the existing realm to which to add the user.")
|
2012-12-06 23:26:47 +01:00
|
|
|
|
2017-10-26 11:35:57 +02:00
|
|
|
def handle(self, *args: Any, **options: Any) -> None:
|
2013-01-08 23:05:42 +01:00
|
|
|
if not options["tos"]:
|
|
|
|
raise CommandError("""You must confirm that this user has accepted the
|
|
|
|
Terms of Service by passing --this-user-has-accepted-the-tos.""")
|
2017-08-07 17:15:11 +02:00
|
|
|
realm = self.get_realm(options)
|
2017-09-26 01:25:39 +02:00
|
|
|
assert realm is not None # Should be ensured by parser
|
|
|
|
|
2012-12-06 23:26:47 +01:00
|
|
|
try:
|
2015-08-21 02:10:41 +02:00
|
|
|
email = options['email']
|
|
|
|
full_name = options['full_name']
|
2012-12-06 23:26:47 +01:00
|
|
|
try:
|
|
|
|
validators.validate_email(email)
|
|
|
|
except ValidationError:
|
|
|
|
raise CommandError("Invalid email address.")
|
2015-08-21 02:10:41 +02:00
|
|
|
except KeyError:
|
|
|
|
if 'email' in options or 'full_name' in options:
|
2013-01-08 23:08:13 +01:00
|
|
|
raise CommandError("""Either specify an email and full name as two
|
|
|
|
parameters, or specify no parameters for interactive user creation.""")
|
2012-12-06 23:26:47 +01:00
|
|
|
else:
|
|
|
|
while True:
|
2015-11-01 17:14:42 +01:00
|
|
|
email = input("Email: ")
|
2012-12-06 23:26:47 +01:00
|
|
|
try:
|
|
|
|
validators.validate_email(email)
|
|
|
|
break
|
|
|
|
except ValidationError:
|
2015-11-01 17:11:06 +01:00
|
|
|
print("Invalid email address.", file=sys.stderr)
|
2015-11-01 17:14:42 +01:00
|
|
|
full_name = input("Full name: ")
|
2012-12-06 23:26:47 +01:00
|
|
|
|
|
|
|
try:
|
2019-06-20 13:57:56 +02:00
|
|
|
if options['password_file']:
|
2020-04-09 21:51:58 +02:00
|
|
|
with open(options['password_file']) as f:
|
2019-07-14 21:37:08 +02:00
|
|
|
pw = f.read()
|
2019-06-20 13:57:56 +02:00
|
|
|
elif options['password']:
|
|
|
|
pw = options['password']
|
2017-07-18 05:50:54 +02:00
|
|
|
else:
|
2018-03-24 23:40:54 +01:00
|
|
|
user_initial_password = initial_password(email)
|
2018-03-24 23:47:11 +01:00
|
|
|
if user_initial_password is None:
|
|
|
|
raise CommandError("Password is unusable.")
|
2019-06-20 14:15:10 +02:00
|
|
|
pw = user_initial_password
|
2020-06-29 13:12:17 +02:00
|
|
|
do_create_user(email, pw, realm, full_name, email_to_username(email), acting_user=None)
|
2012-12-06 23:26:47 +01:00
|
|
|
except IntegrityError:
|
|
|
|
raise CommandError("User already exists.")
|