moment/src/python/matrix_client.py

491 lines
16 KiB
Python
Raw Normal View History

2019-07-08 13:52:41 +10:00
# Copyright 2019 miruka
# This file is part of harmonyqml, licensed under LGPLv3.
import asyncio
import html
import inspect
import json
import logging as log
import platform
from contextlib import suppress
2019-07-04 11:20:49 +10:00
from datetime import datetime
2019-07-16 06:14:08 +10:00
from enum import Enum
from pathlib import Path
from types import ModuleType
2019-07-16 06:14:08 +10:00
from typing import DefaultDict, Dict, Optional, Type, Union
from uuid import uuid4
2019-07-16 06:14:08 +10:00
import filetype
import nio
from nio.rooms import MatrixRoom
from . import __about__
from .events import rooms, users
from .events.rooms import TimelineEventReceived
from .html_filter import HTML_FILTER
2019-07-16 06:14:08 +10:00
class UploadError(Enum):
forbidden = "M_FORBIDDEN"
too_large = "M_TOO_LARGE"
unknown = "UNKNOWN"
class MatrixClient(nio.AsyncClient):
def __init__(self,
backend,
user: str,
homeserver: str = "https://matrix.org",
device_id: Optional[str] = None) -> None:
# TODO: ensure homeserver starts with a scheme://
2019-07-04 11:20:49 +10:00
from .backend import Backend
self.backend: Backend = backend
self.sync_task: Optional[asyncio.Future] = None
2019-07-04 11:20:49 +10:00
self.send_locks: DefaultDict[str, asyncio.Lock] = \
DefaultDict(asyncio.Lock) # {room_id: lock}
2019-07-09 03:08:46 +10:00
# TODO: pass a ClientConfig with a pickle key
super().__init__(
homeserver = homeserver,
user = user,
device_id = device_id,
store_path = self.backend.app.appdirs.user_data_dir,
)
self.connect_callbacks()
def __repr__(self) -> str:
return "%s(user_id=%r, homeserver=%r, device_id=%r)" % (
type(self).__name__, self.user_id, self.homeserver, self.device_id
)
@staticmethod
def _classes_defined_in(module: ModuleType) -> Dict[str, Type]:
return {
m[0]: m[1] for m in inspect.getmembers(module, inspect.isclass)
if not m[0].startswith("_") and
m[1].__module__.startswith(module.__name__)
}
def connect_callbacks(self) -> None:
for name, class_ in self._classes_defined_in(nio.responses).items():
with suppress(AttributeError):
self.add_response_callback(getattr(self, f"on{name}"), class_)
for name, class_ in self._classes_defined_in(nio.events).items():
with suppress(AttributeError):
self.add_event_callback(getattr(self, f"on{name}"), class_)
async def start_syncing(self) -> None:
self.sync_task = asyncio.ensure_future(
self.sync_forever(timeout=10_000)
)
def callback(task):
raise task.exception()
self.sync_task.add_done_callback(callback)
@property
def default_device_name(self) -> str:
os_ = f" on {platform.system()}".rstrip()
os_ = f"{os_} {platform.release()}".rstrip() if os_ != " on" else ""
return f"{__about__.__pretty_name__}{os_}"
async def login(self, password: str, device_name: str = "") -> None:
response = await super().login(
password, device_name or self.default_device_name
)
if isinstance(response, nio.LoginError):
print(response)
else:
await self.start_syncing()
async def resume(self, user_id: str, token: str, device_id: str) -> None:
response = nio.LoginResponse(user_id, device_id, token)
await self.receive_response(response)
await self.start_syncing()
async def logout(self) -> None:
if self.sync_task:
self.sync_task.cancel()
with suppress(asyncio.CancelledError):
await self.sync_task
await self.close()
async def request_user_update_event(self, user_id: str) -> None:
if user_id in self.backend.pending_profile_requests:
return
self.backend.pending_profile_requests.add(user_id)
response = await self.get_profile(user_id)
if isinstance(response, nio.ProfileGetError):
2019-07-05 16:45:30 +10:00
log.warning("%s: %s", user_id, response)
users.UserUpdated(
user_id = user_id,
display_name = getattr(response, "displayname", "") or "",
avatar_url = getattr(response, "avatar_url", "") or "",
)
self.backend.pending_profile_requests.discard(user_id)
2019-07-05 16:45:30 +10:00
@property
def all_rooms(self) -> Dict[str, MatrixRoom]:
return {**self.invited_rooms, **self.rooms}
2019-07-04 11:20:49 +10:00
async def send_markdown(self, room_id: str, text: str) -> None:
2019-07-20 10:55:52 +10:00
escape = False
if text.startswith("//") or text.startswith(r"\/"):
escape = True
text = text[1:]
if text.startswith("/me ") and not escape:
event_type = nio.RoomMessageEmote
text = text[len("/me "): ]
content = {"body": text, "msgtype": "m.emote"}
2019-07-22 07:41:43 +10:00
to_html = HTML_FILTER.from_markdown_inline(text, outgoing=True)
echo_html = HTML_FILTER.from_markdown_inline(text)
2019-07-20 10:55:52 +10:00
else:
event_type = nio.RoomMessageText
content = {"body": text, "msgtype": "m.text"}
2019-07-22 07:41:43 +10:00
to_html = HTML_FILTER.from_markdown(text, outgoing=True)
echo_html = HTML_FILTER.from_markdown(text)
2019-07-22 07:41:43 +10:00
if to_html not in (html.escape(text), f"<p>{html.escape(text)}</p>"):
content["format"] = "org.matrix.custom.html"
content["formatted_body"] = to_html
2019-07-04 11:20:49 +10:00
TimelineEventReceived(
2019-07-20 10:55:52 +10:00
event_type = event_type,
2019-07-04 11:20:49 +10:00
room_id = room_id,
event_id = f"local_echo.{uuid4()}",
2019-07-04 11:20:49 +10:00
sender_id = self.user_id,
date = datetime.now(),
2019-07-22 07:41:43 +10:00
content = echo_html,
2019-07-04 11:20:49 +10:00
is_local_echo = True,
)
async with self.send_locks[room_id]:
2019-07-09 03:08:46 +10:00
response = await self.room_send(
room_id = room_id,
message_type = "m.room.message",
content = content,
ignore_unverified_devices = True,
)
2019-07-05 12:25:06 +10:00
if isinstance(response, nio.RoomSendError):
log.error("Failed to send message: %s", response)
2019-07-04 11:20:49 +10:00
async def load_past_events(self, room_id: str, limit: int = 25) -> bool:
2019-07-05 16:45:30 +10:00
if room_id in self.backend.fully_loaded_rooms:
return False
response = await self.room_messages(
room_id = room_id,
start = self.backend.past_tokens[room_id],
limit = limit,
)
more_to_load = True
if self.backend.past_tokens[room_id] == response.end:
self.backend.fully_loaded_rooms.add(room_id)
more_to_load = False
self.backend.past_tokens[room_id] = response.end
for event in response.chunk:
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, from_past=True
)
return more_to_load
2019-07-08 12:19:17 +10:00
async def room_forget(self, room_id: str) -> None:
await super().room_forget(room_id)
rooms.RoomForgotten(user_id=self.user_id, room_id=room_id)
2019-07-16 06:14:08 +10:00
async def upload_file(self, path: Union[Path, str]) -> str:
path = Path(path)
with open(path, "rb") as file:
mime = filetype.guess_mime(file)
file.seek(0, 0)
resp = await self.upload(file, mime, path.name)
if not isinstance(resp, nio.ErrorResponse):
return resp.content_uri
if resp.status_code == 403:
return UploadError.forbidden.value
if resp.status_code == 413:
return UploadError.too_large.value
return UploadError.unknown.value
async def set_avatar_from_file(self, path: Union[Path, str]
) -> Union[bool, str]:
resp = await self.upload_file(path)
if resp in (i.value for i in UploadError):
return resp
await self.set_avatar(resp)
return True
# Callbacks for nio responses
async def onSyncResponse(self, resp: nio.SyncResponse) -> None:
up = rooms.RoomUpdated.from_nio
for room_id, info in resp.rooms.invite.items():
room = self.invited_rooms[room_id]
for member in room.users.values():
users.UserUpdated.from_nio(member)
up(self.user_id, "Invites", room, info)
for room_id, info in resp.rooms.join.items():
room = self.rooms[room_id]
for member in room.users.values():
users.UserUpdated.from_nio(member)
2019-07-05 16:45:30 +10:00
if room_id not in self.backend.past_tokens:
self.backend.past_tokens[room_id] = info.timeline.prev_batch
up(self.user_id, "Rooms", room, info)
for room_id, info in resp.rooms.leave.items():
# TODO: handle in nio, these are rooms that were left before
# starting the client.
if room_id not in self.rooms:
continue
# TODO: handle left events in nio async client
for ev in info.timeline.events:
if isinstance(ev, nio.RoomMemberEvent):
await self.onRoomMemberEvent(self.rooms[room_id], ev)
up(self.user_id, "Left", self.rooms[room_id], info)
2019-07-07 07:29:00 +10:00
async def onErrorResponse(self, resp: nio.ErrorResponse) -> None:
2019-07-05 12:25:06 +10:00
# TODO: show something in the client
2019-07-24 04:48:37 +10:00
try:
log.warning("%s - %s", resp, json.dumps(resp.__dict__, indent=4))
except Exception:
log.warning(repr(resp))
2019-07-05 12:25:06 +10:00
# Callbacks for nio events
# Content: %1 is the sender, %2 the target (ev.state_key).
2019-07-05 16:45:30 +10:00
# pylint: disable=unused-argument
async def onRoomMessageText(self, room, ev, from_past=False) -> None:
co = HTML_FILTER.filter(
ev.formatted_body
if ev.format == "org.matrix.custom.html" else html.escape(ev.body)
)
TimelineEventReceived.from_nio(room, ev, content=co)
2019-07-20 09:19:43 +10:00
async def onRoomMessageEmote(self, room, ev, from_past=False) -> None:
co = HTML_FILTER.filter_inline(
2019-07-20 09:19:43 +10:00
ev.formatted_body
if ev.format == "org.matrix.custom.html" else html.escape(ev.body)
)
2019-07-20 09:19:43 +10:00
TimelineEventReceived.from_nio(room, ev, content=co)
# async def onRoomMessageImage(self, room, ev, from_past=False) -> None:
# import json; print("RMI", json.dumps( ev.__dict__ , indent=4))
# async def onRoomEncryptedImage(self, room, ev, from_past=False) -> None:
# import json; print("REI", json.dumps( ev.__dict__ , indent=4))
2019-07-05 16:45:30 +10:00
async def onRoomCreateEvent(self, room, ev, from_past=False) -> None:
co = "%1 allowed users on other matrix servers to join this room." \
if ev.federate else \
"%1 blocked users on other matrix servers from joining this room."
TimelineEventReceived.from_nio(room, ev, content=co)
2019-07-05 16:45:30 +10:00
async def onRoomGuestAccessEvent(self, room, ev, from_past=False) -> None:
allowed = "allowed" if ev.guest_access else "forbad"
co = f"%1 {allowed} guests to join the room."
TimelineEventReceived.from_nio(room, ev, content=co)
2019-07-05 16:45:30 +10:00
async def onRoomJoinRulesEvent(self, room, ev, from_past=False) -> None:
access = "public" if ev.join_rule == "public" else "invite-only"
co = f"%1 made the room {access}."
TimelineEventReceived.from_nio(room, ev, content=co)
2019-07-05 16:45:30 +10:00
async def onRoomHistoryVisibilityEvent(self, room, ev, from_past=False
) -> None:
if ev.history_visibility == "shared":
to = "all room members"
elif ev.history_visibility == "world_readable":
to = "any member or outsider"
elif ev.history_visibility == "joined":
to = "all room members, since the time they joined"
elif ev.history_visibility == "invited":
to = "all room members, since the time they were invited"
else:
to = "???"
log.warning("Invalid visibility - %s",
json.dumps(ev.__dict__, indent=4))
co = f"%1 made future room history visible to {to}."
TimelineEventReceived.from_nio(room, ev, content=co)
2019-07-05 16:45:30 +10:00
async def onPowerLevelsEvent(self, room, ev, from_past=False) -> None:
co = "%1 changed the room's permissions." # TODO: improve
TimelineEventReceived.from_nio(room, ev, content=co)
async def get_room_member_event_content(self, ev) -> Optional[str]:
prev = ev.prev_content
now = ev.content
2019-07-07 12:38:56 +10:00
membership = ev.membership
prev_membership = ev.prev_membership
if not prev or membership != prev_membership:
reason = f" Reason: {now['reason']}" if now.get("reason") else ""
if membership == "join":
return (
"%1 accepted their invitation."
if prev and prev_membership == "invite" else
"%1 joined the room."
)
if membership == "invite":
return "%1 invited %2 to the room."
if membership == "leave":
if ev.state_key == ev.sender:
return (
f"%1 declined their invitation.{reason}"
if prev and prev_membership == "invite" else
f"%1 left the room.{reason}"
)
return (
f"%1 withdrew %2's invitation.{reason}"
if prev and prev_membership == "invite" else
f"%1 unbanned %2 from the room.{reason}"
if prev and prev_membership == "ban" else
f"%1 kicked out %2 from the room.{reason}"
)
if membership == "ban":
return f"%1 banned %2 from the room.{reason}"
if ev.sender in self.backend.clients:
# Don't put our own name/avatar changes in the timeline
return None
changed = []
if prev and now["avatar_url"] != prev["avatar_url"]:
changed.append("profile picture") # TODO: <img>s
if prev and now["displayname"] != prev["displayname"]:
changed.append('display name from "{}" to "{}"'.format(
prev["displayname"] or ev.state_key,
now["displayname"] or ev.state_key,
))
if changed:
return "%1 changed their {}.".format(" and ".join(changed))
log.warning("Invalid member event - %s",
json.dumps(ev.__dict__, indent=4))
return None
2019-07-05 16:45:30 +10:00
async def onRoomMemberEvent(self, room, ev, from_past=False) -> None:
co = await self.get_room_member_event_content(ev)
if co is not None:
TimelineEventReceived.from_nio(room, ev, content=co)
2019-07-05 16:45:30 +10:00
async def onRoomAliasEvent(self, room, ev, from_past=False) -> None:
co = f"%1 set the room's main address to {ev.canonical_alias}."
TimelineEventReceived.from_nio(room, ev, content=co)
2019-07-05 16:45:30 +10:00
async def onRoomNameEvent(self, room, ev, from_past=False) -> None:
co = f"%1 changed the room's name to \"{ev.name}\"."
TimelineEventReceived.from_nio(room, ev, content=co)
2019-07-05 16:45:30 +10:00
async def onRoomTopicEvent(self, room, ev, from_past=False) -> None:
co = f"%1 changed the room's topic to \"{ev.topic}\"."
TimelineEventReceived.from_nio(room, ev, content=co)
2019-07-05 16:45:30 +10:00
async def onRoomEncryptionEvent(self, room, ev, from_past=False) -> None:
co = f"%1 turned on encryption for this room."
TimelineEventReceived.from_nio(room, ev, content=co)
2019-07-05 16:45:30 +10:00
async def onOlmEvent(self, room, ev, from_past=False) -> None:
co = f"%1 sent an undecryptable olm message."
TimelineEventReceived.from_nio(room, ev, content=co)
2019-07-05 16:45:30 +10:00
async def onMegolmEvent(self, room, ev, from_past=False) -> None:
co = f"%1 sent an undecryptable message."
2019-07-20 09:19:43 +10:00
TimelineEventReceived.from_nio(room, ev, content=co)
2019-07-05 16:45:30 +10:00
async def onBadEvent(self, room, ev, from_past=False) -> None:
co = f"%1 sent a malformed event."
TimelineEventReceived.from_nio(room, ev, content=co)
2019-07-05 16:45:30 +10:00
async def onUnknownBadEvent(self, room, ev, from_past=False) -> None:
co = f"%1 sent an event this client doesn't understand."
TimelineEventReceived.from_nio(room, ev, content=co)