mirror of
				https://github.com/zulip/zulip.git
				synced 2025-11-03 21:43:21 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			141 lines
		
	
	
		
			5.2 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
			
		
		
	
	
			141 lines
		
	
	
		
			5.2 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
import logging
 | 
						|
from typing import TypedDict
 | 
						|
 | 
						|
from django.conf import settings
 | 
						|
 | 
						|
from zerver.lib.exceptions import (
 | 
						|
    InvalidBouncerPublicKeyError,
 | 
						|
    InvalidEncryptedPushRegistrationError,
 | 
						|
    JsonableError,
 | 
						|
    MissingRemoteRealmError,
 | 
						|
    RequestExpiredError,
 | 
						|
)
 | 
						|
from zerver.lib.push_notifications import PushNotificationsDisallowedByBouncerError
 | 
						|
from zerver.lib.remote_server import (
 | 
						|
    PushNotificationBouncerError,
 | 
						|
    PushNotificationBouncerRetryLaterError,
 | 
						|
    PushNotificationBouncerServerError,
 | 
						|
    send_to_push_bouncer,
 | 
						|
)
 | 
						|
from zerver.models import PushDevice
 | 
						|
from zerver.models.users import UserProfile, get_user_profile_by_id
 | 
						|
from zerver.tornado.django_api import send_event_on_commit
 | 
						|
 | 
						|
if settings.ZILENCER_ENABLED:
 | 
						|
    from zilencer.views import do_register_remote_push_device
 | 
						|
 | 
						|
logger = logging.getLogger(__name__)
 | 
						|
 | 
						|
 | 
						|
class RegisterPushDeviceToBouncerQueueItem(TypedDict):
 | 
						|
    user_profile_id: int
 | 
						|
    bouncer_public_key: str
 | 
						|
    encrypted_push_registration: str
 | 
						|
    push_account_id: int
 | 
						|
 | 
						|
 | 
						|
def handle_registration_to_bouncer_failure(
 | 
						|
    user_profile: UserProfile, push_account_id: int, error_code: str
 | 
						|
) -> None:
 | 
						|
    """Handles a failed registration request to the bouncer by
 | 
						|
    notifying or preparing to notify clients.
 | 
						|
 | 
						|
    * Sends a `push_device` event to notify online clients immediately.
 | 
						|
 | 
						|
    * Stores the `error_code` in the `PushDevice` table. This is later
 | 
						|
      used, along with other metadata, to notify offline clients the
 | 
						|
      next time they call `/register`. See the `push_devices` field in
 | 
						|
      the `/register` response.
 | 
						|
    """
 | 
						|
    PushDevice.objects.filter(user=user_profile, push_account_id=push_account_id).update(
 | 
						|
        error_code=error_code
 | 
						|
    )
 | 
						|
    event = dict(
 | 
						|
        type="push_device",
 | 
						|
        push_account_id=str(push_account_id),
 | 
						|
        status="failed",
 | 
						|
        error_code=error_code,
 | 
						|
    )
 | 
						|
    send_event_on_commit(user_profile.realm, event, [user_profile.id])
 | 
						|
 | 
						|
    # Report the `REQUEST_EXPIRED_ERROR` to the server admins as it indicates
 | 
						|
    # a long-lasting outage somewhere between the server and the bouncer,
 | 
						|
    # most likely in either the server or its local network configuration.
 | 
						|
    if error_code == PushDevice.ErrorCode.REQUEST_EXPIRED:
 | 
						|
        logging.error(
 | 
						|
            "Push device registration request for user_id=%s, push_account_id=%s expired.",
 | 
						|
            user_profile.id,
 | 
						|
            push_account_id,
 | 
						|
        )
 | 
						|
 | 
						|
 | 
						|
def handle_register_push_device_to_bouncer(
 | 
						|
    queue_item: RegisterPushDeviceToBouncerQueueItem,
 | 
						|
) -> None:
 | 
						|
    user_profile_id = queue_item["user_profile_id"]
 | 
						|
    user_profile = get_user_profile_by_id(user_profile_id)
 | 
						|
    bouncer_public_key = queue_item["bouncer_public_key"]
 | 
						|
    encrypted_push_registration = queue_item["encrypted_push_registration"]
 | 
						|
    push_account_id = queue_item["push_account_id"]
 | 
						|
 | 
						|
    try:
 | 
						|
        if settings.ZILENCER_ENABLED:
 | 
						|
            device_id = do_register_remote_push_device(
 | 
						|
                bouncer_public_key,
 | 
						|
                encrypted_push_registration,
 | 
						|
                push_account_id,
 | 
						|
                realm=user_profile.realm,
 | 
						|
            )
 | 
						|
        else:
 | 
						|
            post_data: dict[str, str | int] = {
 | 
						|
                "realm_uuid": str(user_profile.realm.uuid),
 | 
						|
                "push_account_id": push_account_id,
 | 
						|
                "encrypted_push_registration": encrypted_push_registration,
 | 
						|
                "bouncer_public_key": bouncer_public_key,
 | 
						|
            }
 | 
						|
            result = send_to_push_bouncer("POST", "push/e2ee/register", post_data)
 | 
						|
            assert isinstance(result["device_id"], int)  # for mypy
 | 
						|
            device_id = result["device_id"]
 | 
						|
    except (
 | 
						|
        PushNotificationBouncerRetryLaterError,
 | 
						|
        PushNotificationBouncerServerError,
 | 
						|
    ) as e:  # nocoverage
 | 
						|
        # Network error or 5xx error response from bouncer server.
 | 
						|
        # Keep retrying to register until `RequestExpiredError` is raised.
 | 
						|
        raise PushNotificationBouncerRetryLaterError(e.msg)
 | 
						|
    except (
 | 
						|
        # Need to resubmit realm info - `manage.py register_server`
 | 
						|
        MissingRemoteRealmError,
 | 
						|
        # Invalid credentials or unexpected status code
 | 
						|
        PushNotificationBouncerError,
 | 
						|
        # Plan doesn't allow sending push notifications
 | 
						|
        PushNotificationsDisallowedByBouncerError,
 | 
						|
    ):
 | 
						|
        # Server admins need to fix these set of errors, report them.
 | 
						|
        # Server should keep retrying to register until `RequestExpiredError` is raised.
 | 
						|
        error_msg = f"Push device registration request for user_id={user_profile.id}, push_account_id={push_account_id} failed."
 | 
						|
        logging.error(error_msg)
 | 
						|
        raise PushNotificationBouncerRetryLaterError(error_msg)
 | 
						|
    except (
 | 
						|
        InvalidBouncerPublicKeyError,
 | 
						|
        InvalidEncryptedPushRegistrationError,
 | 
						|
        RequestExpiredError,
 | 
						|
        # Any future or unexpected exceptions that we add.
 | 
						|
        JsonableError,
 | 
						|
    ) as e:
 | 
						|
        handle_registration_to_bouncer_failure(
 | 
						|
            user_profile, push_account_id, error_code=e.__class__.code.name
 | 
						|
        )
 | 
						|
        return
 | 
						|
 | 
						|
    # Registration successful.
 | 
						|
    PushDevice.objects.filter(user=user_profile, push_account_id=push_account_id).update(
 | 
						|
        bouncer_device_id=device_id
 | 
						|
    )
 | 
						|
    event = dict(
 | 
						|
        type="push_device",
 | 
						|
        push_account_id=str(push_account_id),
 | 
						|
        status="active",
 | 
						|
    )
 | 
						|
    send_event_on_commit(user_profile.realm, event, [user_profile.id])
 |