2019-05-21 10:46:22 -04:00
|
|
|
# Copyright 2019 The Matrix.org Foundation CIC
|
|
|
|
#
|
|
|
|
# Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
# you may not use this file except in compliance with the License.
|
|
|
|
# You may obtain a copy of the License at
|
|
|
|
#
|
|
|
|
# http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
#
|
|
|
|
# Unless required by applicable law or agreed to in writing, software
|
|
|
|
# distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
# See the License for the specific language governing permissions and
|
|
|
|
# limitations under the License.
|
|
|
|
|
2019-04-04 13:39:44 -04:00
|
|
|
import asyncio
|
2019-06-06 05:14:40 -04:00
|
|
|
import os
|
2019-05-21 10:48:46 -04:00
|
|
|
from collections import defaultdict
|
2019-04-04 05:30:42 -04:00
|
|
|
from pprint import pformat
|
2019-04-12 08:19:37 -04:00
|
|
|
from typing import Any, Dict, Optional
|
2019-04-03 10:48:13 -04:00
|
|
|
|
2019-05-09 04:54:01 -04:00
|
|
|
from aiohttp.client_exceptions import ClientConnectionError
|
2019-06-06 05:14:40 -04:00
|
|
|
from jsonschema import Draft4Validator, FormatChecker, validators
|
2019-06-19 06:37:44 -04:00
|
|
|
from nio import (
|
|
|
|
AsyncClient,
|
|
|
|
ClientConfig,
|
|
|
|
EncryptionError,
|
|
|
|
KeysQueryResponse,
|
|
|
|
KeyVerificationEvent,
|
|
|
|
KeyVerificationKey,
|
|
|
|
KeyVerificationMac,
|
|
|
|
KeyVerificationStart,
|
|
|
|
LocalProtocolError,
|
|
|
|
MegolmEvent,
|
|
|
|
RoomContextError,
|
|
|
|
RoomEncryptedEvent,
|
|
|
|
RoomEncryptedMedia,
|
|
|
|
RoomMessageMedia,
|
|
|
|
RoomMessageText,
|
|
|
|
RoomNameEvent,
|
|
|
|
RoomTopicEvent,
|
|
|
|
SyncResponse,
|
|
|
|
)
|
2019-05-13 10:29:59 -04:00
|
|
|
from nio.crypto import Sas
|
2019-05-14 15:50:30 -04:00
|
|
|
from nio.store import SqliteStore
|
2019-04-03 10:48:13 -04:00
|
|
|
|
2019-07-03 11:39:57 -04:00
|
|
|
from pantalaimon.index import INDEXING_ENABLED
|
2019-04-04 05:30:42 -04:00
|
|
|
from pantalaimon.log import logger
|
2019-06-12 09:39:08 -04:00
|
|
|
from pantalaimon.store import FetchTask
|
2019-06-19 06:37:44 -04:00
|
|
|
from pantalaimon.thread_messages import (
|
|
|
|
DaemonResponse,
|
|
|
|
InviteSasSignal,
|
|
|
|
SasDoneSignal,
|
|
|
|
ShowSasSignal,
|
|
|
|
UpdateDevicesMessage,
|
|
|
|
)
|
2019-04-04 05:30:42 -04:00
|
|
|
|
2019-06-06 05:14:40 -04:00
|
|
|
SEARCH_KEYS = ["content.body", "content.name", "content.topic"]
|
|
|
|
|
|
|
|
SEARCH_TERMS_SCHEMA = {
|
|
|
|
"type": "object",
|
|
|
|
"properties": {
|
|
|
|
"search_categories": {
|
|
|
|
"type": "object",
|
|
|
|
"properties": {
|
|
|
|
"room_events": {
|
|
|
|
"type": "object",
|
|
|
|
"properties": {
|
|
|
|
"search_term": {"type": "string"},
|
|
|
|
"keys": {
|
|
|
|
"type": "array",
|
|
|
|
"items": {"type": "string", "enum": SEARCH_KEYS},
|
2019-06-19 06:37:44 -04:00
|
|
|
"default": SEARCH_KEYS,
|
2019-06-06 05:14:40 -04:00
|
|
|
},
|
|
|
|
"order_by": {"type": "string", "default": "rank"},
|
|
|
|
"include_state": {"type": "boolean", "default": False},
|
|
|
|
"filter": {"type": "object", "default": {}},
|
2019-06-07 05:59:53 -04:00
|
|
|
"event_context": {"type": "object"},
|
2019-06-06 05:14:40 -04:00
|
|
|
"groupings": {"type": "object", "default": {}},
|
|
|
|
},
|
2019-06-19 06:37:44 -04:00
|
|
|
"required": ["search_term"],
|
|
|
|
}
|
|
|
|
},
|
2019-06-06 05:14:40 -04:00
|
|
|
},
|
2019-06-19 06:37:44 -04:00
|
|
|
"required": ["room_events"],
|
2019-06-06 05:14:40 -04:00
|
|
|
},
|
2019-06-19 06:37:44 -04:00
|
|
|
"required": ["search_categories"],
|
2019-06-06 05:14:40 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
def extend_with_default(validator_class):
|
|
|
|
validate_properties = validator_class.VALIDATORS["properties"]
|
|
|
|
|
|
|
|
def set_defaults(validator, properties, instance, schema):
|
|
|
|
for prop, subschema in properties.items():
|
|
|
|
if "default" in subschema:
|
|
|
|
instance.setdefault(prop, subschema["default"])
|
|
|
|
|
2019-06-19 06:37:44 -04:00
|
|
|
for error in validate_properties(validator, properties, instance, schema):
|
2019-06-06 05:14:40 -04:00
|
|
|
yield error
|
|
|
|
|
|
|
|
return validators.extend(validator_class, {"properties": set_defaults})
|
|
|
|
|
|
|
|
|
|
|
|
Validator = extend_with_default(Draft4Validator)
|
|
|
|
|
|
|
|
|
|
|
|
def validate_json(instance, schema):
|
|
|
|
"""Validate a dictionary using the provided json schema."""
|
|
|
|
Validator(schema, format_checker=FormatChecker()).validate(instance)
|
|
|
|
|
|
|
|
|
|
|
|
class UnknownRoomError(Exception):
|
|
|
|
pass
|
|
|
|
|
2019-06-11 05:24:18 -04:00
|
|
|
|
2019-06-10 09:55:37 -04:00
|
|
|
class InvalidOrderByError(Exception):
|
|
|
|
pass
|
|
|
|
|
2019-06-13 07:01:58 -04:00
|
|
|
|
2019-06-13 06:32:21 -04:00
|
|
|
class InvalidLimit(Exception):
|
|
|
|
pass
|
2019-04-03 10:48:13 -04:00
|
|
|
|
2019-06-13 07:01:58 -04:00
|
|
|
|
2019-04-11 08:22:55 -04:00
|
|
|
class PanClient(AsyncClient):
|
2019-04-03 10:48:13 -04:00
|
|
|
"""A wrapper class around a nio AsyncClient extending its functionality."""
|
|
|
|
|
2019-04-04 13:39:44 -04:00
|
|
|
def __init__(
|
2019-06-19 06:37:44 -04:00
|
|
|
self,
|
|
|
|
server_name,
|
|
|
|
pan_store,
|
|
|
|
pan_conf,
|
|
|
|
homeserver,
|
|
|
|
queue=None,
|
|
|
|
user_id="",
|
|
|
|
device_id="",
|
|
|
|
store_path="",
|
|
|
|
config=None,
|
|
|
|
ssl=None,
|
|
|
|
proxy=None,
|
2019-04-04 13:39:44 -04:00
|
|
|
):
|
2019-04-11 08:21:39 -04:00
|
|
|
config = config or ClientConfig(store=SqliteStore, store_name="pan.db")
|
2019-06-19 06:37:44 -04:00
|
|
|
super().__init__(homeserver, user_id, device_id, store_path, config, ssl, proxy)
|
2019-04-04 13:39:44 -04:00
|
|
|
|
2019-06-06 05:14:40 -04:00
|
|
|
index_dir = os.path.join(store_path, server_name, user_id)
|
|
|
|
|
|
|
|
try:
|
|
|
|
os.makedirs(index_dir)
|
|
|
|
except OSError:
|
|
|
|
pass
|
|
|
|
|
|
|
|
self.server_name = server_name
|
|
|
|
self.pan_store = pan_store
|
2019-06-17 06:26:38 -04:00
|
|
|
self.pan_conf = pan_conf
|
2019-07-03 11:39:57 -04:00
|
|
|
|
|
|
|
if INDEXING_ENABLED:
|
|
|
|
logger.info("Indexing enabled.")
|
|
|
|
from pantalaimon.index import IndexStore
|
|
|
|
|
|
|
|
self.index = IndexStore(self.user_id, index_dir)
|
|
|
|
else:
|
|
|
|
logger.info("Indexing disabled.")
|
|
|
|
self.index = None
|
|
|
|
|
2019-04-11 08:20:09 -04:00
|
|
|
self.task = None
|
2019-04-18 05:43:07 -04:00
|
|
|
self.queue = queue
|
2019-04-04 13:39:44 -04:00
|
|
|
|
2019-06-12 09:33:27 -04:00
|
|
|
self.room_members_fetched = defaultdict(bool)
|
|
|
|
|
2019-05-21 04:25:59 -04:00
|
|
|
self.send_semaphores = defaultdict(asyncio.Semaphore)
|
|
|
|
self.send_decision_queues = dict() # type: asyncio.Queue
|
2019-07-03 11:39:57 -04:00
|
|
|
self.last_sync_token = None
|
2019-05-21 04:25:59 -04:00
|
|
|
|
2019-06-11 07:54:56 -04:00
|
|
|
self.history_fetcher_task = None
|
|
|
|
self.history_fetch_queue = asyncio.Queue()
|
|
|
|
|
2019-06-19 06:37:44 -04:00
|
|
|
self.add_to_device_callback(self.key_verification_cb, KeyVerificationEvent)
|
|
|
|
self.add_event_callback(self.undecrypted_event_cb, MegolmEvent)
|
2019-07-03 11:39:57 -04:00
|
|
|
|
|
|
|
if INDEXING_ENABLED:
|
|
|
|
self.add_event_callback(
|
|
|
|
self.store_message_cb,
|
|
|
|
(
|
|
|
|
RoomMessageText,
|
|
|
|
RoomMessageMedia,
|
|
|
|
RoomEncryptedMedia,
|
|
|
|
RoomTopicEvent,
|
|
|
|
RoomNameEvent,
|
|
|
|
),
|
|
|
|
)
|
|
|
|
|
2019-06-19 06:37:44 -04:00
|
|
|
self.add_response_callback(self.keys_query_cb, KeysQueryResponse)
|
|
|
|
self.add_response_callback(self.sync_tasks, SyncResponse)
|
2019-05-07 08:44:29 -04:00
|
|
|
|
2019-06-06 05:14:40 -04:00
|
|
|
def store_message_cb(self, room, event):
|
2019-07-03 11:39:57 -04:00
|
|
|
assert INDEXING_ENABLED
|
|
|
|
|
2019-06-06 05:14:40 -04:00
|
|
|
display_name = room.user_name(event.sender)
|
|
|
|
avatar_url = room.avatar_url(event.sender)
|
|
|
|
|
2019-06-17 09:23:06 -04:00
|
|
|
if not room.encrypted and self.pan_conf.index_encrypted_only:
|
|
|
|
return
|
|
|
|
|
2019-06-14 08:53:25 -04:00
|
|
|
self.index.add_event(event, room.room_id, display_name, avatar_url)
|
2019-06-06 05:14:40 -04:00
|
|
|
|
2019-05-02 07:27:50 -04:00
|
|
|
@property
|
|
|
|
def unable_to_decrypt(self):
|
|
|
|
"""Room event signaling that the message couldn't be decrypted."""
|
|
|
|
return {
|
|
|
|
"type": "m.room.message",
|
|
|
|
"content": {
|
|
|
|
"msgtype": "m.text",
|
2019-06-19 06:37:44 -04:00
|
|
|
"body": (
|
|
|
|
"** Unable to decrypt: The sender's device has not "
|
|
|
|
"sent us the keys for this message. **"
|
|
|
|
),
|
|
|
|
},
|
2019-05-02 07:27:50 -04:00
|
|
|
}
|
|
|
|
|
2019-05-13 10:29:59 -04:00
|
|
|
async def send_message(self, message):
|
|
|
|
"""Send a thread message to the UI thread."""
|
2019-05-03 07:46:19 -04:00
|
|
|
await self.queue.put(message)
|
|
|
|
|
2019-07-01 10:44:39 -04:00
|
|
|
async def send_update_devices(self, devices):
|
|
|
|
"""Send a dictionary of devices to the UI thread."""
|
|
|
|
dict_devices = defaultdict(dict)
|
|
|
|
|
|
|
|
for user_devices in devices.values():
|
|
|
|
for device in user_devices.values():
|
|
|
|
# Turn the OlmDevice type into a dictionary, flatten the
|
|
|
|
# keys dict and remove the deleted key/value.
|
|
|
|
# Since all the keys and values are strings this also
|
|
|
|
# copies them making it thread safe.
|
|
|
|
device_dict = device.as_dict()
|
|
|
|
device_dict = {**device_dict, **device_dict["keys"]}
|
|
|
|
device_dict.pop("keys")
|
|
|
|
display_name = device_dict.pop("display_name")
|
|
|
|
device_dict["device_display_name"] = display_name
|
|
|
|
dict_devices[device.user_id][device.id] = device_dict
|
|
|
|
|
|
|
|
message = UpdateDevicesMessage(self.user_id, dict_devices)
|
2019-05-17 08:31:38 -04:00
|
|
|
await self.queue.put(message)
|
|
|
|
|
2019-07-01 10:44:39 -04:00
|
|
|
async def send_update_device(self, device):
|
|
|
|
"""Send a single device to the UI thread to be updated."""
|
|
|
|
await self.send_update_devices({device.user_id: {device.id: device}})
|
|
|
|
|
2019-06-12 09:39:08 -04:00
|
|
|
def delete_fetcher_task(self, task):
|
2019-06-19 06:37:44 -04:00
|
|
|
self.pan_store.delete_fetcher_task(self.server_name, self.user_id, task)
|
2019-06-12 09:39:08 -04:00
|
|
|
|
2019-06-11 07:54:56 -04:00
|
|
|
async def fetcher_loop(self):
|
2019-07-03 11:39:57 -04:00
|
|
|
assert INDEXING_ENABLED
|
|
|
|
|
2019-06-19 06:37:44 -04:00
|
|
|
for t in self.pan_store.load_fetcher_tasks(self.server_name, self.user_id):
|
2019-06-12 09:39:08 -04:00
|
|
|
await self.history_fetch_queue.put(t)
|
|
|
|
|
2019-06-11 07:54:56 -04:00
|
|
|
while True:
|
|
|
|
try:
|
2019-06-18 10:30:38 -04:00
|
|
|
await asyncio.sleep(self.pan_conf.history_fetch_delay)
|
2019-06-11 07:54:56 -04:00
|
|
|
fetch_task = await self.history_fetch_queue.get()
|
|
|
|
|
|
|
|
try:
|
|
|
|
room = self.rooms[fetch_task.room_id]
|
|
|
|
except KeyError:
|
|
|
|
# The room is missing from our client, we probably left the
|
|
|
|
# room.
|
2019-06-12 09:39:08 -04:00
|
|
|
self.delete_fetcher_task(fetch_task)
|
2019-06-11 07:54:56 -04:00
|
|
|
continue
|
|
|
|
|
|
|
|
try:
|
2019-06-19 06:37:44 -04:00
|
|
|
logger.debug(
|
|
|
|
"Fetching room history for {}".format(room.display_name)
|
|
|
|
)
|
2019-06-17 10:59:54 -04:00
|
|
|
response = await self.room_messages(
|
|
|
|
fetch_task.room_id,
|
|
|
|
fetch_task.token,
|
2019-06-19 06:37:44 -04:00
|
|
|
limit=self.pan_conf.indexing_batch_size,
|
2019-06-17 10:59:54 -04:00
|
|
|
)
|
2019-06-11 07:54:56 -04:00
|
|
|
except ClientConnectionError:
|
|
|
|
self.history_fetch_queue.put(fetch_task)
|
|
|
|
|
2019-06-12 09:39:08 -04:00
|
|
|
# The chunk was empty, we're at the start of the timeline.
|
2019-06-11 07:54:56 -04:00
|
|
|
if not response.chunk:
|
2019-06-12 09:39:08 -04:00
|
|
|
self.delete_fetcher_task(fetch_task)
|
2019-06-11 07:54:56 -04:00
|
|
|
continue
|
|
|
|
|
|
|
|
for event in response.chunk:
|
2019-06-19 06:37:44 -04:00
|
|
|
if not isinstance(
|
|
|
|
event,
|
|
|
|
(
|
2019-06-11 07:54:56 -04:00
|
|
|
RoomMessageText,
|
|
|
|
RoomMessageMedia,
|
|
|
|
RoomEncryptedMedia,
|
|
|
|
RoomTopicEvent,
|
2019-06-19 06:37:44 -04:00
|
|
|
RoomNameEvent,
|
|
|
|
),
|
|
|
|
):
|
2019-06-11 07:54:56 -04:00
|
|
|
continue
|
|
|
|
|
2019-06-14 08:53:25 -04:00
|
|
|
display_name = room.user_name(event.sender)
|
|
|
|
avatar_url = room.avatar_url(event.sender)
|
2019-06-19 06:37:44 -04:00
|
|
|
self.index.add_event(event, room.room_id, display_name, avatar_url)
|
2019-06-14 08:53:25 -04:00
|
|
|
|
|
|
|
last_event = response.chunk[-1]
|
|
|
|
|
2019-06-19 06:37:44 -04:00
|
|
|
if not self.index.event_in_store(last_event.event_id, room.room_id):
|
2019-06-11 07:54:56 -04:00
|
|
|
# There may be even more events to fetch, add a new task to
|
|
|
|
# the queue.
|
2019-06-12 09:39:08 -04:00
|
|
|
task = FetchTask(room.room_id, response.end)
|
2019-06-19 06:37:44 -04:00
|
|
|
self.pan_store.save_fetcher_task(
|
|
|
|
self.server_name, self.user_id, task
|
|
|
|
)
|
2019-06-12 09:39:08 -04:00
|
|
|
await self.history_fetch_queue.put(task)
|
|
|
|
|
2019-06-14 08:53:25 -04:00
|
|
|
await self.index.commit_events()
|
2019-06-12 09:39:08 -04:00
|
|
|
self.delete_fetcher_task(fetch_task)
|
2019-06-18 10:31:17 -04:00
|
|
|
except asyncio.CancelledError:
|
2019-06-11 07:54:56 -04:00
|
|
|
return
|
|
|
|
|
2019-05-07 08:44:29 -04:00
|
|
|
async def sync_tasks(self, response):
|
2019-07-03 11:39:57 -04:00
|
|
|
if self.index:
|
|
|
|
await self.index.commit_events()
|
|
|
|
|
2019-06-19 06:37:44 -04:00
|
|
|
self.pan_store.save_token(self.server_name, self.user_id, self.next_batch)
|
2019-06-11 10:06:35 -04:00
|
|
|
|
2019-06-17 09:23:06 -04:00
|
|
|
for room_id, room_info in response.rooms.join.items():
|
|
|
|
if room_info.timeline.limited:
|
|
|
|
room = self.rooms[room_id]
|
|
|
|
|
|
|
|
if not room.encrypted and self.pan_conf.index_encrypted_only:
|
|
|
|
continue
|
|
|
|
|
2019-06-19 06:37:44 -04:00
|
|
|
logger.info(
|
|
|
|
"Room {} had a limited timeline, queueing "
|
|
|
|
"room for history fetching.".format(room.display_name)
|
|
|
|
)
|
2019-06-17 09:23:06 -04:00
|
|
|
task = FetchTask(room_id, room_info.timeline.prev_batch)
|
2019-06-19 06:37:44 -04:00
|
|
|
self.pan_store.save_fetcher_task(self.server_name, self.user_id, task)
|
2019-06-12 09:39:08 -04:00
|
|
|
|
|
|
|
await self.history_fetch_queue.put(task)
|
2019-06-11 07:54:56 -04:00
|
|
|
|
2019-05-14 06:58:16 -04:00
|
|
|
async def keys_query_cb(self, response):
|
2019-07-01 12:19:36 -04:00
|
|
|
if response.changed:
|
|
|
|
await self.send_update_devices(response.changed)
|
2019-05-07 08:44:29 -04:00
|
|
|
|
2019-07-03 03:50:45 -04:00
|
|
|
async def undecrypted_event_cb(self, room, event):
|
2019-06-19 06:37:44 -04:00
|
|
|
logger.info(
|
|
|
|
"Unable to decrypt event from {} via {}.".format(
|
|
|
|
event.sender, event.device_id
|
|
|
|
)
|
|
|
|
)
|
2019-05-02 06:09:49 -04:00
|
|
|
|
|
|
|
if event.session_id not in self.outgoing_key_requests:
|
|
|
|
logger.info("Requesting room key for undecrypted event.")
|
|
|
|
|
2019-07-03 03:50:45 -04:00
|
|
|
# TODO we may want to retry this
|
|
|
|
try:
|
|
|
|
await self.request_room_key(event)
|
|
|
|
except ClientConnectionError:
|
|
|
|
pass
|
2019-04-28 15:17:10 -04:00
|
|
|
|
2019-07-03 03:50:45 -04:00
|
|
|
async def key_verification_cb(self, event):
|
|
|
|
logger.info("Received key verification event: {}".format(event))
|
2019-04-28 15:17:10 -04:00
|
|
|
if isinstance(event, KeyVerificationStart):
|
2019-06-19 06:37:44 -04:00
|
|
|
logger.info(
|
|
|
|
f"{event.sender} via {event.from_device} has started "
|
|
|
|
f"a key verification process."
|
|
|
|
)
|
2019-05-10 05:21:54 -04:00
|
|
|
|
2019-05-13 10:29:59 -04:00
|
|
|
message = InviteSasSignal(
|
2019-06-19 06:37:44 -04:00
|
|
|
self.user_id, event.sender, event.from_device, event.transaction_id
|
2019-05-10 05:21:54 -04:00
|
|
|
)
|
|
|
|
|
2019-07-03 03:50:45 -04:00
|
|
|
await self.queue.put(message)
|
2019-04-28 15:17:10 -04:00
|
|
|
|
|
|
|
elif isinstance(event, KeyVerificationKey):
|
|
|
|
sas = self.key_verifications.get(event.transaction_id, None)
|
|
|
|
if not sas:
|
|
|
|
return
|
|
|
|
|
|
|
|
device = sas.other_olm_device
|
2019-05-03 07:46:19 -04:00
|
|
|
emoji = sas.get_emoji()
|
2019-04-28 15:17:10 -04:00
|
|
|
|
2019-05-13 10:29:59 -04:00
|
|
|
message = ShowSasSignal(
|
2019-06-19 06:37:44 -04:00
|
|
|
self.user_id, device.user_id, device.id, sas.transaction_id, emoji
|
2019-04-28 15:17:10 -04:00
|
|
|
)
|
|
|
|
|
2019-07-03 03:50:45 -04:00
|
|
|
await self.queue.put(message)
|
2019-04-28 15:17:10 -04:00
|
|
|
|
2019-05-03 07:46:19 -04:00
|
|
|
elif isinstance(event, KeyVerificationMac):
|
|
|
|
sas = self.key_verifications.get(event.transaction_id, None)
|
|
|
|
if not sas:
|
|
|
|
return
|
|
|
|
device = sas.other_olm_device
|
|
|
|
|
|
|
|
if sas.verified:
|
2019-07-03 03:50:45 -04:00
|
|
|
await self.send_message(
|
|
|
|
SasDoneSignal(
|
|
|
|
self.user_id, device.user_id, device.id, sas.transaction_id
|
2019-05-13 10:29:59 -04:00
|
|
|
)
|
2019-06-19 06:37:44 -04:00
|
|
|
)
|
2019-07-03 03:50:45 -04:00
|
|
|
await self.send_update_device(device)
|
2019-05-03 07:46:19 -04:00
|
|
|
|
2019-07-03 11:42:14 -04:00
|
|
|
def start_loop(self, loop_sleep_time=None):
|
2019-04-04 13:39:44 -04:00
|
|
|
"""Start a loop that runs forever and keeps on syncing with the server.
|
|
|
|
|
|
|
|
The loop can be stopped with the stop_loop() method.
|
|
|
|
"""
|
2019-05-07 08:44:29 -04:00
|
|
|
assert not self.task
|
|
|
|
|
|
|
|
logger.info(f"Starting sync loop for {self.user_id}")
|
|
|
|
|
2019-04-11 08:20:09 -04:00
|
|
|
loop = asyncio.get_event_loop()
|
2019-06-11 09:00:35 -04:00
|
|
|
|
2019-07-03 11:39:57 -04:00
|
|
|
if INDEXING_ENABLED:
|
|
|
|
self.history_fetcher_task = loop.create_task(self.fetcher_loop())
|
2019-06-11 09:00:35 -04:00
|
|
|
|
2019-05-07 08:44:29 -04:00
|
|
|
timeout = 30000
|
2019-06-19 06:37:44 -04:00
|
|
|
sync_filter = {"room": {"state": {"lazy_load_members": True}}}
|
2019-06-12 10:24:03 -04:00
|
|
|
next_batch = self.pan_store.load_token(self.server_name, self.user_id)
|
2019-07-03 11:42:14 -04:00
|
|
|
self.last_sync_token = next_batch
|
2019-06-11 10:06:35 -04:00
|
|
|
|
2019-06-11 09:00:35 -04:00
|
|
|
# We don't store any room state so initial sync needs to be with the
|
|
|
|
# full_state parameter. Subsequent ones are normal.
|
2019-06-12 10:24:03 -04:00
|
|
|
task = loop.create_task(
|
2019-07-03 11:42:14 -04:00
|
|
|
self.sync_forever(
|
|
|
|
timeout,
|
|
|
|
sync_filter,
|
|
|
|
full_state=True,
|
|
|
|
since=next_batch,
|
|
|
|
loop_sleep_time=loop_sleep_time
|
|
|
|
)
|
2019-06-12 10:24:03 -04:00
|
|
|
)
|
|
|
|
self.task = task
|
2019-06-11 09:00:35 -04:00
|
|
|
|
2019-06-12 10:24:03 -04:00
|
|
|
return task
|
2019-06-11 07:54:56 -04:00
|
|
|
|
2019-05-15 05:51:33 -04:00
|
|
|
async def start_sas(self, message, device):
|
|
|
|
try:
|
|
|
|
await self.start_key_verification(device)
|
|
|
|
await self.send_message(
|
|
|
|
DaemonResponse(
|
|
|
|
message.message_id,
|
|
|
|
self.user_id,
|
|
|
|
"m.ok",
|
2019-06-19 06:37:44 -04:00
|
|
|
"Successfully started the key verification request",
|
|
|
|
)
|
|
|
|
)
|
2019-05-15 05:51:33 -04:00
|
|
|
except ClientConnectionError as e:
|
|
|
|
await self.send_message(
|
|
|
|
DaemonResponse(
|
2019-06-19 06:37:44 -04:00
|
|
|
message.message_id, self.user_id, "m.connection_error", str(e)
|
|
|
|
)
|
|
|
|
)
|
2019-05-15 05:51:33 -04:00
|
|
|
|
2019-05-10 05:21:54 -04:00
|
|
|
async def accept_sas(self, message):
|
|
|
|
user_id = message.user_id
|
|
|
|
device_id = message.device_id
|
|
|
|
|
|
|
|
sas = self.get_active_sas(user_id, device_id)
|
|
|
|
|
|
|
|
if not sas:
|
2019-05-13 10:29:59 -04:00
|
|
|
await self.send_message(
|
|
|
|
DaemonResponse(
|
|
|
|
message.message_id,
|
|
|
|
self.user_id,
|
|
|
|
Sas._txid_error[0],
|
2019-06-19 06:37:44 -04:00
|
|
|
Sas._txid_error[1],
|
2019-05-13 10:29:59 -04:00
|
|
|
)
|
|
|
|
)
|
2019-05-10 05:21:54 -04:00
|
|
|
return
|
|
|
|
|
|
|
|
try:
|
|
|
|
await self.accept_key_verification(sas.transaction_id)
|
2019-05-13 10:29:59 -04:00
|
|
|
await self.send_message(
|
|
|
|
DaemonResponse(
|
|
|
|
message.message_id,
|
|
|
|
self.user_id,
|
|
|
|
"m.ok",
|
2019-06-19 06:37:44 -04:00
|
|
|
"Successfully accepted the key verification request",
|
|
|
|
)
|
|
|
|
)
|
2019-05-13 10:29:59 -04:00
|
|
|
except LocalProtocolError as e:
|
|
|
|
await self.send_message(
|
|
|
|
DaemonResponse(
|
|
|
|
message.message_id,
|
|
|
|
self.user_id,
|
|
|
|
Sas._unexpected_message_error[0],
|
2019-06-19 06:37:44 -04:00
|
|
|
str(e),
|
|
|
|
)
|
|
|
|
)
|
2019-05-13 10:29:59 -04:00
|
|
|
except ClientConnectionError as e:
|
|
|
|
await self.send_message(
|
|
|
|
DaemonResponse(
|
2019-06-19 06:37:44 -04:00
|
|
|
message.message_id, self.user_id, "m.connection_error", str(e)
|
|
|
|
)
|
|
|
|
)
|
2019-05-10 05:21:54 -04:00
|
|
|
|
2019-05-15 05:51:33 -04:00
|
|
|
async def cancel_sas(self, message):
|
|
|
|
user_id = message.user_id
|
|
|
|
device_id = message.device_id
|
|
|
|
|
|
|
|
sas = self.get_active_sas(user_id, device_id)
|
|
|
|
|
|
|
|
if not sas:
|
|
|
|
await self.send_message(
|
|
|
|
DaemonResponse(
|
|
|
|
message.message_id,
|
|
|
|
self.user_id,
|
|
|
|
Sas._txid_error[0],
|
2019-06-19 06:37:44 -04:00
|
|
|
Sas._txid_error[1],
|
2019-05-15 05:51:33 -04:00
|
|
|
)
|
|
|
|
)
|
|
|
|
return
|
|
|
|
|
|
|
|
try:
|
|
|
|
await self.cancel_key_verification(sas.transaction_id)
|
|
|
|
await self.send_message(
|
|
|
|
DaemonResponse(
|
|
|
|
message.message_id,
|
|
|
|
self.user_id,
|
|
|
|
"m.ok",
|
2019-06-19 06:37:44 -04:00
|
|
|
"Successfully canceled the key verification request",
|
|
|
|
)
|
|
|
|
)
|
2019-05-15 05:51:33 -04:00
|
|
|
except ClientConnectionError as e:
|
|
|
|
await self.send_message(
|
|
|
|
DaemonResponse(
|
2019-06-19 06:37:44 -04:00
|
|
|
message.message_id, self.user_id, "m.connection_error", str(e)
|
|
|
|
)
|
|
|
|
)
|
2019-05-15 05:51:33 -04:00
|
|
|
|
2019-05-03 08:01:01 -04:00
|
|
|
async def confirm_sas(self, message):
|
2019-05-03 07:46:19 -04:00
|
|
|
user_id = message.user_id
|
|
|
|
device_id = message.device_id
|
|
|
|
|
2019-05-03 10:09:06 -04:00
|
|
|
sas = self.get_active_sas(user_id, device_id)
|
2019-05-03 07:46:19 -04:00
|
|
|
|
|
|
|
if not sas:
|
2019-05-13 10:29:59 -04:00
|
|
|
await self.send_message(
|
|
|
|
DaemonResponse(
|
|
|
|
message.message_id,
|
|
|
|
self.user_id,
|
|
|
|
Sas._txid_error[0],
|
2019-06-19 06:37:44 -04:00
|
|
|
Sas._txid_error[1],
|
2019-05-13 10:29:59 -04:00
|
|
|
)
|
|
|
|
)
|
2019-05-03 07:46:19 -04:00
|
|
|
return
|
|
|
|
|
2019-05-09 04:54:01 -04:00
|
|
|
try:
|
|
|
|
await self.confirm_short_auth_string(sas.transaction_id)
|
|
|
|
except ClientConnectionError as e:
|
2019-05-13 10:29:59 -04:00
|
|
|
await self.send_message(
|
|
|
|
DaemonResponse(
|
2019-06-19 06:37:44 -04:00
|
|
|
message.message_id, self.user_id, "m.connection_error", str(e)
|
|
|
|
)
|
|
|
|
)
|
2019-05-13 10:29:59 -04:00
|
|
|
|
|
|
|
return
|
2019-05-03 07:46:19 -04:00
|
|
|
|
2019-05-03 10:09:06 -04:00
|
|
|
device = sas.other_olm_device
|
2019-05-13 10:29:59 -04:00
|
|
|
|
2019-05-03 07:46:19 -04:00
|
|
|
if sas.verified:
|
2019-07-01 10:44:39 -04:00
|
|
|
await self.send_update_device(device)
|
2019-05-13 10:29:59 -04:00
|
|
|
await self.send_message(
|
|
|
|
SasDoneSignal(
|
2019-06-19 06:37:44 -04:00
|
|
|
self.user_id, device.user_id, device.id, sas.transaction_id
|
2019-05-13 10:29:59 -04:00
|
|
|
)
|
|
|
|
)
|
2019-05-03 07:46:19 -04:00
|
|
|
else:
|
2019-05-13 10:29:59 -04:00
|
|
|
await self.send_message(
|
|
|
|
DaemonResponse(
|
|
|
|
message.message_id,
|
|
|
|
self.user_id,
|
|
|
|
"m.ok",
|
2019-06-19 06:37:44 -04:00
|
|
|
f"Waiting for {device.user_id} to confirm.",
|
|
|
|
)
|
|
|
|
)
|
2019-05-03 07:46:19 -04:00
|
|
|
|
2019-04-04 13:39:44 -04:00
|
|
|
async def loop_stop(self):
|
2019-04-11 08:20:09 -04:00
|
|
|
"""Stop the client loop."""
|
2019-05-07 08:44:29 -04:00
|
|
|
logger.info("Stopping the sync loop")
|
|
|
|
|
2019-06-11 07:54:56 -04:00
|
|
|
if self.task and not self.task.done():
|
|
|
|
self.task.cancel()
|
2019-06-18 10:31:17 -04:00
|
|
|
|
|
|
|
try:
|
|
|
|
await self.task
|
|
|
|
except KeyboardInterrupt:
|
|
|
|
pass
|
|
|
|
|
2019-06-11 07:54:56 -04:00
|
|
|
self.task = None
|
2019-04-04 13:39:44 -04:00
|
|
|
|
2019-06-11 07:54:56 -04:00
|
|
|
if self.history_fetcher_task and not self.history_fetcher_task.done():
|
|
|
|
self.history_fetcher_task.cancel()
|
2019-06-18 10:31:17 -04:00
|
|
|
|
|
|
|
try:
|
|
|
|
await self.history_fetcher_task
|
|
|
|
except KeyboardInterrupt:
|
|
|
|
pass
|
|
|
|
|
2019-06-11 07:54:56 -04:00
|
|
|
self.history_fetcher_task = None
|
2019-04-04 13:39:44 -04:00
|
|
|
|
2019-06-12 09:39:08 -04:00
|
|
|
self.history_fetch_queue = asyncio.Queue()
|
|
|
|
|
2019-06-19 06:37:44 -04:00
|
|
|
def pan_decrypt_event(self, event_dict, room_id=None, ignore_failures=True):
|
2019-05-02 07:27:50 -04:00
|
|
|
# type: (Dict[Any, Any], Optional[str], bool) -> (bool)
|
2019-04-12 08:19:37 -04:00
|
|
|
event = RoomEncryptedEvent.parse_event(event_dict)
|
|
|
|
|
|
|
|
if not isinstance(event, MegolmEvent):
|
2019-06-19 06:37:44 -04:00
|
|
|
logger.warn(
|
|
|
|
"Encrypted event is not a megolm event:"
|
|
|
|
"\n{}".format(pformat(event_dict))
|
|
|
|
)
|
2019-04-30 08:47:21 -04:00
|
|
|
return False
|
2019-04-12 08:19:37 -04:00
|
|
|
|
2019-04-28 15:16:14 -04:00
|
|
|
if not event.room_id:
|
|
|
|
event.room_id = room_id
|
|
|
|
|
2019-04-12 08:19:37 -04:00
|
|
|
try:
|
|
|
|
decrypted_event = self.decrypt_event(event)
|
|
|
|
logger.info("Decrypted event: {}".format(decrypted_event))
|
|
|
|
|
2019-04-30 08:47:21 -04:00
|
|
|
event_dict.update(decrypted_event.source)
|
2019-04-12 08:19:37 -04:00
|
|
|
event_dict["decrypted"] = True
|
|
|
|
event_dict["verified"] = decrypted_event.verified
|
|
|
|
|
2019-04-30 08:47:21 -04:00
|
|
|
return True
|
|
|
|
|
2019-04-12 08:19:37 -04:00
|
|
|
except EncryptionError as error:
|
|
|
|
logger.warn(error)
|
2019-05-02 07:27:50 -04:00
|
|
|
|
|
|
|
if ignore_failures:
|
|
|
|
event_dict.update(self.unable_to_decrypt)
|
|
|
|
else:
|
|
|
|
raise
|
|
|
|
|
2019-04-30 08:47:21 -04:00
|
|
|
return False
|
2019-04-12 08:19:37 -04:00
|
|
|
|
2019-05-07 12:37:32 -04:00
|
|
|
def decrypt_messages_body(self, body, ignore_failures=True):
|
2019-05-14 10:11:39 -04:00
|
|
|
# type: (Dict[Any, Any], bool) -> Dict[Any, Any]
|
2019-04-12 08:19:37 -04:00
|
|
|
"""Go through a messages response and decrypt megolm encrypted events.
|
|
|
|
|
|
|
|
Args:
|
|
|
|
body (Dict[Any, Any]): The dictionary of a Sync response.
|
|
|
|
|
|
|
|
Returns the json response with decrypted events.
|
|
|
|
"""
|
|
|
|
if "chunk" not in body:
|
|
|
|
return body
|
|
|
|
|
|
|
|
logger.info("Decrypting room messages")
|
|
|
|
|
|
|
|
for event in body["chunk"]:
|
|
|
|
if "type" not in event:
|
|
|
|
continue
|
|
|
|
|
|
|
|
if event["type"] != "m.room.encrypted":
|
2019-06-19 06:37:44 -04:00
|
|
|
logger.debug("Event is not encrypted: " "\n{}".format(pformat(event)))
|
2019-04-12 08:19:37 -04:00
|
|
|
continue
|
|
|
|
|
2019-05-07 12:37:32 -04:00
|
|
|
self.pan_decrypt_event(event, ignore_failures=ignore_failures)
|
2019-04-12 08:19:37 -04:00
|
|
|
|
|
|
|
return body
|
|
|
|
|
2019-05-02 07:27:50 -04:00
|
|
|
def decrypt_sync_body(self, body, ignore_failures=True):
|
2019-05-14 10:11:39 -04:00
|
|
|
# type: (Dict[Any, Any], bool) -> Dict[Any, Any]
|
2019-04-03 10:48:13 -04:00
|
|
|
"""Go through a json sync response and decrypt megolm encrypted events.
|
|
|
|
|
|
|
|
Args:
|
|
|
|
body (Dict[Any, Any]): The dictionary of a Sync response.
|
|
|
|
|
2019-04-11 08:23:48 -04:00
|
|
|
Returns the json response with decrypted events.
|
2019-04-03 10:48:13 -04:00
|
|
|
"""
|
2019-04-12 08:19:37 -04:00
|
|
|
logger.info("Decrypting sync")
|
2019-04-03 10:48:13 -04:00
|
|
|
for room_id, room_dict in body["rooms"]["join"].items():
|
2019-04-05 12:35:54 -04:00
|
|
|
try:
|
|
|
|
if not self.rooms[room_id].encrypted:
|
2019-06-19 06:37:44 -04:00
|
|
|
logger.info(
|
|
|
|
"Room {} is not encrypted skipping...".format(
|
|
|
|
self.rooms[room_id].display_name
|
|
|
|
)
|
|
|
|
)
|
2019-04-05 12:35:54 -04:00
|
|
|
continue
|
|
|
|
except KeyError:
|
|
|
|
logger.info("Unknown room {} skipping...".format(room_id))
|
2019-04-03 10:48:13 -04:00
|
|
|
continue
|
|
|
|
|
|
|
|
for event in room_dict["timeline"]["events"]:
|
2019-05-07 12:38:08 -04:00
|
|
|
if "type" not in event:
|
|
|
|
continue
|
|
|
|
|
|
|
|
if event["type"] != "m.room.encrypted":
|
|
|
|
continue
|
|
|
|
|
2019-05-02 07:27:50 -04:00
|
|
|
self.pan_decrypt_event(event, room_id, ignore_failures)
|
2019-04-03 10:48:13 -04:00
|
|
|
|
|
|
|
return body
|
2019-06-06 05:14:40 -04:00
|
|
|
|
|
|
|
async def search(self, search_terms):
|
|
|
|
# type: (Dict[Any, Any]) -> Dict[Any, Any]
|
2019-07-03 11:39:57 -04:00
|
|
|
assert INDEXING_ENABLED
|
|
|
|
|
2019-06-07 05:59:53 -04:00
|
|
|
state_cache = dict()
|
|
|
|
|
2019-06-14 08:53:25 -04:00
|
|
|
async def add_context(event_dict, room_id, event_id, include_state):
|
2019-06-07 05:59:53 -04:00
|
|
|
try:
|
2019-06-14 08:53:25 -04:00
|
|
|
context = await self.room_context(room_id, event_id, limit=0)
|
2019-06-07 05:59:53 -04:00
|
|
|
except ClientConnectionError:
|
|
|
|
return
|
|
|
|
|
|
|
|
if isinstance(context, RoomContextError):
|
|
|
|
return
|
|
|
|
|
|
|
|
if include_state:
|
|
|
|
state_cache[room_id] = [e.source for e in context.state]
|
|
|
|
|
2019-06-14 08:53:25 -04:00
|
|
|
event_dict["context"]["start"] = context.start
|
|
|
|
event_dict["context"]["end"] = context.end
|
2019-06-06 05:14:40 -04:00
|
|
|
|
|
|
|
search_terms = search_terms["search_categories"]["room_events"]
|
|
|
|
|
|
|
|
term = search_terms["search_term"]
|
2019-06-07 05:59:53 -04:00
|
|
|
search_filter = search_terms["filter"]
|
|
|
|
limit = search_filter.get("limit", 10)
|
2019-06-13 06:32:21 -04:00
|
|
|
|
|
|
|
if limit <= 0:
|
2019-06-14 08:53:25 -04:00
|
|
|
raise InvalidLimit("The limit must be strictly greater than 0.")
|
2019-06-13 06:32:21 -04:00
|
|
|
|
2019-06-13 06:32:52 -04:00
|
|
|
rooms = search_filter.get("rooms", [])
|
|
|
|
|
|
|
|
room_id = rooms[0] if len(rooms) == 1 else None
|
|
|
|
|
2019-06-10 09:55:37 -04:00
|
|
|
order_by = search_terms.get("order_by")
|
|
|
|
|
|
|
|
if order_by not in ["rank", "recent"]:
|
|
|
|
raise InvalidOrderByError(f"Invalid order by: {order_by}")
|
2019-06-07 05:59:53 -04:00
|
|
|
|
2019-06-14 08:53:25 -04:00
|
|
|
order_by_recent = order_by == "recent"
|
2019-06-10 10:12:45 -04:00
|
|
|
|
2019-06-07 05:59:53 -04:00
|
|
|
before_limit = 0
|
|
|
|
after_limit = 0
|
|
|
|
include_profile = False
|
|
|
|
|
|
|
|
event_context = search_terms.get("event_context")
|
|
|
|
include_state = search_terms.get("include_state")
|
|
|
|
|
|
|
|
if event_context:
|
|
|
|
before_limit = event_context.get("before_limit", 5)
|
|
|
|
after_limit = event_context.get("before_limit", 5)
|
2019-06-06 05:14:40 -04:00
|
|
|
|
2019-06-14 08:53:25 -04:00
|
|
|
if before_limit < 0 or after_limit < 0:
|
2019-06-19 06:37:44 -04:00
|
|
|
raise InvalidLimit(
|
|
|
|
"Invalid context limit, the limit must be a " "positive number"
|
|
|
|
)
|
2019-06-14 08:53:25 -04:00
|
|
|
|
|
|
|
response_dict = await self.index.search(
|
|
|
|
term,
|
|
|
|
room=room_id,
|
|
|
|
max_results=limit,
|
|
|
|
order_by_recent=order_by_recent,
|
|
|
|
include_profile=include_profile,
|
|
|
|
before_limit=before_limit,
|
2019-06-19 06:37:44 -04:00
|
|
|
after_limit=after_limit,
|
2019-06-14 08:53:25 -04:00
|
|
|
)
|
2019-06-06 05:14:40 -04:00
|
|
|
|
2019-06-17 06:26:38 -04:00
|
|
|
if (event_context or include_state) and self.pan_conf.search_requests:
|
2019-06-17 04:29:23 -04:00
|
|
|
for event_dict in response_dict["results"]:
|
|
|
|
await add_context(
|
|
|
|
event_dict,
|
|
|
|
event_dict["result"]["room_id"],
|
|
|
|
event_dict["result"]["event_id"],
|
2019-06-19 06:37:44 -04:00
|
|
|
include_state,
|
2019-06-17 04:29:23 -04:00
|
|
|
)
|
2019-06-06 05:14:40 -04:00
|
|
|
|
2019-06-10 09:53:19 -04:00
|
|
|
if include_state:
|
2019-06-14 08:53:25 -04:00
|
|
|
response_dict["state"] = state_cache
|
2019-06-10 09:53:19 -04:00
|
|
|
|
2019-06-19 06:37:44 -04:00
|
|
|
return {"search_categories": {"room_events": response_dict}}
|