2021-04-06 09:38:30 -04:00
|
|
|
#
|
2023-11-21 15:29:58 -05:00
|
|
|
# This file is licensed under the Affero General Public License (AGPL) version 3.
|
|
|
|
#
|
2024-01-23 06:26:48 -05:00
|
|
|
# Copyright 2021 The Matrix.org Foundation C.I.C.
|
2023-11-21 15:29:58 -05:00
|
|
|
# Copyright (C) 2023 New Vector, Ltd
|
|
|
|
#
|
|
|
|
# This program is free software: you can redistribute it and/or modify
|
|
|
|
# it under the terms of the GNU Affero General Public License as
|
|
|
|
# published by the Free Software Foundation, either version 3 of the
|
|
|
|
# License, or (at your option) any later version.
|
|
|
|
#
|
|
|
|
# See the GNU Affero General Public License for more details:
|
|
|
|
# <https://www.gnu.org/licenses/agpl-3.0.html>.
|
|
|
|
#
|
|
|
|
# Originally licensed under the Apache License, Version 2.0:
|
|
|
|
# <http://www.apache.org/licenses/LICENSE-2.0>.
|
|
|
|
#
|
|
|
|
# [This file includes modifications made by New Vector Limited]
|
2021-04-06 09:38:30 -04:00
|
|
|
#
|
|
|
|
#
|
2021-08-17 09:22:45 -04:00
|
|
|
import logging
|
|
|
|
from typing import (
|
|
|
|
TYPE_CHECKING,
|
2021-10-13 07:24:07 -04:00
|
|
|
Any,
|
2021-08-17 09:22:45 -04:00
|
|
|
Awaitable,
|
|
|
|
Callable,
|
|
|
|
Dict,
|
|
|
|
Iterable,
|
|
|
|
List,
|
|
|
|
Optional,
|
|
|
|
Set,
|
2022-05-09 06:27:39 -04:00
|
|
|
TypeVar,
|
2021-08-17 09:22:45 -04:00
|
|
|
Union,
|
|
|
|
)
|
2021-04-06 09:38:30 -04:00
|
|
|
|
2022-05-09 06:27:39 -04:00
|
|
|
from typing_extensions import ParamSpec
|
|
|
|
|
2022-05-09 07:31:14 -04:00
|
|
|
from twisted.internet.defer import CancelledError
|
|
|
|
|
2021-04-06 09:38:30 -04:00
|
|
|
from synapse.api.presence import UserPresenceState
|
2022-05-09 07:31:14 -04:00
|
|
|
from synapse.util.async_helpers import delay_cancellation, maybe_awaitable
|
2021-04-06 09:38:30 -04:00
|
|
|
|
|
|
|
if TYPE_CHECKING:
|
|
|
|
from synapse.server import HomeServer
|
|
|
|
|
2021-08-17 09:22:45 -04:00
|
|
|
GET_USERS_FOR_STATES_CALLBACK = Callable[
|
|
|
|
[Iterable[UserPresenceState]], Awaitable[Dict[str, Set[UserPresenceState]]]
|
|
|
|
]
|
2021-10-13 07:24:07 -04:00
|
|
|
# This must either return a set of strings or the constant PresenceRouter.ALL_USERS.
|
|
|
|
GET_INTERESTED_USERS_CALLBACK = Callable[[str], Awaitable[Union[Set[str], str]]]
|
2021-08-17 09:22:45 -04:00
|
|
|
|
|
|
|
logger = logging.getLogger(__name__)
|
|
|
|
|
|
|
|
|
2022-05-09 06:27:39 -04:00
|
|
|
P = ParamSpec("P")
|
|
|
|
R = TypeVar("R")
|
|
|
|
|
|
|
|
|
2021-10-13 07:24:07 -04:00
|
|
|
def load_legacy_presence_router(hs: "HomeServer") -> None:
|
2021-08-17 09:22:45 -04:00
|
|
|
"""Wrapper that loads a presence router module configured using the old
|
|
|
|
configuration, and registers the hooks they implement.
|
|
|
|
"""
|
|
|
|
|
2021-09-29 06:44:15 -04:00
|
|
|
if hs.config.server.presence_router_module_class is None:
|
2021-08-17 09:22:45 -04:00
|
|
|
return
|
|
|
|
|
2021-09-29 06:44:15 -04:00
|
|
|
module = hs.config.server.presence_router_module_class
|
|
|
|
config = hs.config.server.presence_router_config
|
2021-08-17 09:22:45 -04:00
|
|
|
api = hs.get_module_api()
|
|
|
|
|
|
|
|
presence_router = module(config=config, module_api=api)
|
|
|
|
|
|
|
|
# The known hooks. If a module implements a method which name appears in this set,
|
|
|
|
# we'll want to register it.
|
|
|
|
presence_router_methods = {
|
|
|
|
"get_users_for_states",
|
|
|
|
"get_interested_users",
|
|
|
|
}
|
|
|
|
|
|
|
|
# All methods that the module provides should be async, but this wasn't enforced
|
|
|
|
# in the old module system, so we wrap them if needed
|
2022-05-09 06:27:39 -04:00
|
|
|
def async_wrapper(
|
2024-09-02 07:39:04 -04:00
|
|
|
f: Optional[Callable[P, R]],
|
2022-05-09 06:27:39 -04:00
|
|
|
) -> Optional[Callable[P, Awaitable[R]]]:
|
2021-08-17 09:22:45 -04:00
|
|
|
# f might be None if the callback isn't implemented by the module. In this
|
|
|
|
# case we don't want to register a callback at all so we return None.
|
|
|
|
if f is None:
|
|
|
|
return None
|
|
|
|
|
2022-05-09 06:27:39 -04:00
|
|
|
def run(*args: P.args, **kwargs: P.kwargs) -> Awaitable[R]:
|
2021-10-13 07:24:07 -04:00
|
|
|
# Assertion required because mypy can't prove we won't change `f`
|
|
|
|
# back to `None`. See
|
|
|
|
# https://mypy.readthedocs.io/en/latest/common_issues.html#narrowing-and-inner-functions
|
2021-08-17 09:22:45 -04:00
|
|
|
assert f is not None
|
|
|
|
|
|
|
|
return maybe_awaitable(f(*args, **kwargs))
|
|
|
|
|
|
|
|
return run
|
|
|
|
|
|
|
|
# Register the hooks through the module API.
|
2022-05-09 06:27:39 -04:00
|
|
|
hooks: Dict[str, Optional[Callable[..., Any]]] = {
|
2021-08-17 09:22:45 -04:00
|
|
|
hook: async_wrapper(getattr(presence_router, hook, None))
|
|
|
|
for hook in presence_router_methods
|
|
|
|
}
|
|
|
|
|
|
|
|
api.register_presence_router_callbacks(**hooks)
|
|
|
|
|
2021-04-06 09:38:30 -04:00
|
|
|
|
|
|
|
class PresenceRouter:
|
|
|
|
"""
|
|
|
|
A module that the homeserver will call upon to help route user presence updates to
|
2021-08-17 09:22:45 -04:00
|
|
|
additional destinations.
|
2021-04-06 09:38:30 -04:00
|
|
|
"""
|
|
|
|
|
|
|
|
ALL_USERS = "ALL"
|
|
|
|
|
|
|
|
def __init__(self, hs: "HomeServer"):
|
2021-08-17 09:22:45 -04:00
|
|
|
# Initially there are no callbacks
|
|
|
|
self._get_users_for_states_callbacks: List[GET_USERS_FOR_STATES_CALLBACK] = []
|
|
|
|
self._get_interested_users_callbacks: List[GET_INTERESTED_USERS_CALLBACK] = []
|
2021-04-06 09:38:30 -04:00
|
|
|
|
2021-08-17 09:22:45 -04:00
|
|
|
def register_presence_router_callbacks(
|
|
|
|
self,
|
|
|
|
get_users_for_states: Optional[GET_USERS_FOR_STATES_CALLBACK] = None,
|
|
|
|
get_interested_users: Optional[GET_INTERESTED_USERS_CALLBACK] = None,
|
2021-10-13 07:24:07 -04:00
|
|
|
) -> None:
|
2021-08-17 09:22:45 -04:00
|
|
|
# PresenceRouter modules are required to implement both of these methods
|
|
|
|
# or neither of them as they are assumed to act in a complementary manner
|
|
|
|
paired_methods = [get_users_for_states, get_interested_users]
|
|
|
|
if paired_methods.count(None) == 1:
|
|
|
|
raise RuntimeError(
|
|
|
|
"PresenceRouter modules must register neither or both of the paired callbacks: "
|
|
|
|
"[get_users_for_states, get_interested_users]"
|
2021-04-06 09:38:30 -04:00
|
|
|
)
|
|
|
|
|
2021-08-17 09:22:45 -04:00
|
|
|
# Append the methods provided to the lists of callbacks
|
|
|
|
if get_users_for_states is not None:
|
|
|
|
self._get_users_for_states_callbacks.append(get_users_for_states)
|
|
|
|
|
|
|
|
if get_interested_users is not None:
|
|
|
|
self._get_interested_users_callbacks.append(get_interested_users)
|
2021-04-06 09:38:30 -04:00
|
|
|
|
|
|
|
async def get_users_for_states(
|
|
|
|
self,
|
|
|
|
state_updates: Iterable[UserPresenceState],
|
|
|
|
) -> Dict[str, Set[UserPresenceState]]:
|
|
|
|
"""
|
|
|
|
Given an iterable of user presence updates, determine where each one
|
|
|
|
needs to go.
|
|
|
|
|
|
|
|
Args:
|
|
|
|
state_updates: An iterable of user presence state updates.
|
|
|
|
|
|
|
|
Returns:
|
|
|
|
A dictionary of user_id -> set of UserPresenceState, indicating which
|
|
|
|
presence updates each user should receive.
|
|
|
|
"""
|
|
|
|
|
2021-08-17 09:22:45 -04:00
|
|
|
# Bail out early if we don't have any callbacks to run.
|
|
|
|
if len(self._get_users_for_states_callbacks) == 0:
|
|
|
|
# Don't include any extra destinations for presence updates
|
|
|
|
return {}
|
|
|
|
|
2021-10-13 07:24:07 -04:00
|
|
|
users_for_states: Dict[str, Set[UserPresenceState]] = {}
|
2021-08-17 09:22:45 -04:00
|
|
|
# run all the callbacks for get_users_for_states and combine the results
|
|
|
|
for callback in self._get_users_for_states_callbacks:
|
|
|
|
try:
|
2022-05-06 08:35:20 -04:00
|
|
|
# Note: result is an object here, because we don't trust modules to
|
|
|
|
# return the types they're supposed to.
|
2022-05-09 07:31:14 -04:00
|
|
|
result: object = await delay_cancellation(callback(state_updates))
|
|
|
|
except CancelledError:
|
|
|
|
raise
|
2021-08-17 09:22:45 -04:00
|
|
|
except Exception as e:
|
|
|
|
logger.warning("Failed to run module API callback %s: %s", callback, e)
|
|
|
|
continue
|
|
|
|
|
|
|
|
if not isinstance(result, Dict):
|
|
|
|
logger.warning(
|
|
|
|
"Wrong type returned by module API callback %s: %s, expected Dict",
|
|
|
|
callback,
|
|
|
|
result,
|
|
|
|
)
|
|
|
|
continue
|
|
|
|
|
|
|
|
for key, new_entries in result.items():
|
|
|
|
if not isinstance(new_entries, Set):
|
|
|
|
logger.warning(
|
|
|
|
"Wrong type returned by module API callback %s: %s, expected Set",
|
|
|
|
callback,
|
|
|
|
new_entries,
|
|
|
|
)
|
|
|
|
break
|
|
|
|
users_for_states.setdefault(key, set()).update(new_entries)
|
|
|
|
|
|
|
|
return users_for_states
|
2021-04-06 09:38:30 -04:00
|
|
|
|
2021-10-13 07:24:07 -04:00
|
|
|
async def get_interested_users(self, user_id: str) -> Union[Set[str], str]:
|
2021-04-06 09:38:30 -04:00
|
|
|
"""
|
|
|
|
Retrieve a list of users that `user_id` is interested in receiving the
|
|
|
|
presence of. This will be in addition to those they share a room with.
|
|
|
|
Optionally, the object PresenceRouter.ALL_USERS can be returned to indicate
|
|
|
|
that this user should receive all incoming local and remote presence updates.
|
|
|
|
|
|
|
|
Note that this method will only be called for local users, but can return users
|
|
|
|
that are local or remote.
|
|
|
|
|
|
|
|
Args:
|
|
|
|
user_id: A user requesting presence updates.
|
|
|
|
|
|
|
|
Returns:
|
|
|
|
A set of user IDs to return presence updates for, or ALL_USERS to return all
|
|
|
|
known updates.
|
|
|
|
"""
|
|
|
|
|
2021-08-17 09:22:45 -04:00
|
|
|
# Bail out early if we don't have any callbacks to run.
|
|
|
|
if len(self._get_interested_users_callbacks) == 0:
|
|
|
|
# Don't report any additional interested users
|
|
|
|
return set()
|
|
|
|
|
|
|
|
interested_users = set()
|
|
|
|
# run all the callbacks for get_interested_users and combine the results
|
|
|
|
for callback in self._get_interested_users_callbacks:
|
|
|
|
try:
|
2022-05-09 07:31:14 -04:00
|
|
|
result = await delay_cancellation(callback(user_id))
|
|
|
|
except CancelledError:
|
|
|
|
raise
|
2021-08-17 09:22:45 -04:00
|
|
|
except Exception as e:
|
|
|
|
logger.warning("Failed to run module API callback %s: %s", callback, e)
|
|
|
|
continue
|
|
|
|
|
|
|
|
# If one of the callbacks returns ALL_USERS then we can stop calling all
|
|
|
|
# of the other callbacks, since the set of interested_users is already as
|
|
|
|
# large as it can possibly be
|
|
|
|
if result == PresenceRouter.ALL_USERS:
|
|
|
|
return PresenceRouter.ALL_USERS
|
|
|
|
|
|
|
|
if not isinstance(result, Set):
|
|
|
|
logger.warning(
|
|
|
|
"Wrong type returned by module API callback %s: %s, expected set",
|
|
|
|
callback,
|
|
|
|
result,
|
|
|
|
)
|
|
|
|
continue
|
|
|
|
|
|
|
|
# Add the new interested users to the set
|
|
|
|
interested_users.update(result)
|
|
|
|
|
|
|
|
return interested_users
|