mirror of
				https://github.com/zulip/zulip.git
				synced 2025-11-04 05:53:43 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			123 lines
		
	
	
		
			4.7 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
			
		
		
	
	
			123 lines
		
	
	
		
			4.7 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
import zoneinfo
 | 
						|
from typing import Any
 | 
						|
 | 
						|
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_rollback_unsafe
 | 
						|
from zerver.lib.send_email import EMAIL_DATE_FORMAT, FromAddress
 | 
						|
from zerver.lib.timestamp import format_datetime_to_string
 | 
						|
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)))
 | 
						|
        context["login_time"] = format_datetime_to_string(local_time, user.twenty_four_hour_time)
 | 
						|
        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,
 | 
						|
            "date": local_time.strftime(EMAIL_DATE_FORMAT),
 | 
						|
        }
 | 
						|
        queue_json_publish_rollback_unsafe("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)
 |