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)