# maubot - A plugin-based Matrix bot system. # Copyright (C) 2018 Tulir Asokan # # 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 . from typing import Type from sqlalchemy import (Column, String, Boolean, ForeignKey, Text, TypeDecorator) from sqlalchemy.orm import Query from sqlalchemy.ext.declarative import declarative_base import json from mautrix.types import UserID, FilterID, SyncToken, ContentURI from mautrix.client.api.types.util import Serializable from mautrix import ClientStore from .command_spec import CommandSpec Base: declarative_base = declarative_base() def make_serializable_alchemy(serializable_type: Type[Serializable]): class SerializableAlchemy(TypeDecorator): impl = Text @property def python_type(self): return serializable_type def process_literal_param(self, value: Serializable, _) -> str: return json.dumps(value.serialize()) if value is not None else None def process_bind_param(self, value: Serializable, _) -> str: return json.dumps(value.serialize()) if value is not None else None def process_result_value(self, value: str, _) -> serializable_type: return serializable_type.deserialize(json.loads(value)) if value is not None else None return SerializableAlchemy class DBPlugin(Base): query: Query __tablename__ = "plugin" id: str = Column(String(255), primary_key=True) type: str = Column(String(255), nullable=False) enabled: bool = Column(Boolean, nullable=False, default=False) primary_user: UserID = Column(String(255), ForeignKey("client.id", onupdate="CASCADE", ondelete="RESTRICT"), nullable=False) class DBClient(ClientStore, Base): query: Query __tablename__ = "client" id: UserID = Column(String(255), primary_key=True) homeserver: str = Column(String(255), nullable=False) access_token: str = Column(String(255), nullable=False) next_batch: SyncToken = Column(String(255), nullable=False, default="") filter_id: FilterID = Column(String(255), nullable=False, default="") sync: bool = Column(Boolean, nullable=False, default=True) autojoin: bool = Column(Boolean, nullable=False, default=True) displayname: str = Column(String(255), nullable=False, default="") avatar_url: ContentURI = Column(String(255), nullable=False, default="") class DBCommandSpec(Base): query: Query __tablename__ = "command_spec" plugin: str = Column(String(255), ForeignKey("plugin.id", onupdate="CASCADE", ondelete="CASCADE"), primary_key=True) client: UserID = Column(String(255), ForeignKey("client.id", onupdate="CASCADE", ondelete="CASCADE"), primary_key=True) spec: CommandSpec = Column(make_serializable_alchemy(CommandSpec), nullable=False)