mirror of
https://github.com/zhenxun-org/zhenxun_bot.git
synced 2025-12-15 06:12:53 +08:00
242 lines
7.2 KiB
Python
242 lines
7.2 KiB
Python
import os
|
||
import random
|
||
import re
|
||
|
||
import ujson as json
|
||
from nonebot_plugin_alconna import UniMessage, UniMsg
|
||
|
||
from zhenxun.configs.config import BotConfig, Config
|
||
from zhenxun.configs.path_config import DATA_PATH, IMAGE_PATH
|
||
from zhenxun.services.log import logger
|
||
from zhenxun.utils.http_utils import AsyncHttpx
|
||
from zhenxun.utils.message import MessageUtils
|
||
|
||
from .utils import ai_message_manager
|
||
|
||
url = "http://openapi.tuling123.com/openapi/api/v2"
|
||
|
||
check_url = "https://v2.alapi.cn/api/censor/text"
|
||
|
||
index = 0
|
||
|
||
anime_data = json.load(open(DATA_PATH / "anime.json", "r", encoding="utf8"))
|
||
|
||
|
||
async def get_chat_result(
|
||
message: UniMsg, user_id: str, nickname: str
|
||
) -> UniMessage | None:
|
||
"""获取 AI 返回值,顺序: 特殊回复 -> 图灵 -> 青云客
|
||
|
||
参数:
|
||
text: 问题
|
||
img_url: 图片链接
|
||
user_id: 用户id
|
||
nickname: 用户昵称
|
||
|
||
返回
|
||
str: 回答
|
||
"""
|
||
global index
|
||
text = message.extract_plain_text()
|
||
ai_message_manager.add_message(user_id, text)
|
||
special_rst = await ai_message_manager.get_result(user_id, nickname)
|
||
if special_rst:
|
||
ai_message_manager.add_result(user_id, special_rst)
|
||
return MessageUtils.build_message(special_rst)
|
||
if index == 5:
|
||
index = 0
|
||
if len(text) < 6 and random.random() < 0.6:
|
||
keys = anime_data.keys()
|
||
for key in keys:
|
||
if text.find(key) != -1:
|
||
return random.choice(anime_data[key]).replace("你", nickname)
|
||
rst = await tu_ling(text, "", user_id)
|
||
if not rst:
|
||
rst = await xie_ai(text)
|
||
if not rst:
|
||
return None
|
||
if nickname:
|
||
if len(nickname) < 5:
|
||
if random.random() < 0.5:
|
||
nickname = "~".join(nickname) + "~"
|
||
if random.random() < 0.2:
|
||
if nickname.find("大人") == -1:
|
||
nickname += "大~人~"
|
||
rst = str(rst).replace("小主人", nickname).replace("小朋友", nickname)
|
||
ai_message_manager.add_result(user_id, rst)
|
||
for t in Config.get_config("ai", "TEXT_FILTER"):
|
||
rst = rst.replace(t, "*")
|
||
return MessageUtils.build_message(rst)
|
||
|
||
|
||
# 图灵接口
|
||
async def tu_ling(text: str, img_url: str, user_id: str) -> str | None:
|
||
"""获取图灵接口的回复
|
||
|
||
参数:
|
||
text: 问题
|
||
img_url: 图片链接
|
||
user_id: 用户id
|
||
|
||
返回
|
||
str: 图灵回复
|
||
"""
|
||
global index
|
||
TL_KEY = Config.get_config("ai", "TL_KEY")
|
||
req = None
|
||
if not TL_KEY:
|
||
return None
|
||
try:
|
||
if text:
|
||
req = {
|
||
"perception": {
|
||
"inputText": {"text": text},
|
||
"selfInfo": {
|
||
"location": {
|
||
"city": "陨石坑",
|
||
"province": "火星",
|
||
"street": "第5坑位",
|
||
}
|
||
},
|
||
},
|
||
"userInfo": {"apiKey": TL_KEY[index], "userId": str(user_id)},
|
||
}
|
||
elif img_url:
|
||
req = {
|
||
"reqType": 1,
|
||
"perception": {
|
||
"inputImage": {"url": img_url},
|
||
"selfInfo": {
|
||
"location": {
|
||
"city": "陨石坑",
|
||
"province": "火星",
|
||
"street": "第5坑位",
|
||
}
|
||
},
|
||
},
|
||
"userInfo": {"apiKey": TL_KEY[index], "userId": str(user_id)},
|
||
}
|
||
except IndexError:
|
||
index = 0
|
||
return None
|
||
text = ""
|
||
response = await AsyncHttpx.post(url, json=req)
|
||
if response.status_code != 200:
|
||
return None
|
||
resp_payload = json.loads(response.text)
|
||
if int(resp_payload["intent"]["code"]) in [4003]:
|
||
return None
|
||
if resp_payload["results"]:
|
||
for result in resp_payload["results"]:
|
||
if result["resultType"] == "text":
|
||
text = result["values"]["text"]
|
||
if "请求次数超过" in text:
|
||
text = ""
|
||
return text
|
||
|
||
|
||
# 屑 AI
|
||
async def xie_ai(text: str) -> str:
|
||
"""获取青云客回复
|
||
|
||
参数:
|
||
text: 问题
|
||
|
||
返回:
|
||
str: 青云可回复
|
||
"""
|
||
res = await AsyncHttpx.get(
|
||
f"http://api.qingyunke.com/api.php?key=free&appid=0&msg={text}"
|
||
)
|
||
content = ""
|
||
try:
|
||
data = json.loads(res.text)
|
||
if data["result"] == 0:
|
||
content = data["content"]
|
||
if "菲菲" in content:
|
||
content = content.replace("菲菲", BotConfig.self_nickname)
|
||
if "艳儿" in content:
|
||
content = content.replace("艳儿", BotConfig.self_nickname)
|
||
if "公众号" in content:
|
||
content = ""
|
||
if "{br}" in content:
|
||
content = content.replace("{br}", "\n")
|
||
if "提示" in content:
|
||
content = content[: content.find("提示")]
|
||
if "淘宝" in content or "taobao.com" in content:
|
||
return ""
|
||
while True:
|
||
r = re.search("{face:(.*)}", content)
|
||
if r:
|
||
id_ = r.group(1)
|
||
content = content.replace("{" + f"face:{id_}" + "}", "")
|
||
else:
|
||
break
|
||
return (
|
||
content
|
||
if not content and not Config.get_config("ai", "ALAPI_AI_CHECK")
|
||
else await check_text(content)
|
||
)
|
||
except Exception as e:
|
||
logger.error(f"Ai xie_ai 发生错误", e=e)
|
||
return ""
|
||
|
||
|
||
def hello() -> UniMessage:
|
||
"""一些打招呼的内容"""
|
||
result = random.choice(
|
||
(
|
||
"哦豁?!",
|
||
"你好!Ov<",
|
||
f"库库库,呼唤{BotConfig.self_nickname}做什么呢",
|
||
"我在呢!",
|
||
"呼呼,叫俺干嘛",
|
||
)
|
||
)
|
||
img = random.choice(os.listdir(IMAGE_PATH / "zai"))
|
||
return MessageUtils.build_message([IMAGE_PATH / "zai" / img, result])
|
||
|
||
|
||
def no_result() -> UniMessage:
|
||
"""
|
||
没有回答时的回复
|
||
"""
|
||
return MessageUtils.build_message(
|
||
[
|
||
random.choice(
|
||
[
|
||
"你在说啥子?",
|
||
f"纯洁的{BotConfig.self_nickname}没听懂",
|
||
"下次再告诉你(下次一定)",
|
||
"你觉得我听懂了吗?嗯?",
|
||
"我!不!知!道!",
|
||
]
|
||
),
|
||
IMAGE_PATH
|
||
/ "noresult"
|
||
/ random.choice(os.listdir(IMAGE_PATH / "noresult")),
|
||
]
|
||
)
|
||
|
||
|
||
async def check_text(text: str) -> str:
|
||
"""ALAPI文本检测,主要针对青云客API,检测为恶俗文本改为无回复的回答
|
||
|
||
参数:
|
||
text: 回复
|
||
|
||
返回:
|
||
str: 检测文本
|
||
"""
|
||
if not Config.get_config("alapi", "ALAPI_TOKEN"):
|
||
return text
|
||
params = {"token": Config.get_config("alapi", "ALAPI_TOKEN"), "text": text}
|
||
try:
|
||
data = (await AsyncHttpx.get(check_url, timeout=2, params=params)).json()
|
||
if data["code"] == 200:
|
||
if data["data"]["conclusion_type"] == 2:
|
||
return ""
|
||
except Exception as e:
|
||
logger.error(f"检测违规文本错误...", e=e)
|
||
return text
|