# -*- coding: future_fstrings -*- # mautrix-telegram - A Matrix-Telegram puppeting bridge # Copyright (C) 2018 Tulir Asokan # # This program is free software: you can redistribute it and/or modify # it under the terms of the GNU 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 General Public License for more details. # # You should have received a copy of the GNU General Public License # along with this program. If not, see . from html import escape import logging from telethon.tl.types import * from mautrix_appservice import MatrixRequestError from .. import user as u, puppet as pu, portal as po from ..db import Message as DBMessage from .util import add_surrogates, remove_surrogates log = logging.getLogger("mau.fmt.tg") def telegram_reply_to_matrix(evt, source): if evt.reply_to_msg_id: space = (evt.to_id.channel_id if isinstance(evt, Message) and isinstance(evt.to_id, PeerChannel) else source.tgid) msg = DBMessage.query.get((evt.reply_to_msg_id, space)) if msg: return { "m.in_reply_to": { "event_id": msg.mxid, "room_id": msg.mx_room, } } return {} async def _add_forward_header(source, text, html, fwd_from_id): if not html: html = escape(text) user = u.User.get_by_tgid(fwd_from_id) if user: fwd_from = f"{user.mxid}" else: puppet = pu.Puppet.get(fwd_from_id, create=False) if puppet and puppet.displayname: fwd_from = f"{puppet.displayname}" else: user = await source.client.get_entity(fwd_from_id) if user: fwd_from = f"{pu.Puppet.get_displayname(user, format=False)}" else: fwd_from = None if not fwd_from: fwd_from = "Unknown user" text = f"Forwarded from {fwd_from}:\n{text}" html = (f"Forwarded message from {fwd_from}
" f"
{html}
") return text, html async def _add_reply_header(source, text, html, evt, relates_to, native_replies, message_link_in_reply, main_intent, reply_text): space = (evt.to_id.channel_id if isinstance(evt, Message) and isinstance(evt.to_id, PeerChannel) else source.tgid) msg = DBMessage.query.get((evt.reply_to_msg_id, space)) if not msg: return text, html if native_replies: relates_to["m.in_reply_to"] = { "event_id": msg.mxid, "room_id": msg.mx_room, } if reply_text == "Edit": html = f"Edit: {html or escape(text)}" text = f"Edit: {text}" return text, html reply_displayname = "unknown user" try: event = await main_intent.get_event(msg.mx_room, msg.mxid) content = event["content"] body = (content["formatted_body"] if "formatted_body" in content else content["body"]) sender = event['sender'] puppet = pu.Puppet.get_by_mxid(sender, create=False) reply_displayname = puppet.displayname if puppet else sender reply_to_user = f"{reply_displayname}" reply_to_msg = (("{reply_text}") if message_link_in_reply else "Reply") quote = f"{reply_to_msg} to {reply_to_user}
{body}
" except (ValueError, KeyError, MatrixRequestError): quote = f"{reply_text} to unknown user (Failed to fetch message):
" if not html: html = escape(text) html = quote + html text = f"{reply_text} to {reply_displayname}:\n{text}" return text, html async def telegram_to_matrix(evt, source, native_replies=False, message_link_in_reply=False, main_intent=None, reply_text="Reply"): text = add_surrogates(evt.message) html = _telegram_entities_to_matrix_catch(text, evt.entities) if evt.entities else None relates_to = {} if evt.fwd_from: text, html = await _add_forward_header(source, text, html, evt.fwd_from.from_id) if evt.reply_to_msg_id: text, html = await _add_reply_header(source, text, html, evt, relates_to, native_replies, message_link_in_reply, main_intent, reply_text) if isinstance(evt, Message) and evt.post and evt.post_author: if not html: html = escape(text) text += f"\n- {evt.post_author}" html += f"
- {evt.post_author}" if html: html = html.replace("\n", "
") return remove_surrogates(text), remove_surrogates(html), relates_to def _telegram_entities_to_matrix_catch(text, entities): try: return _telegram_entities_to_matrix(text, entities) except Exception: log.exception("Failed to convert Telegram format:\n" "message=%s\n" "entities=%s", text, entities) def _telegram_entities_to_matrix(text, entities): if not entities: return text html = [] last_offset = 0 for entity in entities: if entity.offset > last_offset: html.append(escape(text[last_offset:entity.offset])) elif entity.offset < last_offset: continue skip_entity = False entity_text = escape(text[entity.offset:entity.offset + entity.length]) entity_type = type(entity) if entity_type == MessageEntityBold: html.append(f"{entity_text}") elif entity_type == MessageEntityItalic: html.append(f"{entity_text}") elif entity_type == MessageEntityCode: html.append(f"{entity_text}") elif entity_type == MessageEntityPre: skip_entity = _parse_pre(html, entity_text, entity.language) elif entity_type == MessageEntityMention: skip_entity = _parse_mention(html, entity_text) elif entity_type == MessageEntityMentionName: skip_entity = _parse_name_mention(html, entity_text, entity.user_id) elif entity_type == MessageEntityEmail: html.append(f"{entity_text}") elif entity_type in {MessageEntityTextUrl, MessageEntityUrl}: skip_entity = _parse_url(html, entity_text, entity.url if entity_type == MessageEntityTextUrl else None) elif entity_type == MessageEntityBotCommand: html.append(f"!{entity_text[1:]}") elif entity_type == MessageEntityHashtag: html.append(f"{entity_text}") else: skip_entity = True last_offset = entity.offset + (0 if skip_entity else entity.length) html.append(text[last_offset:]) return "".join(html) def _parse_pre(html, entity_text, language): if language: html.append("
"
                    f"{entity_text}"
                    "
") else: html.append(f"
{entity_text}
") return False def _parse_mention(html, entity_text): username = entity_text[1:] user = u.User.find_by_username(username) or pu.Puppet.find_by_username(username) if user: mxid = user.mxid else: portal = po.Portal.find_by_username(username) mxid = portal.alias or portal.mxid if portal else None if mxid: html.append(f"{entity_text}") else: return True return False def _parse_name_mention(html, entity_text, user_id): user = u.User.get_by_tgid(user_id) if user: mxid = user.mxid else: puppet = pu.Puppet.get(user_id, create=False) mxid = puppet.mxid if puppet else None if mxid: html.append(f"{entity_text}") else: return True return False def _parse_url(html, entity_text, url): url = escape(url) if url else entity_text if not url.startswith(("https://", "http://", "ftp://", "magnet://")): url = "http://" + url html.append(f"{entity_text}") return False