Archived
1
0
Fork 0
This repository has been archived on 2024-04-26. You can view files and clone it, but cannot push or open issues or pull requests.
akari-bot/bots/discord/message.py

252 lines
10 KiB
Python
Raw Normal View History

2021-12-25 14:01:33 +00:00
import datetime
2021-08-07 07:56:48 +00:00
import re
2021-08-02 15:35:45 +00:00
import traceback
2021-12-31 14:44:34 +00:00
from typing import List, Union
2021-08-02 15:35:45 +00:00
import discord
import filetype
2021-08-07 07:56:48 +00:00
2022-06-12 07:07:53 +00:00
from bots.discord.client import client
2022-07-14 13:23:15 +00:00
from config import Config
2023-02-05 14:33:33 +00:00
from core.builtins import Bot, Plain, Image, MessageSession as MS
from core.builtins.message.chain import MessageChain
2023-06-29 14:53:46 +00:00
from core.builtins.message.internal import Embed, ErrorMessage, Voice
2022-01-18 12:32:43 +00:00
from core.logger import Logger
2023-02-05 14:33:33 +00:00
from core.types import MsgInfo, Session, FetchTarget as FT, \
FetchedSession as FS, FinishedSession as FinS
from core.utils.http import download_to_cache
2021-08-30 18:53:39 +00:00
from database import BotDBUtil
2021-08-02 15:35:45 +00:00
2022-07-14 13:23:15 +00:00
enable_analytics = Config('enable_analytics')
2022-01-13 12:08:14 +00:00
async def convert_embed(embed: Embed):
2021-12-25 14:01:33 +00:00
if isinstance(embed, Embed):
2022-01-13 12:08:14 +00:00
files = []
2021-12-25 14:01:33 +00:00
embeds = discord.Embed(title=embed.title if embed.title is not None else discord.Embed.Empty,
description=embed.description if embed.description is not None else discord.Embed.Empty,
color=embed.color if embed.color is not None else discord.Embed.Empty,
url=embed.url if embed.url is not None else discord.Embed.Empty,
2021-12-31 14:44:34 +00:00
timestamp=datetime.datetime.fromtimestamp(
embed.timestamp) if embed.timestamp is not None else discord.Embed.Empty, )
2022-01-13 12:09:51 +00:00
if embed.image is not None:
2022-01-13 12:08:14 +00:00
upload = discord.File(await embed.image.get(), filename="image.png")
files.append(upload)
embeds.set_image(url="attachment://image.png")
2021-12-25 14:01:33 +00:00
if embed.thumbnail is not None:
2022-01-13 12:08:14 +00:00
upload = discord.File(await embed.thumbnail.get(), filename="thumbnail.png")
files.append(upload)
embeds.set_thumbnail(url="attachment://thumbnail.png")
2021-12-25 14:01:33 +00:00
if embed.author is not None:
embeds.set_author(name=embed.author)
if embed.footer is not None:
embeds.set_footer(text=embed.footer)
if embed.fields is not None:
for field in embed.fields:
embeds.add_field(name=field.name, value=field.value, inline=field.inline)
2022-01-13 12:08:14 +00:00
return embeds, files
2021-12-25 14:01:33 +00:00
2022-01-15 14:28:35 +00:00
class FinishedSession(FinS):
async def delete(self):
"""
用于删除这条消息
"""
try:
for x in self.result:
await x.delete()
except Exception:
2022-01-20 13:31:50 +00:00
Logger.error(traceback.format_exc())
2022-01-15 14:28:35 +00:00
2021-08-07 07:56:48 +00:00
class MessageSession(MS):
2021-08-02 15:35:45 +00:00
class Feature:
image = True
2023-06-29 14:53:46 +00:00
voice = True
2022-01-16 13:24:15 +00:00
embed = True
2021-10-11 14:45:28 +00:00
forward = False
2021-11-16 14:19:48 +00:00
delete = True
2022-01-16 13:24:15 +00:00
quote = True
wait = True
2021-08-02 15:35:45 +00:00
2022-08-27 14:02:26 +00:00
async def sendMessage(self, msgchain, quote=True, disable_secret_check=False, allow_split_image=True
) -> FinishedSession:
2021-12-31 14:44:34 +00:00
msgchain = MessageChain(msgchain)
2022-01-08 08:08:25 +00:00
if not msgchain.is_safe and not disable_secret_check:
2023-03-05 11:15:44 +00:00
return await self.sendMessage(Plain(ErrorMessage(self.locale.t("error.message.chain.unsafe"))))
2022-06-26 06:06:26 +00:00
self.sent.append(msgchain)
2021-12-31 14:44:34 +00:00
count = 0
send = []
for x in msgchain.asSendable():
if isinstance(x, Plain):
send_ = await self.session.target.send(x.text,
reference=self.session.message if quote and count == 0
2023-04-30 03:30:59 +00:00
and self.session.message else None)
2022-01-18 12:32:43 +00:00
Logger.info(f'[Bot] -> [{self.target.targetId}]: {x.text}')
2021-12-31 14:44:34 +00:00
elif isinstance(x, Image):
send_ = await self.session.target.send(file=discord.File(await x.get()),
reference=self.session.message if quote and count == 0
2023-04-30 03:30:59 +00:00
and self.session.message else None)
2022-01-18 12:32:43 +00:00
Logger.info(f'[Bot] -> [{self.target.targetId}]: Image: {str(x.__dict__)}')
2023-06-29 14:53:46 +00:00
elif isinstance(x, Voice):
send_ = await self.session.target.send(file=discord.File(x.path),
reference=self.session.message if quote and count == 0
and self.session.message else None)
Logger.info(f'[Bot] -> [{self.target.targetId}]: Voice: {str(x.__dict__)}')
2021-12-31 14:44:34 +00:00
elif isinstance(x, Embed):
2022-01-13 12:08:14 +00:00
embeds, files = await convert_embed(x)
send_ = await self.session.target.send(embed=embeds,
2021-12-31 14:44:34 +00:00
reference=self.session.message if quote and count == 0
2023-04-30 03:30:59 +00:00
and self.session.message else None,
2022-01-13 12:08:14 +00:00
files=files)
2022-01-18 12:32:43 +00:00
Logger.info(f'[Bot] -> [{self.target.targetId}]: Embed: {str(x.__dict__)}')
2021-12-31 14:44:34 +00:00
else:
send_ = False
if send_:
send.append(send_)
count += 1
2022-06-28 06:11:03 +00:00
msgIds = []
for x in send:
msgIds.append(x.id)
2023-02-06 10:48:09 +00:00
return FinishedSession(self, msgIds, send)
2021-08-02 15:35:45 +00:00
async def checkPermission(self):
2021-08-07 07:56:48 +00:00
if self.session.message.channel.permissions_for(self.session.message.author).administrator \
2023-04-30 03:30:59 +00:00
or isinstance(self.session.message.channel, discord.DMChannel) \
or self.target.senderInfo.query.isSuperUser \
or self.target.senderId in self.custom_admins:
2021-08-02 15:35:45 +00:00
return True
return False
async def checkNativePermission(self):
if self.session.message.channel.permissions_for(self.session.message.author).administrator \
2023-04-30 03:30:59 +00:00
or isinstance(self.session.message.channel, discord.DMChannel):
return True
return False
async def toMessageChain(self):
lst = []
lst.append(Plain(self.session.message.content))
for x in self.session.message.attachments:
d = await download_to_cache(x.url)
if filetype.is_image(d):
lst.append(Image(d))
return MessageChain(lst)
2023-03-04 09:40:38 +00:00
def asDisplay(self, text_only=False):
msg = self.session.message.content
2023-01-31 13:10:51 +00:00
msg = re.sub(r'<@(.*?)>', r'Discord|Client|\1', msg)
return msg
2021-08-02 15:35:45 +00:00
async def delete(self):
try:
2022-08-04 15:52:53 +00:00
await self.session.message.delete()
2021-08-23 12:44:31 +00:00
except Exception:
2022-01-20 13:31:50 +00:00
Logger.error(traceback.format_exc())
2021-08-02 15:35:45 +00:00
class Typing:
2021-08-07 07:56:48 +00:00
def __init__(self, msg: MS):
2021-08-02 15:35:45 +00:00
self.msg = msg
async def __aenter__(self):
async with self.msg.session.target.typing() as typing:
return typing
async def __aexit__(self, exc_type, exc_val, exc_tb):
2021-08-07 07:56:48 +00:00
pass
2022-01-15 13:46:23 +00:00
class FetchedSession(FS):
def __init__(self, targetFrom, targetId):
self.target = MsgInfo(targetId=f'{targetFrom}|{targetId}',
senderId=f'{targetFrom}|{targetId}',
targetFrom=targetFrom,
senderFrom=targetFrom,
2022-06-28 06:11:03 +00:00
senderName='', clientName='Discord', messageId=0, replyId=None)
2022-01-15 13:46:23 +00:00
self.session = Session(message=False, target=targetId, sender=targetId)
self.parent = MessageSession(self.target, self.session)
async def sendDirectMessage(self, msgchain, disable_secret_check=False):
2022-01-15 13:46:23 +00:00
"""
用于向获取对象发送消息
:param msgchain: 消息链若传入str则自动创建一条带有Plain元素的消息链
:param disable_secret_check: 是否禁用消息检查默认为False
:return: 被发送的消息链
"""
try:
getChannel = await client.fetch_channel(self.session.target)
except Exception:
2022-01-20 13:31:50 +00:00
Logger.error(traceback.format_exc())
2022-01-15 13:46:23 +00:00
return False
self.session.target = self.session.sender = self.parent.session.target = self.parent.session.sender = getChannel
return await self.parent.sendDirectMessage(msgchain, disable_secret_check=disable_secret_check)
2022-01-15 13:46:23 +00:00
2021-08-21 15:58:07 +00:00
class FetchTarget(FT):
2022-01-05 11:14:45 +00:00
name = 'Discord'
2021-08-07 07:56:48 +00:00
@staticmethod
2022-01-15 13:46:23 +00:00
async def fetch_target(targetId) -> Union[FetchedSession, bool]:
2021-08-22 14:55:25 +00:00
matchChannel = re.match(r'^(Discord\|(?:DM\||)Channel)\|(.*)', targetId)
2021-08-07 07:56:48 +00:00
if matchChannel:
2022-01-15 13:46:23 +00:00
return FetchedSession(matchChannel.group(1), matchChannel.group(2))
2021-08-07 07:56:48 +00:00
else:
return False
2021-08-30 18:53:39 +00:00
2021-10-08 11:54:27 +00:00
@staticmethod
2022-01-15 13:46:23 +00:00
async def fetch_target_list(targetList: list) -> List[FetchedSession]:
2021-10-08 11:54:27 +00:00
lst = []
for x in targetList:
fet = await FetchTarget.fetch_target(x)
if fet:
lst.append(fet)
return lst
2021-08-30 18:53:39 +00:00
@staticmethod
2023-03-29 14:42:53 +00:00
async def post_message(module_name, message, user_list: List[FetchedSession] = None, i18n=False, **kwargs):
2021-08-30 18:53:39 +00:00
if user_list is not None:
for x in user_list:
try:
2023-03-29 14:42:53 +00:00
if i18n:
2023-04-05 05:36:11 +00:00
if isinstance(message, dict):
if (gm := message.get(x.parent.locale.locale)) is not None:
await x.sendDirectMessage(gm)
else:
await x.sendDirectMessage(message.get('fallback'))
else:
await x.sendDirectMessage(x.parent.locale.t(message, **kwargs))
2023-03-29 14:42:53 +00:00
else:
await x.sendDirectMessage(message)
2022-07-14 13:23:15 +00:00
if enable_analytics:
BotDBUtil.Analytics(x).add('', module_name, 'schedule')
2021-08-30 18:53:39 +00:00
except Exception:
2022-01-20 13:31:50 +00:00
Logger.error(traceback.format_exc())
2021-08-30 18:53:39 +00:00
else:
2023-04-05 05:36:11 +00:00
get_target_id = BotDBUtil.TargetInfo.get_enabled_this(module_name, "Telegram")
2021-08-30 18:53:39 +00:00
for x in get_target_id:
2022-08-29 15:54:20 +00:00
fetch = await FetchTarget.fetch_target(x.targetId)
2021-08-30 18:53:39 +00:00
if fetch:
try:
2023-03-29 14:42:53 +00:00
if i18n:
2023-04-05 05:36:11 +00:00
if isinstance(message, dict):
if (gm := message.get(fetch.parent.locale.locale)) is not None:
await fetch.sendDirectMessage(gm)
else:
await fetch.sendDirectMessage(message.get('fallback'))
else:
await fetch.sendDirectMessage(fetch.parent.locale.t(message, **kwargs))
2023-03-29 14:42:53 +00:00
else:
await fetch.sendDirectMessage(message)
2022-07-14 13:23:15 +00:00
if enable_analytics:
BotDBUtil.Analytics(fetch).add('', module_name, 'schedule')
2021-08-30 18:53:39 +00:00
except Exception:
2022-01-20 13:31:50 +00:00
Logger.error(traceback.format_exc())
2023-02-04 14:42:21 +00:00
Bot.MessageSession = MessageSession
Bot.FetchTarget = FetchTarget