from typing import Any import zoneinfo from django.conf import settings from django.contrib.auth.signals import user_logged_in, user_logged_out from django.dispatch import receiver from django.utils.timezone import get_current_timezone_name as timezone_get_current_timezone_name from django.utils.timezone import now as timezone_now from django.utils.translation import gettext as _ from confirmation.models import one_click_unsubscribe_link from zerver.lib.queue import queue_json_publish from zerver.lib.send_email import FromAddress from zerver.lib.timezone import canonicalize_timezone from zerver.models import UserProfile JUST_CREATED_THRESHOLD = 60 def get_device_browser(user_agent: str) -> str | None: user_agent = user_agent.lower() if "zulip" in user_agent: return "Zulip" elif "edge" in user_agent: return "Edge" elif "opera" in user_agent or "opr/" in user_agent: return "Opera" elif ("chrome" in user_agent or "crios" in user_agent) and "chromium" not in user_agent: return "Chrome" elif "firefox" in user_agent and "seamonkey" not in user_agent and "chrome" not in user_agent: return "Firefox" elif "chromium" in user_agent: return "Chromium" elif "safari" in user_agent and "chrome" not in user_agent and "chromium" not in user_agent: return "Safari" elif "msie" in user_agent or "trident" in user_agent: return "Internet Explorer" else: return None def get_device_os(user_agent: str) -> str | None: user_agent = user_agent.lower() if "windows" in user_agent: return "Windows" elif "macintosh" in user_agent: return "macOS" elif "linux" in user_agent and "android" not in user_agent: return "Linux" elif "android" in user_agent: return "Android" elif "ios" in user_agent: return "iOS" elif "like mac os x" in user_agent: return "iOS" elif " cros " in user_agent: return "ChromeOS" else: return None @receiver(user_logged_in, dispatch_uid="only_on_login") def email_on_new_login(sender: Any, user: UserProfile, request: Any, **kwargs: Any) -> None: if not user.enable_login_emails: return if user.delivery_email == "": # Do not attempt to send new login emails for users without an email address. # The assertions here are to help document the only circumstance under which # this condition should be possible. assert ( user.realm.demo_organization_scheduled_deletion_date is not None and user.is_realm_owner ) return # We import here to minimize the dependencies of this module, # since it runs as part of `manage.py` initialization from zerver.context_processors import common_context if not settings.SEND_LOGIN_EMAILS: return if request: # If the user's account was just created, avoid sending an email. if (timezone_now() - user.date_joined).total_seconds() <= JUST_CREATED_THRESHOLD: return user_agent = request.headers.get("User-Agent", "").lower() context = common_context(user) context["user_email"] = user.delivery_email user_tz = user.timezone if user_tz == "": user_tz = timezone_get_current_timezone_name() local_time = timezone_now().astimezone(zoneinfo.ZoneInfo(canonicalize_timezone(user_tz))) if user.twenty_four_hour_time: hhmm_string = local_time.strftime("%H:%M") else: hhmm_string = local_time.strftime("%I:%M %p") context["login_time"] = local_time.strftime(f"%A, %B %d, %Y at {hhmm_string} %Z") context["device_ip"] = request.META.get("REMOTE_ADDR") or _("Unknown IP address") context["device_os"] = get_device_os(user_agent) or _("an unknown operating system") context["device_browser"] = get_device_browser(user_agent) or _("An unknown browser") context["unsubscribe_link"] = one_click_unsubscribe_link(user, "login") email_dict = { "template_prefix": "zerver/emails/notify_new_login", "to_user_ids": [user.id], "from_name": FromAddress.security_email_from_name(user_profile=user), "from_address": FromAddress.NOREPLY, "context": context, } queue_json_publish("email_senders", email_dict) @receiver(user_logged_out) def clear_zoom_token_on_logout( sender: object, *, user: UserProfile | None, **kwargs: object ) -> None: # Loaded lazily so django.setup() succeeds before static asset generation from zerver.actions.video_calls import do_set_zoom_token if user is not None and user.zoom_token is not None: do_set_zoom_token(user, None)