2018-02-13 11:36:00 +01:00
|
|
|
# -*- coding: utf-8 -*-
|
|
|
|
|
|
|
|
# Copyright © 2018 Damir Jelić <poljar@termina.org.uk>
|
|
|
|
#
|
|
|
|
# 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.
|
|
|
|
|
|
|
|
from __future__ import unicode_literals
|
|
|
|
from builtins import str
|
|
|
|
|
2018-02-13 15:50:30 +01:00
|
|
|
import time
|
2018-02-16 15:15:56 +01:00
|
|
|
from functools import partial
|
2018-02-13 15:50:30 +01:00
|
|
|
|
2018-02-13 11:36:00 +01:00
|
|
|
from matrix.globals import W, OPTIONS
|
2018-02-22 21:50:13 +01:00
|
|
|
from matrix.utils import (color_for_tags, tags_for_message, sanitize_id,
|
|
|
|
sanitize_token, sanitize_text)
|
|
|
|
from matrix.rooms import (matrix_create_room_buffer, RoomInfo, RoomMessageEvent,
|
|
|
|
RoomRedactedMessageEvent)
|
2018-02-16 15:15:56 +01:00
|
|
|
|
|
|
|
|
2018-02-13 11:36:00 +01:00
|
|
|
class MatrixEvent():
|
2018-02-22 13:39:37 +01:00
|
|
|
|
2018-02-13 11:36:00 +01:00
|
|
|
def __init__(self, server):
|
|
|
|
self.server = server
|
|
|
|
|
|
|
|
def execute(self):
|
|
|
|
pass
|
|
|
|
|
|
|
|
|
2018-02-13 13:06:30 +01:00
|
|
|
class MatrixErrorEvent(MatrixEvent):
|
2018-02-22 13:39:37 +01:00
|
|
|
|
2018-02-13 13:06:30 +01:00
|
|
|
def __init__(self, server, error_message, fatal=False):
|
|
|
|
self.error_message = error_message
|
|
|
|
self.fatal = fatal
|
|
|
|
MatrixEvent.__init__(self, server)
|
|
|
|
|
|
|
|
def execute(self):
|
|
|
|
message = ("{prefix}matrix: {error}").format(
|
2018-02-22 13:39:37 +01:00
|
|
|
prefix=W.prefix("error"), error=self.error_message)
|
2018-02-13 13:06:30 +01:00
|
|
|
|
|
|
|
W.prnt(self.server.server_buffer, message)
|
|
|
|
|
|
|
|
if self.fatal:
|
|
|
|
self.server.disconnect(reconnect=False)
|
|
|
|
|
2018-02-14 11:11:54 +01:00
|
|
|
@classmethod
|
|
|
|
def from_dict(cls, server, error_prefix, fatal, parsed_dict):
|
|
|
|
try:
|
2018-02-14 14:53:07 +01:00
|
|
|
message = "{prefix}: {error}".format(
|
2018-02-22 14:06:25 +01:00
|
|
|
prefix=error_prefix, error=sanitize_text(parsed_dict["error"]))
|
2018-02-22 13:39:37 +01:00
|
|
|
return cls(server, message, fatal=fatal)
|
2018-02-14 11:11:54 +01:00
|
|
|
except KeyError:
|
|
|
|
return cls(
|
2018-02-22 13:39:37 +01:00
|
|
|
server, ("{prefix}: Invalid JSON response "
|
|
|
|
"from server.").format(prefix=error_prefix),
|
2018-02-14 11:11:54 +01:00
|
|
|
fatal=fatal)
|
|
|
|
|
2018-02-13 13:06:30 +01:00
|
|
|
|
|
|
|
class MatrixLoginEvent(MatrixEvent):
|
2018-02-22 13:39:37 +01:00
|
|
|
|
2018-02-13 11:36:00 +01:00
|
|
|
def __init__(self, server, user_id, access_token):
|
|
|
|
self.user_id = user_id
|
|
|
|
self.access_token = access_token
|
|
|
|
MatrixEvent.__init__(self, server)
|
|
|
|
|
|
|
|
def execute(self):
|
|
|
|
self.server.access_token = self.access_token
|
|
|
|
self.server.user_id = self.user_id
|
|
|
|
self.server.client.access_token = self.access_token
|
|
|
|
|
|
|
|
self.server.sync()
|
2018-02-13 13:06:30 +01:00
|
|
|
|
|
|
|
@classmethod
|
|
|
|
def from_dict(cls, server, parsed_dict):
|
|
|
|
try:
|
2018-02-22 13:39:37 +01:00
|
|
|
return cls(server, sanitize_id(parsed_dict["user_id"]),
|
|
|
|
sanitize_token(parsed_dict["access_token"]))
|
2018-02-16 15:15:56 +01:00
|
|
|
except (KeyError, TypeError, ValueError):
|
2018-02-22 13:39:37 +01:00
|
|
|
return MatrixErrorEvent.from_dict(server, "Error logging in", True,
|
|
|
|
parsed_dict)
|
2018-02-13 15:50:30 +01:00
|
|
|
|
|
|
|
|
|
|
|
class MatrixSendEvent(MatrixEvent):
|
2018-02-22 13:39:37 +01:00
|
|
|
|
2018-02-13 15:50:30 +01:00
|
|
|
def __init__(self, server, room_id, event_id, message):
|
|
|
|
self.room_id = room_id
|
|
|
|
self.event_id = event_id
|
|
|
|
self.message = message
|
|
|
|
MatrixEvent.__init__(self, server)
|
|
|
|
|
|
|
|
def execute(self):
|
|
|
|
room_id = self.room_id
|
|
|
|
author = self.server.user
|
|
|
|
event_id = self.event_id
|
|
|
|
weechat_message = self.message.to_weechat()
|
|
|
|
|
|
|
|
date = int(time.time())
|
|
|
|
|
|
|
|
# This message will be part of the next sync, we already printed it out
|
|
|
|
# so ignore it in the sync.
|
|
|
|
self.server.ignore_event_list.append(event_id)
|
|
|
|
|
|
|
|
tag = ("notify_none,no_highlight,self_msg,log1,nick_{a},"
|
|
|
|
"prefix_nick_{color},matrix_id_{event_id},"
|
|
|
|
"matrix_message").format(
|
|
|
|
a=author,
|
|
|
|
color=color_for_tags("weechat.color.chat_nick_self"),
|
|
|
|
event_id=event_id)
|
|
|
|
|
|
|
|
message = "{author}\t{msg}".format(author=author, msg=weechat_message)
|
|
|
|
|
|
|
|
buf = self.server.buffers[room_id]
|
|
|
|
W.prnt_date_tags(buf, date, tag, message)
|
|
|
|
|
|
|
|
@classmethod
|
|
|
|
def from_dict(cls, server, room_id, message, parsed_dict):
|
|
|
|
try:
|
2018-02-22 13:39:37 +01:00
|
|
|
return cls(server, room_id, sanitize_id(parsed_dict["event_id"]),
|
|
|
|
message)
|
2018-02-16 15:15:56 +01:00
|
|
|
except (KeyError, TypeError, ValueError):
|
2018-02-22 13:39:37 +01:00
|
|
|
return MatrixErrorEvent.from_dict(server, "Error sending message",
|
|
|
|
False, parsed_dict)
|
2018-02-14 11:51:57 +01:00
|
|
|
|
|
|
|
|
|
|
|
class MatrixTopicEvent(MatrixEvent):
|
2018-02-22 13:39:37 +01:00
|
|
|
|
2018-02-14 11:51:57 +01:00
|
|
|
def __init__(self, server, room_id, event_id, topic):
|
|
|
|
self.room_id = room_id
|
|
|
|
self.topic = topic
|
|
|
|
self.event_id = event_id
|
|
|
|
MatrixEvent.__init__(self, server)
|
|
|
|
|
|
|
|
@classmethod
|
|
|
|
def from_dict(cls, server, room_id, topic, parsed_dict):
|
|
|
|
try:
|
2018-02-22 13:39:37 +01:00
|
|
|
return cls(server, room_id, sanitize_id(parsed_dict["event_id"]),
|
|
|
|
topic)
|
2018-02-16 15:15:56 +01:00
|
|
|
except (KeyError, TypeError, ValueError):
|
2018-02-22 13:39:37 +01:00
|
|
|
return MatrixErrorEvent.from_dict(server, "Error setting topic",
|
|
|
|
False, parsed_dict)
|
2018-02-14 13:00:18 +01:00
|
|
|
|
|
|
|
|
|
|
|
class MatrixRedactEvent(MatrixEvent):
|
2018-02-22 13:39:37 +01:00
|
|
|
|
2018-02-14 13:00:18 +01:00
|
|
|
def __init__(self, server, room_id, event_id, reason):
|
|
|
|
self.room_id = room_id
|
|
|
|
self.topic = reason
|
|
|
|
self.event_id = event_id
|
|
|
|
MatrixEvent.__init__(self, server)
|
|
|
|
|
|
|
|
@classmethod
|
|
|
|
def from_dict(cls, server, room_id, reason, parsed_dict):
|
|
|
|
try:
|
2018-02-22 13:39:37 +01:00
|
|
|
return cls(server, room_id, sanitize_id(parsed_dict["event_id"]),
|
|
|
|
reason)
|
2018-02-16 15:15:56 +01:00
|
|
|
except (KeyError, TypeError, ValueError):
|
2018-02-22 13:39:37 +01:00
|
|
|
return MatrixErrorEvent.from_dict(server, "Error redacting message",
|
|
|
|
False, parsed_dict)
|
2018-02-14 13:48:00 +01:00
|
|
|
|
|
|
|
|
|
|
|
class MatrixJoinEvent(MatrixEvent):
|
2018-02-22 13:39:37 +01:00
|
|
|
|
2018-02-14 14:21:56 +01:00
|
|
|
def __init__(self, server, room, room_id):
|
|
|
|
self.room = room
|
2018-02-14 13:48:00 +01:00
|
|
|
self.room_id = room_id
|
|
|
|
MatrixEvent.__init__(self, server)
|
|
|
|
|
|
|
|
@classmethod
|
|
|
|
def from_dict(cls, server, room_id, parsed_dict):
|
|
|
|
try:
|
|
|
|
return cls(
|
|
|
|
server,
|
|
|
|
room_id,
|
2018-02-14 14:21:56 +01:00
|
|
|
sanitize_id(parsed_dict["room_id"]),
|
2018-02-14 13:48:00 +01:00
|
|
|
)
|
2018-02-16 15:15:56 +01:00
|
|
|
except (KeyError, TypeError, ValueError):
|
2018-02-22 13:39:37 +01:00
|
|
|
return MatrixErrorEvent.from_dict(server, "Error joining room",
|
|
|
|
False, parsed_dict)
|
2018-02-14 14:21:56 +01:00
|
|
|
|
|
|
|
|
|
|
|
class MatrixPartEvent(MatrixEvent):
|
2018-02-22 13:39:37 +01:00
|
|
|
|
2018-02-14 14:21:56 +01:00
|
|
|
def __init__(self, server, room_id):
|
|
|
|
self.room_id = room_id
|
|
|
|
MatrixEvent.__init__(self, server)
|
|
|
|
|
|
|
|
@classmethod
|
|
|
|
def from_dict(cls, server, room_id, parsed_dict):
|
|
|
|
try:
|
2018-02-14 14:22:20 +01:00
|
|
|
if parsed_dict == {}:
|
2018-02-22 13:39:37 +01:00
|
|
|
return cls(server, room_id)
|
2018-02-14 14:21:56 +01:00
|
|
|
|
|
|
|
raise KeyError
|
|
|
|
except KeyError:
|
2018-02-22 13:39:37 +01:00
|
|
|
return MatrixErrorEvent.from_dict(server, "Error leaving room",
|
|
|
|
False, parsed_dict)
|
2018-02-14 14:52:21 +01:00
|
|
|
|
|
|
|
|
|
|
|
class MatrixInviteEvent(MatrixEvent):
|
2018-02-22 13:39:37 +01:00
|
|
|
|
2018-02-14 14:52:21 +01:00
|
|
|
def __init__(self, server, room_id, user_id):
|
|
|
|
self.room_id = room_id
|
|
|
|
self.user_id = user_id
|
|
|
|
MatrixEvent.__init__(self, server)
|
|
|
|
|
|
|
|
@classmethod
|
|
|
|
def from_dict(cls, server, room_id, user_id, parsed_dict):
|
|
|
|
try:
|
|
|
|
if parsed_dict == {}:
|
2018-02-22 13:39:37 +01:00
|
|
|
return cls(server, room_id, user_id)
|
2018-02-14 14:52:21 +01:00
|
|
|
|
|
|
|
raise KeyError
|
|
|
|
except KeyError:
|
2018-02-22 13:39:37 +01:00
|
|
|
return MatrixErrorEvent.from_dict(server, "Error inviting user",
|
|
|
|
False, parsed_dict)
|
2018-02-16 15:15:56 +01:00
|
|
|
|
|
|
|
|
|
|
|
class MatrixBacklogEvent(MatrixEvent):
|
2018-02-22 13:39:37 +01:00
|
|
|
|
2018-02-22 21:38:08 +01:00
|
|
|
def __init__(self, server, room_id, end_token, events):
|
2018-02-16 15:15:56 +01:00
|
|
|
self.room_id = room_id
|
|
|
|
self.end_token = end_token
|
2018-02-22 21:38:08 +01:00
|
|
|
self.events = events
|
2018-02-16 15:15:56 +01:00
|
|
|
MatrixEvent.__init__(self, server)
|
|
|
|
|
|
|
|
@staticmethod
|
2018-02-22 21:38:08 +01:00
|
|
|
def _room_event_from_dict(room_id, event_dict):
|
|
|
|
if room_id != event_dict["room_id"]:
|
2018-02-16 15:15:56 +01:00
|
|
|
raise ValueError
|
|
|
|
|
2018-02-22 21:38:08 +01:00
|
|
|
if "redacted_by" in event_dict["unsigned"]:
|
|
|
|
return RoomRedactedMessageEvent.from_dict(event_dict)
|
2018-02-16 15:15:56 +01:00
|
|
|
|
2018-02-22 21:38:08 +01:00
|
|
|
return RoomMessageEvent.from_dict(event_dict)
|
2018-02-16 15:15:56 +01:00
|
|
|
|
|
|
|
@classmethod
|
|
|
|
def from_dict(cls, server, room_id, parsed_dict):
|
|
|
|
try:
|
|
|
|
end_token = sanitize_id(parsed_dict["end"])
|
|
|
|
|
2018-02-22 14:10:47 +01:00
|
|
|
if not parsed_dict["chunk"]:
|
|
|
|
return cls(server, room_id, end_token, [])
|
|
|
|
|
2018-02-22 21:38:08 +01:00
|
|
|
event_func = partial(MatrixBacklogEvent._room_event_from_dict,
|
|
|
|
room_id)
|
2018-02-16 15:15:56 +01:00
|
|
|
|
2018-02-22 13:39:37 +01:00
|
|
|
message_events = list(
|
|
|
|
filter(lambda event: event["type"] == "m.room.message",
|
|
|
|
parsed_dict["chunk"]))
|
2018-02-16 15:15:56 +01:00
|
|
|
|
2018-02-22 21:38:08 +01:00
|
|
|
events = [event_func(m) for m in message_events]
|
2018-02-16 15:15:56 +01:00
|
|
|
|
2018-02-22 21:38:08 +01:00
|
|
|
return cls(server, room_id, end_token, events)
|
2018-02-16 15:15:56 +01:00
|
|
|
except (KeyError, ValueError, TypeError):
|
2018-02-22 13:39:37 +01:00
|
|
|
return MatrixErrorEvent.from_dict(server, "Error fetching backlog",
|
|
|
|
False, parsed_dict)
|
2018-02-16 15:15:56 +01:00
|
|
|
|
|
|
|
def execute(self):
|
|
|
|
room = self.server.rooms[self.room_id]
|
|
|
|
buf = self.server.buffers[self.room_id]
|
|
|
|
tags = tags_for_message("backlog")
|
|
|
|
|
2018-02-22 21:38:08 +01:00
|
|
|
for event in self.events:
|
2018-02-23 15:21:13 +01:00
|
|
|
event.execute(room, buf, list(tags))
|
2018-02-16 15:15:56 +01:00
|
|
|
|
|
|
|
room.prev_batch = self.end_token
|
2018-02-22 14:26:54 +01:00
|
|
|
room.backlog_pending = False
|
2018-02-22 15:02:42 +01:00
|
|
|
W.bar_item_update("buffer_modes")
|
2018-02-16 15:15:56 +01:00
|
|
|
|
|
|
|
|
2018-02-22 13:39:37 +01:00
|
|
|
class MatrixSyncEvent(MatrixEvent):
|
|
|
|
|
|
|
|
def __init__(self, server, next_batch, room_infos, invited_infos):
|
|
|
|
self.next_batch = next_batch
|
|
|
|
self.joined_room_infos = room_infos
|
|
|
|
self.invited_room_infos = invited_infos
|
|
|
|
|
|
|
|
MatrixEvent.__init__(self, server)
|
|
|
|
|
|
|
|
@staticmethod
|
|
|
|
def _infos_from_dict(parsed_dict):
|
|
|
|
join_infos = []
|
|
|
|
invite_infos = []
|
|
|
|
|
|
|
|
for room_id, room_dict in parsed_dict['join'].items():
|
|
|
|
if not room_id:
|
|
|
|
continue
|
|
|
|
|
|
|
|
join_infos.append(RoomInfo.from_dict(room_id, room_dict))
|
|
|
|
|
|
|
|
return (join_infos, invite_infos)
|
|
|
|
|
|
|
|
@classmethod
|
|
|
|
def from_dict(cls, server, parsed_dict):
|
|
|
|
try:
|
2018-02-22 14:06:25 +01:00
|
|
|
next_batch = sanitize_id(parsed_dict["next_batch"])
|
2018-02-22 13:39:37 +01:00
|
|
|
room_info_dict = parsed_dict["rooms"]
|
|
|
|
|
|
|
|
join_infos, invite_infos = MatrixSyncEvent._infos_from_dict(
|
|
|
|
room_info_dict)
|
|
|
|
|
|
|
|
return cls(server, next_batch, join_infos, invite_infos)
|
|
|
|
except (KeyError, ValueError, TypeError):
|
|
|
|
return MatrixErrorEvent.from_dict(server, "Error syncing", False,
|
|
|
|
parsed_dict)
|
|
|
|
|
|
|
|
def _execute_joined_info(self, info):
|
|
|
|
server = self.server
|
|
|
|
|
|
|
|
if info.room_id not in server.buffers:
|
|
|
|
matrix_create_room_buffer(server, info.room_id)
|
|
|
|
|
|
|
|
room = server.rooms[info.room_id]
|
|
|
|
buf = server.buffers[info.room_id]
|
|
|
|
|
|
|
|
if not room.prev_batch:
|
|
|
|
room.prev_batch = info.prev_batch
|
|
|
|
|
|
|
|
tags = tags_for_message("message")
|
2018-02-23 18:52:10 +01:00
|
|
|
|
|
|
|
for event in info.membership_events:
|
|
|
|
event.execute(server, room, buf, list(tags))
|
|
|
|
|
2018-02-23 15:21:13 +01:00
|
|
|
for event in info.events:
|
|
|
|
event.execute(server, room, buf, list(tags))
|
2018-02-22 13:39:37 +01:00
|
|
|
|
|
|
|
def execute(self):
|
|
|
|
server = self.server
|
|
|
|
|
|
|
|
# we got the same batch again, nothing to do
|
|
|
|
if self.next_batch == server.next_batch:
|
|
|
|
server.sync()
|
|
|
|
return
|
|
|
|
|
|
|
|
map(self._execute_joined_info, self.joined_room_infos)
|
|
|
|
|
|
|
|
server.next_batch = self.next_batch
|
|
|
|
server.sync()
|