mirror of
https://github.com/zulip/zulip.git
synced 2025-11-07 23:43:43 +00:00
The presence and user status update events are only sent to accessible users, i.e. guests do not receive presence and user status updates for users they cannot access.
225 lines
8.0 KiB
Python
225 lines
8.0 KiB
Python
import datetime
|
|
import time
|
|
from collections import defaultdict
|
|
from typing import Any, Dict, Mapping, Optional, Sequence, Set
|
|
|
|
from django.conf import settings
|
|
from django.utils.timezone import now as timezone_now
|
|
|
|
from zerver.lib.timestamp import datetime_to_timestamp
|
|
from zerver.lib.users import check_user_can_access_all_users, get_accessible_user_ids
|
|
from zerver.models import PushDeviceToken, Realm, UserPresence, UserProfile, query_for_ids
|
|
|
|
|
|
def get_presence_dicts_for_rows(
|
|
all_rows: Sequence[Mapping[str, Any]], mobile_user_ids: Set[int], slim_presence: bool
|
|
) -> Dict[str, Dict[str, Any]]:
|
|
if slim_presence:
|
|
# Stringify user_id here, since it's gonna be turned
|
|
# into a string anyway by JSON, and it keeps mypy happy.
|
|
get_user_key = lambda row: str(row["user_profile_id"])
|
|
get_user_presence_info = get_modern_user_presence_info
|
|
else:
|
|
get_user_key = lambda row: row["user_profile__email"]
|
|
get_user_presence_info = get_legacy_user_presence_info
|
|
|
|
user_statuses: Dict[str, Dict[str, Any]] = {}
|
|
|
|
for presence_row in all_rows:
|
|
user_key = get_user_key(presence_row)
|
|
|
|
last_active_time = user_presence_datetime_with_date_joined_default(
|
|
presence_row["last_active_time"], presence_row["user_profile__date_joined"]
|
|
)
|
|
last_connected_time = user_presence_datetime_with_date_joined_default(
|
|
presence_row["last_connected_time"], presence_row["user_profile__date_joined"]
|
|
)
|
|
|
|
info = get_user_presence_info(
|
|
last_active_time,
|
|
last_connected_time,
|
|
)
|
|
user_statuses[user_key] = info
|
|
|
|
return user_statuses
|
|
|
|
|
|
def user_presence_datetime_with_date_joined_default(
|
|
dt: Optional[datetime.datetime], date_joined: datetime.datetime
|
|
) -> datetime.datetime:
|
|
"""
|
|
Our data models support UserPresence objects not having None
|
|
values for last_active_time/last_connected_time. The legacy API
|
|
however has always sent timestamps, so for backward
|
|
compatibility we cannot send such values through the API and need
|
|
to default to a sane datetime.
|
|
|
|
This helper functions expects to take a last_active_time or
|
|
last_connected_time value and the date_joined of the user, which
|
|
will serve as the default value if the first argument is None.
|
|
"""
|
|
if dt is None:
|
|
return date_joined
|
|
|
|
return dt
|
|
|
|
|
|
def get_modern_user_presence_info(
|
|
last_active_time: datetime.datetime, last_connected_time: datetime.datetime
|
|
) -> Dict[str, Any]:
|
|
# TODO: Do further bandwidth optimizations to this structure.
|
|
result = {}
|
|
result["active_timestamp"] = datetime_to_timestamp(last_active_time)
|
|
result["idle_timestamp"] = datetime_to_timestamp(last_connected_time)
|
|
return result
|
|
|
|
|
|
def get_legacy_user_presence_info(
|
|
last_active_time: datetime.datetime, last_connected_time: datetime.datetime
|
|
) -> Dict[str, Any]:
|
|
"""
|
|
Reformats the modern UserPresence data structure so that legacy
|
|
API clients can still access presence data.
|
|
We expect this code to remain mostly unchanged until we can delete it.
|
|
"""
|
|
|
|
# Now we put things together in the legacy presence format with
|
|
# one client + an `aggregated` field.
|
|
#
|
|
# TODO: Look at whether we can drop to just the "aggregated" field
|
|
# if no clients look at the rest.
|
|
most_recent_info = format_legacy_presence_dict(last_active_time, last_connected_time)
|
|
|
|
result = {}
|
|
|
|
# The word "aggregated" here is possibly misleading.
|
|
# It's really just the most recent client's info.
|
|
result["aggregated"] = dict(
|
|
client=most_recent_info["client"],
|
|
status=most_recent_info["status"],
|
|
timestamp=most_recent_info["timestamp"],
|
|
)
|
|
|
|
result["website"] = most_recent_info
|
|
|
|
return result
|
|
|
|
|
|
def format_legacy_presence_dict(
|
|
last_active_time: datetime.datetime, last_connected_time: datetime.datetime
|
|
) -> Dict[str, Any]:
|
|
"""
|
|
This function assumes it's being called right after the presence object was updated,
|
|
and is not meant to be used on old presence data.
|
|
"""
|
|
if (
|
|
last_active_time
|
|
+ datetime.timedelta(seconds=settings.PRESENCE_LEGACY_EVENT_OFFSET_FOR_ACTIVITY_SECONDS)
|
|
>= last_connected_time
|
|
):
|
|
status = UserPresence.LEGACY_STATUS_ACTIVE
|
|
timestamp = datetime_to_timestamp(last_active_time)
|
|
else:
|
|
status = UserPresence.LEGACY_STATUS_IDLE
|
|
timestamp = datetime_to_timestamp(last_connected_time)
|
|
|
|
# This field was never used by clients of the legacy API, so we
|
|
# just set it to a fixed value for API format compatibility.
|
|
pushable = False
|
|
|
|
return dict(client="website", status=status, timestamp=timestamp, pushable=pushable)
|
|
|
|
|
|
def get_presence_for_user(
|
|
user_profile_id: int, slim_presence: bool = False
|
|
) -> Dict[str, Dict[str, Any]]:
|
|
query = UserPresence.objects.filter(user_profile_id=user_profile_id).values(
|
|
"last_active_time",
|
|
"last_connected_time",
|
|
"user_profile__email",
|
|
"user_profile_id",
|
|
"user_profile__enable_offline_push_notifications",
|
|
"user_profile__date_joined",
|
|
)
|
|
presence_rows = list(query)
|
|
|
|
mobile_user_ids: Set[int] = set()
|
|
if PushDeviceToken.objects.filter(user_id=user_profile_id).exists(): # nocoverage
|
|
# TODO: Add a test, though this is low priority, since we don't use mobile_user_ids yet.
|
|
mobile_user_ids.add(user_profile_id)
|
|
|
|
return get_presence_dicts_for_rows(presence_rows, mobile_user_ids, slim_presence)
|
|
|
|
|
|
def get_presence_dict_by_realm(
|
|
realm: Realm, slim_presence: bool = False, requesting_user_profile: Optional[UserProfile] = None
|
|
) -> Dict[str, Dict[str, Any]]:
|
|
two_weeks_ago = timezone_now() - datetime.timedelta(weeks=2)
|
|
query = UserPresence.objects.filter(
|
|
realm_id=realm.id,
|
|
last_connected_time__gte=two_weeks_ago,
|
|
user_profile__is_active=True,
|
|
user_profile__is_bot=False,
|
|
)
|
|
|
|
if settings.CAN_ACCESS_ALL_USERS_GROUP_LIMITS_PRESENCE and not check_user_can_access_all_users(
|
|
requesting_user_profile
|
|
):
|
|
assert requesting_user_profile is not None
|
|
accessible_user_ids = get_accessible_user_ids(realm, requesting_user_profile)
|
|
query = query.filter(user_profile_id__in=accessible_user_ids)
|
|
|
|
presence_rows = list(
|
|
query.values(
|
|
"last_active_time",
|
|
"last_connected_time",
|
|
"user_profile__email",
|
|
"user_profile_id",
|
|
"user_profile__enable_offline_push_notifications",
|
|
"user_profile__date_joined",
|
|
)
|
|
)
|
|
|
|
mobile_query = PushDeviceToken.objects.distinct("user_id").values_list(
|
|
"user_id",
|
|
flat=True,
|
|
)
|
|
|
|
user_profile_ids = [presence_row["user_profile_id"] for presence_row in presence_rows]
|
|
if len(user_profile_ids) == 0:
|
|
# This conditional is necessary because query_for_ids
|
|
# throws an exception if passed an empty list.
|
|
#
|
|
# It's not clear this condition is actually possible,
|
|
# though, because it shouldn't be possible to end up with
|
|
# a realm with 0 active users.
|
|
return {}
|
|
|
|
mobile_query_ids = query_for_ids(
|
|
query=mobile_query,
|
|
user_ids=user_profile_ids,
|
|
field="user_id",
|
|
)
|
|
mobile_user_ids = set(mobile_query_ids)
|
|
|
|
return get_presence_dicts_for_rows(presence_rows, mobile_user_ids, slim_presence)
|
|
|
|
|
|
def get_presences_for_realm(
|
|
realm: Realm, slim_presence: bool, requesting_user_profile: UserProfile
|
|
) -> Dict[str, Dict[str, Dict[str, Any]]]:
|
|
if realm.presence_disabled:
|
|
# Return an empty dict if presence is disabled in this realm
|
|
return defaultdict(dict)
|
|
|
|
return get_presence_dict_by_realm(realm, slim_presence, requesting_user_profile)
|
|
|
|
|
|
def get_presence_response(
|
|
requesting_user_profile: UserProfile, slim_presence: bool
|
|
) -> Dict[str, Any]:
|
|
realm = requesting_user_profile.realm
|
|
server_timestamp = time.time()
|
|
presences = get_presences_for_realm(realm, slim_presence, requesting_user_profile)
|
|
return dict(presences=presences, server_timestamp=server_timestamp)
|