mirror of
				https://github.com/zulip/zulip.git
				synced 2025-10-31 12:03:46 +00:00 
			
		
		
		
	With an extensive docstring by tabbott documenting its limitations. Co-authored-by: Tim Abbott <tabbott@zulip.com>
		
			
				
	
	
		
			242 lines
		
	
	
		
			8.7 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
			
		
		
	
	
			242 lines
		
	
	
		
			8.7 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
| from datetime import datetime
 | |
| from email.headerregistry import Address
 | |
| from typing import Optional, Union
 | |
| 
 | |
| import orjson
 | |
| from django.contrib.auth.models import UserManager
 | |
| from django.utils.timezone import now as timezone_now
 | |
| 
 | |
| from zerver.lib.hotspots import copy_hotspots
 | |
| from zerver.lib.timezone import canonicalize_timezone
 | |
| from zerver.lib.upload import copy_avatar
 | |
| from zerver.lib.utils import generate_api_key
 | |
| from zerver.models import (
 | |
|     Realm,
 | |
|     RealmUserDefault,
 | |
|     Recipient,
 | |
|     Stream,
 | |
|     Subscription,
 | |
|     UserBaseSettings,
 | |
|     UserProfile,
 | |
|     get_fake_email_domain,
 | |
| )
 | |
| 
 | |
| 
 | |
| def copy_default_settings(
 | |
|     settings_source: Union[UserProfile, RealmUserDefault], target_profile: UserProfile
 | |
| ) -> None:
 | |
|     # Important note: Code run from here to configure the user's
 | |
|     # settings should not call send_event, as that would cause clients
 | |
|     # to throw an exception (we haven't sent the realm_user/add event
 | |
|     # yet, so that event will include the updated details of target_profile).
 | |
|     #
 | |
|     # Note that this function will do at least one save() on target_profile.
 | |
|     for settings_name in UserBaseSettings.property_types:
 | |
|         if settings_name in ["default_language", "enable_login_emails"] and isinstance(
 | |
|             settings_source, RealmUserDefault
 | |
|         ):
 | |
|             continue
 | |
| 
 | |
|         if settings_name == "email_address_visibility":
 | |
|             # For email_address_visibility, the value selected in registration form
 | |
|             # is preferred over the realm-level default value and value of source
 | |
|             # profile.
 | |
|             continue
 | |
|         value = getattr(settings_source, settings_name)
 | |
|         setattr(target_profile, settings_name, value)
 | |
| 
 | |
|     if isinstance(settings_source, RealmUserDefault):
 | |
|         target_profile.save()
 | |
|         return
 | |
| 
 | |
|     target_profile.full_name = settings_source.full_name
 | |
|     target_profile.timezone = canonicalize_timezone(settings_source.timezone)
 | |
|     target_profile.save()
 | |
| 
 | |
|     if settings_source.avatar_source == UserProfile.AVATAR_FROM_USER:
 | |
|         from zerver.actions.user_settings import do_change_avatar_fields
 | |
| 
 | |
|         do_change_avatar_fields(
 | |
|             target_profile,
 | |
|             UserProfile.AVATAR_FROM_USER,
 | |
|             skip_notify=True,
 | |
|             acting_user=target_profile,
 | |
|         )
 | |
|         copy_avatar(settings_source, target_profile)
 | |
| 
 | |
|     copy_hotspots(settings_source, target_profile)
 | |
| 
 | |
| 
 | |
| def get_display_email_address(user_profile: UserProfile) -> str:
 | |
|     if not user_profile.email_address_is_realm_public():
 | |
|         return Address(
 | |
|             username=f"user{user_profile.id}", domain=get_fake_email_domain(user_profile.realm)
 | |
|         ).addr_spec
 | |
|     return user_profile.delivery_email
 | |
| 
 | |
| 
 | |
| # create_user_profile is based on Django's User.objects.create_user,
 | |
| # except that we don't save to the database so it can used in
 | |
| # bulk_creates
 | |
| #
 | |
| # Only use this for bulk_create -- for normal usage one should use
 | |
| # create_user (below) which will also make the Subscription and
 | |
| # Recipient objects
 | |
| def create_user_profile(
 | |
|     realm: Realm,
 | |
|     email: str,
 | |
|     password: Optional[str],
 | |
|     active: bool,
 | |
|     bot_type: Optional[int],
 | |
|     full_name: str,
 | |
|     bot_owner: Optional[UserProfile],
 | |
|     is_mirror_dummy: bool,
 | |
|     tos_version: Optional[str],
 | |
|     timezone: str,
 | |
|     default_language: str = "en",
 | |
|     tutorial_status: str = UserProfile.TUTORIAL_WAITING,
 | |
|     force_id: Optional[int] = None,
 | |
|     force_date_joined: Optional[datetime] = None,
 | |
|     *,
 | |
|     email_address_visibility: int,
 | |
| ) -> UserProfile:
 | |
|     if force_date_joined is None:
 | |
|         date_joined = timezone_now()
 | |
|     else:
 | |
|         date_joined = force_date_joined
 | |
| 
 | |
|     email = UserManager.normalize_email(email)
 | |
| 
 | |
|     extra_kwargs = {}
 | |
|     if force_id is not None:
 | |
|         extra_kwargs["id"] = force_id
 | |
| 
 | |
|     user_profile = UserProfile(
 | |
|         is_staff=False,
 | |
|         is_active=active,
 | |
|         full_name=full_name,
 | |
|         last_login=date_joined,
 | |
|         date_joined=date_joined,
 | |
|         realm=realm,
 | |
|         is_bot=bool(bot_type),
 | |
|         bot_type=bot_type,
 | |
|         bot_owner=bot_owner,
 | |
|         is_mirror_dummy=is_mirror_dummy,
 | |
|         tos_version=tos_version,
 | |
|         timezone=timezone,
 | |
|         tutorial_status=tutorial_status,
 | |
|         onboarding_steps=orjson.dumps([]).decode(),
 | |
|         default_language=default_language,
 | |
|         delivery_email=email,
 | |
|         email_address_visibility=email_address_visibility,
 | |
|         **extra_kwargs,
 | |
|     )
 | |
