mirror of
https://github.com/langbot-app/LangBot.git
synced 2026-06-02 12:05:54 +00:00
* feat: add voice and file supports for wecom * feat: add and in query variables * feat: supports for lark recv file message * feat: kook recv voice msg * feat: supports for Voice and File in discord * chore: remove debug msg * perf: remove unnecessary bot logs * feat: implement bot log filtering and per label color (#1839) * feat: add sender_name and group_name in query variables
208 lines
8.0 KiB
Python
208 lines
8.0 KiB
Python
from __future__ import annotations
|
||
import typing
|
||
import asyncio
|
||
import traceback
|
||
|
||
import datetime
|
||
|
||
from langbot.libs.slack_api.api import SlackClient
|
||
import langbot_plugin.api.definition.abstract.platform.adapter as abstract_platform_adapter
|
||
from langbot.libs.slack_api.slackevent import SlackEvent
|
||
import langbot_plugin.api.entities.builtin.platform.events as platform_events
|
||
import langbot_plugin.api.entities.builtin.platform.message as platform_message
|
||
import langbot_plugin.api.entities.builtin.platform.entities as platform_entities
|
||
from langbot_plugin.api.entities.builtin.command import errors as command_errors
|
||
from langbot.pkg.utils import image
|
||
from langbot.pkg.platform.logger import EventLogger
|
||
|
||
|
||
class SlackMessageConverter(abstract_platform_adapter.AbstractMessageConverter):
|
||
@staticmethod
|
||
async def yiri2target(message_chain: platform_message.MessageChain):
|
||
content_list = []
|
||
for msg in message_chain:
|
||
if type(msg) is platform_message.Plain:
|
||
content_list.append(
|
||
{
|
||
'type': 'text',
|
||
'content': msg.text,
|
||
}
|
||
)
|
||
elif type(msg) is platform_message.Image:
|
||
# Slack supports images via unfurling URLs
|
||
# Include image URL in the message so Slack can unfurl it
|
||
if msg.url:
|
||
content_list.append(
|
||
{
|
||
'type': 'image',
|
||
'content': msg.url,
|
||
}
|
||
)
|
||
|
||
return content_list
|
||
|
||
@staticmethod
|
||
async def target2yiri(message: str, message_id: str, pic_url: str, bot: SlackClient):
|
||
yiri_msg_list = []
|
||
yiri_msg_list.append(platform_message.Source(id=message_id, time=datetime.datetime.now()))
|
||
if pic_url is not None:
|
||
base64_url = await image.get_slack_image_to_base64(pic_url=pic_url, bot_token=bot.bot_token)
|
||
yiri_msg_list.append(platform_message.Image(base64=base64_url))
|
||
|
||
yiri_msg_list.append(platform_message.Plain(text=message))
|
||
chain = platform_message.MessageChain(yiri_msg_list)
|
||
return chain
|
||
|
||
|
||
class SlackEventConverter(abstract_platform_adapter.AbstractEventConverter):
|
||
@staticmethod
|
||
async def yiri2target(event: platform_events.MessageEvent) -> SlackEvent:
|
||
return event.source_platform_object
|
||
|
||
@staticmethod
|
||
async def target2yiri(event: SlackEvent, bot: SlackClient):
|
||
yiri_chain = await SlackMessageConverter.target2yiri(
|
||
message=event.text, message_id=event.message_id, pic_url=event.pic_url, bot=bot
|
||
)
|
||
|
||
if event.type == 'channel':
|
||
yiri_chain.insert(0, platform_message.At(target='SlackBot'))
|
||
|
||
sender = platform_entities.GroupMember(
|
||
id=event.user_id,
|
||
member_name=str(event.sender_name),
|
||
permission='MEMBER',
|
||
group=platform_entities.Group(
|
||
id=event.channel_id, name='MEMBER', permission=platform_entities.Permission.Member
|
||
),
|
||
special_title='',
|
||
)
|
||
time = int(datetime.datetime.utcnow().timestamp())
|
||
return platform_events.GroupMessage(
|
||
sender=sender, message_chain=yiri_chain, time=time, source_platform_object=event
|
||
)
|
||
|
||
if event.type == 'im':
|
||
return platform_events.FriendMessage(
|
||
sender=platform_entities.Friend(id=event.user_id, nickname=event.sender_name, remark=''),
|
||
message_chain=yiri_chain,
|
||
time=float(datetime.datetime.now().timestamp()),
|
||
source_platform_object=event,
|
||
)
|
||
|
||
|
||
class SlackAdapter(abstract_platform_adapter.AbstractMessagePlatformAdapter):
|
||
bot: SlackClient
|
||
bot_account_id: str
|
||
bot_uuid: str = None
|
||
message_converter: SlackMessageConverter = SlackMessageConverter()
|
||
event_converter: SlackEventConverter = SlackEventConverter()
|
||
config: dict
|
||
|
||
def __init__(self, config: dict, logger: EventLogger):
|
||
required_keys = [
|
||
'bot_token',
|
||
'signing_secret',
|
||
]
|
||
missing_keys = [key for key in required_keys if key not in config]
|
||
if missing_keys:
|
||
raise command_errors.ParamNotEnoughError('Slack机器人缺少相关配置项,请查看文档或联系管理员')
|
||
|
||
bot = SlackClient(
|
||
bot_token=config['bot_token'], signing_secret=config['signing_secret'], logger=logger, unified_mode=True
|
||
)
|
||
|
||
super().__init__(
|
||
config=config,
|
||
logger=logger,
|
||
bot=bot,
|
||
bot_account_id=config['bot_token'],
|
||
)
|
||
|
||
async def reply_message(
|
||
self,
|
||
message_source: platform_events.MessageEvent,
|
||
message: platform_message.MessageChain,
|
||
quote_origin: bool = False,
|
||
):
|
||
slack_event = await SlackEventConverter.yiri2target(message_source)
|
||
|
||
content_list = await SlackMessageConverter.yiri2target(message)
|
||
|
||
for content in content_list:
|
||
# Both text and image (URL) are sent as text messages
|
||
# Slack will auto-unfurl image URLs
|
||
message_content = content['content']
|
||
if slack_event.type == 'channel':
|
||
await self.bot.send_message_to_channel(message_content, slack_event.channel_id)
|
||
if slack_event.type == 'im':
|
||
await self.bot.send_message_to_one(message_content, slack_event.user_id)
|
||
|
||
async def send_message(self, target_type: str, target_id: str, message: platform_message.MessageChain):
|
||
content_list = await SlackMessageConverter.yiri2target(message)
|
||
for content in content_list:
|
||
# Both text and image (URL) are sent as text messages
|
||
# Slack will auto-unfurl image URLs
|
||
message_content = content['content']
|
||
if target_type == 'person':
|
||
await self.bot.send_message_to_one(message_content, target_id)
|
||
if target_type == 'group':
|
||
await self.bot.send_message_to_channel(message_content, target_id)
|
||
|
||
def register_listener(
|
||
self,
|
||
event_type: typing.Type[platform_events.Event],
|
||
callback: typing.Callable[
|
||
[platform_events.Event, abstract_platform_adapter.AbstractMessagePlatformAdapter], None
|
||
],
|
||
):
|
||
async def on_message(event: SlackEvent):
|
||
self.bot_account_id = 'SlackBot'
|
||
try:
|
||
return await callback(await self.event_converter.target2yiri(event, self.bot), self)
|
||
except Exception:
|
||
await self.logger.error(f'Error in slack callback: {traceback.format_exc()}')
|
||
|
||
if event_type == platform_events.FriendMessage:
|
||
self.bot.on_message('im')(on_message)
|
||
elif event_type == platform_events.GroupMessage:
|
||
self.bot.on_message('channel')(on_message)
|
||
|
||
def set_bot_uuid(self, bot_uuid: str):
|
||
"""设置 bot UUID(用于生成 webhook URL)"""
|
||
self.bot_uuid = bot_uuid
|
||
|
||
async def handle_unified_webhook(self, bot_uuid: str, path: str, request):
|
||
"""处理统一 webhook 请求。
|
||
|
||
Args:
|
||
bot_uuid: Bot 的 UUID
|
||
path: 子路径(如果有的话)
|
||
request: Quart Request 对象
|
||
|
||
Returns:
|
||
响应数据
|
||
"""
|
||
return await self.bot.handle_unified_webhook(request)
|
||
|
||
async def run_async(self):
|
||
# 统一 webhook 模式下,不启动独立的 Quart 应用
|
||
# 保持运行但不启动独立端口
|
||
async def keep_alive():
|
||
while True:
|
||
await asyncio.sleep(1)
|
||
|
||
await keep_alive()
|
||
|
||
async def kill(self) -> bool:
|
||
return False
|
||
|
||
async def unregister_listener(
|
||
self,
|
||
event_type: type,
|
||
callback: typing.Callable[
|
||
[platform_events.Event, abstract_platform_adapter.AbstractMessagePlatformAdapter], None
|
||
],
|
||
):
|
||
return super().unregister_listener(event_type, callback)
|