# Lain Iwakura from io import BytesIO from typing import Tuple from PIL import Image, ImageOps from pillow_heif import HeifImagePlugin from typing import Type from maubot import Plugin, MessageEvent from maubot.handlers import command from mautrix.client import Client as MatrixClient from mautrix.crypto import attachments from mautrix.types import EncryptedFile, ImageInfo, MediaMessageEventContent, MessageType from mautrix.util.config import BaseProxyConfig, ConfigUpdateHelper from mautrix.errors import MForbidden class Config(BaseProxyConfig): def do_update(self, helper: ConfigUpdateHelper) -> None: helper.copy("rooms") async def download_encrypted_media( file: EncryptedFile, client: MatrixClient) -> bytes: """ Download an encrypted media file :param file: The `EncryptedFile` instance, from MediaMessageEventContent.file. :param client: The Matrix client. Can be accessed via MessageEvent.client :return: The media file as bytes. """ return attachments.decrypt_attachment( await client.download_media(file.url), file.key.key, file.hashes['sha256'], file.iv ) async def download_unencrypted_media( url, client: MatrixClient) -> bytes: """ Download an unencrypted media file :param url: The media file mxc url, from MediaMessageEventContent.url. :param client: The Matrix client. Can be accessed via MessageEvent.client :return: The media file as bytes. """ return await client.download_media(url) async def send_encrypted_message( img_enc, room_id, info, client: MatrixClient) -> None: """ Sends image to an encrypted room :param img_enc: The data to upload. :param room_id: RoomID. :param info: mautrix image info. """ uri = await client.upload_media( img_enc[0], mime_type="image/jpeg", filename="image.jpg" ) content = MediaMessageEventContent( msgtype=MessageType.IMAGE, body="image.jpg", file=EncryptedFile( key=img_enc[1].key, iv=img_enc[1].iv, hashes=img_enc[1].hashes, url=uri, version=img_enc[1].version ), info=ImageInfo( mimetype="image/jpeg", width=info.width, height=info.height ) ) await client.send_message(room_id, content) async def send_unencrypted_message( img, room_id, info, client: MatrixClient) -> None: """ Sends image to an unencrypted room :param img: The data to upload. :param room_id: RoomID. :param info: mautrix image info. """ uri = await client.upload_media( img, mime_type="image/jpeg", filename="image.jpg" ) content = MediaMessageEventContent( msgtype=MessageType.IMAGE, body="image.jpg", url=uri, info=ImageInfo( mimetype="image/jpeg", width=info.width, height=info.height ) ) await client.send_message(room_id, content) class HeicToJpeg(Plugin): async def start(self) -> None: await super().start() self.config.load_and_update() self.rooms = self.config['rooms'] if self.config['rooms'] else None @classmethod def get_config_class(cls) -> Type[BaseProxyConfig]: return Config @command.passive("", msgtypes=(MessageType.IMAGE, MessageType.FILE)) async def hate_heif_message( self, evt: MessageEvent, match: Tuple[str]) -> None: """ If heif = make it jpg. """ # If there are rooms specified in config, serve only these rooms if self.rooms: if evt.room_id not in self.rooms: self.log.debug(f"Current room {evt.room_id} is not listed in {self.rooms} so I ignore it.") return self.log.debug(f"Received message with an image with MIME: {evt.content.info.mimetype}") if evt.content.info.mimetype not in ("image/heic", "image/heif"): return try: await evt.redact(reason="Image converted to less appleish format bellow") except MForbidden: self.log.error("No permission to redact!") return if evt.content.url: # url exists. File is not encrypted. data = await download_unencrypted_media(evt.content.url, evt.client) is_enc = False elif evt.content.file: # file exists. File is encrypted. data = await download_encrypted_media(evt.content.file, evt.client) is_enc = True else: # shouldn't happen self.log.warning("A message with IMAGE type received, but it does not contain a file.") return # de-heif via pillow img_in = Image.open(BytesIO(data)) self.log.debug(f"Received image parameters: {img_in.format} {img_in.size} {img_in.mode}") ImageOps.exif_transpose(img_in, in_place=True) with BytesIO() as img_out: img_in.save(img_out, format="JPEG") img = img_out.getvalue() img_tst = Image.open(BytesIO(img)) self.log.debug(f"Created image parameters: {img_tst.format} {img_tst.size} {img_tst.mode}") evt.content.info.width=img_tst.size[0] evt.content.info.height=img_tst.size[1] if is_enc: img_enc = attachments.encrypt_attachment(img) #self.log.debug(f"{img_enc}") await send_encrypted_message( img_enc, evt.room_id, evt.content.info, evt.client ) else: await send_unencrypted_message( img, evt.room_id, evt.content.info, evt.client )