|
123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137 |
- import asyncio
- import json
- import threading
- from concurrent.futures import ThreadPoolExecutor
-
- import plugins
- from bridge.context import ContextType
- from bridge.reply import Reply, ReplyType
- from channel.chat_message import ChatMessage
- from common.log import logger
- from config import conf, global_config
- from plugins import *
- from .midjourney import MJBot, TaskType
-
- # 任务线程池
- task_thread_pool = ThreadPoolExecutor(max_workers=4)
-
-
- @plugins.register(
- name="linkai",
- desc="A plugin that supports knowledge base and midjourney drawing.",
- version="0.1.0",
- author="https://link-ai.tech",
- )
- class LinkAI(Plugin):
- def __init__(self):
- super().__init__()
- self.handlers[Event.ON_HANDLE_CONTEXT] = self.on_handle_context
- self.config = super().load_config()
- if self.config:
- self.mj_bot = MJBot(self.config.get("midjourney"))
- logger.info("[LinkAI] inited")
-
- def on_handle_context(self, e_context: EventContext):
- """
- 消息处理逻辑
- :param e_context: 消息上下文
- """
- if not self.config:
- return
-
- context = e_context['context']
- if context.type not in [ContextType.TEXT, ContextType.IMAGE, ContextType.IMAGE_CREATE]:
- # filter content no need solve
- return
-
- mj_type = self.mj_bot.judge_mj_task_type(e_context)
- if mj_type:
- # MJ作图任务处理
- self.mj_bot.process_mj_task(mj_type, e_context)
- return
-
- if context.content.startswith(f"{_get_trigger_prefix()}linkai"):
- # 应用管理功能
- self._process_admin_cmd(e_context)
- return
-
- if self._is_chat_task(e_context):
- # 文本对话任务处理
- self._process_chat_task(e_context)
-
- # 插件管理功能
- def _process_admin_cmd(self, e_context: EventContext):
- context = e_context['context']
- cmd = context.content.split()
- if len(cmd) == 1 or (len(cmd) == 2 and cmd[1] == "help"):
- _set_reply_text(self.get_help_text(verbose=True), e_context, level=ReplyType.INFO)
- return
- if len(cmd) == 3 and cmd[1] == "app":
- if not context.kwargs.get("isgroup"):
- _set_reply_text("该指令需在群聊中使用", e_context, level=ReplyType.ERROR)
- return
- if context.kwargs.get("msg").actual_user_id not in global_config["admin_users"]:
- _set_reply_text("需要管理员权限执行", e_context, level=ReplyType.ERROR)
- return
- app_code = cmd[2]
- group_name = context.kwargs.get("msg").from_user_nickname
- group_mapping = self.config.get("group_app_map")
- if group_mapping:
- group_mapping[group_name] = app_code
- else:
- self.config["group_app_map"] = {group_name: app_code}
- # 保存插件配置
- super().save_config(self.config)
- _set_reply_text(f"应用设置成功: {app_code}", e_context, level=ReplyType.INFO)
- else:
- _set_reply_text(f"指令错误,请输入{_get_trigger_prefix()}linkai help 获取帮助", e_context, level=ReplyType.INFO)
- return
-
- # LinkAI 对话任务处理
- def _is_chat_task(self, e_context: EventContext):
- context = e_context['context']
- # 群聊应用管理
- return self.config.get("group_app_map") and context.kwargs.get("isgroup")
-
- def _process_chat_task(self, e_context: EventContext):
- """
- 处理LinkAI对话任务
- :param e_context: 对话上下文
- """
- context = e_context['context']
- # 群聊应用管理
- group_name = context.kwargs.get("msg").from_user_nickname
- app_code = self._fetch_group_app_code(group_name)
- if app_code:
- context.kwargs['app_code'] = app_code
-
- def _fetch_group_app_code(self, group_name: str) -> str:
- """
- 根据群聊名称获取对应的应用code
- :param group_name: 群聊名称
- :return: 应用code
- """
- group_mapping = self.config.get("group_app_map")
- if group_mapping:
- app_code = group_mapping.get(group_name) or group_mapping.get("ALL_GROUP")
- return app_code
-
- def get_help_text(self, verbose=False, **kwargs):
- trigger_prefix = _get_trigger_prefix()
- help_text = "用于集成 LinkAI 提供的知识库、Midjourney绘画等能力。\n\n"
- if not verbose:
- return help_text
- help_text += f'📖 知识库\n - 群聊中指定应用: {trigger_prefix}linkai app 应用编码\n\n例如: \n"$linkai app Kv2fXJcH"\n\n'
- help_text += f"🎨 绘画\n - 生成: {trigger_prefix}mj 描述词1, 描述词2.. \n - 放大: {trigger_prefix}mju 图片ID 图片序号\n\n例如:\n\"{trigger_prefix}mj a little cat, white --ar 9:16\"\n\"{trigger_prefix}mju 1105592717188272288 2\""
- return help_text
-
-
- # 静态方法
- def _set_reply_text(content: str, e_context: EventContext, level: ReplyType = ReplyType.ERROR):
- reply = Reply(level, content)
- e_context["reply"] = reply
- e_context.action = EventAction.BREAK_PASS
-
-
- def _get_trigger_prefix():
- return conf().get("plugin_trigger_prefix", "$")
|