mirror of
https://github.com/langbot-app/LangBot.git
synced 2026-06-09 15:26:03 +00:00
Compare commits
15 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
ffe9c3e0f8 | ||
|
|
e20b79b0ed | ||
|
|
e04d46db2c | ||
|
|
7341435127 | ||
|
|
8b56f94667 | ||
|
|
f5e98d4ebb | ||
|
|
23a0dba470 | ||
|
|
629ebae0e9 | ||
|
|
394d4b3c1b | ||
|
|
a0fd152d19 | ||
|
|
1a62e08bab | ||
|
|
edbc59c117 | ||
|
|
cfdd0f8cb2 | ||
|
|
808f30675d | ||
|
|
46072abb41 |
2
.gitignore
vendored
2
.gitignore
vendored
@@ -19,7 +19,7 @@ cookies.json
|
|||||||
data/labels/announcement_saved.json
|
data/labels/announcement_saved.json
|
||||||
cmdpriv.json
|
cmdpriv.json
|
||||||
tips.py
|
tips.py
|
||||||
.venv
|
venv*
|
||||||
bin/
|
bin/
|
||||||
.vscode
|
.vscode
|
||||||
test_*
|
test_*
|
||||||
|
|||||||
15
README.md
15
README.md
@@ -26,7 +26,7 @@
|
|||||||
[](https://qm.qq.com/q/JLi38whHum)
|
[](https://qm.qq.com/q/JLi38whHum)
|
||||||
[](https://github.com/RockChinQ/LangBot/releases/latest)
|
[](https://github.com/RockChinQ/LangBot/releases/latest)
|
||||||

|

|
||||||
<img src="https://img.shields.io/badge/python-3.10 | 3.11 | 3.12-blue.svg" alt="python">
|
<img src="https://img.shields.io/badge/python-3.10 ~ 3.13 -blue.svg" alt="python">
|
||||||
[](https://gitcode.com/RockChinQ/LangBot)
|
[](https://gitcode.com/RockChinQ/LangBot)
|
||||||
|
|
||||||
[简体中文](README.md) / [English](README_EN.md) / [日本語](README_JP.md)
|
[简体中文](README.md) / [English](README_EN.md) / [日本語](README_JP.md)
|
||||||
@@ -93,6 +93,8 @@
|
|||||||
| 钉钉 | ✅ | |
|
| 钉钉 | ✅ | |
|
||||||
| Discord | ✅ | |
|
| Discord | ✅ | |
|
||||||
| Telegram | ✅ | |
|
| Telegram | ✅ | |
|
||||||
|
| Slack | 🚧 | |
|
||||||
|
| LINE | 🚧 | |
|
||||||
| WhatsApp | 🚧 | |
|
| WhatsApp | 🚧 | |
|
||||||
|
|
||||||
🚧: 正在开发中
|
🚧: 正在开发中
|
||||||
@@ -115,6 +117,7 @@
|
|||||||
| [阿里云百炼](https://bailian.console.aliyun.com/) | ✅ | 大模型聚合平台, LLMOps 平台 |
|
| [阿里云百炼](https://bailian.console.aliyun.com/) | ✅ | 大模型聚合平台, LLMOps 平台 |
|
||||||
| [火山方舟](https://console.volcengine.com/ark/region:ark+cn-beijing/model?vendor=Bytedance&view=LIST_VIEW) | ✅ | 大模型聚合平台, LLMOps 平台 |
|
| [火山方舟](https://console.volcengine.com/ark/region:ark+cn-beijing/model?vendor=Bytedance&view=LIST_VIEW) | ✅ | 大模型聚合平台, LLMOps 平台 |
|
||||||
| [MCP](https://modelcontextprotocol.io/) | ✅ | 支持通过 MCP 协议获取工具 |
|
| [MCP](https://modelcontextprotocol.io/) | ✅ | 支持通过 MCP 协议获取工具 |
|
||||||
|
|
||||||
### TTS
|
### TTS
|
||||||
|
|
||||||
| 平台/模型 | 备注 |
|
| 平台/模型 | 备注 |
|
||||||
@@ -131,9 +134,15 @@
|
|||||||
|
|
||||||
## 😘 社区贡献
|
## 😘 社区贡献
|
||||||
|
|
||||||
LangBot 离不开以下贡献者和社区内所有人的贡献,我们欢迎任何形式的贡献和反馈。
|
感谢以下[代码贡献者](https://github.com/RockChinQ/LangBot/graphs/contributors)和社区里其他成员对 LangBot 的贡献:
|
||||||
|
|
||||||
|
|
||||||
<a href="https://github.com/RockChinQ/LangBot/graphs/contributors">
|
<a href="https://github.com/RockChinQ/LangBot/graphs/contributors">
|
||||||
<img src="https://contrib.rocks/image?repo=RockChinQ/LangBot" />
|
<img src="https://contrib.rocks/image?repo=RockChinQ/LangBot" />
|
||||||
</a>
|
</a>
|
||||||
|
|
||||||
|
以及 LangBot 核心团队成员:
|
||||||
|
|
||||||
|
- [RockChinQ](https://github.com/RockChinQ)
|
||||||
|
- [the-lazy-me](https://github.com/the-lazy-me)
|
||||||
|
- [wangcham](https://github.com/wangcham)
|
||||||
|
- [KaedeSAMA](https://github.com/KaedeSAMA)
|
||||||
12
README_EN.md
12
README_EN.md
@@ -24,7 +24,7 @@
|
|||||||
[](https://discord.gg/wdNEHETs87)
|
[](https://discord.gg/wdNEHETs87)
|
||||||
[](https://github.com/RockChinQ/LangBot/releases/latest)
|
[](https://github.com/RockChinQ/LangBot/releases/latest)
|
||||||
)
|
)
|
||||||
<img src="https://img.shields.io/badge/python-3.10 | 3.11 | 3.12-blue.svg" alt="python">
|
<img src="https://img.shields.io/badge/python-3.10 ~ 3.13 -blue.svg" alt="python">
|
||||||
|
|
||||||
[简体中文](README.md) / [English](README_EN.md) / [日本語](README_JP.md)
|
[简体中文](README.md) / [English](README_EN.md) / [日本語](README_JP.md)
|
||||||
|
|
||||||
@@ -90,6 +90,8 @@ Directly use the released version to run, see the [Manual Deployment](https://do
|
|||||||
| DingTalk | ✅ | |
|
| DingTalk | ✅ | |
|
||||||
| Discord | ✅ | |
|
| Discord | ✅ | |
|
||||||
| Telegram | ✅ | |
|
| Telegram | ✅ | |
|
||||||
|
| Slack | 🚧 | |
|
||||||
|
| LINE | 🚧 | |
|
||||||
| WhatsApp | 🚧 | |
|
| WhatsApp | 🚧 | |
|
||||||
|
|
||||||
🚧: In development
|
🚧: In development
|
||||||
@@ -115,11 +117,15 @@ Directly use the released version to run, see the [Manual Deployment](https://do
|
|||||||
|
|
||||||
## 🤝 Community Contribution
|
## 🤝 Community Contribution
|
||||||
|
|
||||||
Thanks to the following contributors and everyone in the community for their contributions.
|
Thank you for the following [code contributors](https://github.com/RockChinQ/LangBot/graphs/contributors) and other members in the community for their contributions to LangBot:
|
||||||
|
|
||||||
|
|
||||||
<a href="https://github.com/RockChinQ/LangBot/graphs/contributors">
|
<a href="https://github.com/RockChinQ/LangBot/graphs/contributors">
|
||||||
<img src="https://contrib.rocks/image?repo=RockChinQ/LangBot" />
|
<img src="https://contrib.rocks/image?repo=RockChinQ/LangBot" />
|
||||||
</a>
|
</a>
|
||||||
|
|
||||||
|
And the core team members of LangBot:
|
||||||
|
|
||||||
|
- [RockChinQ](https://github.com/RockChinQ)
|
||||||
|
- [the-lazy-me](https://github.com/the-lazy-me)
|
||||||
|
- [wangcham](https://github.com/wangcham)
|
||||||
|
- [KaedeSAMA](https://github.com/KaedeSAMA)
|
||||||
|
|||||||
12
README_JP.md
12
README_JP.md
@@ -23,7 +23,7 @@
|
|||||||
[](https://discord.gg/wdNEHETs87)
|
[](https://discord.gg/wdNEHETs87)
|
||||||
[](https://github.com/RockChinQ/LangBot/releases/latest)
|
[](https://github.com/RockChinQ/LangBot/releases/latest)
|
||||||
)
|
)
|
||||||
<img src="https://img.shields.io/badge/python-3.10 | 3.11 | 3.12-blue.svg" alt="python">
|
<img src="https://img.shields.io/badge/python-3.10 ~ 3.13 -blue.svg" alt="python">
|
||||||
|
|
||||||
[简体中文](README.md) / [English](README_EN.md) / [日本語](README_JP.md)
|
[简体中文](README.md) / [English](README_EN.md) / [日本語](README_JP.md)
|
||||||
|
|
||||||
@@ -89,6 +89,8 @@ LangBotはBTPanelにリストされています。BTPanelをインストール
|
|||||||
| DingTalk | ✅ | |
|
| DingTalk | ✅ | |
|
||||||
| Discord | ✅ | |
|
| Discord | ✅ | |
|
||||||
| Telegram | ✅ | |
|
| Telegram | ✅ | |
|
||||||
|
| Slack | 🚧 | |
|
||||||
|
| LINE | 🚧 | |
|
||||||
| WhatsApp | 🚧 | |
|
| WhatsApp | 🚧 | |
|
||||||
|
|
||||||
🚧: 開発中
|
🚧: 開発中
|
||||||
@@ -114,11 +116,15 @@ LangBotはBTPanelにリストされています。BTPanelをインストール
|
|||||||
|
|
||||||
## 🤝 コミュニティ貢献
|
## 🤝 コミュニティ貢献
|
||||||
|
|
||||||
以下の貢献者とコミュニティの皆さんの貢献に感謝します。
|
LangBot への貢献に対して、以下の [コード貢献者](https://github.com/RockChinQ/LangBot/graphs/contributors) とコミュニティの他のメンバーに感謝します。
|
||||||
|
|
||||||
|
|
||||||
<a href="https://github.com/RockChinQ/LangBot/graphs/contributors">
|
<a href="https://github.com/RockChinQ/LangBot/graphs/contributors">
|
||||||
<img src="https://contrib.rocks/image?repo=RockChinQ/LangBot" />
|
<img src="https://contrib.rocks/image?repo=RockChinQ/LangBot" />
|
||||||
</a>
|
</a>
|
||||||
|
|
||||||
|
LangBot の核心チームメンバー:
|
||||||
|
|
||||||
|
- [RockChinQ](https://github.com/RockChinQ)
|
||||||
|
- [the-lazy-me](https://github.com/the-lazy-me)
|
||||||
|
- [wangcham](https://github.com/wangcham)
|
||||||
|
- [KaedeSAMA](https://github.com/KaedeSAMA)
|
||||||
|
|||||||
@@ -27,7 +27,6 @@ xml_template = """
|
|||||||
</xml>
|
</xml>
|
||||||
"""
|
"""
|
||||||
|
|
||||||
user_msg_queue = {}
|
|
||||||
|
|
||||||
class OAClient():
|
class OAClient():
|
||||||
|
|
||||||
@@ -45,6 +44,7 @@ class OAClient():
|
|||||||
}
|
}
|
||||||
self.access_token_expiry_time = None
|
self.access_token_expiry_time = None
|
||||||
self.msg_id_map = {}
|
self.msg_id_map = {}
|
||||||
|
self.generated_content = {}
|
||||||
|
|
||||||
async def handle_callback_request(self):
|
async def handle_callback_request(self):
|
||||||
|
|
||||||
@@ -87,12 +87,10 @@ class OAClient():
|
|||||||
from_user = root.find("FromUserName").text # 发送者
|
from_user = root.find("FromUserName").text # 发送者
|
||||||
to_user = root.find("ToUserName").text # 机器人
|
to_user = root.find("ToUserName").text # 机器人
|
||||||
|
|
||||||
from pkg.platform.sources import officialaccount
|
|
||||||
|
|
||||||
timeout = 4.80
|
timeout = 4.80
|
||||||
interval = 0.1
|
interval = 0.1
|
||||||
while True:
|
while True:
|
||||||
content = officialaccount.generated_content.pop(message_data["MsgId"], None)
|
content = self.generated_content.pop(message_data["MsgId"], None)
|
||||||
if content:
|
if content:
|
||||||
response_xml = xml_template.format(
|
response_xml = xml_template.format(
|
||||||
to_user=from_user,
|
to_user=from_user,
|
||||||
@@ -172,6 +170,9 @@ class OAClient():
|
|||||||
for handler in self._message_handlers[msg_type]:
|
for handler in self._message_handlers[msg_type]:
|
||||||
await handler(event)
|
await handler(event)
|
||||||
|
|
||||||
|
async def set_message(self,msg_id:int,content:str):
|
||||||
|
self.generated_content[msg_id] = content
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
class OAClientForLongerResponse():
|
class OAClientForLongerResponse():
|
||||||
@@ -190,6 +191,8 @@ class OAClientForLongerResponse():
|
|||||||
}
|
}
|
||||||
self.access_token_expiry_time = None
|
self.access_token_expiry_time = None
|
||||||
self.loading_message = LoadingMessage
|
self.loading_message = LoadingMessage
|
||||||
|
self.msg_queue = {}
|
||||||
|
self.user_msg_queue = {}
|
||||||
|
|
||||||
async def handle_callback_request(self):
|
async def handle_callback_request(self):
|
||||||
try:
|
try:
|
||||||
@@ -222,17 +225,15 @@ class OAClientForLongerResponse():
|
|||||||
from_user = root.find("FromUserName").text
|
from_user = root.find("FromUserName").text
|
||||||
to_user = root.find("ToUserName").text
|
to_user = root.find("ToUserName").text
|
||||||
|
|
||||||
|
|
||||||
from pkg.platform.sources import officialaccount as oa
|
|
||||||
|
|
||||||
|
if self.msg_queue.get(from_user) and self.msg_queue[from_user][0]["content"]:
|
||||||
if oa.msg_queue.get(from_user) and oa.msg_queue[from_user][0]["content"]:
|
|
||||||
queue_top = oa.msg_queue[from_user].pop(0)
|
queue_top = self.msg_queue[from_user].pop(0)
|
||||||
queue_content = queue_top["content"]
|
queue_content = queue_top["content"]
|
||||||
|
|
||||||
# 弹出用户消息
|
# 弹出用户消息
|
||||||
if user_msg_queue.get(from_user) and user_msg_queue[from_user]:
|
if self.user_msg_queue.get(from_user) and self.user_msg_queue[from_user]:
|
||||||
user_msg_queue[from_user].pop(0)
|
self.user_msg_queue[from_user].pop(0)
|
||||||
|
|
||||||
response_xml = xml_template.format(
|
response_xml = xml_template.format(
|
||||||
to_user=from_user,
|
to_user=from_user,
|
||||||
@@ -250,7 +251,7 @@ class OAClientForLongerResponse():
|
|||||||
content=self.loading_message
|
content=self.loading_message
|
||||||
)
|
)
|
||||||
|
|
||||||
if user_msg_queue.get(from_user) and user_msg_queue[from_user][0]["content"]:
|
if self.user_msg_queue.get(from_user) and self.user_msg_queue[from_user][0]["content"]:
|
||||||
return response_xml
|
return response_xml
|
||||||
else:
|
else:
|
||||||
message_data = await self.get_message(xml_msg)
|
message_data = await self.get_message(xml_msg)
|
||||||
@@ -258,7 +259,7 @@ class OAClientForLongerResponse():
|
|||||||
if message_data:
|
if message_data:
|
||||||
event = OAEvent.from_payload(message_data)
|
event = OAEvent.from_payload(message_data)
|
||||||
if event:
|
if event:
|
||||||
user_msg_queue.setdefault(from_user,[]).append(
|
self.user_msg_queue.setdefault(from_user,[]).append(
|
||||||
{
|
{
|
||||||
"content":event.message,
|
"content":event.message,
|
||||||
}
|
}
|
||||||
@@ -318,6 +319,18 @@ class OAClientForLongerResponse():
|
|||||||
for handler in self._message_handlers[msg_type]:
|
for handler in self._message_handlers[msg_type]:
|
||||||
await handler(event)
|
await handler(event)
|
||||||
|
|
||||||
|
async def set_message(self,from_user:int,message_id:int,content:str):
|
||||||
|
if from_user not in self.msg_queue:
|
||||||
|
self.msg_queue[from_user] = []
|
||||||
|
|
||||||
|
self.msg_queue[from_user].append(
|
||||||
|
{
|
||||||
|
"msg_id":message_id,
|
||||||
|
"content":content,
|
||||||
|
}
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|||||||
@@ -66,8 +66,42 @@ class HTTPController:
|
|||||||
|
|
||||||
@self.quart_app.route("/")
|
@self.quart_app.route("/")
|
||||||
async def index():
|
async def index():
|
||||||
return await quart.send_from_directory(frontend_path, "index.html")
|
return await quart.send_from_directory(
|
||||||
|
frontend_path,
|
||||||
|
"index.html",
|
||||||
|
mimetype="text/html"
|
||||||
|
)
|
||||||
|
|
||||||
@self.quart_app.route("/<path:path>")
|
@self.quart_app.route("/<path:path>")
|
||||||
async def static_file(path: str):
|
async def static_file(path: str):
|
||||||
return await quart.send_from_directory(frontend_path, path)
|
|
||||||
|
mimetype = None
|
||||||
|
|
||||||
|
if path.endswith(".html"):
|
||||||
|
mimetype = "text/html"
|
||||||
|
elif path.endswith(".js"):
|
||||||
|
mimetype = "application/javascript"
|
||||||
|
elif path.endswith(".css"):
|
||||||
|
mimetype = "text/css"
|
||||||
|
elif path.endswith(".png"):
|
||||||
|
mimetype = "image/png"
|
||||||
|
elif path.endswith(".jpg"):
|
||||||
|
mimetype = "image/jpeg"
|
||||||
|
elif path.endswith(".jpeg"):
|
||||||
|
mimetype = "image/jpeg"
|
||||||
|
elif path.endswith(".gif"):
|
||||||
|
mimetype = "image/gif"
|
||||||
|
elif path.endswith(".svg"):
|
||||||
|
mimetype = "image/svg+xml"
|
||||||
|
elif path.endswith(".ico"):
|
||||||
|
mimetype = "image/x-icon"
|
||||||
|
elif path.endswith(".json"):
|
||||||
|
mimetype = "application/json"
|
||||||
|
elif path.endswith(".txt"):
|
||||||
|
mimetype = "text/plain"
|
||||||
|
|
||||||
|
return await quart.send_from_directory(
|
||||||
|
frontend_path,
|
||||||
|
path,
|
||||||
|
mimetype=mimetype
|
||||||
|
)
|
||||||
@@ -110,7 +110,7 @@ class PlatformManager:
|
|||||||
if len(self.adapters) == 0:
|
if len(self.adapters) == 0:
|
||||||
self.ap.logger.warning('未运行平台适配器,请根据文档配置并启用平台适配器。')
|
self.ap.logger.warning('未运行平台适配器,请根据文档配置并启用平台适配器。')
|
||||||
|
|
||||||
async def write_back_config(self, adapter_name: str, adapter_inst: msadapter.MessagePlatformAdapter, config: dict):
|
def write_back_config(self, adapter_name: str, adapter_inst: msadapter.MessagePlatformAdapter, config: dict):
|
||||||
index = -2
|
index = -2
|
||||||
|
|
||||||
for i, adapter in enumerate(self.adapters):
|
for i, adapter in enumerate(self.adapters):
|
||||||
@@ -137,7 +137,7 @@ class PlatformManager:
|
|||||||
**config
|
**config
|
||||||
}
|
}
|
||||||
self.ap.platform_cfg.data['platform-adapters'][real_index] = new_cfg
|
self.ap.platform_cfg.data['platform-adapters'][real_index] = new_cfg
|
||||||
await self.ap.platform_cfg.dump_config()
|
self.ap.platform_cfg.dump_config_sync()
|
||||||
|
|
||||||
async def send(self, event: platform_events.MessageEvent, msg: platform_message.MessageChain, adapter: msadapter.MessagePlatformAdapter):
|
async def send(self, event: platform_events.MessageEvent, msg: platform_message.MessageChain, adapter: msadapter.MessagePlatformAdapter):
|
||||||
|
|
||||||
|
|||||||
@@ -57,7 +57,7 @@ class AiocqhttpMessageConverter(adapter.MessageConverter):
|
|||||||
elif msg.path:
|
elif msg.path:
|
||||||
arg = msg.path
|
arg = msg.path
|
||||||
msg_list.append(aiocqhttp.MessageSegment.record(msg.path))
|
msg_list.append(aiocqhttp.MessageSegment.record(msg.path))
|
||||||
elif type(msg) is forward.Forward:
|
elif type(msg) is platform_message.Forward:
|
||||||
|
|
||||||
for node in msg.node_list:
|
for node in msg.node_list:
|
||||||
msg_list.extend((await AiocqhttpMessageConverter.yiri2target(node.message_chain))[0])
|
msg_list.extend((await AiocqhttpMessageConverter.yiri2target(node.message_chain))[0])
|
||||||
@@ -101,69 +101,8 @@ class AiocqhttpMessageConverter(adapter.MessageConverter):
|
|||||||
class AiocqhttpEventConverter(adapter.EventConverter):
|
class AiocqhttpEventConverter(adapter.EventConverter):
|
||||||
|
|
||||||
@staticmethod
|
@staticmethod
|
||||||
async def yiri2target(event: platform_events.Event, bot_account_id: int):
|
async def yiri2target(event: platform_events.MessageEvent, bot_account_id: int):
|
||||||
|
return event.source_platform_object
|
||||||
msg, msg_id, msg_time = await AiocqhttpMessageConverter.yiri2target(event.message_chain)
|
|
||||||
|
|
||||||
if type(event) is platform_events.GroupMessage:
|
|
||||||
role = "member"
|
|
||||||
|
|
||||||
if event.sender.permission == "ADMINISTRATOR":
|
|
||||||
role = "admin"
|
|
||||||
elif event.sender.permission == "OWNER":
|
|
||||||
role = "owner"
|
|
||||||
|
|
||||||
payload = {
|
|
||||||
"post_type": "message",
|
|
||||||
"message_type": "group",
|
|
||||||
"time": int(msg_time.timestamp()),
|
|
||||||
"self_id": bot_account_id,
|
|
||||||
"sub_type": "normal",
|
|
||||||
"anonymous": None,
|
|
||||||
"font": 0,
|
|
||||||
"message": str(msg),
|
|
||||||
"raw_message": str(msg),
|
|
||||||
"sender": {
|
|
||||||
"age": 0,
|
|
||||||
"area": "",
|
|
||||||
"card": "",
|
|
||||||
"level": "",
|
|
||||||
"nickname": event.sender.member_name,
|
|
||||||
"role": role,
|
|
||||||
"sex": "unknown",
|
|
||||||
"title": "",
|
|
||||||
"user_id": event.sender.id,
|
|
||||||
},
|
|
||||||
"user_id": event.sender.id,
|
|
||||||
"message_id": msg_id,
|
|
||||||
"group_id": event.group.id,
|
|
||||||
"message_seq": 0,
|
|
||||||
}
|
|
||||||
|
|
||||||
return aiocqhttp.Event.from_payload(payload)
|
|
||||||
elif type(event) is platform_events.FriendMessage:
|
|
||||||
|
|
||||||
payload = {
|
|
||||||
"post_type": "message",
|
|
||||||
"message_type": "private",
|
|
||||||
"time": int(msg_time.timestamp()),
|
|
||||||
"self_id": bot_account_id,
|
|
||||||
"sub_type": "friend",
|
|
||||||
"target_id": bot_account_id,
|
|
||||||
"message": str(msg),
|
|
||||||
"raw_message": str(msg),
|
|
||||||
"font": 0,
|
|
||||||
"sender": {
|
|
||||||
"age": 0,
|
|
||||||
"nickname": event.sender.nickname,
|
|
||||||
"sex": "unknown",
|
|
||||||
"user_id": event.sender.id,
|
|
||||||
},
|
|
||||||
"message_id": msg_id,
|
|
||||||
"user_id": event.sender.id,
|
|
||||||
}
|
|
||||||
|
|
||||||
return aiocqhttp.Event.from_payload(payload)
|
|
||||||
|
|
||||||
@staticmethod
|
@staticmethod
|
||||||
async def target2yiri(event: aiocqhttp.Event):
|
async def target2yiri(event: aiocqhttp.Event):
|
||||||
@@ -196,6 +135,7 @@ class AiocqhttpEventConverter(adapter.EventConverter):
|
|||||||
),
|
),
|
||||||
message_chain=yiri_chain,
|
message_chain=yiri_chain,
|
||||||
time=event.time,
|
time=event.time,
|
||||||
|
source_platform_object=event
|
||||||
)
|
)
|
||||||
return converted_event
|
return converted_event
|
||||||
elif event.message_type == "private":
|
elif event.message_type == "private":
|
||||||
@@ -207,6 +147,7 @@ class AiocqhttpEventConverter(adapter.EventConverter):
|
|||||||
),
|
),
|
||||||
message_chain=yiri_chain,
|
message_chain=yiri_chain,
|
||||||
time=event.time,
|
time=event.time,
|
||||||
|
source_platform_object=event
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|
||||||
|
|||||||
@@ -310,6 +310,9 @@ class GeWeChatAdapter(adapter.MessagePlatformAdapter):
|
|||||||
async def gewechat_callback():
|
async def gewechat_callback():
|
||||||
data = await quart.request.json
|
data = await quart.request.json
|
||||||
# print(json.dumps(data, indent=4, ensure_ascii=False))
|
# print(json.dumps(data, indent=4, ensure_ascii=False))
|
||||||
|
self.ap.logger.debug(
|
||||||
|
f"Gewechat callback event: {data}"
|
||||||
|
)
|
||||||
|
|
||||||
if 'data' in data:
|
if 'data' in data:
|
||||||
data['Data'] = data['data']
|
data['Data'] = data['data']
|
||||||
@@ -428,26 +431,28 @@ class GeWeChatAdapter(adapter.MessagePlatformAdapter):
|
|||||||
self.config["token"]
|
self.config["token"]
|
||||||
)
|
)
|
||||||
|
|
||||||
app_id, error_msg = self.bot.login(self.config["app_id"])
|
def gewechat_login_process():
|
||||||
if error_msg:
|
|
||||||
raise Exception(f"Gewechat 登录失败: {error_msg}")
|
|
||||||
|
|
||||||
self.config["app_id"] = app_id
|
app_id, error_msg = self.bot.login(self.config["app_id"])
|
||||||
|
if error_msg:
|
||||||
|
raise Exception(f"Gewechat 登录失败: {error_msg}")
|
||||||
|
|
||||||
self.ap.logger.info(f"Gewechat 登录成功,app_id: {app_id}")
|
self.config["app_id"] = app_id
|
||||||
|
|
||||||
await self.ap.platform_mgr.write_back_config('gewechat', self, self.config)
|
self.ap.logger.info(f"Gewechat 登录成功,app_id: {app_id}")
|
||||||
|
|
||||||
# 获取 nickname
|
self.ap.platform_mgr.write_back_config('gewechat', self, self.config)
|
||||||
profile = self.bot.get_profile(self.config["app_id"])
|
|
||||||
self.bot_account_id = profile["data"]["nickName"]
|
# 获取 nickname
|
||||||
|
profile = self.bot.get_profile(self.config["app_id"])
|
||||||
|
self.bot_account_id = profile["data"]["nickName"]
|
||||||
|
|
||||||
|
time.sleep(2)
|
||||||
|
|
||||||
def thread_set_callback():
|
|
||||||
time.sleep(3)
|
|
||||||
ret = self.bot.set_callback(self.config["token"], self.config["callback_url"])
|
ret = self.bot.set_callback(self.config["token"], self.config["callback_url"])
|
||||||
print('设置 Gewechat 回调:', ret)
|
print('设置 Gewechat 回调:', ret)
|
||||||
|
|
||||||
threading.Thread(target=thread_set_callback).start()
|
threading.Thread(target=gewechat_login_process).start()
|
||||||
|
|
||||||
async def shutdown_trigger_placeholder():
|
async def shutdown_trigger_placeholder():
|
||||||
while True:
|
while True:
|
||||||
|
|||||||
@@ -328,6 +328,10 @@ class LarkAdapter(adapter.MessagePlatformAdapter):
|
|||||||
try:
|
try:
|
||||||
data = await quart.request.json
|
data = await quart.request.json
|
||||||
|
|
||||||
|
self.ap.logger.debug(
|
||||||
|
f"Lark callback event: {data}"
|
||||||
|
)
|
||||||
|
|
||||||
if 'encrypt' in data:
|
if 'encrypt' in data:
|
||||||
cipher = AESCipher(self.config['encrypt-key'])
|
cipher = AESCipher(self.config['encrypt-key'])
|
||||||
data = cipher.decrypt_string(data['encrypt'])
|
data = cipher.decrypt_string(data['encrypt'])
|
||||||
|
|||||||
@@ -22,10 +22,6 @@ from ..types import entities as platform_entities
|
|||||||
from ...command.errors import ParamNotEnoughError
|
from ...command.errors import ParamNotEnoughError
|
||||||
|
|
||||||
|
|
||||||
# 生成的ai回答
|
|
||||||
generated_content = {}
|
|
||||||
msg_queue = {}
|
|
||||||
|
|
||||||
class OAMessageConverter(adapter.MessageConverter):
|
class OAMessageConverter(adapter.MessageConverter):
|
||||||
@staticmethod
|
@staticmethod
|
||||||
async def yiri2target(message_chain: platform_message.MessageChain):
|
async def yiri2target(message_chain: platform_message.MessageChain):
|
||||||
@@ -69,7 +65,7 @@ class OAEventConverter(adapter.EventConverter):
|
|||||||
|
|
||||||
class OfficialAccountAdapter(adapter.MessagePlatformAdapter):
|
class OfficialAccountAdapter(adapter.MessagePlatformAdapter):
|
||||||
|
|
||||||
bot : OAClient
|
bot : OAClient | OAClientForLongerResponse
|
||||||
ap : app.Application
|
ap : app.Application
|
||||||
bot_account_id: str
|
bot_account_id: str
|
||||||
message_converter: OAMessageConverter = OAMessageConverter()
|
message_converter: OAMessageConverter = OAMessageConverter()
|
||||||
@@ -114,26 +110,15 @@ class OfficialAccountAdapter(adapter.MessagePlatformAdapter):
|
|||||||
|
|
||||||
|
|
||||||
async def reply_message(self, message_source: platform_events.FriendMessage, message: platform_message.MessageChain, quote_origin: bool = False):
|
async def reply_message(self, message_source: platform_events.FriendMessage, message: platform_message.MessageChain, quote_origin: bool = False):
|
||||||
global generated_content
|
|
||||||
|
|
||||||
content = await OAMessageConverter.yiri2target(
|
content = await OAMessageConverter.yiri2target(
|
||||||
message
|
message
|
||||||
)
|
)
|
||||||
|
if type(self.bot) == OAClient:
|
||||||
generated_content[message_source.message_chain.message_id] = content
|
await self.bot.set_message(message_source.message_chain.message_id,content)
|
||||||
|
if type(self.bot) == OAClientForLongerResponse:
|
||||||
from_user = message_source.sender.id
|
from_user = message_source.sender.id
|
||||||
|
await self.bot.set_message(from_user,message_source.message_chain.message_id,content)
|
||||||
|
|
||||||
if from_user not in msg_queue:
|
|
||||||
msg_queue[from_user] = []
|
|
||||||
|
|
||||||
msg_queue[from_user].append(
|
|
||||||
{
|
|
||||||
"msg_id":message_source.message_chain.message_id,
|
|
||||||
"content":content,
|
|
||||||
}
|
|
||||||
)
|
|
||||||
|
|
||||||
|
|
||||||
async def send_message(
|
async def send_message(
|
||||||
|
|||||||
@@ -25,7 +25,7 @@ class AnthropicMessages(requester.LLMAPIRequester):
|
|||||||
async def initialize(self):
|
async def initialize(self):
|
||||||
|
|
||||||
httpx_client = anthropic._base_client.AsyncHttpxClientWrapper(
|
httpx_client = anthropic._base_client.AsyncHttpxClientWrapper(
|
||||||
base_url=self.ap.provider_cfg.data['requester']['anthropic-messages']['base-url'],
|
base_url=self.ap.provider_cfg.data['requester']['anthropic-messages']['base-url'].replace(' ', ''),
|
||||||
# cast to a valid type because mypy doesn't understand our type narrowing
|
# cast to a valid type because mypy doesn't understand our type narrowing
|
||||||
timeout=typing.cast(httpx.Timeout, self.ap.provider_cfg.data['requester']['anthropic-messages']['timeout']),
|
timeout=typing.cast(httpx.Timeout, self.ap.provider_cfg.data['requester']['anthropic-messages']['timeout']),
|
||||||
limits=anthropic._constants.DEFAULT_CONNECTION_LIMITS,
|
limits=anthropic._constants.DEFAULT_CONNECTION_LIMITS,
|
||||||
@@ -59,9 +59,11 @@ class AnthropicMessages(requester.LLMAPIRequester):
|
|||||||
if m.role == "system":
|
if m.role == "system":
|
||||||
system_role_message = m
|
system_role_message = m
|
||||||
|
|
||||||
messages.pop(i)
|
|
||||||
break
|
break
|
||||||
|
|
||||||
|
if system_role_message:
|
||||||
|
messages.pop(i)
|
||||||
|
|
||||||
if isinstance(system_role_message, llm_entities.Message) \
|
if isinstance(system_role_message, llm_entities.Message) \
|
||||||
and isinstance(system_role_message.content, str):
|
and isinstance(system_role_message.content, str):
|
||||||
args['system'] = system_role_message.content
|
args['system'] = system_role_message.content
|
||||||
|
|||||||
@@ -36,7 +36,7 @@ class OpenAIChatCompletions(requester.LLMAPIRequester):
|
|||||||
|
|
||||||
self.client = openai.AsyncClient(
|
self.client = openai.AsyncClient(
|
||||||
api_key="",
|
api_key="",
|
||||||
base_url=self.requester_cfg['base-url'],
|
base_url=self.requester_cfg['base-url'].replace(' ', ''),
|
||||||
timeout=self.requester_cfg['timeout'],
|
timeout=self.requester_cfg['timeout'],
|
||||||
http_client=httpx.AsyncClient(
|
http_client=httpx.AsyncClient(
|
||||||
trust_env=True,
|
trust_env=True,
|
||||||
@@ -47,8 +47,9 @@ class OpenAIChatCompletions(requester.LLMAPIRequester):
|
|||||||
async def _req(
|
async def _req(
|
||||||
self,
|
self,
|
||||||
args: dict,
|
args: dict,
|
||||||
|
extra_body: dict = {},
|
||||||
) -> chat_completion.ChatCompletion:
|
) -> chat_completion.ChatCompletion:
|
||||||
return await self.client.chat.completions.create(**args)
|
return await self.client.chat.completions.create(**args, extra_body=extra_body)
|
||||||
|
|
||||||
async def _make_msg(
|
async def _make_msg(
|
||||||
self,
|
self,
|
||||||
@@ -73,7 +74,7 @@ class OpenAIChatCompletions(requester.LLMAPIRequester):
|
|||||||
) -> llm_entities.Message:
|
) -> llm_entities.Message:
|
||||||
self.client.api_key = use_model.token_mgr.get_token()
|
self.client.api_key = use_model.token_mgr.get_token()
|
||||||
|
|
||||||
args = self.requester_cfg['args'].copy()
|
args = {}
|
||||||
args["model"] = use_model.name if use_model.model_name is None else use_model.model_name
|
args["model"] = use_model.name if use_model.model_name is None else use_model.model_name
|
||||||
|
|
||||||
if use_funcs:
|
if use_funcs:
|
||||||
@@ -99,7 +100,7 @@ class OpenAIChatCompletions(requester.LLMAPIRequester):
|
|||||||
args["messages"] = messages
|
args["messages"] = messages
|
||||||
|
|
||||||
# 发送请求
|
# 发送请求
|
||||||
resp = await self._req(args)
|
resp = await self._req(args, extra_body=self.requester_cfg['args'])
|
||||||
|
|
||||||
# 处理请求结果
|
# 处理请求结果
|
||||||
message = await self._make_msg(resp)
|
message = await self._make_msg(resp)
|
||||||
|
|||||||
@@ -23,7 +23,7 @@ class DeepseekChatCompletions(chatcmpl.OpenAIChatCompletions):
|
|||||||
) -> llm_entities.Message:
|
) -> llm_entities.Message:
|
||||||
self.client.api_key = use_model.token_mgr.get_token()
|
self.client.api_key = use_model.token_mgr.get_token()
|
||||||
|
|
||||||
args = self.requester_cfg['args'].copy()
|
args = {}
|
||||||
args["model"] = use_model.name if use_model.model_name is None else use_model.model_name
|
args["model"] = use_model.name if use_model.model_name is None else use_model.model_name
|
||||||
|
|
||||||
if use_funcs:
|
if use_funcs:
|
||||||
@@ -43,7 +43,7 @@ class DeepseekChatCompletions(chatcmpl.OpenAIChatCompletions):
|
|||||||
args["messages"] = messages
|
args["messages"] = messages
|
||||||
|
|
||||||
# 发送请求
|
# 发送请求
|
||||||
resp = await self._req(args)
|
resp = await self._req(args, extra_body=self.requester_cfg['args'])
|
||||||
|
|
||||||
if resp is None:
|
if resp is None:
|
||||||
raise errors.RequesterError('接口返回为空,请确定模型提供商服务是否正常')
|
raise errors.RequesterError('接口返回为空,请确定模型提供商服务是否正常')
|
||||||
|
|||||||
@@ -30,7 +30,7 @@ class GiteeAIChatCompletions(chatcmpl.OpenAIChatCompletions):
|
|||||||
) -> llm_entities.Message:
|
) -> llm_entities.Message:
|
||||||
self.client.api_key = use_model.token_mgr.get_token()
|
self.client.api_key = use_model.token_mgr.get_token()
|
||||||
|
|
||||||
args = self.requester_cfg['args'].copy()
|
args = {}
|
||||||
args["model"] = use_model.name if use_model.model_name is None else use_model.model_name
|
args["model"] = use_model.name if use_model.model_name is None else use_model.model_name
|
||||||
|
|
||||||
if use_funcs:
|
if use_funcs:
|
||||||
@@ -46,7 +46,7 @@ class GiteeAIChatCompletions(chatcmpl.OpenAIChatCompletions):
|
|||||||
|
|
||||||
args["messages"] = req_messages
|
args["messages"] = req_messages
|
||||||
|
|
||||||
resp = await self._req(args)
|
resp = await self._req(args, extra_body=self.requester_cfg['args'])
|
||||||
|
|
||||||
message = await self._make_msg(resp)
|
message = await self._make_msg(resp)
|
||||||
|
|
||||||
|
|||||||
@@ -25,7 +25,7 @@ class MoonshotChatCompletions(chatcmpl.OpenAIChatCompletions):
|
|||||||
) -> llm_entities.Message:
|
) -> llm_entities.Message:
|
||||||
self.client.api_key = use_model.token_mgr.get_token()
|
self.client.api_key = use_model.token_mgr.get_token()
|
||||||
|
|
||||||
args = self.requester_cfg['args'].copy()
|
args = {}
|
||||||
args["model"] = use_model.name if use_model.model_name is None else use_model.model_name
|
args["model"] = use_model.name if use_model.model_name is None else use_model.model_name
|
||||||
|
|
||||||
if use_funcs:
|
if use_funcs:
|
||||||
@@ -48,7 +48,7 @@ class MoonshotChatCompletions(chatcmpl.OpenAIChatCompletions):
|
|||||||
args["messages"] = messages
|
args["messages"] = messages
|
||||||
|
|
||||||
# 发送请求
|
# 发送请求
|
||||||
resp = await self._req(args)
|
resp = await self._req(args, extra_body=self.requester_cfg['args'])
|
||||||
|
|
||||||
# 处理请求结果
|
# 处理请求结果
|
||||||
message = await self._make_msg(resp)
|
message = await self._make_msg(resp)
|
||||||
|
|||||||
@@ -225,6 +225,8 @@ class DifyServiceAPIRunner(runner.RequestRunner):
|
|||||||
role="assistant",
|
role="assistant",
|
||||||
content=[llm_entities.ContentElement.from_image_url(image_url)],
|
content=[llm_entities.ContentElement.from_image_url(image_url)],
|
||||||
)
|
)
|
||||||
|
if chunk['event'] == 'error':
|
||||||
|
raise errors.DifyAPIError("dify 服务错误: " + chunk['message'])
|
||||||
|
|
||||||
query.session.using_conversation.uuid = chunk["conversation_id"]
|
query.session.using_conversation.uuid = chunk["conversation_id"]
|
||||||
|
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
semantic_version = "v3.4.11"
|
semantic_version = "v3.4.11.2"
|
||||||
|
|
||||||
debug_mode = False
|
debug_mode = False
|
||||||
|
|
||||||
|
|||||||
Reference in New Issue
Block a user