@@ -46,6 +46,8 @@ DEMO视频:https://cdn.link-ai.tech/doc/cow_demo.mp4 | |||||
# 🏷 更新日志 | # 🏷 更新日志 | ||||
>**2024.08.02:** [1.7.0版本](https://github.com/zhayujie/chatgpt-on-wechat/releases/tag/1.6.9) 新增 讯飞4.0 模型、知识库引用来源展示、相关插件优化 | |||||
>**2024.07.19:** [1.6.9版本](https://github.com/zhayujie/chatgpt-on-wechat/releases/tag/1.6.9) 新增 gpt-4o-mini 模型、阿里语音识别、企微应用渠道路由优化 | >**2024.07.19:** [1.6.9版本](https://github.com/zhayujie/chatgpt-on-wechat/releases/tag/1.6.9) 新增 gpt-4o-mini 模型、阿里语音识别、企微应用渠道路由优化 | ||||
>**2024.07.05:** [1.6.8版本](https://github.com/zhayujie/chatgpt-on-wechat/releases/tag/1.6.8) 和 [1.6.7版本](https://github.com/zhayujie/chatgpt-on-wechat/releases/tag/1.6.7),Claude3.5, Gemini 1.5 Pro, MiniMax模型、工作流图片输入、模型列表完善 | >**2024.07.05:** [1.6.8版本](https://github.com/zhayujie/chatgpt-on-wechat/releases/tag/1.6.8) 和 [1.6.7版本](https://github.com/zhayujie/chatgpt-on-wechat/releases/tag/1.6.7),Claude3.5, Gemini 1.5 Pro, MiniMax模型、工作流图片输入、模型列表完善 | ||||
@@ -41,14 +41,15 @@ class XunFeiBot(Bot): | |||||
self.api_key = conf().get("xunfei_api_key") | self.api_key = conf().get("xunfei_api_key") | ||||
self.api_secret = conf().get("xunfei_api_secret") | self.api_secret = conf().get("xunfei_api_secret") | ||||
# 默认使用v2.0版本: "generalv2" | # 默认使用v2.0版本: "generalv2" | ||||
# v1.5版本为 "general" | |||||
# v3.0版本为: "generalv3" | |||||
self.domain = "generalv3" | |||||
# 默认使用v2.0版本: "ws://spark-api.xf-yun.com/v2.1/chat" | |||||
# v1.5版本为: "ws://spark-api.xf-yun.com/v1.1/chat" | |||||
# v3.0版本为: "ws://spark-api.xf-yun.com/v3.1/chat" | |||||
# v3.5版本为: "wss://spark-api.xf-yun.com/v3.5/chat" | |||||
self.spark_url = "wss://spark-api.xf-yun.com/v3.5/chat" | |||||
# Spark Lite请求地址(spark_url): wss://spark-api.xf-yun.com/v1.1/chat, 对应的domain参数为: "general" | |||||
# Spark V2.0请求地址(spark_url): wss://spark-api.xf-yun.com/v2.1/chat, 对应的domain参数为: "generalv2" | |||||
# Spark Pro 请求地址(spark_url): wss://spark-api.xf-yun.com/v3.1/chat, 对应的domain参数为: "generalv3" | |||||
# Spark Pro-128K请求地址(spark_url): wss://spark-api.xf-yun.com/chat/pro-128k, 对应的domain参数为: "pro-128k" | |||||
# Spark Max 请求地址(spark_url): wss://spark-api.xf-yun.com/v3.5/chat, 对应的domain参数为: "generalv3.5" | |||||
# Spark4.0 Ultra 请求地址(spark_url): wss://spark-api.xf-yun.com/v4.0/chat, 对应的domain参数为: "4.0Ultra" | |||||
# 后续模型更新,对应的参数可以参考官网文档获取:https://www.xfyun.cn/doc/spark/Web.html | |||||
self.domain = conf().get("xunfei_domain", "generalv3.5") | |||||
self.spark_url = conf().get("xunfei_spark_url", "wss://spark-api.xf-yun.com/v3.5/chat") | |||||
self.host = urlparse(self.spark_url).netloc | self.host = urlparse(self.spark_url).netloc | ||||
self.path = urlparse(self.spark_url).path | self.path = urlparse(self.spark_url).path | ||||
# 和wenxin使用相同的session机制 | # 和wenxin使用相同的session机制 | ||||
@@ -100,7 +100,7 @@ class DingTalkChanel(ChatChannel, dingtalk_stream.ChatbotHandler): | |||||
super(dingtalk_stream.ChatbotHandler, self).__init__() | super(dingtalk_stream.ChatbotHandler, self).__init__() | ||||
self.logger = self.setup_logger() | self.logger = self.setup_logger() | ||||
# 历史消息id暂存,用于幂等控制 | # 历史消息id暂存,用于幂等控制 | ||||
self.receivedMsgs = ExpiredDict(conf().get("expires_in_seconds")) | |||||
self.receivedMsgs = ExpiredDict(conf().get("expires_in_seconds", 3600)) | |||||
logger.info("[DingTalk] client_id={}, client_secret={} ".format( | logger.info("[DingTalk] client_id={}, client_secret={} ".format( | ||||
self.dingtalk_client_id, self.dingtalk_client_secret)) | self.dingtalk_client_id, self.dingtalk_client_secret)) | ||||
# 无需群校验和前缀 | # 无需群校验和前缀 | ||||
@@ -9,7 +9,6 @@ import json | |||||
import os | import os | ||||
import threading | import threading | ||||
import time | import time | ||||
import requests | import requests | ||||
from bridge.context import * | from bridge.context import * | ||||
@@ -21,6 +20,7 @@ from common.expired_dict import ExpiredDict | |||||
from common.log import logger | from common.log import logger | ||||
from common.singleton import singleton | from common.singleton import singleton | ||||
from common.time_check import time_checker | from common.time_check import time_checker | ||||
from common.utils import convert_webp_to_png | |||||
from config import conf, get_appdata_dir | from config import conf, get_appdata_dir | ||||
from lib import itchat | from lib import itchat | ||||
from lib.itchat.content import * | from lib.itchat.content import * | ||||
@@ -109,7 +109,7 @@ class WechatChannel(ChatChannel): | |||||
def __init__(self): | def __init__(self): | ||||
super().__init__() | super().__init__() | ||||
self.receivedMsgs = ExpiredDict(conf().get("expires_in_seconds")) | |||||
self.receivedMsgs = ExpiredDict(conf().get("expires_in_seconds", 3600)) | |||||
self.auto_login_times = 0 | self.auto_login_times = 0 | ||||
def startup(self): | def startup(self): | ||||
@@ -229,6 +229,12 @@ class WechatChannel(ChatChannel): | |||||
image_storage.write(block) | image_storage.write(block) | ||||
logger.info(f"[WX] download image success, size={size}, img_url={img_url}") | logger.info(f"[WX] download image success, size={size}, img_url={img_url}") | ||||
image_storage.seek(0) | image_storage.seek(0) | ||||
if ".webp" in img_url: | |||||
try: | |||||
image_storage = convert_webp_to_png(image_storage) | |||||
except Exception as e: | |||||
logger.error(f"Failed to convert image: {e}") | |||||
return | |||||
itchat.send_image(image_storage, toUserName=receiver) | itchat.send_image(image_storage, toUserName=receiver) | ||||
logger.info("[WX] sendImage url={}, receiver={}".format(img_url, receiver)) | logger.info("[WX] sendImage url={}, receiver={}".format(img_url, receiver)) | ||||
elif reply.type == ReplyType.IMAGE: # 从文件读取图片 | elif reply.type == ReplyType.IMAGE: # 从文件读取图片 | ||||
@@ -266,6 +272,7 @@ def _send_login_success(): | |||||
except Exception as e: | except Exception as e: | ||||
pass | pass | ||||
def _send_logout(): | def _send_logout(): | ||||
try: | try: | ||||
from common.linkai_client import chat_client | from common.linkai_client import chat_client | ||||
@@ -274,6 +281,7 @@ def _send_logout(): | |||||
except Exception as e: | except Exception as e: | ||||
pass | pass | ||||
def _send_qr_code(qrcode_list: list): | def _send_qr_code(qrcode_list: list): | ||||
try: | try: | ||||
from common.linkai_client import chat_client | from common.linkai_client import chat_client | ||||
@@ -281,3 +289,4 @@ def _send_qr_code(qrcode_list: list): | |||||
chat_client.send_qrcode(qrcode_list) | chat_client.send_qrcode(qrcode_list) | ||||
except Exception as e: | except Exception as e: | ||||
pass | pass | ||||
@@ -17,7 +17,7 @@ from channel.wechatcom.wechatcomapp_client import WechatComAppClient | |||||
from channel.wechatcom.wechatcomapp_message import WechatComAppMessage | from channel.wechatcom.wechatcomapp_message import WechatComAppMessage | ||||
from common.log import logger | from common.log import logger | ||||
from common.singleton import singleton | from common.singleton import singleton | ||||
from common.utils import compress_imgfile, fsize, split_string_by_utf8_length | |||||
from common.utils import compress_imgfile, fsize, split_string_by_utf8_length, convert_webp_to_png | |||||
from config import conf, subscribe_msg | from config import conf, subscribe_msg | ||||
from voice.audio_convert import any_to_amr, split_audio | from voice.audio_convert import any_to_amr, split_audio | ||||
@@ -99,6 +99,12 @@ class WechatComAppChannel(ChatChannel): | |||||
image_storage = compress_imgfile(image_storage, 10 * 1024 * 1024 - 1) | image_storage = compress_imgfile(image_storage, 10 * 1024 * 1024 - 1) | ||||
logger.info("[wechatcom] image compressed, sz={}".format(fsize(image_storage))) | logger.info("[wechatcom] image compressed, sz={}".format(fsize(image_storage))) | ||||
image_storage.seek(0) | image_storage.seek(0) | ||||
if ".webp" in img_url: | |||||
try: | |||||
image_storage = convert_webp_to_png(image_storage) | |||||
except Exception as e: | |||||
logger.error(f"Failed to convert image: {e}") | |||||
return | |||||
try: | try: | ||||
response = self.client.media.upload("image", image_storage) | response = self.client.media.upload("image", image_storage) | ||||
logger.debug("[wechatcom] upload image response: {}".format(response)) | logger.debug("[wechatcom] upload image response: {}".format(response)) | ||||
@@ -2,7 +2,7 @@ import io | |||||
import os | import os | ||||
from urllib.parse import urlparse | from urllib.parse import urlparse | ||||
from PIL import Image | from PIL import Image | ||||
from common.log import logger | |||||
def fsize(file): | def fsize(file): | ||||
if isinstance(file, io.BytesIO): | if isinstance(file, io.BytesIO): | ||||
@@ -54,3 +54,17 @@ def split_string_by_utf8_length(string, max_length, max_split=0): | |||||
def get_path_suffix(path): | def get_path_suffix(path): | ||||
path = urlparse(path).path | path = urlparse(path).path | ||||
return os.path.splitext(path)[-1].lstrip('.') | return os.path.splitext(path)[-1].lstrip('.') | ||||
def convert_webp_to_png(webp_image): | |||||
from PIL import Image | |||||
try: | |||||
webp_image.seek(0) | |||||
img = Image.open(webp_image).convert("RGBA") | |||||
png_image = io.BytesIO() | |||||
img.save(png_image, format="PNG") | |||||
png_image.seek(0) | |||||
return png_image | |||||
except Exception as e: | |||||
logger.error(f"Failed to convert WEBP to PNG: {e}") | |||||
raise |
@@ -73,6 +73,8 @@ available_setting = { | |||||
"xunfei_app_id": "", # 讯飞应用ID | "xunfei_app_id": "", # 讯飞应用ID | ||||
"xunfei_api_key": "", # 讯飞 API key | "xunfei_api_key": "", # 讯飞 API key | ||||
"xunfei_api_secret": "", # 讯飞 API secret | "xunfei_api_secret": "", # 讯飞 API secret | ||||
"xunfei_domain": "", # 讯飞模型对应的domain参数,Spark4.0 Ultra为 4.0Ultra,其他模型详见: https://www.xfyun.cn/doc/spark/Web.html | |||||
"xunfei_spark_url": "", # 讯飞模型对应的请求地址,Spark4.0 Ultra为 wss://spark-api.xf-yun.com/v4.0/chat,其他模型参考详见: https://www.xfyun.cn/doc/spark/Web.html | |||||
# claude 配置 | # claude 配置 | ||||
"claude_api_cookie": "", | "claude_api_cookie": "", | ||||
"claude_uuid": "", | "claude_uuid": "", | ||||
@@ -55,7 +55,7 @@ class Keyword(Plugin): | |||||
reply_text = self.keyword[content] | reply_text = self.keyword[content] | ||||
# 判断匹配内容的类型 | # 判断匹配内容的类型 | ||||
if (reply_text.startswith("http://") or reply_text.startswith("https://")) and any(reply_text.endswith(ext) for ext in [".jpg", ".jpeg", ".png", ".gif", ".img"]): | |||||
if (reply_text.startswith("http://") or reply_text.startswith("https://")) and any(reply_text.endswith(ext) for ext in [".jpg", ".webp", ".jpeg", ".png", ".gif", ".img"]): | |||||
# 如果是以 http:// 或 https:// 开头,且".jpg", ".jpeg", ".png", ".gif", ".img"结尾,则认为是图片 URL。 | # 如果是以 http:// 或 https:// 开头,且".jpg", ".jpeg", ".png", ".gif", ".img"结尾,则认为是图片 URL。 | ||||
reply = Reply() | reply = Reply() | ||||
reply.type = ReplyType.IMAGE_URL | reply.type = ReplyType.IMAGE_URL | ||||
@@ -99,7 +99,8 @@ class Role(Plugin): | |||||
if e_context["context"].type != ContextType.TEXT: | if e_context["context"].type != ContextType.TEXT: | ||||
return | return | ||||
btype = Bridge().get_bot_type("chat") | btype = Bridge().get_bot_type("chat") | ||||
if btype not in [const.OPEN_AI, const.CHATGPT, const.CHATGPTONAZURE, const.LINKAI]: | |||||
if btype not in [const.OPEN_AI, const.CHATGPT, const.CHATGPTONAZURE, const.QWEN_DASHSCOPE, const.XUNFEI, const.BAIDU, const.ZHIPU_AI, const.MOONSHOT, const.MiniMax, const.LINKAI]: | |||||
logger.debug(f'不支持的bot: {btype}') | |||||
return | return | ||||
bot = Bridge().get_bot("chat") | bot = Bridge().get_bot("chat") | ||||
content = e_context["context"].content[:] | content = e_context["context"].content[:] | ||||
@@ -22,11 +22,13 @@ class Tool(Plugin): | |||||
def __init__(self): | def __init__(self): | ||||
super().__init__() | super().__init__() | ||||
self.handlers[Event.ON_HANDLE_CONTEXT] = self.on_handle_context | self.handlers[Event.ON_HANDLE_CONTEXT] = self.on_handle_context | ||||
self.app = self._reset_app() | self.app = self._reset_app() | ||||
if not self.tool_config.get("tools"): | |||||
logger.warn("[tool] init failed, ignore ") | |||||
raise Exception("config.json not found") | |||||
logger.info("[tool] inited") | logger.info("[tool] inited") | ||||
def get_help_text(self, verbose=False, **kwargs): | def get_help_text(self, verbose=False, **kwargs): | ||||
help_text = "这是一个能让chatgpt联网,搜索,数字运算的插件,将赋予强大且丰富的扩展能力。" | help_text = "这是一个能让chatgpt联网,搜索,数字运算的插件,将赋予强大且丰富的扩展能力。" | ||||
trigger_prefix = conf().get("plugin_trigger_prefix", "$") | trigger_prefix = conf().get("plugin_trigger_prefix", "$") | ||||