|     if bot_type or not active:
 | |
|         password = None
 | |
|     if user_profile.email_address_is_realm_public():
 | |
|         # If emails are visible to everyone, we can set this here and save a DB query
 | |
|         user_profile.email = get_display_email_address(user_profile)
 | |
|     user_profile.set_password(password)
 | |
|     user_profile.api_key = generate_api_key()
 | |
|     return user_profile
 | |
| 
 | |
| 
 | |
| def create_user(
 | |
|     email: str,
 | |
|     password: Optional[str],
 | |
|     realm: Realm,
 | |
|     full_name: str,
 | |
|     active: bool = True,
 | |
|     role: Optional[int] = None,
 | |
|     bot_type: Optional[int] = None,
 | |
|     bot_owner: Optional[UserProfile] = None,
 | |
|     tos_version: Optional[str] = None,
 | |
|     timezone: str = "",
 | |
|     avatar_source: str = UserProfile.AVATAR_FROM_GRAVATAR,
 | |
|     is_mirror_dummy: bool = False,
 | |
|     default_language: str = "en",
 | |
|     default_sending_stream: Optional[Stream] = None,
 | |
|     default_events_register_stream: Optional[Stream] = None,
 | |
|     default_all_public_streams: Optional[bool] = None,
 | |
|     source_profile: Optional[UserProfile] = None,
 | |
|     force_id: Optional[int] = None,
 | |
|     force_date_joined: Optional[datetime] = None,
 | |
|     create_personal_recipient: bool = True,
 | |
|     enable_marketing_emails: Optional[bool] = None,
 | |
|     email_address_visibility: Optional[int] = None,
 | |
| ) -> UserProfile:
 | |
|     realm_user_default = RealmUserDefault.objects.get(realm=realm)
 | |
|     if bot_type is None:
 | |
|         if email_address_visibility is not None:
 | |
|             user_email_address_visibility = email_address_visibility
 | |
|         else:
 | |
|             user_email_address_visibility = realm_user_default.email_address_visibility
 | |
|     else:
 | |
|         # There is no privacy motivation for limiting access to bot email addresses,
 | |
|         # so we hardcode them to EMAIL_ADDRESS_VISIBILITY_EVERYONE.
 | |
|         user_email_address_visibility = UserProfile.EMAIL_ADDRESS_VISIBILITY_EVERYONE
 | |
| 
 | |
|     user_profile = create_user_profile(
 | |
|         realm,
 | |
|         email,
 | |
|         password,
 | |
|         active,
 | |
|         bot_type,
 | |
|         full_name,
 | |
|         bot_owner,
 | |
|         is_mirror_dummy,
 | |
|         tos_version,
 | |
|         timezone,
 | |
|         default_language,
 | |
|         force_id=force_id,
 | |
|         force_date_joined=force_date_joined,
 | |
|         email_address_visibility=user_email_address_visibility,
 | |
|     )
 | |
|     user_profile.avatar_source = avatar_source
 | |
|     user_profile.timezone = timezone
 | |
|     user_profile.default_sending_stream = default_sending_stream
 | |
|     user_profile.default_events_register_stream = default_events_register_stream
 | |
|     if role is not None:
 | |
|         user_profile.role = role
 | |
|     # Allow the ORM default to be used if not provided
 | |
|     if default_all_public_streams is not None:
 | |
|         user_profile.default_all_public_streams = default_all_public_streams
 | |
|     # If a source profile was specified, we copy settings from that
 | |
|     # user.  Note that this is positioned in a way that overrides
 | |
|     # other arguments passed in, which is correct for most defaults
 | |
|     # like time zone where the source profile likely has a better value
 | |
|     # than the guess. As we decide on details like avatars and full
 | |
|     # names for this feature, we may want to move it.
 | |
|     if source_profile is not None:
 | |
|         # copy_default_settings saves the attribute values so a secondary
 | |
|         # save is not required.
 | |
|         copy_default_settings(source_profile, user_profile)
 | |
|     elif bot_type is None:
 | |
|         copy_default_settings(realm_user_default, user_profile)
 | |
|     else:
 | |
|         # This will be executed only for bots.
 | |
|         user_profile.save()
 | |
| 
 | |
|     if bot_type is None and enable_marketing_emails is not None:
 | |
|         user_profile.enable_marketing_emails = enable_marketing_emails
 | |
|         user_profile.save(update_fields=["enable_marketing_emails"])
 | |
| 
 | |
|     if not user_profile.email_address_is_realm_public():
 | |
|         # With restricted access to email addresses, we can't generate
 | |
|         # the fake email addresses we use for display purposes without
 | |
|         # a User ID, which isn't generated until the .save() above.
 | |
|         user_profile.email = get_display_email_address(user_profile)
 | |
|         user_profile.save(update_fields=["email"])
 | |
| 
 | |
|     if not create_personal_recipient:
 | |
|         return user_profile
 | |
| 
 | |
|     recipient = Recipient.objects.create(type_id=user_profile.id, type=Recipient.PERSONAL)
 | |
|     user_profile.recipient = recipient
 | |
|     user_profile.save(update_fields=["recipient"])
 | |
| 
 | |
|     Subscription.objects.create(
 | |
|         user_profile=user_profile, recipient=recipient, is_user_active=user_profile.is_active
 | |
|     )
 | |
|     return user_profile
 |