mirror of
https://github.com/zhenxun-org/zhenxun_bot.git
synced 2025-12-15 06:12:53 +08:00
216 lines
6.7 KiB
Python
Executable File
216 lines
6.7 KiB
Python
Executable File
import os
|
||
import random
|
||
import re
|
||
from utils.http_utils import AsyncHttpx
|
||
from configs.path_config import IMAGE_PATH, DATA_PATH
|
||
from services.log import logger
|
||
from utils.message_builder import image, face
|
||
from configs.config import Config, NICKNAME
|
||
from .utils import ai_message_manager
|
||
|
||
try:
|
||
import ujson as json
|
||
except ModuleNotFoundError:
|
||
import json
|
||
|
||
|
||
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(text: str, img_url: str, user_id: int, nickname: str) -> str:
|
||
"""
|
||
获取 AI 返回值,顺序: 特殊回复 -> 图灵 -> 青云客
|
||
:param text: 问题
|
||
:param img_url: 图片链接
|
||
:param user_id: 用户id
|
||
:param nickname: 用户昵称
|
||
:return: 回答
|
||
"""
|
||
global index
|
||
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 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, img_url, user_id)
|
||
if not rst:
|
||
rst = await xie_ai(text)
|
||
if not rst:
|
||
return no_result()
|
||
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)
|
||
return rst
|
||
|
||
|
||
# 图灵接口
|
||
async def tu_ling(text: str, img_url: str, user_id: int) -> str:
|
||
"""
|
||
获取图灵接口的回复
|
||
:param text: 问题
|
||
:param img_url: 图片链接
|
||
:param user_id: 用户id
|
||
:return: 图灵回复
|
||
"""
|
||
global index
|
||
TL_KEY = Config.get_config("ai", "TL_KEY")
|
||
req = None
|
||
if not TL_KEY:
|
||
return ""
|
||
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 ""
|
||
text = ""
|
||
response = await AsyncHttpx.post(url, json=req)
|
||
if response.status_code != 200:
|
||
return no_result()
|
||
resp_payload = json.loads(response.text)
|
||
if int(resp_payload["intent"]["code"]) in [4003]:
|
||
return ""
|
||
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:
|
||
"""
|
||
获取青云客回复
|
||
:param text: 问题
|
||
:return: 青云可回复
|
||
"""
|
||
res = await AsyncHttpx.get(f"http://api.qingyunke.com/api.php?key=free&appid=0&msg={text}")
|
||
content = ""
|
||
data = json.loads(res.text)
|
||
if data["result"] == 0:
|
||
content = data["content"]
|
||
if "菲菲" in content:
|
||
content = content.replace("菲菲", NICKNAME)
|
||
if "艳儿" in content:
|
||
content = content.replace("艳儿", 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_}" + "}", str(face(int(id_)))
|
||
)
|
||
else:
|
||
break
|
||
return (
|
||
content
|
||
if not content and not Config.get_config("ai", "ALAPI_AI_CHECK")
|
||
else await check_text(content)
|
||
)
|
||
|
||
|
||
def hello() -> str:
|
||
"""
|
||
一些打招呼的内容
|
||
"""
|
||
result = random.choice(
|
||
(
|
||
"哦豁?!",
|
||
"你好!Ov<",
|
||
f"库库库,呼唤{NICKNAME}做什么呢",
|
||
"我在呢!",
|
||
"呼呼,叫俺干嘛",
|
||
)
|
||
)
|
||
img = random.choice(os.listdir(IMAGE_PATH / "zai"))
|
||
if img[-4:] == ".gif":
|
||
result += image(img, "zai")
|
||
else:
|
||
result += image(img, "zai")
|
||
return result
|
||
|
||
|
||
# 没有回答时回复内容
|
||
def no_result() -> str:
|
||
"""
|
||
没有回答时的回复
|
||
"""
|
||
return (
|
||
random.choice(
|
||
[
|
||
"你在说啥子?",
|
||
f"纯洁的{NICKNAME}没听懂",
|
||
"下次再告诉你(下次一定)",
|
||
"你觉得我听懂了吗?嗯?",
|
||
"我!不!知!道!",
|
||
]
|
||
)
|
||
+ image(random.choice(os.listdir(IMAGE_PATH / "noresult")), "noresult")
|
||
)
|
||
|
||
|
||
async def check_text(text: str) -> str:
|
||
"""
|
||
ALAPI文本检测,主要针对青云客API,检测为恶俗文本改为无回复的回答
|
||
:param text: 回复
|
||
"""
|
||
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"检测违规文本错误...{type(e)}:{e}")
|
||
return text
|