2016-07-26 23:26:39 +02:00
|
|
|
|
|
|
|
import datetime
|
|
|
|
import time
|
2017-03-06 12:25:37 +01:00
|
|
|
|
|
|
|
from django.conf import settings
|
2018-04-24 03:47:28 +02:00
|
|
|
from typing import Any, Dict
|
2016-07-26 23:26:39 +02:00
|
|
|
|
|
|
|
from django.http import HttpRequest, HttpResponse
|
2017-04-16 21:32:57 +02:00
|
|
|
from django.utils.timezone import now as timezone_now
|
2016-07-26 23:26:39 +02:00
|
|
|
from django.utils.translation import ugettext as _
|
|
|
|
|
2017-11-04 00:59:22 +01:00
|
|
|
from zerver.decorator import human_users_only
|
2018-12-17 22:04:07 +01:00
|
|
|
from zerver.lib.actions import (
|
|
|
|
do_revoke_away_status,
|
|
|
|
do_set_away_status,
|
|
|
|
get_status_dict,
|
|
|
|
update_user_presence,
|
|
|
|
)
|
2016-07-26 23:26:39 +02:00
|
|
|
from zerver.lib.request import has_request_variables, REQ, JsonableError
|
|
|
|
from zerver.lib.response import json_success, json_error
|
2017-03-06 12:25:37 +01:00
|
|
|
from zerver.lib.timestamp import datetime_to_timestamp
|
2016-07-26 23:26:39 +02:00
|
|
|
from zerver.lib.validator import check_bool
|
2018-12-07 00:05:57 +01:00
|
|
|
from zerver.models import UserActivity, UserPresence, UserProfile, \
|
|
|
|
get_active_user_by_delivery_email
|
2016-07-26 23:26:39 +02:00
|
|
|
|
2017-10-27 02:18:49 +02:00
|
|
|
def get_status_list(requesting_user_profile: UserProfile) -> Dict[str, Any]:
|
2016-07-26 23:26:39 +02:00
|
|
|
return {'presences': get_status_dict(requesting_user_profile),
|
|
|
|
'server_timestamp': time.time()}
|
|
|
|
|
2017-10-27 02:18:49 +02:00
|
|
|
def get_presence_backend(request: HttpRequest, user_profile: UserProfile,
|
2018-04-24 03:47:28 +02:00
|
|
|
email: str) -> HttpResponse:
|
2017-02-11 08:38:16 +01:00
|
|
|
try:
|
2018-12-07 00:05:57 +01:00
|
|
|
target = get_active_user_by_delivery_email(email, user_profile.realm)
|
2017-02-11 08:38:16 +01:00
|
|
|
except UserProfile.DoesNotExist:
|
|
|
|
return json_error(_('No such user'))
|
|
|
|
if target.is_bot:
|
2017-04-12 20:50:23 +02:00
|
|
|
return json_error(_('Presence is not supported for bot users.'))
|
2017-02-11 08:38:16 +01:00
|
|
|
|
|
|
|
presence_dict = UserPresence.get_status_dict_by_user(target)
|
|
|
|
if len(presence_dict) == 0:
|
|
|
|
return json_error(_('No presence data for %s' % (target.email,)))
|
|
|
|
|
|
|
|
# For initial version, we just include the status and timestamp keys
|
|
|
|
result = dict(presence=presence_dict[target.email])
|
2017-03-06 12:25:37 +01:00
|
|
|
aggregated_info = result['presence']['aggregated']
|
2017-04-16 21:32:57 +02:00
|
|
|
aggr_status_duration = datetime_to_timestamp(timezone_now()) - aggregated_info['timestamp']
|
2017-03-06 12:25:37 +01:00
|
|
|
if aggr_status_duration > settings.OFFLINE_THRESHOLD_SECS:
|
|
|
|
aggregated_info['status'] = 'offline'
|
2017-02-11 08:38:16 +01:00
|
|
|
for val in result['presence'].values():
|
2017-03-02 09:52:17 +01:00
|
|
|
val.pop('client', None)
|
|
|
|
val.pop('pushable', None)
|
2017-02-11 08:38:16 +01:00
|
|
|
return json_success(result)
|
|
|
|
|
2018-12-17 22:04:07 +01:00
|
|
|
@human_users_only
|
|
|
|
@has_request_variables
|
|
|
|
def update_user_status_backend(request: HttpRequest,
|
|
|
|
user_profile: UserProfile,
|
|
|
|
away: bool=REQ(validator=check_bool),
|
|
|
|
) -> HttpResponse:
|
|
|
|
if away:
|
|
|
|
do_set_away_status(
|
|
|
|
user_profile=user_profile,
|
|
|
|
client_id=request.client.id,
|
|
|
|
)
|
|
|
|
else:
|
|
|
|
do_revoke_away_status(
|
|
|
|
user_profile=user_profile,
|
|
|
|
)
|
|
|
|
|
|
|
|
result = dict()
|
|
|
|
return json_success(result)
|
|
|
|
|
2017-04-15 20:51:51 +02:00
|
|
|
@human_users_only
|
2016-07-26 23:26:39 +02:00
|
|
|
@has_request_variables
|
2017-12-29 14:34:49 +01:00
|
|
|
def update_active_status_backend(request: HttpRequest, user_profile: UserProfile,
|
|
|
|
status: str=REQ(),
|
|
|
|
ping_only: bool=REQ(validator=check_bool, default=False),
|
|
|
|
new_user_input: bool=REQ(validator=check_bool, default=False)
|
|
|
|
) -> HttpResponse:
|
2016-07-26 23:26:39 +02:00
|
|
|
status_val = UserPresence.status_from_string(status)
|
|
|
|
if status_val is None:
|
2017-01-29 00:08:08 +01:00
|
|
|
raise JsonableError(_("Invalid status: %s") % (status,))
|
2016-07-26 23:26:39 +02:00
|
|
|
else:
|
2017-04-16 21:32:57 +02:00
|
|
|
update_user_presence(user_profile, request.client, timezone_now(),
|
2016-11-06 02:49:37 +01:00
|
|
|
status_val, new_user_input)
|
2016-07-26 23:26:39 +02:00
|
|
|
|
2017-03-31 01:46:45 +02:00
|
|
|
if ping_only:
|
2017-05-17 22:12:46 +02:00
|
|
|
ret = {} # type: Dict[str, Any]
|
2017-03-31 01:46:45 +02:00
|
|
|
else:
|
|
|
|
ret = get_status_list(user_profile)
|
|
|
|
|
2016-07-27 01:45:29 +02:00
|
|
|
if user_profile.realm.is_zephyr_mirror_realm:
|
2016-09-13 23:45:27 +02:00
|
|
|
# In zephyr mirroring realms, users can't see the presence of other
|
|
|
|
# users, but each user **is** interested in whether their mirror bot
|
|
|
|
# (running as their user) has been active.
|
2016-07-26 23:26:39 +02:00
|
|
|
try:
|
|
|
|
activity = UserActivity.objects.get(user_profile = user_profile,
|
2018-07-13 13:10:12 +02:00
|
|
|
query="get_events",
|
2016-07-26 23:26:39 +02:00
|
|
|
client__name="zephyr_mirror")
|
|
|
|
|
|
|
|
ret['zephyr_mirror_active'] = \
|
2017-04-16 21:32:57 +02:00
|
|
|
(activity.last_visit > timezone_now() - datetime.timedelta(minutes=5))
|
2016-07-26 23:26:39 +02:00
|
|
|
except UserActivity.DoesNotExist:
|
|
|
|
ret['zephyr_mirror_active'] = False
|
|
|
|
|
|
|
|
return json_success(ret)
|
2018-10-14 19:22:04 +02:00
|
|
|
|
|
|
|
def get_statuses_for_realm(request: HttpRequest, user_profile: UserProfile) -> HttpResponse:
|
|
|
|
return json_success(get_status_list(user_profile))
|