Skip to content
Snippets Groups Projects
portal.py 37.4 KiB
Newer Older
  • Learn to ignore specific revisions
  • Tulir Asokan's avatar
    Tulir Asokan committed
    # mautrix-facebook - A Matrix-Facebook Messenger puppeting bridge
    
    # Copyright (C) 2020 Tulir Asokan
    
    Tulir Asokan's avatar
    Tulir Asokan committed
    #
    # This program is free software: you can redistribute it and/or modify
    # it under the terms of the GNU Affero General Public License as published by
    # the Free Software Foundation, either version 3 of the License, or
    # (at your option) any later version.
    #
    # This program is distributed in the hope that it will be useful,
    # but WITHOUT ANY WARRANTY; without even the implied warranty of
    # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
    # GNU Affero General Public License for more details.
    #
    # You should have received a copy of the GNU Affero General Public License
    # along with this program.  If not, see <https://www.gnu.org/licenses/>.
    
    from typing import Dict, Deque, Optional, Tuple, Union, Set, Iterator, TYPE_CHECKING
    
    from collections import deque
    
    Tulir Asokan's avatar
    Tulir Asokan committed
    import asyncio
    
    
    from yarl import URL
    import aiohttp
    
    Tulir Asokan's avatar
    Tulir Asokan committed
    import fbchat
    
    from mautrix.types import (RoomID, EventType, ContentURI, MessageEventContent, EventID,
    
                               ImageInfo, MessageType, LocationMessageEventContent, LocationInfo,
    
                               ThumbnailInfo, FileInfo, AudioInfo, Format, RelatesTo, RelationType,
    
                               TextMessageEventContent, MediaMessageEventContent, Membership,
    
                               EncryptedFile)
    from mautrix.appservice import IntentAPI
    
    from mautrix.errors import MForbidden, IntentError, MatrixError
    
    from mautrix.bridge import BasePortal
    
    Tulir Asokan's avatar
    Tulir Asokan committed
    
    
    from .formatter import facebook_to_matrix, matrix_to_facebook
    
    Tulir Asokan's avatar
    Tulir Asokan committed
    from .config import Config
    
    from .db import (Portal as DBPortal, Message as DBMessage, Reaction as DBReaction,
    
    Tulir Asokan's avatar
    Tulir Asokan committed
                     UserPortal as DBUserPortal, ThreadType)
    
    Tulir Asokan's avatar
    Tulir Asokan committed
    from . import puppet as p, user as u
    
    if TYPE_CHECKING:
        from .context import Context
    
        from .matrix import MatrixHandler
    
    try:
        from nio.crypto import decrypt_attachment, encrypt_attachment
    except ImportError:
        decrypt_attachment = encrypt_attachment = None
    
    Tulir Asokan's avatar
    Tulir Asokan committed
    
    config: Config
    
    
    Tulir Asokan's avatar
    Tulir Asokan committed
    ThreadClass = Union[fbchat.UserData, fbchat.GroupData, fbchat.PageData]
    AttachmentClass = Union[fbchat.AudioAttachment, fbchat.VideoAttachment, fbchat.FileAttachment,
                            fbchat.ImageAttachment, fbchat.LocationAttachment,
                            fbchat.ShareAttachment]
    
    Tulir Asokan's avatar
    Tulir Asokan committed
    
    
    
    class FakeLock:
        async def __aenter__(self) -> None:
            pass
    
        async def __aexit__(self, exc_type, exc, tb) -> None:
            pass
    
    
    
    class Portal(BasePortal):
    
        invite_own_puppet_to_pm: bool = False
    
    Tulir Asokan's avatar
    Tulir Asokan committed
        by_mxid: Dict[RoomID, 'Portal'] = {}
    
        by_fbid: Dict[Tuple[str, str], 'Portal'] = {}
    
        matrix: 'MatrixHandler'
    
    Tulir Asokan's avatar
    Tulir Asokan committed
    
        fbid: str
    
        fb_receiver: str
    
    Tulir Asokan's avatar
    Tulir Asokan committed
        fb_type: ThreadType
        mxid: Optional[RoomID]
    
        encrypted: bool
    
    Tulir Asokan's avatar
    Tulir Asokan committed
    
        name: str
    
    Tulir Asokan's avatar
    Tulir Asokan committed
    
    
        _db_instance: DBPortal
    
    Tulir Asokan's avatar
    Tulir Asokan committed
        _main_intent: Optional[IntentAPI]
    
        _create_room_lock: asyncio.Lock
    
        _last_bridged_mxid: Optional[EventID]
    
        _dedup: Deque[str]
    
        _avatar_uri: Optional[ContentURI]
    
        _send_locks: Dict[str, asyncio.Lock]
    
        _noop_lock: FakeLock = FakeLock()
    
        _typing: Set['u.User']
    
    Tulir Asokan's avatar
    Tulir Asokan committed
    
    
        def __init__(self, fbid: str, fb_receiver: str, fb_type: ThreadType,
    
                     mxid: Optional[RoomID] = None, encrypted: bool = False, name: str = "",
                     photo_id: str = "", db_instance: Optional[DBPortal] = None) -> None:
    
    Tulir Asokan's avatar
    Tulir Asokan committed
            self.fbid = fbid
    
            self.fb_receiver = fb_receiver
    
    Tulir Asokan's avatar
    Tulir Asokan committed
            self.fb_type = fb_type
            self.mxid = mxid
    
            self.encrypted = encrypted
    
            self.photo_id = photo_id
    
    
            self._db_instance = db_instance
    
    Tulir Asokan's avatar
    Tulir Asokan committed
    
            self._main_intent = None
    
            self._create_room_lock = asyncio.Lock()
    
            self._last_bridged_mxid = None
    
            self._dedup = deque(maxlen=100)
            self._avatar_uri = None
    
            self._send_locks = {}
    
            self.log = self.log.getChild(self.fbid_log)
    
            self.by_fbid[self.fbid_full] = self
            if self.mxid:
                self.by_mxid[self.mxid] = self
    
    Tulir Asokan's avatar
    Tulir Asokan committed
    
    
        # region DB conversion
    
        @property
        def db_instance(self) -> DBPortal:
            if not self._db_instance:
                self._db_instance = DBPortal(fbid=self.fbid, fb_receiver=self.fb_receiver,
                                             fb_type=self.fb_type, mxid=self.mxid, name=self.name,
    
                                             encrypted=self.encrypted, photo_id=self.photo_id)
    
            return self._db_instance
    
    Tulir Asokan's avatar
    Tulir Asokan committed
    
        @classmethod
    
        def from_db(cls, db_portal: DBPortal) -> 'Portal':
            return Portal(fbid=db_portal.fbid, fb_receiver=db_portal.fb_receiver,
                          fb_type=db_portal.fb_type, mxid=db_portal.mxid, name=db_portal.name,
    
                          encrypted=db_portal.encrypted, photo_id=db_portal.photo_id,
                          db_instance=db_portal)
    
    
        def save(self) -> None:
    
            self.db_instance.edit(mxid=self.mxid, name=self.name, photo_id=self.photo_id,
                                  encrypted=self.encrypted)
    
        def delete(self) -> None:
            self.by_fbid.pop(self.fbid_full, None)
            self.by_mxid.pop(self.mxid, None)
            if self._db_instance:
                self._db_instance.delete()
    
    
        # endregion
        # region Properties
    
    Tulir Asokan's avatar
    Tulir Asokan committed
    
    
        @property
        def fbid_full(self) -> Tuple[str, str]:
            return self.fbid, self.fb_receiver
    
        @property
        def fbid_log(self) -> str:
            if self.is_direct:
                return f"{self.fbid}<->{self.fb_receiver}"
            return self.fbid
    
    
    Tulir Asokan's avatar
    Tulir Asokan committed
        def thread_for(self, user: 'u.User') -> Union[fbchat.User, fbchat.Group, fbchat.Page]:
            if self.fb_type == ThreadType.USER:
                return fbchat.User(session=user.session, id=self.fbid)
            elif self.fb_type == ThreadType.GROUP:
                return fbchat.Group(session=user.session, id=self.fbid)
            elif self.fb_type == ThreadType.PAGE:
                return fbchat.Page(session=user.session, id=self.fbid)
            else:
                raise ValueError("Unsupported thread type")
    
    
    Tulir Asokan's avatar
    Tulir Asokan committed
        @property
        def is_direct(self) -> bool:
            return self.fb_type == ThreadType.USER
    
        @property
        def main_intent(self) -> IntentAPI:
            if not self._main_intent:
    
                self._main_intent = (p.Puppet.get_by_fbid(self.fbid).default_mxid_intent
    
                                     if self.is_direct else self.az.intent)
    
    Tulir Asokan's avatar
    Tulir Asokan committed
    
            return self._main_intent
    
    
        # endregion
        # region Chat info updating
    
    
    Tulir Asokan's avatar
    Tulir Asokan committed
        async def update_info(self, source: Optional['u.User'] = None,
                              info: Optional[ThreadClass] = None) -> ThreadClass:
            if not info:
    
    Tulir Asokan's avatar
    Tulir Asokan committed
                info = await source.client.fetch_thread_info(self.fbid).__anext__()
                # TODO validate that we got some sane info?
    
            changed = any(await asyncio.gather(self._update_name(info.name),
                                               self._update_photo(info.photo),
                                               self._update_participants(source, info),
                                               loop=self.loop))
            if changed:
                self.save()
    
    Tulir Asokan's avatar
    Tulir Asokan committed
            return info
    
    
    Tulir Asokan's avatar
    Tulir Asokan committed
        def _get_photo_id(photo: Optional[Union[fbchat.Image, str]]) -> Optional[str]:
            if not photo:
    
    Tulir Asokan's avatar
    Tulir Asokan committed
            elif isinstance(photo, fbchat.Image):
                photo = photo.url
            path = URL(photo).path
    
            return path[path.rfind("/") + 1:]
    
        @staticmethod
    
        async def _reupload_fb_file(url: str, intent: IntentAPI, filename: Optional[str] = None,
                                    encrypt: bool = False
                                    ) -> Tuple[ContentURI, str, int, Optional[EncryptedFile]]:
    
            if not url:
                raise ValueError('URL not provided')
    
            async with aiohttp.ClientSession() as session:
                resp = await session.get(url)
                data = await resp.read()
    
            mime = magic.from_buffer(data, mime=True)
    
            upload_mime_type = mime
            decryption_info = None
            if encrypt and encrypt_attachment:
                data, decryption_info_dict = encrypt_attachment(data)
                decryption_info = EncryptedFile.deserialize(decryption_info_dict)
                upload_mime_type = "application/octet-stream"
            url = await intent.upload_media(data, mime_type=upload_mime_type, filename=filename)
            if decryption_info:
                decryption_info.url = url
            return url, mime, len(data), decryption_info
    
        async def _update_name(self, name: str) -> bool:
    
    Tulir Asokan's avatar
    Tulir Asokan committed
            if self.name != name:
                self.name = name
    
                if self.mxid and (self.encrypted or not self.is_direct):
    
    Tulir Asokan's avatar
    Tulir Asokan committed
                    await self.main_intent.set_room_name(self.mxid, self.name)
    
                return True
            return False
    
    Tulir Asokan's avatar
    Tulir Asokan committed
    
    
    Tulir Asokan's avatar
    Tulir Asokan committed
        async def _update_photo(self, photo: fbchat.Image) -> bool:
    
            if self.is_direct and not self.encrypted:
    
    Tulir Asokan's avatar
    Tulir Asokan committed
            photo_id = self._get_photo_id(photo)
    
            if self.photo_id != photo_id:
    
                self.photo_id = photo_id
    
    Tulir Asokan's avatar
    Tulir Asokan committed
                if photo:
                    self._avatar_uri, *_ = await self._reupload_fb_file(photo.url, self.main_intent)
    
                else:
                    self._avatar_uri = ContentURI("")
                if self.mxid:
                    await self.main_intent.set_room_avatar(self.mxid, self._avatar_uri)
                return True
            return False
    
    Tulir Asokan's avatar
    Tulir Asokan committed
    
        async def _update_participants(self, source: 'u.User', info: ThreadClass) -> None:
    
    Tulir Asokan's avatar
    Tulir Asokan committed
                await p.Puppet.get_by_fbid(info.id).update_info(source=source, info=info)
    
    Tulir Asokan's avatar
    Tulir Asokan committed
                return
    
            elif not self.mxid:
                return
    
    Tulir Asokan's avatar
    Tulir Asokan committed
            # TODO maybe change this back to happen simultaneously
            async for user in source.client.fetch_thread_info([user.id for user in info.participants]):
                if not isinstance(user, fbchat.UserData):
                    # TODO log
                    continue
                puppet = p.Puppet.get_by_fbid(user.id)
                await puppet.update_info(source, user)
                await puppet.intent_for(self).ensure_joined(self.mxid)
    
    Tulir Asokan's avatar
    Tulir Asokan committed
    
    
        # endregion
        # region Matrix room creation
    
    
        async def _update_matrix_room(self, source: 'u.User',
                                      info: Optional[ThreadClass] = None) -> None:
    
            await self.main_intent.invite_user(self.mxid, source.mxid, check_cache=True)
    
            puppet = p.Puppet.get_by_custom_mxid(source.mxid)
            if puppet:
                await puppet.intent.ensure_joined(self.mxid)
    
            await self.update_info(source, info)
    
            up = DBUserPortal.get(source.fbid, self.fbid, self.fb_receiver)
            if not up:
                in_community = await source._community_helper.add_room(source._community_id, self.mxid)
                DBUserPortal(user=source.fbid, portal=self.fbid, portal_receiver=self.fb_receiver,
                             in_community=in_community).insert()
            elif not up.in_community:
                in_community = await source._community_helper.add_room(source._community_id, self.mxid)
                up.edit(in_community=in_community)
    
    
        async def create_matrix_room(self, source: 'u.User', info: Optional[ThreadClass] = None
    
                                     ) -> Optional[RoomID]:
    
                try:
                    await self._update_matrix_room(source, info)
                except Exception:
                    self.log.exception("Failed to update portal")
    
                return self.mxid
            async with self._create_room_lock:
    
                try:
                    return await self._create_matrix_room(source, info)
                except Exception:
                    self.log.exception("Failed to create portal")
                    return None
    
    
        async def _create_matrix_room(self, source: 'u.User', info: Optional[ThreadClass] = None
                                      ) -> RoomID:
    
    Tulir Asokan's avatar
    Tulir Asokan committed
            if self.mxid:
    
                await self._update_matrix_room(source, info)
    
    Tulir Asokan's avatar
    Tulir Asokan committed
                return self.mxid
    
            info = await self.update_info(source=source, info=info)
            self.log.debug(f"Creating Matrix room")
            name: Optional[str] = None
    
            initial_state = [{
                "type": "m.bridge",
                "state_key": f"net.maunium.facebook://facebook/{self.fbid}",
                "content": {
                    "bridgebot": self.az.bot_mxid,
                    "creator": self.main_intent.mxid,
                    "protocol": {
                        "id": "facebook",
                        "displayname": "Facebook Messenger",
                        "avatar_url": config["appservice.bot_avatar"],
                    },
                    "channel": {
                        "id": self.fbid
                    }
                }
            }]
    
            invites = [source.mxid]
            if config["bridge.encryption.default"] and self.matrix.e2ee:
                self.encrypted = True
                initial_state.append({
                    "type": "m.room.encryption",
                    "content": {"algorithm": "m.megolm.v1.aes-sha2"},
                })
                if self.is_direct:
                    invites.append(self.az.bot_mxid)
            if self.encrypted or not self.is_direct:
    
    Tulir Asokan's avatar
    Tulir Asokan committed
                name = self.name
                initial_state.append({"type": str(EventType.ROOM_AVATAR),
    
                                      "content": {"avatar_url": self._avatar_uri}})
    
            if config["appservice.community_id"]:
                initial_state.append({
                    "type": "m.room.related_groups",
                    "content": {"groups": [config["appservice.community_id"]]},
                })
    
    Tulir Asokan's avatar
    Tulir Asokan committed
            self.mxid = await self.main_intent.create_room(name=name, is_direct=self.is_direct,
                                                           initial_state=initial_state,
    
                                                           invitees=invites)
            if not self.mxid:
                raise Exception("Failed to create room: no mxid returned")
    
            if self.encrypted and self.matrix.e2ee:
                members = [self.main_intent.mxid]
                if self.is_direct:
                    # This isn't very accurate, but let's do it anyway
                    members += [source.mxid]
                    try:
                        await self.az.intent.join_room_by_id(self.mxid)
                        members += [self.az.intent.mxid]
                    except Exception:
                        self.log.warning(f"Failed to add bridge bot to new private chat {self.mxid}")
                await self.matrix.e2ee.add_room(self.mxid, members=members, encrypted=True)
    
    
    Tulir Asokan's avatar
    Tulir Asokan committed
            self.log.debug(f"Matrix room created: {self.mxid}")
            self.by_mxid[self.mxid] = self
    
            if not self.is_direct:
                await self._update_participants(source, info)
    
            else:
                puppet = p.Puppet.get_by_custom_mxid(source.mxid)
                if puppet:
                    await puppet.intent.ensure_joined(self.mxid)
    
    
            in_community = await source._community_helper.add_room(source._community_id, self.mxid)
            DBUserPortal(user=source.fbid, portal=self.fbid, portal_receiver=self.fb_receiver,
                         in_community=in_community).upsert()
    
    
            return self.mxid
    
    Tulir Asokan's avatar
    Tulir Asokan committed
    
    
        # endregion
        # region Matrix room cleanup
    
        @staticmethod
        async def cleanup_room(intent: IntentAPI, room_id: RoomID, message: str = "Portal deleted",
                               puppets_only: bool = False) -> None:
            try:
                members = await intent.get_room_members(room_id)
            except MatrixError:
                members = []
            for user_id in members:
                puppet = p.Puppet.get_by_mxid(user_id, create=False)
                if user_id != intent.mxid and (not puppets_only or puppet):
                    try:
                        if puppet:
                            await puppet.intent.leave_room(room_id)
                        else:
                            await intent.kick_user(room_id, user_id, message)
                    except MatrixError:
                        pass
            try:
                await intent.leave_room(room_id)
            except MatrixError:
                pass
    
        async def unbridge(self) -> None:
            await self.cleanup_room(self.main_intent, self.mxid, "Room unbridged", puppets_only=True)
            self.delete()
    
        async def cleanup_and_delete(self) -> None:
            await self.cleanup_room(self.main_intent, self.mxid)
            self.delete()
    
    
        # region Matrix event handling
    
    
        def require_send_lock(self, user_id: str) -> asyncio.Lock:
    
            try:
                lock = self._send_locks[user_id]
            except KeyError:
                lock = asyncio.Lock()
                self._send_locks[user_id] = lock
    
        def optional_send_lock(self, user_id: str) -> Union[asyncio.Lock, FakeLock]:
    
                return self._send_locks[user_id]
    
            except KeyError:
    
                pass
            return self._noop_lock
    
    Tulir Asokan's avatar
    Tulir Asokan committed
        async def handle_matrix_message(self, sender: 'u.User', message: MessageEventContent,
    
                                        event_id: EventID) -> None:
    
            puppet = p.Puppet.get_by_custom_mxid(sender.mxid)
            if puppet and message.get("net.maunium.facebook.puppet", False):
                self.log.debug(f"Ignoring puppet-sent message by confirmed puppet user {sender.mxid}")
                return
    
            # TODO this probably isn't nice for bridging images, it really only needs to lock the
            #      actual message send call and dedup queue append.
    
    Tulir Asokan's avatar
    Tulir Asokan committed
            async with self.require_send_lock(sender.fbid):
    
                if message.msgtype == MessageType.TEXT or message.msgtype == MessageType.NOTICE:
                    fbid = await self._handle_matrix_text(sender, message)
                elif message.msgtype == MessageType.IMAGE:
                    fbid = await self._handle_matrix_image(sender, message)
                elif message.msgtype == MessageType.LOCATION:
                    fbid = await self._handle_matrix_location(sender, message)
                else:
    
                    self.log.warning(f"Unsupported msgtype in {message}")
    
                    return
                if not fbid:
                    return
                self._dedup.appendleft(fbid)
                DBMessage(mxid=event_id, mx_room=self.mxid,
                          fbid=fbid, fb_receiver=self.fb_receiver,
                          index=0).insert()
                self._last_bridged_mxid = event_id
    
        async def _handle_matrix_text(self, sender: 'u.User', message: TextMessageEventContent) -> str:
    
    Tulir Asokan's avatar
    Tulir Asokan committed
            return await self.thread_for(sender).send_text(**matrix_to_facebook(message, self.mxid))
    
        async def _handle_matrix_image(self, sender: 'u.User',
    
                                       message: MediaMessageEventContent) -> Optional[str]:
            if message.file and decrypt_attachment:
                data = await self.main_intent.download_media(message.file.url)
                data = decrypt_attachment(data, message.file.key.key,
                                          message.file.hashes.get("sha256"), message.file.iv)
            elif message.url:
                data = await self.main_intent.download_media(message.url)
            else:
                return None
    
            mime = message.info.mimetype or magic.from_buffer(data, mime=True)
    
    Tulir Asokan's avatar
    Tulir Asokan committed
            files = await sender.client.upload([(message.body, data, mime)])
            return await self.thread_for(sender).send_files(files)
    
        async def _handle_matrix_location(self, sender: 'u.User',
                                          message: LocationMessageEventContent) -> str:
    
        async def handle_matrix_redaction(self, sender: 'u.User', event_id: EventID) -> None:
            if not self.mxid:
                return
    
            message = DBMessage.get_by_mxid(event_id, self.mxid)
    
            if message:
                try:
                    message.delete()
    
    Tulir Asokan's avatar
    Tulir Asokan committed
                    await fbchat.Message(thread=self.thread_for(sender), id=message.fbid).unsend()
    
                except Exception:
                    self.log.exception("Unsend failed")
    
    
            reaction = DBReaction.get_by_mxid(event_id, self.mxid)
            if reaction:
                try:
                    reaction.delete()
    
    Tulir Asokan's avatar
    Tulir Asokan committed
                    await fbchat.Message(thread=self.thread_for(sender),
                                         id=reaction.fb_msgid).react(None)
    
                except Exception:
                    self.log.exception("Removing reaction failed")
    
        async def handle_matrix_reaction(self, sender: 'u.User', event_id: EventID,
    
    Tulir Asokan's avatar
    Tulir Asokan committed
                                         reacting_to: EventID, reaction: str) -> None:
            async with self.require_send_lock(sender.fbid):
    
                message = DBMessage.get_by_mxid(reacting_to, self.mxid)
                if not message:
                    self.log.debug(f"Ignoring reaction to unknown event {reacting_to}")
                    return
    
    
    Tulir Asokan's avatar
    Tulir Asokan committed
                existing = DBReaction.get_by_fbid(message.fbid, self.fb_receiver, sender.fbid)
                if existing and existing.reaction == reaction:
    
    Tulir Asokan's avatar
    Tulir Asokan committed
                # TODO normalize reaction emoji bytes and maybe pre-reject invalid emojis
                await fbchat.Message(thread=self.thread_for(sender), id=message.fbid).react(reaction)
    
                await self._upsert_reaction(existing, self.main_intent, event_id, message, sender,
    
    Tulir Asokan's avatar
    Tulir Asokan committed
                                            reaction)
    
        async def handle_matrix_leave(self, user: 'u.User') -> None:
            if self.is_direct:
                self.log.info(f"{user.mxid} left private chat portal with {self.fbid},"
                              " cleaning up and deleting...")
                await self.cleanup_and_delete()
            else:
                self.log.debug(f"{user.mxid} left portal to {self.fbid}")
    
    
        async def handle_matrix_typing(self, users: Set['u.User']) -> None:
    
    Tulir Asokan's avatar
    Tulir Asokan committed
            stopped_typing = [self.thread_for(user).stop_typing() for user in self._typing - users]
            started_typing = [self.thread_for(user).start_typing() for user in self._typing - users]
    
            self._typing = users
            await asyncio.gather(*stopped_typing, *started_typing, loop=self.loop)
    
    
        # endregion
        # region Facebook event handling
    
    
        async def _bridge_own_message_pm(self, source: 'u.User', sender: 'p.Puppet', mid: str) -> bool:
    
    Tulir Asokan's avatar
    Tulir Asokan committed
            if self.is_direct and sender.fbid == source.fbid and not sender.is_real_user:
    
                if self.invite_own_puppet_to_pm:
                    await self.main_intent.invite_user(self.mxid, sender.mxid)
                elif self.az.state_store.get_membership(self.mxid, sender.mxid) != Membership.JOIN:
    
                    self.log.warning(f"Ignoring own {mid} in private chat because own puppet is not in"
                                     " room.")
    
                    return False
            return True
    
    
        async def handle_facebook_message(self, source: 'u.User', sender: 'p.Puppet',
    
    Tulir Asokan's avatar
    Tulir Asokan committed
                                          message: fbchat.MessageData) -> None:
    
            async with self.optional_send_lock(sender.fbid):
    
    Tulir Asokan's avatar
    Tulir Asokan committed
                if message.id in self._dedup:
                    await source.client.mark_as_delivered(message)
    
    Tulir Asokan's avatar
    Tulir Asokan committed
                self._dedup.appendleft(message.id)
    
                mxid = await self.create_matrix_room(source)
                if not mxid:
                    # Failed to create
                    return
    
    Tulir Asokan's avatar
    Tulir Asokan committed
            if not await self._bridge_own_message_pm(source, sender, f"message {message.id}"):
    
            intent = sender.intent_for(self)
    
            if message.sticker:
    
                event_ids = [await self._handle_facebook_sticker(intent, message.sticker,
                                                                 message.reply_to_id)]
    
            elif len(message.attachments) > 0:
                attach_ids = await asyncio.gather(
                    *[self._handle_facebook_attachment(intent, attachment, message.reply_to_id)
                      for attachment in message.attachments])
                event_ids += [attach_id for attach_id in attach_ids if attach_id]
    
            if not event_ids:
    
                if message.text or any(x for x in message.attachments
    
    Tulir Asokan's avatar
    Tulir Asokan committed
                                       if isinstance(x, fbchat.ShareAttachment)):
    
                    event_ids = [await self._handle_facebook_text(intent, message)]
                else:
                    self.log.warning(f"Unhandled Messenger message: {message}")
    
    Tulir Asokan's avatar
    Tulir Asokan committed
            DBMessage.bulk_create(fbid=message.id, fb_receiver=self.fb_receiver, mx_room=self.mxid,
    
                                  event_ids=[event_id for event_id in event_ids if event_id])
    
    Tulir Asokan's avatar
    Tulir Asokan committed
            await source.client.mark_as_delivered(message)
    
    Tulir Asokan's avatar
    Tulir Asokan committed
    
    
        async def _add_facebook_reply(self, content: TextMessageEventContent, reply: str) -> None:
            if reply:
                message = DBMessage.get_by_fbid(reply, self.fb_receiver)
                if message:
                    evt = await self.main_intent.get_event(message.mx_room, message.mxid)
                    if evt:
    
                        if isinstance(evt.content, TextMessageEventContent):
                            evt.content.trim_reply_fallback()
    
                        content.set_reply(evt)
    
        def _get_facebook_reply(self, reply: str) -> Optional[RelatesTo]:
            if reply:
                message = DBMessage.get_by_fbid(reply, self.fb_receiver)
                if message:
                    return RelatesTo(rel_type=RelationType.REFERENCE, event_id=message.mxid)
            return None
    
    
        async def _send_message(self, intent: IntentAPI, content: MessageEventContent,
                                event_type: EventType = EventType.ROOM_MESSAGE, **kwargs) -> EventID:
            if self.encrypted and self.matrix.e2ee:
    
                if intent.api.is_real_user:
                    content[intent.api.real_user_content_key] = True
    
                event_type, content = await self.matrix.e2ee.encrypt(self.mxid, event_type, content)
            return await intent.send_message_event(self.mxid, event_type, content, **kwargs)
    
    
    Tulir Asokan's avatar
    Tulir Asokan committed
        async def _handle_facebook_text(self, intent: IntentAPI, message: fbchat.Message) -> EventID:
    
            content = facebook_to_matrix(message)
            await self._add_facebook_reply(content, message.reply_to_id)
    
            return await self._send_message(intent, content)
    
    Tulir Asokan's avatar
    Tulir Asokan committed
        async def _handle_facebook_sticker(self, intent: IntentAPI, sticker: fbchat.Sticker,
    
                                           reply_to: str) -> EventID:
    
            # TODO handle animated stickers?
    
            mxc, mime, size, decryption_info = await self._reupload_fb_file(
                sticker.url, intent, encrypt=self.encrypted)
            return await self._send_message(intent, event_type=EventType.STICKER,
                                            content=MediaMessageEventContent(
                                                url=mxc, file=decryption_info,
                                                msgtype=MessageType.STICKER, body=sticker.label,
                                                info=ImageInfo(width=sticker.width, size=size,
                                                               height=sticker.height, mimetype=mime),
                                                relates_to=self._get_facebook_reply(reply_to)))
    
        async def _handle_facebook_attachment(self, intent: IntentAPI, attachment: AttachmentClass,
    
                                              reply_to: str) -> Optional[EventID]:
    
    Tulir Asokan's avatar
    Tulir Asokan committed
            if isinstance(attachment, fbchat.AudioAttachment):
    
                mxc, mime, size, decryption_info = await self._reupload_fb_file(
                    attachment.url, intent, attachment.filename, encrypt=self.encrypted)
                event_id = await self._send_message(intent, MediaMessageEventContent(
                    url=mxc, file=decryption_info, msgtype=MessageType.AUDIO, body=attachment.filename,
                    info=AudioInfo(size=size, mimetype=mime, duration=attachment.duration),
                    relates_to=self._get_facebook_reply(reply_to)))
    
    Tulir Asokan's avatar
    Tulir Asokan committed
            # elif isinstance(attachment, fbchat.VideoAttachment):
    
    Tulir Asokan's avatar
    Tulir Asokan committed
            elif isinstance(attachment, fbchat.FileAttachment):
    
                mxc, mime, size, decryption_info = await self._reupload_fb_file(
                    attachment.url, intent, attachment.name, encrypt=self.encrypted)
                event_id = await self._send_message(intent, MediaMessageEventContent(
                    url=mxc, file=decryption_info, msgtype=MessageType.FILE, body=attachment.name,
                    info=FileInfo(size=size, mimetype=mime),
                    relates_to=self._get_facebook_reply(reply_to)))
    
    Tulir Asokan's avatar
    Tulir Asokan committed
            elif isinstance(attachment, fbchat.ImageAttachment):
    
                mxc, mime, size, decryption_info = await self._reupload_fb_file(
                    attachment.large_preview_url or attachment.preview_url, intent,
                    encrypt=self.encrypted)
                event_id = await self._send_message(intent, MediaMessageEventContent(
                    url=mxc, file=decryption_info, msgtype=MessageType.IMAGE,
                    body=f"image.{attachment.original_extension}",
                    info=ImageInfo(size=size, mimetype=mime, width=attachment.large_preview_width,
                                   height=attachment.large_preview_height),
                    relates_to=self._get_facebook_reply(reply_to)))
    
    Tulir Asokan's avatar
    Tulir Asokan committed
            elif isinstance(attachment, fbchat.LocationAttachment):
    
                content = await self._convert_facebook_location(intent, attachment)
    
                content.relates_to = self._get_facebook_reply(reply_to)
    
                event_id = await self._send_message(intent, content)
    
    Tulir Asokan's avatar
    Tulir Asokan committed
            elif isinstance(attachment, fbchat.ShareAttachment):
    
                # These are handled in the text formatter
                return None
    
                self.log.warning(f"Unsupported attachment type: {attachment}")
    
            self._last_bridged_mxid = event_id
            return event_id
    
    Tulir Asokan's avatar
    Tulir Asokan committed
        async def _convert_facebook_location(self, intent: IntentAPI,
                                             location: fbchat.LocationAttachment
    
                                             ) -> LocationMessageEventContent:
            long, lat = location.longitude, location.latitude
            long_char = "E" if long > 0 else "W"
            lat_char = "N" if lat > 0 else "S"
            rounded_long = round(long, 5)
            rounded_lat = round(lat, 5)
    
            text = f"{rounded_lat}° {lat_char}, {rounded_long}° {long_char}"
            url = f"https://maps.google.com/?q={lat},{long}"
    
    
            thumbnail_url, mime, size, decryption_info = await self._reupload_fb_file(
                location.image_url, intent, encrypt=True)
    
            thumbnail_info = ThumbnailInfo(mimetype=mime, width=location.image_width,
                                           height=location.image_height, size=size)
            content = LocationMessageEventContent(
                body=f"{location.address}\nLocation: {text}\n{url}", geo_uri=f"geo:{lat},{long}",
                msgtype=MessageType.LOCATION, info=LocationInfo(thumbnail_url=thumbnail_url,
    
                                                                thumbnail_file=decryption_info,
    
                                                                thumbnail_info=thumbnail_info))
            # Some clients support formatted body in m.location, so add that as well.
            content["format"] = Format.HTML
            content["formatted_body"] = (f"<p>{location.address}</p>"
                                         f"<p>Location: <a href='{url}'>{text}</a></p")
            return content
    
        async def handle_facebook_unsend(self, source: 'u.User', sender: 'p.Puppet', message_id: str
                                         ) -> None:
            if not self.mxid:
                return
    
            for message in DBMessage.get_all_by_fbid(message_id, self.fb_receiver):
    
                    await sender.intent_for(self).redact(message.mx_room, message.mxid)
    
                except MForbidden:
                    await self.main_intent.redact(message.mx_room, message.mxid)
                message.delete()
    
        async def handle_facebook_seen(self, source: 'u.User', sender: 'p.Puppet') -> None:
    
            if not self.mxid or not self._last_bridged_mxid:
    
            await sender.intent_for(self).mark_read(self.mxid, self._last_bridged_mxid)
    
    
        async def handle_facebook_typing(self, source: 'u.User', sender: 'p.Puppet') -> None:
    
            await sender.intent.set_typing(self.mxid, is_typing=True)
    
        async def handle_facebook_photo(self, source: 'u.User', sender: 'p.Puppet', new_photo_id: str,
                                        message_id: str) -> None:
            if not self.mxid or self.is_direct:
                return
            if message_id in self._dedup:
                return
            self._dedup.appendleft(message_id)
            # When we fetch thread info manually, we only get the URL instead of the ID,
            # so we can't use the actual ID here either.
    
            # self.photo_id = new_photo_id
    
    Tulir Asokan's avatar
    Tulir Asokan committed
            photo_url = await source.client.fetch_image_url(new_photo_id)
    
            photo_id = self._get_photo_id(photo_url)
            if self.photo_id == photo_id:
                return
            self.photo_id = photo_id
    
            self._avatar_uri, *_ = await self._reupload_fb_file(photo_url, sender.intent)
    
            try:
                event_id = await sender.intent.set_room_avatar(self.mxid, self._avatar_uri)
            except IntentError:
                event_id = await self.main_intent.set_room_avatar(self.mxid, self._avatar_uri)
            DBMessage(mxid=event_id, mx_room=self.mxid,
                      fbid=message_id, fb_receiver=self.fb_receiver,
                      index=0).insert()
    
        async def handle_facebook_name(self, source: 'u.User', sender: 'p.Puppet', new_name: str,
                                       message_id: str) -> None:
            if self.name == new_name or message_id in self._dedup:
                return
            self._dedup.appendleft(message_id)
            self.name = new_name
            if not self.mxid or self.is_direct:
                return
            try:
                event_id = await sender.intent.set_room_name(self.mxid, self.name)
            except IntentError:
                event_id = await self.main_intent.set_room_name(self.mxid, self.name)
            DBMessage(mxid=event_id, mx_room=self.mxid,
                      fbid=message_id, fb_receiver=self.fb_receiver,
                      index=0).insert()
    
        async def handle_facebook_reaction_add(self, source: 'u.User', sender: 'p.Puppet',
                                               message_id: str, reaction: str) -> None:
            dedup_id = f"react_{message_id}_{sender}_{reaction}"
            async with self.optional_send_lock(sender.fbid):
                if dedup_id in self._dedup:
                    return
                self._dedup.appendleft(dedup_id)
    
            existing = DBReaction.get_by_fbid(message_id, self.fb_receiver, sender.fbid)
            if existing and existing.reaction == reaction:
                return
    
            if not await self._bridge_own_message_pm(source, sender, f"reaction to {message_id}"):
                return
    
            intent = sender.intent_for(self)
    
            message = DBMessage.get_by_fbid(message_id, self.fb_receiver)
            if not message:
                self.log.debug(f"Ignoring reaction to unknown message {message}")
                return
    
            mxid = await intent.react(message.mx_room, message.mxid, reaction)
            self.log.debug(f"Reacted to {message.mxid}, got {mxid}")
    
            await self._upsert_reaction(existing, intent, mxid, message, sender, reaction)
    
        async def _upsert_reaction(self, existing: DBReaction, intent: IntentAPI, mxid: EventID,
                                   message: DBMessage, sender: Union['u.User', 'p.Puppet'],
                                   reaction: str) -> None:
            if existing:
                self.log.debug(f"_upsert_reaction redacting {existing.mxid} and inserting {mxid}"
                               f" (message: {message.mxid})")
                await intent.redact(existing.mx_room, existing.mxid)
                existing.edit(reaction=reaction, mxid=mxid, mx_room=message.mx_room)
            else:
                self.log.debug(f"_upsert_reaction inserting {mxid} (message: {message.mxid})")
                DBReaction(mxid=mxid, mx_room=message.mx_room, fb_msgid=message.fbid,
                           fb_receiver=self.fb_receiver, fb_sender=sender.fbid,
                           reaction=reaction).insert()
    
        async def handle_facebook_reaction_remove(self, source: 'u.User', sender: 'p.Puppet',
                                                  message_id: str) -> None:
            if not self.mxid:
                return
            reaction = DBReaction.get_by_fbid(message_id, self.fb_receiver, sender.fbid)
            if reaction:
                self.log.debug(f"redacting {reaction.mxid}")
                try:
                    await sender.intent_for(self).redact(reaction.mx_room, reaction.mxid)
                except MForbidden:
                    await self.main_intent.redact(reaction.mx_room, reaction.mxid)
                reaction.delete()
    
    
    Tulir Asokan's avatar
    Tulir Asokan committed
        @classmethod
        def get_by_mxid(cls, mxid: RoomID) -> Optional['Portal']:
            try:
                return cls.by_mxid[mxid]
            except KeyError:
                pass
    
    
            db_portal = DBPortal.get_by_mxid(mxid)
            if db_portal:
                return cls.from_db(db_portal)
    
    
    Tulir Asokan's avatar
    Tulir Asokan committed
            return None
    
        @classmethod
    
        def get_by_fbid(cls, fbid: str, fb_receiver: Optional[str] = None,
                        fb_type: Optional[ThreadType] = None) -> Optional['Portal']:
    
            if fb_type:
                fb_receiver = fb_receiver if fb_type == ThreadType.USER else fbid
            else:
                fb_receiver = fb_receiver or fbid
    
            fbid_full = (fbid, fb_receiver)
    
    Tulir Asokan's avatar
    Tulir Asokan committed
            try:
    
                return cls.by_fbid[fbid_full]
    
    Tulir Asokan's avatar
    Tulir Asokan committed
            except KeyError:
    
                pass
    
            db_portal = DBPortal.get_by_fbid(fbid, fb_receiver)
            if db_portal:
                return cls.from_db(db_portal)
    
            if fb_type:
                portal = cls(fbid=fbid, fb_receiver=fb_receiver, fb_type=fb_type)
                portal.db_instance.insert()
                return portal
    
    
    Tulir Asokan's avatar
    Tulir Asokan committed
            return None
    
    
        @classmethod
        def get_all_by_receiver(cls, fb_receiver: str) -> Iterator['Portal']:
            for db_portal in DBPortal.get_all_by_receiver(fb_receiver):
                try:
                    yield cls.by_fbid[(db_portal.fbid, db_portal.fb_receiver)]
                except KeyError:
                    yield cls.from_db(db_portal)
    
    
    Tulir Asokan's avatar
    Tulir Asokan committed
        @classmethod
    
    Tulir Asokan's avatar
    Tulir Asokan committed
        def get_by_thread(cls, thread: fbchat.ThreadABC, fb_receiver: Optional[str] = None
                          ) -> 'Portal':
            return cls.get_by_fbid(thread.id, fb_receiver, ThreadType.from_thread(thread))
    
    Tulir Asokan's avatar
    Tulir Asokan committed
    
    
    Tulir Asokan's avatar
    Tulir Asokan committed
    
    def init(context: 'Context') -> None:
        global config
        Portal.az, config, Portal.loop = context.core
    
        Portal.matrix = context.mx
    
        Portal.invite_own_puppet_to_pm = config["bridge.invite_own_puppet_to_pm"]