|
| 1 | +from typing import TYPE_CHECKING |
| 2 | +import asyncio |
| 3 | + |
| 4 | +from sqlalchemy import MetaData, Table, Column, Text |
| 5 | +from sqlalchemy.engine.base import Engine |
| 6 | + |
| 7 | +from mautrix.types import ContentURI |
| 8 | + |
| 9 | +if TYPE_CHECKING: |
| 10 | + from .bot import LinearBot |
| 11 | + |
| 12 | + |
| 13 | +class AvatarManager: |
| 14 | + bot: 'LinearBot' |
| 15 | + _avatars: dict[str, ContentURI] |
| 16 | + _table: Table |
| 17 | + _db: Engine |
| 18 | + _lock: asyncio.Lock |
| 19 | + |
| 20 | + def __init__(self, bot: 'LinearBot', metadata: MetaData) -> None: |
| 21 | + self.bot = bot |
| 22 | + self._db = bot.database |
| 23 | + self._table = Table("avatar", metadata, |
| 24 | + Column("url", Text, primary_key=True), |
| 25 | + Column("mxc", Text, nullable=False)) |
| 26 | + self._lock = asyncio.Lock() |
| 27 | + self._avatars = {} |
| 28 | + |
| 29 | + def load_db(self) -> None: |
| 30 | + self._avatars = {url: ContentURI(mxc) |
| 31 | + for url, mxc |
| 32 | + in self._db.execute(self._table.select())} |
| 33 | + |
| 34 | + async def get_mxc(self, url: str) -> ContentURI: |
| 35 | + try: |
| 36 | + return self._avatars[url] |
| 37 | + except KeyError: |
| 38 | + pass |
| 39 | + async with self.bot.http.get(url) as resp: |
| 40 | + resp.raise_for_status() |
| 41 | + data = await resp.read() |
| 42 | + async with self._lock: |
| 43 | + try: |
| 44 | + return self._avatars[url] |
| 45 | + except KeyError: |
| 46 | + pass |
| 47 | + mxc = await self.bot.client.upload_media(data) |
| 48 | + self._avatars[url] = mxc |
| 49 | + with self._db.begin() as conn: |
| 50 | + conn.execute(self._table.insert().values(url=url, mxc=mxc)) |
| 51 | + return mxc |
0 commit comments