# Copyright © 2018 Damir Jelić <poljar@termina.org.uk>
# Copyright © 2020-2021 Famedly GmbH
#
# Permission to use, copy, modify, and/or distribute this software for
# any purpose with or without fee is hereby granted, provided that the
# above copyright notice and this permission notice appear in all copies.
#
# THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
# WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
# MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY
# SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER
# RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF
# CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF OR IN
# CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
"""nio api module.
This module contains primitives to build Matrix API http requests.
In general these functions are not directly called. One should use an existing
client like AsyncClient or HttpClient.
"""
from __future__ import annotations
import json
import os
from collections import defaultdict
from collections.abc import Iterable
from enum import Enum, unique
from typing import (
TYPE_CHECKING,
Any,
DefaultDict,
Dict,
List,
Optional,
Sequence,
Tuple,
Union,
)
from uuid import UUID
if TYPE_CHECKING:
from .events.account_data import PushAction, PushCondition
try:
from urllib.parse import quote, urlencode, urlparse
except ImportError:
from urllib import quote, urlencode # type: ignore
from urlparse import urlparse # type: ignore
MATRIX_API_PATH: str = "/_matrix/client/r0"
MATRIX_MEDIA_API_PATH: str = "/_matrix/media/r0"
_FilterT = Union[None, str, Dict[Any, Any]]
@unique
class MessageDirection(Enum):
"""Enum representing the direction messages should be fetched from."""
back = 0
front = 1
@unique
class ResizingMethod(Enum):
"""Enum representing the desired resizing method for a thumbnail.
"scale" maintains the original aspect ratio of the image,
"crop" provides an image in the aspect ratio of the requested size.
"""
scale = "scale"
crop = "crop"
@unique
class RoomVisibility(Enum):
"""Enum representing the desired visibility when creating a room.
"public" means the room will be shown in the server's room directory.
"private" will hide the room from the server's room directory.
"""
private = "private"
public = "public"
@unique
class RoomPreset(Enum):
"""Enum representing the available rule presets when creating a room.
"private_chat" makes the room invite-only and allows guests.
"trusted_private_chat" is the same as above, but also gives all invitees
the same power level as the room's creator.
"public_chat" makes the room joinable by anyone without invitations, and
forbid guests.
"""
private_chat = "private_chat"
trusted_private_chat = "trusted_private_chat"
public_chat = "public_chat"
@unique
class EventFormat(Enum):
"""Available formats in which a filter can make the server return events.
"client" will return the events in a format suitable for clients.
"federation" will return the raw event as received over federation.
"""
client = "client"
federation = "federation"
@unique
class PushRuleKind(Enum):
"""Push rule kinds defined by the Matrix spec, ordered by priority."""
override = "override"
content = "content"
room = "room"
sender = "sender"
underride = "underride"
[docs]
class Api:
"""Matrix API class.
Static methods reflecting the Matrix REST API.
"""
[docs]
@staticmethod
def to_json(content_dict: Dict[Any, Any]) -> str:
"""Turn a dictionary into a json string."""
return json.dumps(content_dict, separators=(",", ":"))
[docs]
@staticmethod
def to_canonical_json(content_dict: Dict[Any, Any]) -> str:
"""Turn a dictionary into a canonical json string."""
return json.dumps(
content_dict,
ensure_ascii=False,
separators=(",", ":"),
sort_keys=True,
)
[docs]
@staticmethod
def mimetype_to_msgtype(mimetype: str) -> str:
"""Turn a mimetype into a matrix message type."""
if mimetype.startswith("image"):
return "m.image"
elif mimetype.startswith("video"):
return "m.video"
elif mimetype.startswith("audio"):
return "m.audio"
return "m.file"
[docs]
@staticmethod
def mxc_to_http(mxc: str, homeserver: Optional[str] = None) -> Optional[str]:
"""Convert a matrix content URI to a HTTP URI."""
url = urlparse(mxc)
if url.scheme != "mxc":
return None
if not url.netloc or not url.path:
return None
parsed_homeserver = urlparse(homeserver) if homeserver else None
http_url = (
"{homeserver}/_matrix/media/r0/download/" "{server_name}{mediaId}"
).format(
homeserver=(
parsed_homeserver.geturl()
if parsed_homeserver
else f"https://{url.netloc}"
),
server_name=url.hostname,
mediaId=url.path,
)
return http_url
[docs]
@staticmethod
def encrypted_mxc_to_plumb(
mxc,
key: str,
hash: str,
iv: str,
homeserver: Optional[str] = None,
mimetype: Optional[str] = None,
) -> Optional[str]:
"""Convert a matrix content URI to a encrypted mxc URI.
The return value of this function will have a URI schema of emxc://.
The path of the URI will be converted just like the mxc_to_http()
function does, but it will also contain query parameters that are
necessary to decrypt the payload the URI is pointing to.
This function is useful to present a clickable URI that can be passed
to a plumber program that will download and decrypt the content that
the matrix content URI is pointing to.
The returned URI should never be converted to http and opened directly,
as that would expose the decryption parameters to any middleman or ISP.
Args:
mxc (str): The matrix content URI.
key (str): The encryption key that can be used to decrypt the
payload the URI is pointing to.
hash (str): The hash of the payload.
iv (str): The initial value needed to decrypt the payload.
mimetype (str): The mimetype of the payload.
"""
url = urlparse(mxc)
if url.scheme != "mxc":
return None
if not url.netloc or not url.path:
return None
parsed_homeserver = urlparse(homeserver) if homeserver else None
host = (
parsed_homeserver._replace(scheme="emxc").geturl()
if parsed_homeserver
else None
)
plumb_url = (
"{homeserver}/_matrix/media/r0/download/" "{server_name}{mediaId}"
).format(
homeserver=host if host else f"emxc://{url.netloc}",
server_name=url.hostname,
mediaId=url.path,
)
query_parameters = {
"key": key,
"hash": hash,
"iv": iv,
}
if mimetype is not None:
query_parameters["mimetype"] = mimetype
plumb_url += f"?{urlencode(query_parameters)}"
return plumb_url
@staticmethod
def _build_path(
path: List[str],
query_parameters: Optional[Dict] = None,
base_path: str = MATRIX_API_PATH,
) -> str:
"""Builds a percent-encoded path from a list of strings.
For example, turns ["hello", "wo/rld"] into "/hello/wo%2Frld".
All special characters are percent encoded,
including the forward slash (/).
Args:
path (List[str]): the list of path elements.
query_parameters (Dict, optional): [description]. Defaults to None.
base_path (str, optional): A base path to be prepended to path. Defaults to MATRIX_API_PATH.
Returns:
str: [description]
"""
quoted_path = ""
if isinstance(path, str):
quoted_path = quote(path, safe="")
elif isinstance(path, List):
quoted_path = "/".join([quote(str(part), safe="") for part in path])
else:
raise AssertionError(
f"'path' must be of type List[str] or str, got {type(path)}"
)
built_path = f"{base_path}/{quoted_path}"
built_path = built_path.rstrip("/")
if query_parameters:
built_path += f"?{urlencode(query_parameters)}"
return built_path
[docs]
@staticmethod
def discovery_info() -> Tuple[str, str]:
"""Get discovery information about a domain.
Returns the HTTP method and HTTP path for the request.
"""
path = Api._build_path(path=[".well-known", "matrix", "client"], base_path="")
return ("GET", path)
[docs]
@staticmethod
def login_info() -> Tuple[str, str]:
"""Get the homeserver's supported login types
Returns the HTTP method and HTTP path for the request.
"""
path = Api._build_path(path=["login"])
return "GET", path
[docs]
@staticmethod
def register(
user: str,
password: Optional[str] = None,
device_name: Optional[str] = "",
device_id: Optional[str] = "",
auth_dict: Optional[dict[str, Any]] = None,
):
"""Register a new user.
Args:
user (str): The fully qualified user ID or just local part of the
user ID, to log in.
password (str): The user's password.
device_name (str): A display name to assign to a newly-created
device. Ignored if device_id corresponds to a known device
device_id (str): ID of the client device. If this does not
correspond to a known client device, a new device will be
created.
auth_dict (Dict[str, Any, optional): The authentication dictionary
containing the elements for a particular registration flow.
If not provided, then m.login.dummy is used.
See the example below and here
https://spec.matrix.org/latest/client-server-api/#account-registration-and-management
for detailed documentation
Example:
>>> auth_dict = {
>>> "type": "m.login.registration_token",
>>> "registration_token": "REGISTRATIONTOKEN",
>>> "session": "session-id-from-homeserver"
>>> }
"""
path = Api._build_path(["register"])
content_dict = {
"username": user,
"password": password,
"auth": auth_dict or {"type": "m.login.dummy"},
}
if device_id:
content_dict["device_id"] = device_id
if device_name:
content_dict["initial_device_display_name"] = device_name
return "POST", path, Api.to_json(content_dict)
[docs]
@staticmethod
def login(
user: str,
password: Optional[str] = None,
device_name: Optional[str] = "",
device_id: Optional[str] = "",
token: Optional[str] = None,
) -> Tuple[str, str, str]:
"""Authenticate the user.
Returns the HTTP method, HTTP path and data for the request.
Args:
user (str): The fully qualified user ID or just local part of the
user ID, to log in. If the user ID contains an '@', but no ':',
the user ID will be considered to be an email address.
password (str): The user's password.
device_name (str): A display name to assign to a newly-created
device. Ignored if device_id corresponds to a known device
device_id (str): ID of the client device. If this does not
correspond to a known client device, a new device will be
created.
token (str): Token for token-based login.
"""
path = Api._build_path(path=["login"])
if password is not None:
identifier = {}
if "@" in user and not user.startswith("@"):
identifier = {
"type": "m.id.thirdparty",
"medium": "email",
"address": user,
}
else:
# As per spec, a user can login with either their localpart (that
# cannot contain an @) or their full Matrix ID, starting with an @.
identifier = {
"type": "m.id.user",
"user": user,
}
content_dict = {
"type": "m.login.password",
"identifier": identifier,
"password": password,
}
elif token is not None:
content_dict = {
"type": "m.login.token",
"token": token,
}
else:
raise ValueError("Neither a password nor a token was provided")
if device_id:
content_dict["device_id"] = device_id
if device_name:
content_dict["initial_device_display_name"] = device_name
return "POST", path, Api.to_json(content_dict)
[docs]
@staticmethod
def login_raw(
auth_dict: Dict[str, Any],
) -> Tuple[str, str, str]:
"""Login to the homeserver using a raw dictionary.
Returns the HTTP method, HTTP path and data for the request.
Args:
auth_dict (Dict[str, Any): The authentication dictionary
containing the elements for the logon.
See the example below and here
https://matrix.org/docs/spec/client_server/r0.6.0#authentication-types
for detailed documentation
Example:
>>> auth_dict = {
>>> "type": "m.login.password",
>>> "identifier": {
>>> "type": "m.id.thirdparty",
>>> "medium": "email",
>>> "address": "testemail@mail.org"
>>> },
>>> "password": "PASSWORDABCD",
>>> "initial_device_display_name": "Test user"
>>> }
"""
if auth_dict is None or auth_dict == {}:
raise ValueError("Auth dictionary shall not be empty")
path = Api._build_path(path=["login"])
return "POST", path, Api.to_json(auth_dict)
[docs]
@staticmethod
def logout(
access_token: str,
all_devices: bool = False,
):
"""Logout the session.
Returns nothing.
Args:
access_token (str): the access token to be used with the request.
all_devices (bool): Logout all sessions from all devices if set to True.
"""
query_parameters = {"access_token": access_token}
if all_devices:
api_path = ["logout", "all"]
else:
api_path = ["logout"]
content_dict: Dict = {}
return (
"POST",
Api._build_path(api_path, query_parameters),
Api.to_json(content_dict),
)
[docs]
@staticmethod
def sync(
access_token: str,
since: Optional[str] = None,
timeout: Optional[int] = None,
filter: Optional[_FilterT] = None,
full_state: Optional[bool] = None,
set_presence: Optional[str] = None,
) -> Tuple[str, str]:
"""Synchronise the client's state with the latest state on the server.
Returns the HTTP method and HTTP path for the request.
Args:
access_token (str): The access token to be used with the request.
since (str): The room id of the room where the event will be sent
to.
timeout (int): The maximum time to wait, in milliseconds, before
returning this request.
filter (Union[None, str, Dict[Any, Any]):
A filter ID or dict that should be used for this sync request.
full_state (bool, optional): Controls whether to include the full
state for all rooms the user is a member of. If this is set to
true, then all state events will be returned, even if since is
non-empty. The timeline will still be limited by the since
parameter.
set_presence (str, optional): Controls whether the client is automatically
marked as online by polling this API. If this parameter is omitted
then the client is automatically marked as online when it uses this API.
Otherwise if the parameter is set to "offline" then the client is not
marked as being online when it uses this API. When set to "unavailable",
the client is marked as being idle.
One of: ["offline", "online", "unavailable"]
"""
query_parameters = {"access_token": access_token}
if since:
query_parameters["since"] = since
if full_state is not None:
query_parameters["full_state"] = str(full_state).lower()
if timeout is not None:
query_parameters["timeout"] = str(timeout)
if set_presence:
query_parameters["set_presence"] = set_presence
if isinstance(filter, dict):
filter_json = json.dumps(filter, separators=(",", ":"))
query_parameters["filter"] = filter_json
elif isinstance(filter, str):
query_parameters["filter"] = filter
return "GET", Api._build_path(["sync"], query_parameters)
[docs]
@staticmethod
def room_send(
access_token: str,
room_id: str,
event_type: str,
body: Dict[Any, Any],
tx_id: Union[str, UUID],
) -> Tuple[str, str, str]:
"""Send a message event to a room.
Returns the HTTP method, HTTP path and data for the request.
Args:
access_token (str): The access token to be used with the request.
room_id (str): The room id of the room where the event will be sent
to.
event_type (str): The type of the message that will be sent.
body(Dict): The body of the event. The fields in this
object will vary depending on the type of event.
tx_id (str): The transaction ID for this event.
"""
query_parameters = {"access_token": access_token}
path = ["rooms", room_id, "send", event_type, str(tx_id)]
return ("PUT", Api._build_path(path, query_parameters), Api.to_json(body))
[docs]
@staticmethod
def space_get_hierarchy(
access_token: str,
space_id: str,
from_page: Optional[str] = None,
limit: Optional[int] = None,
max_depth: Optional[int] = None,
suggested_only: bool = False,
) -> Tuple[str, str]:
"""Get rooms/spaces that are a part of the provided space.
Returns the HTTP method and HTTP path for the request.
Args:
access_token (str): The access token to be used with the request.
space_id (str): The ID of the space to get the hierarchy for.
from_page (str, optional): Pagination token from a previous request
to this endpoint.
limit (int, optional): The maximum number of rooms to return.
max_depth (int, optional): The maximum depth of the returned tree.
suggested_only (bool, optional): Whether to only return
rooms that are considered suggested. Defaults to False.
"""
query_parameters = {"access_token": access_token}
if from_page:
query_parameters["from"] = from_page
if limit:
query_parameters["limit"] = limit
if max_depth:
query_parameters["max_depth"] = max_depth
if suggested_only:
query_parameters["suggested_only"] = suggested_only
path = ["rooms", space_id, "hierarchy"]
return ("GET", Api._build_path(path, query_parameters, "/_matrix/client/v1"))
[docs]
@staticmethod
def direct_room_list(access_token: str, user_id: str) -> Tuple[str, str]:
"""Lists all rooms flagged as direct the client is participating in.
Returns the HTTP method and HTTP path for the request.
Args:
access_token (str): The access token to be used within the request
user_id (str): The user id of the user to get the direct rooms for
"""
query_parameters = {"access_token": access_token}
path = ["user", user_id, "account_data", "m.direct"]
return ("GET", Api._build_path(path, query_parameters))
[docs]
@staticmethod
def room_get_event(
access_token: str, room_id: str, event_id: str
) -> Tuple[str, str]:
"""Get a single event based on roomId/eventId.
Returns the HTTP method and HTTP path for the request.
Args:
access_token (str): The access token to be used with the request.
room_id (str): The room id of the room where the event is in.
event_id (str): The event id to get.
"""
query_parameters = {"access_token": access_token}
path = ["rooms", room_id, "event", event_id]
return ("GET", Api._build_path(path, query_parameters))
[docs]
@staticmethod
def room_put_state(
access_token: str,
room_id: str,
event_type: str,
body: Dict[Any, Any],
state_key: str = "",
) -> Tuple[str, str, str]:
"""Send a state event.
Returns the HTTP method, HTTP path and data for the request.
Args:
access_token (str): The access token to be used with the request.
room_id (str): The room id of the room where the event will be sent
to.
event_type (str): The type of the event that will be sent.
body(Dict): The body of the event. The fields in this
object will vary depending on the type of event.
state_key: The key of the state to look up. Defaults to an empty
string.
"""
query_parameters = {"access_token": access_token}
path = ["rooms", room_id, "state", event_type, state_key]
return ("PUT", Api._build_path(path, query_parameters), Api.to_json(body))
[docs]
@staticmethod
def room_get_state_event(
access_token, room_id: str, event_type: str, state_key: str = ""
) -> Tuple[str, str]:
"""Fetch a state event.
Returns the HTTP method and HTTP path for the request.
Args:
access_token (str): The access token to be used with the request.
room_id (str): The room id of the room where the state is fetched
from.
event_type (str): The type of the event that will be fetched.
state_key: The key of the state to look up. Defaults to an empty
string.
"""
query_parameters = {"access_token": access_token}
path = ["rooms", room_id, "state", event_type, state_key]
return ("GET", Api._build_path(path, query_parameters))
[docs]
@staticmethod
def room_get_state(access_token: str, room_id: str) -> Tuple[str, str]:
"""Fetch the current state for a room.
Returns the HTTP method and HTTP path for the request.
Args:
access_token (str): The access token to be used with the request.
room_id (str): The room id of the room where the state is fetched
from.
"""
query_parameters = {"access_token": access_token}
path = ["rooms", room_id, "state"]
return ("GET", Api._build_path(path, query_parameters))
[docs]
@staticmethod
def room_redact(
access_token: str,
room_id: str,
event_id: str,
tx_id: Union[str, UUID],
reason: Optional[str] = None,
) -> Tuple[str, str, str]:
"""Strip information out of an event.
Returns the HTTP method, HTTP path and data for the request.
Args:
access_token (str): The access token to be used with the request.
room_id (str): The room id of the room that contains the event that
will be redacted.
event_id (str): The ID of the event that will be redacted.
tx_id (str/UUID, optional): A transaction ID for this event.
reason(str, optional): A description explaining why the
event was redacted.
"""
query_parameters = {"access_token": access_token}
body = {}
if reason:
body["reason"] = reason
path = ["rooms", room_id, "redact", event_id, str(tx_id)]
return ("PUT", Api._build_path(path, query_parameters), Api.to_json(body))
[docs]
@staticmethod
def room_kick(
access_token: str, room_id: str, user_id: str, reason: Optional[str] = None
) -> Tuple[str, str, str]:
"""Kick a user from a room, or withdraw their invitation.
Returns the HTTP method, HTTP path and data for the request.
Args:
access_token (str): The access token to be used with the request.
room_id (str): The room id of the room that the user will be
kicked from.
user_id (str): The user_id of the user that should be kicked.
reason (str, optional): A reason for which the user is kicked.
"""
query_parameters = {"access_token": access_token}
body = {"user_id": user_id}
if reason:
body["reason"] = reason
path = ["rooms", room_id, "kick"]
return ("POST", Api._build_path(path, query_parameters), Api.to_json(body))
[docs]
@staticmethod
def room_ban(
access_token: str,
room_id: str,
user_id: str,
reason: Optional[str] = None,
) -> Tuple[str, str, str]:
"""Ban a user from a room.
Returns the HTTP method, HTTP path and data for the request.
Args:
access_token (str): The access token to be used with the request.
room_id (str): The room id of the room that the user will be
banned from.
user_id (str): The user_id of the user that should be banned.
reason (str, optional): A reason for which the user is banned.
"""
path = ["rooms", room_id, "ban"]
query_parameters = {"access_token": access_token}
body = {"user_id": user_id}
if reason:
body["reason"] = reason
return (
"POST",
Api._build_path(path, query_parameters),
Api.to_json(body),
)
[docs]
@staticmethod
def room_unban(
access_token: str,
room_id: str,
user_id: str,
) -> Tuple[str, str, str]:
"""Unban a user from a room.
Returns the HTTP method, HTTP path and data for the request.
Args:
access_token (str): The access token to be used with the request.
room_id (str): The room id of the room that the user will be
unbanned from.
user_id (str): The user_id of the user that should be unbanned.
"""
path = ["rooms", room_id, "unban"]
query_parameters = {"access_token": access_token}
body = {"user_id": user_id}
return (
"POST",
Api._build_path(path, query_parameters),
Api.to_json(body),
)
[docs]
@staticmethod
def room_knock(
access_token: str,
room_id: str,
reason: Optional[str] = None,
) -> Tuple[str, str, str]:
"""Knocks on a room for the user.
Returns the HTTP method, HTTP path and data for the request.
Args:
access_token (str): The access token to be used with the request.
room_id (str): The room id of the room that the user will be
knocking on.
reason (str, optional): The reason the user is knocking.
"""
path = ["knock", room_id]
query_parameters = {"access_token": access_token}
body = {}
if reason:
body["reason"] = reason
return (
"POST",
Api._build_path(path, query_parameters),
Api.to_json(body),
)
[docs]
@staticmethod
def room_invite(
access_token: str, room_id: str, user_id: str
) -> Tuple[str, str, str]:
"""Invite a user to a room.
Returns the HTTP method, HTTP path and data for the request.
Args:
access_token (str): The access token to be used with the request.
room_id (str): The room id of the room that the user will be
invited to.
user_id (str): The user id of the user that should be invited.
"""
path = ["rooms", room_id, "invite"]
query_parameters = {"access_token": access_token}
body = {"user_id": user_id}
return ("POST", Api._build_path(path, query_parameters), Api.to_json(body))
[docs]
@staticmethod
def room_create(
access_token: str,
visibility: RoomVisibility = RoomVisibility.private,
alias: Optional[str] = None,
name: Optional[str] = None,
topic: Optional[str] = None,
room_version: Optional[str] = None,
room_type: Optional[str] = None,
federate: bool = True,
is_direct: bool = False,
preset: Optional[RoomPreset] = None,
invite: Sequence[str] = (),
initial_state: Sequence[Dict[str, Any]] = (),
power_level_override: Optional[Dict[str, Any]] = None,
predecessor: Optional[Dict[str, Any]] = None,
space: bool = False,
) -> Tuple[str, str, str]:
"""Create a new room.
Returns the HTTP method, HTTP path and data for the request.
Args:
access_token (str): The access token to be used with the request.
visibility (RoomVisibility): whether to have the room published in
the server's room directory or not.
Defaults to ``RoomVisibility.private``.
alias (str, optional): The desired canonical alias local part.
For example, if set to "foo" and the room is created on the
"example.com" server, the room alias will be
"#foo:example.com".
name (str, optional): A name to set for the room.
topic (str, optional): A topic to set for the room.
room_version (str, optional): The room version to set.
If not specified, the homeserver will use its default setting.
If a version not supported by the homeserver is specified,
a 400 ``M_UNSUPPORTED_ROOM_VERSION`` error will be returned.
room_type (str, optional): The room type to set.
If not specified, the homeserver will use its default setting.
In spec v1.2 the following room types are specified:
- ``m.space``
Unspecified room types are permitted through the use of Namespaced Identifiers.
federate (bool): Whether to allow users from other homeservers from
joining the room. Defaults to ``True``.
Cannot be changed later.
is_direct (bool): If this should be considered a
direct messaging room.
If ``True``, the server will set the ``is_direct`` flag on
``m.room.member events`` sent to the users in ``invite``.
Defaults to ``False``.
preset (RoomPreset, optional): The selected preset will set various
rules for the room.
If unspecified, the server will choose a preset from the
``visibility``: ``RoomVisibility.public`` equates to
``RoomPreset.public_chat``, and
``RoomVisibility.private`` equates to a
``RoomPreset.private_chat``.
invite (list): A list of user id to invite to the room.
initial_state (list): A list of state event dicts to send when
the room is created.
For example, a room could be made encrypted immediately by
having a ``m.room.encryption`` event dict.
power_level_override (dict): A ``m.room.power_levels content`` dict
to override the default.
The dict will be applied on top of the generated
``m.room.power_levels`` event before it is sent to the room.
space (bool): Create as a Space (defaults to False).
"""
path = ["createRoom"]
query_parameters = {"access_token": access_token}
body = {
"visibility": visibility.value,
"creation_content": {"m.federate": federate},
"is_direct": is_direct,
}
if alias:
body["room_alias_name"] = alias
if name:
body["name"] = name
if topic:
body["topic"] = topic
if room_version:
body["room_version"] = room_version
if room_type:
body["creation_content"]["type"] = room_type
if preset:
body["preset"] = preset.value
if invite:
body["invite"] = list(invite)
if initial_state:
body["initial_state"] = list(initial_state)
if power_level_override:
body["power_level_content_override"] = power_level_override
if predecessor:
body["creation_content"]["predecessor"] = predecessor
if space:
body["creation_content"]["type"] = "m.space"
return ("POST", Api._build_path(path, query_parameters), Api.to_json(body))
[docs]
@staticmethod
def join(access_token: str, room_id: str) -> Tuple[str, str, str]:
"""Join a room.
Returns the HTTP method, HTTP path and data for the request.
Args:
access_token (str): The access token to be used with the request.
room_id (str): The room identifier or alias to join.
"""
path = ["join", room_id]
query_parameters = {"access_token": access_token}
body = {}
return ("POST", Api._build_path(path, query_parameters), Api.to_json(body))
[docs]
@staticmethod
def room_leave(access_token: str, room_id: str) -> Tuple[str, str, str]:
"""Leave a room.
Returns the HTTP method, HTTP path and data for the request.
Args:
access_token (str): The access token to be used with the request.
room_id (str): The room id of the room that will be left.
"""
path = ["rooms", room_id, "leave"]
query_parameters = {"access_token": access_token}
body = {}
return ("POST", Api._build_path(path, query_parameters), Api.to_json(body))
[docs]
@staticmethod
def room_forget(access_token: str, room_id: str) -> Tuple[str, str, str]:
"""Forget a room.
Returns the HTTP method, HTTP path and data for the request.
Args:
access_token (str): The access token to be used with the request.
room_id (str): The room id of the room that will be forgotten.
"""
path = ["rooms", room_id, "forget"]
query_parameters = {"access_token": access_token}
body = {}
return ("POST", Api._build_path(path, query_parameters), Api.to_json(body))
[docs]
@staticmethod
def room_messages(
access_token: str,
room_id: str,
start: str,
end: Optional[str] = None,
direction: MessageDirection = MessageDirection.back,
limit: int = 10,
message_filter: Optional[Dict[Any, Any]] = None,
) -> Tuple[str, str]:
"""Get room messages.
Returns the HTTP method and HTTP path for the request.
Args:
access_token (str): The access token to be used with the request.
room_id (str): room id of the room for which to download the
messages
start (str): The token to start returning events from.
end (str): The token to stop returning events at.
direction (MessageDirection): The direction to return events from.
limit (int): The maximum number of events to return.
message_filter (Optional[Dict[Any, Any]]):
A filter dict that should be used for this room messages
request.
"""
query_parameters = {
"access_token": access_token,
"from": start,
"limit": limit,
}
if end:
query_parameters["to"] = end
if isinstance(direction, str):
if direction in ("b", "back"):
direction = MessageDirection.back
elif direction in ("f", "front"):
direction = MessageDirection.front
else:
raise ValueError("Invalid direction")
if direction is MessageDirection.front:
query_parameters["dir"] = "f"
else:
query_parameters["dir"] = "b"
if isinstance(message_filter, dict):
filter_json = json.dumps(message_filter, separators=(",", ":"))
query_parameters["filter"] = filter_json
path = ["rooms", room_id, "messages"]
return "GET", Api._build_path(path, query_parameters)
[docs]
@staticmethod
def keys_upload(
access_token: str, key_dict: Dict[str, Any]
) -> Tuple[str, str, str]:
"""Publish end-to-end encryption keys.
Returns the HTTP method, HTTP path and data for the request.
Args:
access_token (str): The access token to be used with the request.
key_dict (Dict): The dictionary containing device and one-time
keys that will be published to the server.
"""
query_parameters = {"access_token": access_token}
body = key_dict
path = ["keys", "upload"]
return ("POST", Api._build_path(path, query_parameters), Api.to_json(body))
[docs]
@staticmethod
def keys_query(
access_token: str, user_set: Iterable[str], token: Optional[str] = None
) -> Tuple[str, str, str]:
"""Query the current devices and identity keys for the given users.
Returns the HTTP method, HTTP path and data for the request.
Args:
access_token (str): The access token to be used with the request.
user_set (Set[str]): The users for which the keys should be
downloaded.
token (Optional[str]): If the client is fetching keys as a result
of a device update received in a sync request, this should be
the 'since' token of that sync request, or any later sync
token.
"""
query_parameters = {"access_token": access_token}
path = ["keys", "query"]
content: Dict[str, Dict[str, List]] = {
"device_keys": {user: [] for user in user_set}
}
if token:
content["token"] = token # type: ignore
return ("POST", Api._build_path(path, query_parameters), Api.to_json(content))
[docs]
@staticmethod
def keys_claim(
access_token: str, user_set: Dict[str, Iterable[str]]
) -> Tuple[str, str, str]:
"""Claim one-time keys for use in Olm pre-key messages.
Returns the HTTP method, HTTP path and data for the request.
Args:
access_token (str): The access token to be used with the request.
user_set (Dict[str, List[str]]): The users and devices for which to
claim one-time keys to be claimed. A map from user ID, to a
list of device IDs.
"""
query_parameters = {"access_token": access_token}
path = ["keys", "claim"]
payload: DefaultDict[str, Dict[str, str]] = defaultdict(dict)
for user_id, device_list in user_set.items():
for device_id in device_list:
payload[user_id][device_id] = "signed_curve25519"
content = {"one_time_keys": payload}
return ("POST", Api._build_path(path, query_parameters), Api.to_json(content))
[docs]
@staticmethod
def to_device(
access_token: str,
event_type: str,
content: Dict[Any, Any],
tx_id: Union[str, UUID],
) -> Tuple[str, str, str]:
r"""Send to-device events to a set of client devices.
Returns the HTTP method, HTTP path and data for the request.
Args:
access_token (str): The access token to be used with the request.
event_type (str): The type of the event which will be sent.
content (Dict): The messages to send. A map from user ID, to a map
from device ID to message body. The device ID may also be \*,
meaning all known devices for the user.
tx_id (str): The transaction ID for this event.
"""
query_parameters = {"access_token": access_token}
path = ["sendToDevice", event_type, str(tx_id)]
return ("PUT", Api._build_path(path, query_parameters), Api.to_json(content))
[docs]
@staticmethod
def devices(access_token: str) -> Tuple[str, str]:
"""Get the list of devices for the current user.
Returns the HTTP method and HTTP path for the request.
Args:
access_token (str): The access token to be used with the request.
"""
query_parameters = {"access_token": access_token}
path = ["devices"]
return "GET", Api._build_path(path, query_parameters)
[docs]
@staticmethod
def update_device(
access_token: str, device_id: str, content: Dict[str, str]
) -> Tuple[str, str, str]:
"""Update the metadata of the given device.
Returns the HTTP method, HTTP path and data for the request.
Args:
access_token (str): The access token to be used with the request.
device_id (str): The device for which the metadata will be updated.
content (Dict): A dictionary of metadata values that will be
updated for the device.
"""
query_parameters = {"access_token": access_token}
path = ["devices", device_id]
return ("PUT", Api._build_path(path, query_parameters), Api.to_json(content))
[docs]
@staticmethod
def delete_devices(
access_token: str,
devices: List[str],
auth_dict: Optional[Dict[str, str]] = None,
) -> Tuple[str, str, str]:
"""Delete a device.
This API endpoint uses the User-Interactive Authentication API.
This tells the server to delete the given devices and invalidate their
associated access tokens.
Should first be called with no additional authentication information.
Returns the HTTP method, HTTP path and data for the request.
Args:
access_token (str): The access token to be used with the request.
devices (List[str]): A list of devices which will be deleted.
auth_dict (Dict): Additional authentication information for
the user-interactive authentication API.
"""
query_parameters = {"access_token": access_token}
path = ["delete_devices"]
content: Dict[str, Any] = {"devices": devices}
if auth_dict:
content["auth"] = auth_dict
return ("POST", Api._build_path(path, query_parameters), Api.to_json(content))
[docs]
@staticmethod
def joined_members(access_token: str, room_id: str) -> Tuple[str, str]:
"""Get the list of joined members for a room.
Returns the HTTP method and HTTP path for the request.
Args:
access_token (str): The access token to be used with the request.
room_id (str): Room id of the room where the user is typing.
"""
query_parameters = {"access_token": access_token}
path = ["rooms", room_id, "joined_members"]
return "GET", Api._build_path(path, query_parameters)
[docs]
@staticmethod
def joined_rooms(access_token: str) -> Tuple[str, str]:
"""Get the list of joined rooms for the logged in account.
Returns the HTTP method and HTTP path for the request.
Args:
access_token (str): The access token to be used with the request.
"""
query_parameters = {"access_token": access_token}
path = ["joined_rooms"]
return "GET", Api._build_path(path, query_parameters)
[docs]
@staticmethod
def room_resolve_alias(room_alias: str) -> Tuple[str, str]:
"""Resolve a room alias to a room ID.
Returns the HTTP method and HTTP path for the request.
Args:
room_alias (str): The alias to resolve
"""
path = ["directory", "room", room_alias]
return "GET", Api._build_path(path)
[docs]
@staticmethod
def room_delete_alias(access_token: str, room_alias: str) -> Tuple[str, str]:
"""Delete a room alias.
Returns the HTTP method and HTTP path for the request.
Args:
access_token (str): The access token to be used with the request.
room_alias (str): The alias to delete
"""
query_parameters = {"access_token": access_token}
path = ["directory", "room", room_alias]
return "DELETE", Api._build_path(path, query_parameters)
[docs]
@staticmethod
def room_put_alias(
access_token: str, room_alias: str, room_id: str
) -> Tuple[str, str, str]:
"""Add a room alias.
Returns the HTTP method and HTTP path for the request.
Args:
access_token (str): The access token to be used with the request.
room_alias (str): The alias to add
room_id (str): The room ID to map to
"""
query_parameters = {"access_token": access_token}
path = ["directory", "room", room_alias]
body = {
"room_id": room_id,
}
return "PUT", Api._build_path(path, query_parameters), Api.to_json(body)
[docs]
@staticmethod
def room_get_visibility(room_id: str) -> Tuple[str, str]:
"""Get visibility of a room in the directory.
Returns the HTTP method and HTTP path for the request.
Args:
room_id (str): The room ID to query.
"""
path = ["directory", "list", "room", room_id]
return "GET", Api._build_path(path)
[docs]
@staticmethod
def room_typing(
access_token: str,
room_id: str,
user_id: str,
typing_state: bool = True,
timeout: int = 30000,
) -> Tuple[str, str, str]:
"""Send a typing notice to the server.
This tells the server that the user is typing for the next N
milliseconds or that the user has stopped typing.
Returns the HTTP method, HTTP path and data for the request.
Args:
access_token (str): The access token to be used with the request.
room_id (str): Room id of the room where the user is typing.
user_id (str): The user who has started to type.
typing_state (bool): A flag representing whether the user started
or stopped typing
timeout (int): For how long should the new typing notice be
valid for in milliseconds.
"""
query_parameters = {"access_token": access_token}
path = ["rooms", room_id, "typing", user_id]
content = {"typing": typing_state}
if typing_state:
content["timeout"] = timeout # type: ignore
return ("PUT", Api._build_path(path, query_parameters), Api.to_json(content))
[docs]
@staticmethod
def update_receipt_marker(
access_token: str,
room_id: str,
event_id: str,
receipt_type: str = "m.read",
) -> Tuple[str, str]:
"""Update the marker of given `receipt_type` to specified `event_id`.
Returns the HTTP method and HTTP path for the request.
Args:
access_token (str): The access token to be used with the request.
room_id (str): Room id of the room where the marker should
be updated
event_id (str): The event ID the read marker should be located at
receipt_type (str): The type of receipt to send. Currently, only
`m.read` is supported by the Matrix specification.
"""
query_parameters = {"access_token": access_token}
path = ["rooms", room_id, "receipt", receipt_type, event_id]
return ("POST", Api._build_path(path, query_parameters))
[docs]
@staticmethod
def room_read_markers(
access_token: str,
room_id: str,
fully_read_event: str,
read_event: Optional[str] = None,
) -> Tuple[str, str, str]:
"""Update fully read marker and optionally read marker for a room.
This sets the position of the read marker for a given room,
and optionally the read receipt's location.
Returns the HTTP method, HTTP path and data for the request.
Args:
access_token (str): The access token to be used with the request.
room_id (str): Room id of the room where the read
markers should be updated
fully_read_event (str): The event ID the read marker should be
located at.
read_event (Optional[str]): The event ID to set the read receipt
location at.
"""
query_parameters = {"access_token": access_token}
path = ["rooms", room_id, "read_markers"]
content = {"m.fully_read": fully_read_event}
if read_event:
content["m.read"] = read_event
return ("POST", Api._build_path(path, query_parameters), Api.to_json(content))
[docs]
@staticmethod
def content_repository_config(access_token: str) -> Tuple[str, str]:
"""Get the content repository configuration, such as upload limits.
Returns the HTTP method and HTTP path for the request.
Args:
access_token (str): The access token to be used with the request.
"""
query_parameters = {"access_token": access_token}
path = ["config"]
return (
"GET",
Api._build_path(path, query_parameters, MATRIX_MEDIA_API_PATH),
)
[docs]
@staticmethod
def upload(
access_token: str,
filename: Optional[str] = None,
) -> Tuple[str, str, str]:
"""Upload a file's content to the content repository.
Returns the HTTP method, HTTP path and empty data for the request.
The real data should be read from the file that should be uploaded.
Note: This requests also requires the Content-Type http header to be
set.
Args:
access_token (str): The access token to be used with the request.
filename (str): The name of the file being uploaded
"""
query_parameters = {"access_token": access_token}
path = ["upload"]
if filename:
query_parameters["filename"] = filename
return (
"POST",
Api._build_path(path, query_parameters, MATRIX_MEDIA_API_PATH),
"",
)
[docs]
@staticmethod
def download(
server_name: str,
media_id: str,
filename: Optional[str] = None,
allow_remote: bool = True,
file: Optional[os.PathLike] = None,
) -> Tuple[str, str]:
"""Get the content of a file from the content repository.
Returns the HTTP method and HTTP path for the request.
Args:
server_name (str): The server name from the mxc:// URI.
media_id (str): The media ID from the mxc:// URI.
filename (str, optional): A filename to be returned in the response
by the server. If None (default), the original name of the
file will be returned instead, if there is one.
allow_remote (bool): Indicates to the server that it should not
attempt to fetch the media if it is deemed remote.
This is to prevent routing loops where the server contacts
itself.
file (os.PathLike): The file to stream the downloaded content to.
"""
query_parameters = {
"allow_remote": "true" if allow_remote else "false",
}
end = ""
if filename:
end = filename
path = ["download", server_name, media_id, end]
return ("GET", Api._build_path(path, query_parameters, MATRIX_MEDIA_API_PATH))
[docs]
@staticmethod
def thumbnail(
server_name: str,
media_id: str,
width: int,
height: int,
method=ResizingMethod.scale, # ŧype: ResizingMethod
allow_remote: bool = True,
) -> Tuple[str, str]:
"""Get the thumbnail of a file from the content repository.
Returns the HTTP method and HTTP path for the request.
Note: The actual thumbnail may be larger than the size specified.
Args:
server_name (str): The server name from the mxc:// URI.
media_id (str): The media ID from the mxc:// URI.
width (int): The desired width of the thumbnail.
height (int): The desired height of the thumbnail.
method (ResizingMethod): The desired resizing method.
allow_remote (bool): Indicates to the server that it should not
attempt to fetch the media if it is deemed remote.
This is to prevent routing loops where the server contacts
itself.
"""
query_parameters = {
"width": width,
"height": height,
"method": method.value,
"allow_remote": "true" if allow_remote else "false",
}
path = ["thumbnail", server_name, media_id]
return ("GET", Api._build_path(path, query_parameters, MATRIX_MEDIA_API_PATH))
[docs]
@staticmethod
def profile_get(
user_id: str, access_token: Optional[str] = None
) -> Tuple[str, str]:
"""Get the combined profile information for a user.
Returns the HTTP method and HTTP path for the request.
Args:
user_id (str): User id to get the profile for.
access_token (str): The access token to be used with the request. If
omitted, an unauthenticated request is performed.
"""
assert user_id
query_parameters = {}
if access_token is not None:
query_parameters["access_token"] = access_token
path = ["profile", user_id]
return "GET", Api._build_path(path, query_parameters)
[docs]
@staticmethod
def profile_get_displayname(
user_id: str, access_token: Optional[str] = None
) -> Tuple[str, str]:
"""Get display name.
Returns the HTTP method and HTTP path for the request.
Args:
user_id (str): User id to get display name for.
access_token (str): The access token to be used with the request. If
omitted, an unauthenticated request is performed.
"""
query_parameters = {}
if access_token is not None:
query_parameters["access_token"] = access_token
path = ["profile", user_id, "displayname"]
return "GET", Api._build_path(path, query_parameters)
[docs]
@staticmethod
def profile_set_displayname(
access_token: str, user_id: str, display_name: str
) -> Tuple[str, str, str]:
"""Set display name.
Returns the HTTP method, HTTP path and data for the request.
Args:
access_token (str): The access token to be used with the request.
user_id (str): User id to set display name for.
display_name (str): Display name for user to set.
"""
query_parameters = {"access_token": access_token}
content = {"displayname": display_name}
path = ["profile", user_id, "displayname"]
return ("PUT", Api._build_path(path, query_parameters), Api.to_json(content))
[docs]
@staticmethod
def profile_get_avatar(
user_id: str, access_token: Optional[str] = None
) -> Tuple[str, str]:
"""Get avatar URL.
Returns the HTTP method and HTTP path for the request.
Args:
user_id (str): User id to get avatar for.
access_token (str): The access token to be used with the request. If
omitted, an unauthenticated request is performed.
"""
query_parameters = {}
if access_token is not None:
query_parameters["access_token"] = access_token
path = ["profile", user_id, "avatar_url"]
return "GET", Api._build_path(path, query_parameters)
[docs]
@staticmethod
def profile_set_avatar(
access_token: str, user_id: str, avatar_url: str
) -> Tuple[str, str, str]:
"""Set avatar url.
Returns the HTTP method, HTTP path and data for the request.
Args:
access_token (str): The access token to be used with the request.
user_id (str): User id to set display name for.
avatar_url (str): matrix content URI of the avatar to set.
"""
query_parameters = {"access_token": access_token}
content = {"avatar_url": avatar_url}
path = ["profile", user_id, "avatar_url"]
return ("PUT", Api._build_path(path, query_parameters), Api.to_json(content))
[docs]
@staticmethod
def get_presence(access_token: str, user_id: str) -> Tuple[str, str]:
"""Get the given user's presence state.
Returns the HTTP method and HTTP path for the request.
Args:
access_token (str): The access token to be used with the request.
user_id (str): User id whose presence state to get.
"""
query_parameters = {"access_token": access_token}
path = ["presence", user_id, "status"]
return (
"GET",
Api._build_path(path, query_parameters),
)
[docs]
@staticmethod
def set_presence(
access_token: str, user_id: str, presence: str, status_msg: Optional[str] = None
):
"""This API sets the given user's presence state.
Returns the HTTP method, HTTP path and data for the request.
Args:
access_token (str): The access token to be used with the request.
user_id (str): User id whose presence state to get.
presence (str): The new presence state.
status_msg (str, optional): The status message to attach to this state.
"""
query_parameters = {"access_token": access_token}
content = {"presence": presence}
if status_msg:
content["status_msg"] = status_msg
path = ["presence", user_id, "status"]
return ("PUT", Api._build_path(path, query_parameters), Api.to_json(content))
[docs]
@staticmethod
def whoami(access_token: str) -> Tuple[str, str]:
"""Get information about the owner of a given access token.
Returns the HTTP method, HTTP path and data for the request.
Args:
access_token (str): The access token to be used with the request.
"""
query_parameters = {"access_token": access_token}
path = ["account", "whoami"]
return "GET", Api._build_path(path, query_parameters)
[docs]
@staticmethod
def room_context(
access_token: str, room_id: str, event_id: str, limit: Optional[int] = None
) -> Tuple[str, str]:
"""Fetch a number of events that happened before and after an event.
This allows clients to get the context surrounding an event.
Returns the HTTP method, HTTP path and data for the request.
Args:
access_token (str): The access token to be used with the request.
room_id (str): The room_id of the room that contains the event and
its context.
event_id (str): The event_id of the event that we wish to get the
context for.
limit(int, optional): The maximum number of events to request.
"""
query_parameters = {"access_token": access_token}
if limit:
query_parameters["limit"] = limit
path = ["rooms", room_id, "context", event_id]
return "GET", Api._build_path(path, query_parameters)
[docs]
@staticmethod
def upload_filter(
access_token: str,
user_id: str,
event_fields: Optional[List[str]] = None,
event_format: EventFormat = EventFormat.client,
presence: Optional[Dict[str, Any]] = None,
account_data: Optional[Dict[str, Any]] = None,
room: Optional[Dict[str, Any]] = None,
) -> Tuple[str, str, str]:
"""Upload a new filter definition to the homeserver.
Returns the HTTP method, HTTP path and data for the request.
Args:
access_token (str): The access token to be used with the request.
user_id (str): ID of the user uploading the filter.
event_fields (Optional[List[str]]): List of event fields to
include. If this list is absent then all fields are included.
The entries may include '.' characters to indicate sub-fields.
A literal '.' character in a field name may be escaped
using a '\'.
event_format (EventFormat): The format to use for events.
presence (Dict[str, Any]): The presence updates to include.
The dict corresponds to the `EventFilter` type described
in https://matrix.org/docs/spec/client_server/latest#id240
account_data (Dict[str, Any]): The user account data that isn't
associated with rooms to include.
The dict corresponds to the `EventFilter` type described
in https://matrix.org/docs/spec/client_server/latest#id240
room (Dict[str, Any]): Filters to be applied to room data.
The dict corresponds to the `RoomFilter` type described
in https://matrix.org/docs/spec/client_server/latest#id240
"""
path = ["user", user_id, "filter"]
query_parameters = {"access_token": access_token}
content = {
"event_fields": event_fields,
"event_format": event_format.value,
"presence": presence,
"account_data": account_data,
"room": room,
}
content = {k: v for k, v in content.items() if v is not None}
return (
"POST",
Api._build_path(path, query_parameters),
Api.to_json(content),
)
[docs]
@staticmethod
def get_openid_token(access_token: str, user_id: str):
"""Gets an OpenID token object that the requester may supply to another service
to verify their identity in matrix.
Returns the HTTP method, HTTP path and data for the request.
Args:
access_token (str): The access token to be used with the request.
user_id (str): The user who requested the OpenID token
"""
path = ["user", user_id, "openid", "request_token"]
query_parameters = {"access_token": access_token}
body = {}
return (
"POST",
Api._build_path(path, query_parameters),
Api.to_json(body),
)
[docs]
@staticmethod
def set_pushrule(
access_token: str,
scope: str,
kind: PushRuleKind,
rule_id: str,
before: Optional[str] = None,
after: Optional[str] = None,
actions: Sequence[PushAction] = (),
conditions: Optional[Sequence[PushCondition]] = None,
pattern: Optional[str] = None,
) -> Tuple[str, str, str]:
"""Create or modify an existing user-created push rule.
Returns the HTTP method, HTTP path and data for the request.
Args:
access_token (str): The access token to be used with the request.
scope (str): The scope of this rule, e.g. ``"global"``.
Homeservers currently only process ``global`` rules for
event matching, while ``device`` rules are a planned feature.
It is up to clients to interpret any other scope name.
kind (PushRuleKind): The kind of rule.
rule_id (str): The identifier of the rule. Must be unique
within its scope and kind.
For rules of ``room`` kind, this is the room ID to match for.
For rules of ``sender`` kind, this is the user ID to match.
before (Optional[str]): Position this rule before the one matching
the given rule ID.
The rule ID cannot belong to a predefined server rule.
``before`` and ``after`` cannot be both specified.
after (Optional[str]): Position this rule after the one matching
the given rule ID.
The rule ID cannot belong to a predefined server rule.
``before`` and ``after`` cannot be both specified.
actions (Sequence[PushAction]): Actions to perform when the
conditions for this rule are met. The given actions replace
the existing ones.
conditions (Sequence[PushCondition]): Event conditions that must
hold true for the rule to apply to that event.
A rule with no conditions always hold true.
Only applicable to ``underride`` and ``override`` rules.
pattern (Optional[str]): Glob-style pattern to match against
for the event's content.
Only applicable to ``content`` rules.
"""
path = ["pushrules", scope, kind.value, rule_id]
query_parameters = {"access_token": access_token}
content: Dict[str, Any] = {"actions": [a.as_value for a in actions]}
if before is not None and after is not None:
raise TypeError("before and after cannot be both specified")
elif before is not None:
query_parameters["before"] = before
elif after is not None:
query_parameters["after"] = after
if pattern is not None:
if kind != PushRuleKind.content:
raise TypeError("pattern can only be set for content rules")
content["pattern"] = pattern
if conditions is not None:
if kind not in (PushRuleKind.override, PushRuleKind.underride):
raise TypeError(
"conditions can only be set for override/underride rules",
)
content["conditions"] = [c.as_value for c in conditions]
return (
"PUT",
Api._build_path(path, query_parameters),
Api.to_json(content),
)
[docs]
@staticmethod
def delete_pushrule(
access_token: str,
scope: str,
kind: PushRuleKind,
rule_id: str,
) -> Tuple[str, str]:
"""Delete an existing user-created push rule.
Returns the HTTP method and HTTP path for the request.
Args:
access_token (str): The access token to be used with the request.
scope (str): The scope of this rule, e.g. ``"global"``.
kind (PushRuleKind): The kind of rule.
rule_id (str): The identifier of the rule. Must be unique
within its scope and kind.
"""
path = ["pushrules", scope, kind.value, rule_id]
query_parameters = {"access_token": access_token}
return ("DELETE", Api._build_path(path, query_parameters))
[docs]
@staticmethod
def enable_pushrule(
access_token: str,
scope: str,
kind: PushRuleKind,
rule_id: str,
enable: bool,
) -> Tuple[str, str, str]:
"""Enable or disable an existing built-in or user-created push rule.
Returns the HTTP method, HTTP path and data for the request.
Args:
access_token (str): The access token to be used with the request.
scope (str): The scope of this rule, e.g. ``"global"``.
kind (PushRuleKind): The kind of rule.
rule_id (str): The identifier of the rule. Must be unique
within its scope and kind.
enable (bool): Whether to enable or disable the rule.
"""
path = ["pushrules", scope, kind.value, rule_id, "enabled"]
query_parameters = {"access_token": access_token}
content = {"enabled": enable}
return (
"PUT",
Api._build_path(path, query_parameters),
Api.to_json(content),
)
[docs]
@staticmethod
def set_pushrule_actions(
access_token: str,
scope: str,
kind: PushRuleKind,
rule_id: str,
actions: Sequence[PushAction],
) -> Tuple[str, str, str]:
"""Set the actions for an existing built-in or user-created push rule.
Unlike ``set_pushrule``, this method can edit built-in server rules.
Returns the HTTP method, HTTP path and data for the request.
Args:
access_token (str): The access token to be used with the request.
scope (str): The scope of this rule, e.g. ``"global"``.
kind (PushRuleKind): The kind of rule.
rule_id (str): The identifier of the rule. Must be unique
within its scope and kind.
actions (Sequence[PushAction]): Actions to perform when the
conditions for this rule are met. The given actions replace
the existing ones.
"""
path = ["pushrules", scope, kind.value, rule_id, "actions"]
query_parameters = {"access_token": access_token}
content = {"actions": [a.as_value for a in actions]}
return (
"PUT",
Api._build_path(path, query_parameters),
Api.to_json(content),
)
[docs]
@staticmethod
def delete_room_alias(access_token: str, alias: str) -> Tuple[str, str]:
"""Delete an room alias
Returns the HTTP method and HTTP path for the request.
Args:
access_token (str): The access token to be used with the request.
alias (str): The room alias
"""
query_parameters = {"access_token": access_token}
path = ["directory", "room", alias]
return ("DELETE", Api._build_path(path, query_parameters))
[docs]
@staticmethod
def put_room_alias(
access_token: str, alias: str, room_id: str
) -> Tuple[str, str, str]:
"""Add an room alias
Returns the HTTP method, HTTP path and data for the request.
Args:
access_token (str): The access token to be used with the request.
alias (str): The room alias
room_id (str): The room to point to
"""
query_parameters = {"access_token": access_token}
path = ["directory", "room", alias]
content = {}
content["room_id"] = room_id
return (
"PUT",
Api._build_path(path, query_parameters),
Api.to_json(content),
)