2019-02-13 10:22:16 +01:00
|
|
|
from typing import Any, Dict, List, Tuple
|
2017-10-29 15:52:01 +01:00
|
|
|
|
2017-10-29 15:40:07 +01:00
|
|
|
from django.db.models.query import QuerySet
|
|
|
|
from zerver.models import (
|
|
|
|
Recipient,
|
2017-10-29 20:19:57 +01:00
|
|
|
Stream,
|
2017-10-29 15:40:07 +01:00
|
|
|
Subscription,
|
2017-10-29 17:11:11 +01:00
|
|
|
UserProfile,
|
2017-10-29 15:40:07 +01:00
|
|
|
)
|
|
|
|
|
2018-03-15 00:32:42 +01:00
|
|
|
def get_active_subscriptions_for_stream_id(stream_id: int) -> QuerySet:
|
|
|
|
# TODO: Change return type to QuerySet[Subscription]
|
2017-10-29 15:40:07 +01:00
|
|
|
return Subscription.objects.filter(
|
|
|
|
recipient__type=Recipient.STREAM,
|
|
|
|
recipient__type_id=stream_id,
|
|
|
|
active=True,
|
|
|
|
)
|
|
|
|
|
2018-03-15 00:32:42 +01:00
|
|
|
def get_active_subscriptions_for_stream_ids(stream_ids: List[int]) -> QuerySet:
|
|
|
|
# TODO: Change return type to QuerySet[Subscription]
|
2017-10-29 15:52:01 +01:00
|
|
|
return Subscription.objects.filter(
|
|
|
|
recipient__type=Recipient.STREAM,
|
|
|
|
recipient__type_id__in=stream_ids,
|
|
|
|
active=True
|
|
|
|
)
|
|
|
|
|
perf: Extract get_subscribed_stream_ids_for_user.
This new method prevents us from getting fat
objects from the database.
Instead, now we just get ids from the database
to build our subqueries.
Note that we could also technically eliminate
the `set(...)` wrappers in this code to have
Django make a subquery and save a round trip.
I am postponing that for another commit (since
it's still somewhat coupled to some other
complexity in `do_get_streams` that I am trying
to cut through, plus it's not the main point
of this commit.)
BEFORE:
# old, still in use for other codepaths
def get_stream_subscriptions_for_user(user_profile: UserProfile) -> QuerySet:
# TODO: Change return type to QuerySet[Subscription]
return Subscription.objects.filter(
user_profile=user_profile,
recipient__type=Recipient.STREAM,
)
user_subs = get_stream_subscriptions_for_user(user_profile).filter(
active=True,
).select_related('recipient')
recipient_check = Q(id__in=[sub.recipient.type_id for sub in user_subs])
AFTER:
# newly added
def get_subscribed_stream_ids_for_user(user_profile: UserProfile) -> QuerySet:
return Subscription.objects.filter(
user_profile_id=user_profile,
recipient__type=Recipient.STREAM,
active=True,
).values_list('recipient__type_id', flat=True)
subscribed_stream_ids = get_subscribed_stream_ids_for_user(user_profile)
recipient_check = Q(id__in=set(subscribed_stream_ids))
2020-02-29 18:41:41 +01:00
|
|
|
def get_subscribed_stream_ids_for_user(user_profile: UserProfile) -> QuerySet:
|
|
|
|
return Subscription.objects.filter(
|
|
|
|
user_profile_id=user_profile,
|
|
|
|
recipient__type=Recipient.STREAM,
|
|
|
|
active=True,
|
|
|
|
).values_list('recipient__type_id', flat=True)
|
|
|
|
|
2018-03-15 00:32:42 +01:00
|
|
|
def get_stream_subscriptions_for_user(user_profile: UserProfile) -> QuerySet:
|
|
|
|
# TODO: Change return type to QuerySet[Subscription]
|
2017-10-29 17:11:11 +01:00
|
|
|
return Subscription.objects.filter(
|
|
|
|
user_profile=user_profile,
|
|
|
|
recipient__type=Recipient.STREAM,
|
|
|
|
)
|
|
|
|
|
2018-03-15 00:32:42 +01:00
|
|
|
def get_stream_subscriptions_for_users(user_profiles: List[UserProfile]) -> QuerySet:
|
|
|
|
# TODO: Change return type to QuerySet[Subscription]
|
2017-10-29 19:15:35 +01:00
|
|
|
return Subscription.objects.filter(
|
|
|
|
user_profile__in=user_profiles,
|
|
|
|
recipient__type=Recipient.STREAM,
|
|
|
|
)
|
|
|
|
|
2017-11-05 11:15:10 +01:00
|
|
|
def get_bulk_stream_subscriber_info(
|
|
|
|
user_profiles: List[UserProfile],
|
|
|
|
stream_dict: Dict[int, Stream]) -> Dict[int, List[Tuple[Subscription, Stream]]]:
|
2017-10-29 20:19:57 +01:00
|
|
|
|
|
|
|
stream_ids = stream_dict.keys()
|
|
|
|
|
|
|
|
result = {
|
|
|
|
user_profile.id: []
|
|
|
|
for user_profile in user_profiles
|
|
|
|
} # type: Dict[int, List[Tuple[Subscription, Stream]]]
|
|
|
|
|
|
|
|
subs = Subscription.objects.filter(
|
|
|
|
user_profile__in=user_profiles,
|
|
|
|
recipient__type=Recipient.STREAM,
|
|
|
|
recipient__type_id__in=stream_ids,
|
|
|
|
active=True,
|
|
|
|
).select_related('user_profile', 'recipient')
|
|
|
|
|
|
|
|
for sub in subs:
|
|
|
|
user_profile_id = sub.user_profile_id
|
|
|
|
stream_id = sub.recipient.type_id
|
|
|
|
stream = stream_dict[stream_id]
|
|
|
|
result[user_profile_id].append((sub, stream))
|
|
|
|
|
|
|
|
return result
|
|
|
|
|
2017-11-05 11:15:10 +01:00
|
|
|
def num_subscribers_for_stream_id(stream_id: int) -> int:
|
2017-10-29 15:40:07 +01:00
|
|
|
return get_active_subscriptions_for_stream_id(stream_id).filter(
|
|
|
|
user_profile__is_active=True,
|
|
|
|
).count()
|
2019-02-13 10:22:16 +01:00
|
|
|
|
|
|
|
|
|
|
|
def handle_stream_notifications_compatibility(user_profile: UserProfile,
|
|
|
|
stream_dict: Dict[str, Any],
|
|
|
|
notification_settings_null: bool) -> None:
|
|
|
|
# Old versions of the mobile apps don't support `None` as a
|
|
|
|
# value for the stream-level notifications properties, so we
|
|
|
|
# have to handle the normally frontend-side defaults for these
|
|
|
|
# settings here for those older clients.
|
|
|
|
#
|
|
|
|
# Note that this situation results in these older mobile apps
|
|
|
|
# having a subtle bug where changes to the user-level stream
|
|
|
|
# notification defaults will not properly propagate to the
|
|
|
|
# mobile app "stream notification settings" UI until the app
|
|
|
|
# re-registers. This is an acceptable level of
|
|
|
|
# backwards-compatibility problem in our view.
|
|
|
|
assert not notification_settings_null
|
|
|
|
|
|
|
|
for notification_type in ["desktop_notifications", "audible_notifications",
|
|
|
|
"push_notifications", "email_notifications"]:
|
|
|
|
# Values of true/false are supported by older clients.
|
|
|
|
if stream_dict[notification_type] is not None:
|
|
|
|
continue
|
|
|
|
target_attr = "enable_stream_" + notification_type
|
|
|
|
stream_dict[notification_type] = getattr(user_profile, target_attr)
|