2023-01-27 12:36:41 +00:00
|
|
|
import contextlib
|
|
|
|
from asyncio import sleep
|
|
|
|
from typing import Optional
|
|
|
|
|
|
|
|
from aiohttp import ClientConnectorError
|
|
|
|
from mipa.exception import WebSocketNotConnected
|
2022-12-22 14:18:43 +00:00
|
|
|
from mipa.ext import commands
|
|
|
|
from mipa.router import Router
|
2023-07-03 14:39:52 +00:00
|
|
|
from mipac import (
|
|
|
|
Note,
|
|
|
|
NotificationFollow,
|
|
|
|
NotificationFollowRequest,
|
|
|
|
ChatMessage,
|
|
|
|
NotificationAchievement,
|
|
|
|
)
|
2023-01-27 12:36:41 +00:00
|
|
|
from mipac.client import Client as MisskeyClient
|
2022-12-22 14:18:43 +00:00
|
|
|
|
2023-01-17 17:11:26 +00:00
|
|
|
from defs.chat import send_chat_message
|
2022-12-22 14:18:43 +00:00
|
|
|
from defs.misskey import send_update
|
2023-07-03 14:39:52 +00:00
|
|
|
from defs.notice import (
|
|
|
|
send_user_followed,
|
|
|
|
send_follow_request,
|
|
|
|
send_follow_request_accept,
|
|
|
|
send_achievement_earned,
|
|
|
|
)
|
2023-01-27 12:36:41 +00:00
|
|
|
|
|
|
|
from models.models.user import User, TokenStatusEnum
|
|
|
|
from models.services.user import UserAction
|
|
|
|
|
|
|
|
from init import bot, logs, sqlite
|
2022-12-22 14:18:43 +00:00
|
|
|
|
|
|
|
|
|
|
|
class MisskeyBot(commands.Bot):
|
2023-07-03 14:22:02 +00:00
|
|
|
def __init__(self, user: User):
|
2022-12-22 14:18:43 +00:00
|
|
|
super().__init__()
|
2023-07-03 14:22:02 +00:00
|
|
|
self.user_id: int = user.user_id
|
|
|
|
self.tg_user: User = user
|
2022-12-22 14:18:43 +00:00
|
|
|
|
|
|
|
async def on_ready(self, ws):
|
2022-12-23 03:06:17 +00:00
|
|
|
await Router(ws).connect_channel(["main", "home"])
|
2023-01-27 12:36:41 +00:00
|
|
|
logs.info(f"成功启动 Misskey Bot WS 任务 {self.user_id}")
|
2022-12-22 14:18:43 +00:00
|
|
|
|
2022-12-24 04:09:40 +00:00
|
|
|
async def on_reconnect(self, ws):
|
|
|
|
await Router(ws).connect_channel(["main", "home"])
|
|
|
|
|
2022-12-24 13:19:24 +00:00
|
|
|
async def on_note(self, note: Note):
|
2023-07-20 14:21:37 +00:00
|
|
|
await send_update(
|
|
|
|
self.tg_user.host, self.tg_user.chat_id, note, self.tg_user.timeline_topic
|
|
|
|
)
|
2022-12-22 14:18:43 +00:00
|
|
|
|
2022-12-26 09:48:13 +00:00
|
|
|
async def on_user_followed(self, notice: NotificationFollow):
|
2023-07-20 14:21:37 +00:00
|
|
|
await send_user_followed(
|
|
|
|
self.tg_user.chat_id, notice, self.tg_user.notice_topic
|
|
|
|
)
|
2022-12-26 09:48:13 +00:00
|
|
|
|
|
|
|
async def on_follow_request(self, notice: NotificationFollowRequest):
|
2023-07-20 14:21:37 +00:00
|
|
|
await send_follow_request(
|
|
|
|
self.tg_user.chat_id, notice, self.tg_user.notice_topic
|
|
|
|
)
|
2022-12-26 09:48:13 +00:00
|
|
|
|
|
|
|
async def on_follow_request_accept(self, notice: NotificationFollowRequest):
|
2023-07-20 14:21:37 +00:00
|
|
|
await send_follow_request_accept(
|
|
|
|
self.tg_user.chat_id, notice, self.tg_user.notice_topic
|
|
|
|
)
|
2022-12-26 09:48:13 +00:00
|
|
|
|
2023-01-17 17:11:26 +00:00
|
|
|
async def on_chat(self, message: ChatMessage):
|
2023-07-20 14:21:37 +00:00
|
|
|
await send_chat_message(
|
|
|
|
self.tg_user.host, self.tg_user.chat_id, message, self.tg_user.notice_topic
|
|
|
|
)
|
2023-01-17 17:11:26 +00:00
|
|
|
|
|
|
|
async def on_chat_unread_message(self, message: ChatMessage):
|
|
|
|
await message.api.read()
|
|
|
|
|
2023-01-24 07:23:25 +00:00
|
|
|
async def on_achievement_earned(self, notice: NotificationAchievement):
|
2023-07-20 14:21:37 +00:00
|
|
|
await send_achievement_earned(
|
|
|
|
self.tg_user.chat_id, notice, self.tg_user.notice_topic
|
|
|
|
)
|
2023-01-27 12:36:41 +00:00
|
|
|
|
|
|
|
|
|
|
|
misskey_bot_map: dict[int, MisskeyBot] = {}
|
|
|
|
|
|
|
|
|
|
|
|
def get_misskey_bot(user_id: int) -> Optional[MisskeyBot]:
|
|
|
|
return None if user_id not in misskey_bot_map else misskey_bot_map[user_id]
|
|
|
|
|
|
|
|
|
2023-07-03 14:22:02 +00:00
|
|
|
async def create_or_get_misskey_bot(user: User) -> MisskeyBot:
|
|
|
|
if user.user_id not in misskey_bot_map:
|
|
|
|
misskey_bot_map[user.user_id] = MisskeyBot(user)
|
|
|
|
return misskey_bot_map[user.user_id]
|
2023-01-27 12:36:41 +00:00
|
|
|
|
|
|
|
|
|
|
|
async def run(user: User):
|
2023-07-03 14:22:02 +00:00
|
|
|
misskey = await create_or_get_misskey_bot(user)
|
2023-01-27 12:36:41 +00:00
|
|
|
try:
|
|
|
|
logs.info(f"尝试启动 Misskey Bot WS 任务 {user.user_id}")
|
2023-07-20 14:21:37 +00:00
|
|
|
await misskey.start(f"wss://{user.host}", user.token)
|
2023-01-27 12:36:41 +00:00
|
|
|
except ClientConnectorError:
|
|
|
|
await sleep(3)
|
|
|
|
await run(user)
|
|
|
|
|
|
|
|
|
2023-07-20 14:21:37 +00:00
|
|
|
async def test_token(host: str, token: str) -> bool:
|
2023-01-27 12:36:41 +00:00
|
|
|
try:
|
2023-07-20 14:21:37 +00:00
|
|
|
logs.info(f"验证 Token {host} {token}")
|
|
|
|
client = MisskeyClient(f"https://{host}", token)
|
2023-01-27 12:36:41 +00:00
|
|
|
await client.http.login()
|
|
|
|
await client.http.close_session()
|
|
|
|
return True
|
|
|
|
except Exception:
|
|
|
|
return False
|
|
|
|
|
|
|
|
|
|
|
|
async def rerun_misskey_bot(user_id: int) -> bool:
|
|
|
|
if misskey := get_misskey_bot(user_id):
|
|
|
|
with contextlib.suppress(WebSocketNotConnected):
|
|
|
|
await misskey.disconnect()
|
|
|
|
misskey_bot_map.pop(user_id)
|
|
|
|
user = await UserAction.get_user_if_ok(user_id)
|
|
|
|
if not user:
|
|
|
|
return False
|
2023-07-20 14:21:37 +00:00
|
|
|
if not await test_token(user.host, user.token):
|
2023-01-27 12:36:41 +00:00
|
|
|
await UserAction.set_user_status(user_id, TokenStatusEnum.INVALID_TOKEN)
|
|
|
|
return False
|
|
|
|
bot.loop.create_task(run(user))
|
|
|
|
return True
|
2023-01-24 07:23:25 +00:00
|
|
|
|
2022-12-22 14:18:43 +00:00
|
|
|
|
2023-01-27 12:36:41 +00:00
|
|
|
async def init_misskey_bot():
|
|
|
|
await sqlite.create_db_and_tables()
|
|
|
|
count = 0
|
|
|
|
for user in await UserAction.get_all_token_ok_users():
|
2023-07-20 14:21:37 +00:00
|
|
|
if not await test_token(user.host, user.token):
|
2023-01-27 12:36:41 +00:00
|
|
|
user.status = TokenStatusEnum.INVALID_TOKEN
|
|
|
|
await UserAction.update_user(user)
|
|
|
|
continue
|
|
|
|
count += 1
|
|
|
|
bot.loop.create_task(run(user))
|
|
|
|
logs.info(f"初始化 Misskey Bot 完成,共启动 {count} 个 WS 任务")
|