# SPDX-License-Identifier: LGPL-3.0-or-later
"""Matrix client to interact with a homeserver and other related classes."""
import asyncio
import html
import io
import logging as log
import platform
import re
import sys
import textwrap
import traceback
from contextlib import suppress
from copy import deepcopy
from datetime import datetime, timedelta
from functools import partial
from pathlib import Path
from typing import (
TYPE_CHECKING, Any, ClassVar, Dict, List, NamedTuple, Optional, Set, Tuple,
Type, Union,
)
from urllib.parse import urlparse
from uuid import UUID, uuid4
import cairosvg
from PIL import Image as PILImage
from pymediainfo import MediaInfo
import nio
from nio.crypto import AsyncDataT as UploadData
from nio.crypto import async_generator_from_data
from . import __app_name__, __display_name__, utils
from .errors import (
BadMimeType, InvalidUserId, InvalidUserInContext, MatrixBadGateway,
MatrixError, MatrixNotFound, MatrixTooLarge, MatrixUnauthorized,
UneededThumbnail, UserFromOtherServerDisallowed,
)
from .html_markdown import HTML_PROCESSOR as HTML
from .media_cache import Media, Thumbnail
from .models.items import Event, Member, Room, Upload, UploadStatus, ZeroDate
from .models.model_store import ModelStore
from .nio_callbacks import NioCallbacks
from .pyotherside_events import AlertRequested, LoopException
if TYPE_CHECKING:
from .backend import Backend
if sys.version_info >= (3, 7):
current_task = asyncio.current_task
else:
current_task = asyncio.Task.current_task
CryptDict = Dict[str, Any]
REPLY_FALLBACK = (
""
'In reply to '
'{user_id}'
"
"
"
"
"{content}"
"
{html.escape(text)}
"): content["format"] = "org.matrix.custom.html" content["formatted_body"] = to_html if reply_to_event_id: to: Event = \ self.models[self.user_id, room_id, "events"][reply_to_event_id] source_body = getattr(to.source, "body", "") content["format"] = "org.matrix.custom.html" plain_source_body = "\n".join( f"> <{to.sender_id}> {line}" if i == 0 else f"> {line}" for i, line in enumerate(source_body.splitlines()) ) content["body"] = f"{plain_source_body}\n\n{text}" to_html = REPLY_FALLBACK.format( room_id = room_id, event_id = to.event_id, user_id = to.sender_id, content = getattr(to.source, "formatted_body", "") or source_body or html.escape(to.source.source["type"] if to.source else ""), reply_content = to_html, ) echo_body = HTML.filter(to_html) content["formatted_body"] = HTML.filter(to_html, outgoing=True) content["m.relates_to"] = { "m.in_reply_to": { "event_id": to.event_id }, } # Can't use the standard Matrix transaction IDs; they're only visible # to the sender so our other accounts wouldn't be able to replace # local echoes by real messages. tx_id = uuid4() content[f"{__app_name__}.transaction_id"] = str(tx_id) mentions = HTML.mentions_in_html(echo_body) await self._local_echo( room_id, tx_id, event_type, content = echo_body, mentions = mentions, ) await self._send_message(room_id, content, tx_id) async def toggle_pause_upload( self, room_id: str, uuid: Union[str, UUID], ) -> None: if isinstance(uuid, str): uuid = UUID(uuid) pause = not self.upload_monitors[uuid].pause self.upload_monitors[uuid].pause = pause self.models[room_id, "uploads"][str(uuid)].paused = pause async def cancel_upload(self, uuid: Union[str, UUID]) -> None: if isinstance(uuid, str): uuid = UUID(uuid) self.upload_tasks[uuid].cancel() async def send_file(self, room_id: str, path: Union[Path, str]) -> None: """Send a `m.file`, `m.image`, `m.audio` or `m.video` message.""" item_uuid = uuid4() try: await self._send_file(item_uuid, room_id, path) except (nio.TransferCancelledError, asyncio.CancelledError): log.info("Deleting item for cancelled upload %s", item_uuid) del self.upload_monitors[item_uuid] del self.upload_tasks[item_uuid] del self.models[room_id, "uploads"][str(item_uuid)] async def _send_file( self, item_uuid: UUID, room_id: str, path: Union[Path, str], ) -> None: """Upload and monitor a file + thumbnail and send the built event.""" # TODO: this function is way too complex, and most of it should be # refactored into nio. transaction_id = uuid4() path = Path(path) encrypt = room_id in self.encrypted_rooms try: size = path.resolve().stat().st_size except (PermissionError, FileNotFoundError): # This error will be caught again by the try block later below size = 0 monitor = nio.TransferMonitor(size) upload_item = Upload(item_uuid, path, total_size=size) self.models[room_id, "uploads"][str(item_uuid)] = upload_item self.upload_monitors[item_uuid] = monitor self.upload_tasks[item_uuid] = current_task() # type: ignore def on_transferred(transferred: int) -> None: upload_item.uploaded = transferred def on_speed_changed(speed: float) -> None: upload_item.set_fields( speed = speed, time_left = monitor.remaining_time or timedelta(0), ) monitor.on_transferred = on_transferred monitor.on_speed_changed = on_speed_changed try: url, mime, crypt_dict = await self.upload( lambda *_: path, filename = path.name, filesize = size, encrypt = encrypt, monitor = monitor, ) # FIXME: nio might not catch the cancel in time if monitor.cancel: raise nio.TransferCancelledError() except (MatrixError, OSError) as err: upload_item.set_fields( status = UploadStatus.Error, error = type(err), error_args = err.args, ) # Wait for cancellation from UI, see parent send_file() method while True: await asyncio.sleep(0.1) upload_item.status = UploadStatus.Caching await Media.from_existing_file(self.backend.media_cache, url, path) kind = (mime or "").split("/")[0] thumb_url: str = "" thumb_info: Optional[MatrixImageInfo] = None content: dict = { f"{__app_name__}.transaction_id": str(transaction_id), "body": path.name, "info": { "mimetype": mime, "size": upload_item.total_size, }, } if encrypt: content["file"] = {"url": url, **crypt_dict} else: content["url"] = url if kind == "image": is_svg = mime == "image/svg+xml" event_type = \ nio.RoomEncryptedImage if encrypt else nio.RoomMessageImage content["msgtype"] = "m.image" content["info"]["w"], content["info"]["h"] = ( await utils.svg_dimensions(path) if is_svg else PILImage.open(path).size ) try: thumb_data, thumb_info = await self.generate_thumbnail( path, is_svg=is_svg, ) except UneededThumbnail: pass except Exception: trace = traceback.format_exc().rstrip() log.warning("Failed thumbnailing %s:\n%s", path, trace) else: thumb_ext = "png" if thumb_info.mime == "image/png" else "jpg" thumb_name = f"{path.stem}_thumbnail.{thumb_ext}" upload_item.set_fields( status = UploadStatus.Uploading, filepath = Path(thumb_name), total_size = len(thumb_data), ) try: upload_item.total_size = thumb_info.size monitor = nio.TransferMonitor(thumb_info.size) monitor.on_transferred = on_transferred monitor.on_speed_changed = on_speed_changed self.upload_monitors[item_uuid] = monitor thumb_url, _, thumb_crypt_dict = await self.upload( lambda *_: thumb_data, filename = f"{path.stem}_sample{path.suffix}", filesize = thumb_info.size, encrypt = encrypt, monitor = monitor, ) # FIXME: nio might not catch the cancel in time if monitor.cancel: raise nio.TransferCancelledError() except MatrixError as err: log.warning(f"Failed uploading thumbnail {path}: {err}") else: upload_item.status = UploadStatus.Caching await Thumbnail.from_bytes( self.backend.media_cache, thumb_url, path.name, thumb_data, wanted_size = (content["info"]["w"], content["info"]["h"]), ) if encrypt: content["info"]["thumbnail_file"] = { "url": thumb_url, **thumb_crypt_dict, } else: content["info"]["thumbnail_url"] = thumb_url content["info"]["thumbnail_info"] = thumb_info.as_dict() elif kind == "audio": event_type = \ nio.RoomEncryptedAudio if encrypt else nio.RoomMessageAudio content["msgtype"] = "m.audio" content["info"]["duration"] = getattr( MediaInfo.parse(path).tracks[0], "duration", 0, ) or 0 elif kind == "video": event_type = \ nio.RoomEncryptedVideo if encrypt else nio.RoomMessageVideo content["msgtype"] = "m.video" tracks = MediaInfo.parse(path).tracks content["info"]["duration"] = \ getattr(tracks[0], "duration", 0) or 0 content["info"]["w"] = max( getattr(t, "width", 0) or 0 for t in tracks ) content["info"]["h"] = max( getattr(t, "height", 0) or 0 for t in tracks ) else: event_type = \ nio.RoomEncryptedFile if encrypt else nio.RoomMessageFile content["msgtype"] = "m.file" content["filename"] = path.name del self.upload_monitors[item_uuid] del self.upload_tasks[item_uuid] del self.models[room_id, "uploads"][str(upload_item.id)] await self._local_echo( room_id, transaction_id, event_type, inline_content = content["body"], media_url = url, media_title = path.name, media_width = content["info"].get("w", 0), media_height = content["info"].get("h", 0), media_duration = content["info"].get("duration", 0), media_size = content["info"]["size"], media_mime = content["info"]["mimetype"], thumbnail_url = thumb_url, thumbnail_width = content["info"].get("thumbnail_info", {}).get("w", 0), thumbnail_height = content["info"].get("thumbnail_info", {}).get("h", 0), thumbnail_mime = content["info"].get("thumbnail_info", {}).get("mimetype", ""), ) await self._send_message(room_id, content, transaction_id) async def _local_echo( self, room_id: str, transaction_id: UUID, event_type: Type[nio.Event], **event_fields, ) -> None: """Register a local model `Event` while waiting for the server. When the user sends a message, we want to show instant feedback in the UI timeline without waiting for the servers to receive our message and retransmit it to us. The event will be locally echoed for all our accounts that are members of the `room_id` room. This allows sending messages from other accounts within the same composer without having to go to another page in the UI, and getting direct feedback for these accounts in the timeline. When we do get the real event retransmited by the server, it will replace the local one we registered. """ our_info = self.models["accounts"][self.user_id] content = event_fields.get("content", "").strip() if content and "inline_content" not in event_fields: event_fields["inline_content"] = HTML.filter( content, inline=True, room_id=room_id, ) event = Event( id = f"echo-{transaction_id}", event_id = "", event_type = event_type, date = datetime.now(), sender_id = self.user_id, sender_name = our_info.display_name, sender_avatar = our_info.avatar_url, is_local_echo = True, links = Event.parse_links(content), **event_fields, ) for user_id in self.models["accounts"]: if user_id in self.models[self.user_id, room_id, "members"]: key = f"echo-{transaction_id}" self.models[user_id, room_id, "events"][key] = deepcopy(event) await self.set_room_last_event(room_id, event) async def _send_message( self, room_id: str, content: dict, transaction_id: UUID, ) -> None: """Send a message event with `content` dict to a room.""" self.send_message_tasks[transaction_id] = \ current_task() # type: ignore async with self.backend.send_locks[room_id]: await self.room_send( room_id = room_id, message_type = "m.room.message", content = content, ignore_unverified_devices = True, ) async def load_all_room_members(self, room_id: str) -> None: """Request a room's full member list if it hasn't already been loaded. Member lazy-loading is used to accelerate the initial sync with the server. This method will be called from QML to load a room's entire member list when the user is currently viewing the room. """ # Room may be gone by the time this is called due to room_forget() room = self.all_rooms.get(room_id) if room and not room.members_synced: await super().joined_members(room_id) await self.register_nio_room(room, force_register_members=True) async def load_past_events(self, room_id: str) -> bool: """Ask the server for 100 previous events of the room. Events from before the client was started will be requested and registered into our models. Returns whether there are any messages left to load. """ if room_id in self.fully_loaded_rooms or \ room_id in self.invited_rooms or \ room_id in self.cleared_events_rooms or \ self.models[self.user_id, "rooms"][room_id].left: return False await self.first_sync_done.wait() while not self.past_tokens.get(room_id): # If a new room was added, wait for onSyncResponse to set the token await asyncio.sleep(0.1) response = await self.room_messages( room_id = room_id, start = self.past_tokens[room_id], limit = 100 if room_id in self.loaded_once_rooms else 20, message_filter = self.lazy_load_filter, ) self.loaded_once_rooms.add(room_id) more_to_load = True self.past_tokens[room_id] = response.end for event in response.chunk: if isinstance(event, nio.RoomCreateEvent): self.fully_loaded_rooms.add(room_id) more_to_load = False for cb in self.event_callbacks: if (cb.filter is None or isinstance(event, cb.filter)): await cb.func(self.all_rooms[room_id], event) return more_to_load async def new_direct_chat(self, invite: str, encrypt: bool = False) -> str: """Create a room and invite a single user in it for a direct chat.""" if invite == self.user_id: raise InvalidUserInContext(invite) if not self.user_id_regex.match(invite): raise InvalidUserId(invite) # Raise MatrixNotFound if profile doesn't exist await self.get_profile(invite) response = await super().room_create( invite = [invite], is_direct = True, visibility = nio.RoomVisibility.private, initial_state = [nio.EnableEncryptionBuilder().as_dict()] if encrypt else [], ) return response.room_id async def new_group_chat( self, name: Optional[str] = None, topic: Optional[str] = None, public: bool = False, encrypt: bool = False, federate: bool = True, ) -> str: """Create a new matrix room with the purpose of being a group chat.""" response = await super().room_create( name = name or None, topic = topic or None, federate = federate, visibility = nio.RoomVisibility.public if public else nio.RoomVisibility.private, initial_state = [nio.EnableEncryptionBuilder().as_dict()] if encrypt else [], ) return response.room_id async def room_join(self, alias_or_id_or_url: str) -> str: """Join an existing matrix room.""" string = alias_or_id_or_url.strip() if self.http_s_url_regex.match(string): for part in urlparse(string).fragment.split("/"): if self.room_id_or_alias_regex.match(part): string = part break else: raise ValueError(f"No alias or room id found in url {string}") if not self.room_id_or_alias_regex.match(string): raise ValueError("Not an alias or room id") response = await super().join(string) return response.room_id async def room_forget(self, room_id: str) -> None: """Leave a joined room (or decline an invite) and forget its history. If all the members of a room leave and forget it, that room will be marked as suitable for destruction by the server. """ self.models[self.user_id, "rooms"].pop(room_id, None) self.models.pop((self.user_id, room_id, "events"), None) self.models.pop((self.user_id, room_id, "members"), None) await self.update_account_unread_counts() try: await super().room_leave(room_id) except MatrixNotFound: # already left pass await super().room_forget(room_id) async def room_mass_invite( self, room_id: str, *user_ids: str, ) -> Tuple[List[str], List[Tuple[str, Exception]]]: """Invite users to a room in parallel. Returns a tuple with: - A list of users we successfully invited - A list of `(user_id, Exception)` tuples for those failed to invite. """ user_ids = tuple( uid for uid in user_ids # Server would return a 403 forbidden for users already in the room if uid not in self.all_rooms[room_id].users ) async def invite(user_id: str): if not self.user_id_regex.match(user_id): return InvalidUserId(user_id) if not self.rooms[room_id].federate: _, user_server = user_id.split(":", maxsplit=1) _, room_server = room_id.split(":", maxsplit=1) user_server = re.sub(r":443$", "", user_server) room_server = re.sub(r":443$", "", room_server) if user_server != room_server: return UserFromOtherServerDisallowed(user_id) try: await self.get_profile(user_id) except (MatrixNotFound, MatrixBadGateway) as err: return err return await self.room_invite(room_id, user_id) coros = [invite(uid) for uid in user_ids] successes = [] errors: list = [] responses = await asyncio.gather(*coros) for user_id, response in zip(user_ids, responses): if isinstance(response, nio.RoomInviteError): errors.append((user_id, MatrixError.from_nio(response))) elif isinstance(response, Exception): errors.append((user_id, response)) else: successes.append(user_id) return (successes, errors) async def room_put_state_builder( self, room_id: str, builder: nio.EventBuilder, ) -> str: """Send state event to room based from a `nio.EventBuilder` object.""" dct = builder.as_dict() response = await self.room_put_state( room_id = room_id, event_type = dct["type"], content = dct["content"], state_key = dct["state_key"], ) return response.event_id async def room_set( self, room_id: str, name: Optional[str] = None, topic: Optional[str] = None, encrypt: Optional[bool] = None, require_invite: Optional[bool] = None, forbid_guests: Optional[bool] = None, ) -> None: """Send setting state events for arguments that aren't `None`.""" builders: List[nio.EventBuilder] = [] if name is not None: builders.append(nio.ChangeNameBuilder(name=name)) if topic is not None: builders.append(nio.ChangeTopicBuilder(topic=topic)) if encrypt is False: raise ValueError("Cannot disable encryption in a E2E room") if encrypt is True: builders.append(nio.EnableEncryptionBuilder()) if require_invite is not None: builders.append(nio.ChangeJoinRulesBuilder( rule="invite" if require_invite else "public", )) if forbid_guests is not None: builders.append(nio.ChangeGuestAccessBuilder( access = "forbidden" if forbid_guests else "can_join", )) await asyncio.gather(*[ self.room_put_state_builder(room_id, b) for b in builders ]) async def get_redacted_event_content( self, nio_type: Type[nio.Event], redacter: str, sender: str, reason: str = "", ) -> str: """Get content to be displayed in place of a redacted event.""" kind = ( "message" if issubclass(nio_type, nio.RoomMessage) else "media" if issubclass(nio_type, nio.RoomMessageMedia) else "event" ) content = f"%1 removed this {kind}" if redacter == sender else \ f"%1's {kind} was removed by %2" if reason: content = f"{content}, reason: {html.escape(reason)}" return content async def room_mass_redact( self, room_id: str, reason: str, *event_client_ids: str, ) -> List[nio.RoomRedactResponse]: """Redact events from a room in parallel.""" tasks = [] for user_id in self.backend.clients: for client_id in event_client_ids: event = self.models[user_id, room_id, "events"].get(client_id) if not event: continue if event.is_local_echo: if user_id == self.user_id: uuid = UUID(event.id.replace("echo-", "")) self.send_message_tasks[uuid].cancel() event.is_local_echo = False else: if user_id == self.user_id: tasks.append( self.room_redact(room_id, event.event_id, reason), ) event.is_local_echo = True event.content = await self.get_redacted_event_content( event.event_type, self.user_id, event.sender_id, reason, ) event.event_type = nio.RedactedEvent return await asyncio.gather(*tasks) async def generate_thumbnail( self, data: UploadData, is_svg: bool = False, ) -> Tuple[bytes, MatrixImageInfo]: """Create a thumbnail from an image, return the bytes and info.""" png_modes = ("1", "L", "P", "RGBA") data = b"".join([c async for c in async_generator_from_data(data)]) is_svg = await utils.guess_mime(data) == "image/svg+xml" if is_svg: svg_width, svg_height = await utils.svg_dimensions(data) data = cairosvg.svg2png( bytestring = data, parent_width = svg_width, parent_height = svg_height, ) thumb = PILImage.open(io.BytesIO(data)) small = thumb.width <= 800 and thumb.height <= 600 is_jpg_png = thumb.format in ("JPEG", "PNG") jpgable_png = thumb.format == "PNG" and thumb.mode not in png_modes if small and is_jpg_png and not jpgable_png and not is_svg: raise UneededThumbnail() if not small: thumb.thumbnail((800, 600)) with io.BytesIO() as out: if thumb.mode in png_modes: thumb.save(out, "PNG", optimize=True) mime = "image/png" else: thumb.convert("RGB").save(out, "JPEG", optimize=True) mime = "image/jpeg" thumb_data = out.getvalue() thumb_size = len(thumb_data) if thumb_size >= len(data) and is_jpg_png and not is_svg: raise UneededThumbnail() info = MatrixImageInfo(thumb.width, thumb.height, mime, thumb_size) return (thumb_data, info) async def upload( self, data_provider: nio.DataProvider, filename: Optional[str] = None, filesize: Optional[int] = None, mime: Optional[str] = None, encrypt: bool = False, monitor: Optional[nio.TransferMonitor] = None, ) -> UploadReturn: """Upload a file to the matrix homeserver.""" if filesize > self.models["accounts"][self.user_id].max_upload_size: raise MatrixTooLarge() mime = mime or await utils.guess_mime(data_provider(0, 0)) response, decryption_dict = await super().upload( data_provider = data_provider, content_type = "application/octet-stream" if encrypt else mime, filename = filename, encrypt = encrypt, monitor = monitor, filesize = filesize, ) return UploadReturn(response.content_uri, mime, decryption_dict) async def set_avatar_from_file(self, path: Union[Path, str]) -> None: """Upload an image to the homeserver and set it as our avatar.""" path = Path(path) mime = await utils.guess_mime(path) if mime.split("/")[0] != "image": raise BadMimeType(wanted="image/*", got=mime) mxc, *_ = await self.upload( data_provider = lambda *_: path, filename = path.name, filesize = path.resolve().stat().st_size, mime = mime, ) await self.set_avatar(mxc) async def import_keys(self, infile: str, passphrase: str) -> None: """Import decryption keys from a file, then retry decrypting events.""" await super().import_keys(infile, passphrase) await self.retry_decrypting_events() async def export_keys(self, outfile: str, passphrase: str) -> None: """Export our decryption keys to a file.""" path = Path(outfile) path.parent.mkdir(parents=True, exist_ok=True) # The QML dialog asks the user if he wants to overwrite before this if path.exists(): path.unlink() await super().export_keys(outfile, passphrase) async def retry_decrypting_events(self) -> None: """Retry decrypting room `Event`s in our model we failed to decrypt.""" for sync_id, model in self.models.items(): if not (isinstance(sync_id, tuple) and len(sync_id) == 3 and sync_id[0] == self.user_id and sync_id[2] == "events"): continue _, room_id, _ = sync_id with model.write_lock: for ev in model.values(): room = self.all_rooms[room_id] if isinstance(ev.source, nio.MegolmEvent): try: decrypted = self.decrypt_event(ev.source) if not decrypted: raise nio.EncryptionError() except nio.EncryptionError: continue for callback in self.event_callbacks: filter_ = callback.filter if not filter_ or isinstance(decrypted, filter_): coro = asyncio.coroutine(callback.func) await coro(room, decrypted) async def clear_events(self, room_id: str) -> None: """Remove every `Event` of a room we registered in our model. The events will be gone from the UI, until the client is restarted. """ self.cleared_events_rooms.add(room_id) model = self.models[self.user_id, room_id, "events"] if model: model.clear() self.models[self.user_id, "rooms"][room_id].last_event_date = \ ZeroDate async def devices_info(self) -> List[Dict[str, Any]]: """Get sorted list of devices and their info for our user.""" def get_type(device_id: str) -> str: # Return "current", "no_keys", "verified", "blacklisted", # "ignored" or "unset" if device_id == self.device_id: return "current" if device_id not in self.device_store[self.user_id]: return "no_keys" trust = self.device_store[self.user_id][device_id].trust_state return trust.name def get_ed25519(device_id: str) -> str: key = "" if device_id == self.device_id: key = self.olm.account.identity_keys["ed25519"] elif device_id in self.device_store[self.user_id]: key = self.device_store[self.user_id][device_id].ed25519 return " ".join(textwrap.wrap(key, 4)) devices = [ { "id": device.id, "display_name": device.display_name or "", "last_seen_ip": (device.last_seen_ip or "").strip(" -"), "last_seen_date": device.last_seen_date or ZeroDate, "last_seen_country": "", "type": get_type(device.id), "ed25519_key": get_ed25519(device.id), } for device in (await self.devices()).devices ] # Reversed due to sorted(reverse=True) call below types_order = { "current": 5, "unset": 4, "no_keys": 3, "verified": 2, "ignored": 1, "blacklisted": 0, } # Sort by type, then by descending date return sorted( devices, key = lambda d: (types_order[d["type"]], d["last_seen_date"]), reverse = True, ) async def member_devices(self, user_id: str) -> List[Dict[str, Any]]: """Get list of E2E-aware devices for a user we share a room with.""" devices = [ # types: "verified", "blacklisted", "ignored" or "unset" { "id": device.id, "display_name": device.display_name or "", "type": device.trust_state.name, "ed25519_key": device.ed25519, } for device in self.device_store.active_user_devices(user_id) ] types_order = { "unset": 0, "verified": 1, "ignored": 2, "blacklisted": 3, } # Sort by type, then by display name, then by ID return sorted( devices, key = lambda d: (types_order[d["type"]], d["display_name"], d["id"]), ) async def rename_device(self, device_id: str, name: str) -> bool: """Rename one of our device, return `False` if it doesn't exist.""" try: await self.update_device(device_id, {"display_name": name}) return True except MatrixNotFound: return False async def auto_verify_all_other_accounts(self) -> None: """Automatically verify/blacklist our other accounts's devices.""" for client in self.backend.clients.values(): await self.auto_verify_account(client) async def auto_verify_account(self, client: "MatrixClient") -> None: """Automatically verify/blacklist one of our accounts's devices.""" if client.user_id == self.user_id: return for device in self.device_store.active_user_devices(client.user_id): if device.device_id != client.device_id: continue if device.verified or device.blacklisted: continue if device.ed25519 == client.olm.account.identity_keys["ed25519"]: self.verify_device(device) else: self.blacklist_device(device) async def delete_devices_with_password( self, device_ids: List[str], password: str, ) -> None: """Delete devices, authentifying using the account's password.""" auth = { "type": "m.login.password", "user": self.user_id, "password": password, } resp = await super().delete_devices(device_ids, auth) if isinstance(resp, nio.DeleteDevicesAuthResponse): raise MatrixUnauthorized() # Functions to register/modify data into models async def update_account_unread_counts(self) -> None: """Recalculate total unread notifications/highlights for our account""" unreads = 0 highlights = 0 local_unreads = False local_highlights = False for room in self.models[self.user_id, "rooms"].values(): unreads += room.unreads highlights += room.highlights if room.local_unreads: local_unreads = True if room.local_highlights: local_highlights = True account = self.models["accounts"][self.user_id] account.set_fields( total_unread = unreads, total_highlights = highlights, local_unreads = local_unreads, local_highlights = local_highlights, ) async def event_is_past(self, ev: Union[nio.Event, Event]) -> bool: """Return whether an event was created before this client started.""" if not self.first_sync_date: return True if isinstance(ev, Event): return ev.date < self.first_sync_date date = datetime.fromtimestamp(ev.server_timestamp / 1000) return date < self.first_sync_date async def set_room_last_event(self, room_id: str, item: Event) -> None: """Set the `last_event` for a `Room` using data in our `Event` model. The `last_event` is notably displayed in the UI room subtitles. """ room = self.models[self.user_id, "rooms"][room_id] if item.date > room.last_event_date: room.last_event_date = item.date async def register_nio_room( self, room: nio.MatrixRoom, left: bool = False, force_register_members: bool = False, ) -> None: """Register/update a `nio.MatrixRoom` as a `models.items.Room`.""" # Add room inviter = getattr(room, "inviter", "") or "" levels = room.power_levels can_send_state = partial(levels.can_user_send_state, self.user_id) can_send_msg = partial(levels.can_user_send_message, self.user_id) try: registered = self.models[self.user_id, "rooms"][room.room_id] except KeyError: registered = None last_event_date = datetime.fromtimestamp(0) typing_members = [] local_unreads = False local_highlights = False update_account_unread_counts = True unverified_devices = ( False if isinstance(room, nio.MatrixInvitedRoom) else self.room_contains_unverified(room.room_id) ) else: last_event_date = registered.last_event_date typing_members = registered.typing_members local_unreads = registered.local_unreads local_highlights = registered.local_highlights update_account_unread_counts = ( registered.unreads != room.unread_notifications or registered.highlights != room.unread_highlights ) unverified_devices = registered.unverified_devices room_item = Room( id = room.room_id, for_account = self.user_id, given_name = room.name or "", display_name = room.display_name or "", avatar_url = room.gen_avatar_url or "", plain_topic = room.topic or "", topic = HTML.filter( utils.plain2html(room.topic or ""), inline = True, room_id = room.room_id, ), inviter_id = inviter, inviter_name = room.user_name(inviter) if inviter else "", inviter_avatar = (room.avatar_url(inviter) or "") if inviter else "", left = left, typing_members = typing_members, encrypted = room.encrypted, unverified_devices = unverified_devices, invite_required = room.join_rule == "invite", guests_allowed = room.guest_access == "can_join", can_invite = levels.can_user_invite(self.user), can_kick = levels.can_user_kick(self.user), can_redact_all = levels.can_user_redact(self.user), can_send_messages = can_send_msg(), can_set_name = can_send_state("m.room.name"), can_set_topic = can_send_state("m.room.topic"), can_set_avatar = can_send_state("m.room.avatar"), can_set_encryption = can_send_state("m.room.encryption"), can_set_join_rules = can_send_state("m.room.join_rules"), can_set_guest_access = can_send_state("m.room.guest_access"), last_event_date = last_event_date, unreads = room.unread_notifications, highlights = room.unread_highlights, local_unreads = local_unreads, local_highlights = local_highlights, ) self.models[self.user_id, "rooms"][room.room_id] = room_item if not registered or force_register_members: for user_id in room.users: await self.add_member(room, user_id) if update_account_unread_counts: await self.update_account_unread_counts() async def add_member(self, room: nio.MatrixRoom, user_id: str) -> None: """Register/update a room member into our models.""" member = room.users[user_id] self.models[self.user_id, room.room_id, "members"][user_id] = Member( id = user_id, display_name = room.user_name(user_id) # disambiguated if member.display_name else "", avatar_url = member.avatar_url or "", typing = user_id in room.typing_users, power_level = member.power_level, invited = member.invited, ) if member.display_name: HTML.rooms_user_id_names[room.room_id][user_id] = \ member.display_name async def remove_member(self, room: nio.MatrixRoom, user_id: str) -> None: """Remove a room member from our models.""" self.models[self.user_id, room.room_id, "members"].pop(user_id, None) HTML.rooms_user_id_names[room.room_id].pop(user_id, None) async def get_event_profiles(self, room_id: str, event_id: str) -> None: """Fetch from network an event's sender, target and remover's profile. This should be called from QML, see `MatrixClient.get_member_profile`'s docstring. """ ev: Event = self.models[self.user_id, room_id, "events"][event_id] if not ev.fetch_profile: return get_profile = partial( self.get_member_profile, room_id, can_fetch_from_network=True, ) if not ev.sender_name and not ev.sender_avatar: sender_name, sender_avatar, _ = await get_profile(ev.sender_id) ev.set_fields(sender_name=sender_name, sender_avatar=sender_avatar) if ev.target_id and not ev.target_name and not ev.target_avatar: target_name, target_avatar, _ = await get_profile(ev.target_id) ev.set_fields(target_name=target_name, target_avatar=target_avatar) if ev.redacter_id and not ev.redacter_name: redacter_name, _, _ = await get_profile(ev.target_id) ev.redacter_name = redacter_name ev.fetch_profile = False async def get_member_profile( self, room_id: str, user_id: str, can_fetch_from_network: bool = False, ) -> Tuple[str, str, bool]: """Return a room member's (display_name, avatar, should_lazy_fetch) The returned tuple's last element tells whether `MatrixClient.get_event_profiles()` should be called by QML with `can_fetch_from_network = True` when appropriate, e.g. when this message comes in the user's view. If the member isn't found in the room (e.g. they left) and `can_fetch_from_network` is `True`, their profile is retrieved using `MatrixClient.backend.get_profile()`. """ try: member = self.models[self.user_id, room_id, "members"][user_id] return (member.display_name, member.avatar_url, False) except KeyError: # e.g. member is not in the room anymore if not can_fetch_from_network: return ("", "", True) try: info = await self.backend.get_profile(user_id) return (info.displayname or "", info.avatar_url or "", False) except MatrixError: return ("", "", False) async def register_nio_event( self, room: nio.MatrixRoom, ev: nio.Event, event_id: str = "", override_fetch_profile: Optional[bool] = None, **fields, ) -> None: """Register/update a `nio.Event` as a `models.items.Event` object.""" await self.register_nio_room(room) sender_name, sender_avatar, must_fetch_sender = \ await self.get_member_profile(room.room_id, ev.sender) target_id = getattr(ev, "state_key", "") or "" target_name, target_avatar, must_fetch_target = \ await self.get_member_profile(room.room_id, target_id) \ if target_id else ("", "", False) content = fields.get("content", "").strip() if content and "inline_content" not in fields: fields["inline_content"] = HTML.filter( content, inline=True, room_id=room.room_id, ) # Create Event ModelItem item = Event( id = event_id or ev.event_id, event_id = ev.event_id, event_type = type(ev), source = ev, date = datetime.fromtimestamp(ev.server_timestamp / 1000), sender_id = ev.sender, sender_name = sender_name, sender_avatar = sender_avatar, target_id = target_id, target_name = target_name, target_avatar = target_avatar, links = Event.parse_links(content), fetch_profile = (must_fetch_sender or must_fetch_target) if override_fetch_profile is None else override_fetch_profile, **fields, ) # Add the Event to model model = self.models[self.user_id, room.room_id, "events"] tx_id = ev.source.get("content", {}).get( f"{__app_name__}.transaction_id", ) from_us = ev.sender in self.backend.clients if from_us and tx_id and f"echo-{tx_id}" in model: item.id = f"echo-{tx_id}" model[item.id] = item await self.set_room_last_event(room.room_id, item) # Alerts if from_us or await self.event_is_past(ev): return mentions_us = HTML.user_id_link_in_html(item.content, self.user_id) AlertRequested(high_importance=mentions_us) room_item = self.models[self.user_id, "rooms"][room.room_id] room_item.local_unreads = True if mentions_us: room_item.local_highlights = True await self.update_account_unread_counts()