mirror of
https://github.com/zhenxun-org/zhenxun_bot.git
synced 2025-12-15 14:22:55 +08:00
modified: basic_plugins/admin_bot_manage/admin_config.py modified: basic_plugins/admin_bot_manage/custom_welcome_message.py modified: basic_plugins/admin_bot_manage/timing_task.py modified: basic_plugins/apscheduler/__init__.py modified: basic_plugins/ban/__init__.py modified: basic_plugins/ban/data_source.py modified: basic_plugins/chat_history/chat_message.py modified: basic_plugins/chat_history/chat_message_handle.py modified: basic_plugins/group_handle/__init__.py modified: basic_plugins/hooks/_utils.py modified: basic_plugins/hooks/ban_hook.py modified: basic_plugins/hooks/chkdsk_hook.py modified: basic_plugins/init_plugin_config/__init__.py deleted: basic_plugins/init_plugin_config/init_group_manager.py modified: basic_plugins/invite_manager/__init__.py new file: basic_plugins/invite_manager/utils.py modified: basic_plugins/nickname.py modified: basic_plugins/plugin_shop/__init__.py modified: basic_plugins/plugin_shop/data_source.py modified: basic_plugins/scripts.py modified: basic_plugins/shop/__init__.py modified: basic_plugins/shop/buy.py modified: basic_plugins/shop/gold.py modified: basic_plugins/shop/my_props/__init__.py deleted: basic_plugins/shop/reset_today_gold.py modified: basic_plugins/shop/shop_handle/__init__.py modified: basic_plugins/shop/shop_handle/data_source.py modified: basic_plugins/shop/use/__init__.py modified: basic_plugins/shop/use/data_source.py modified: basic_plugins/super_cmd/__init__.py modified: basic_plugins/super_cmd/bot_friend_group.py modified: basic_plugins/super_cmd/clear_data.py modified: basic_plugins/super_cmd/exec_sql.py modified: basic_plugins/super_cmd/manager_group.py modified: basic_plugins/super_cmd/reload_setting.py modified: basic_plugins/super_cmd/set_admin_permissions.py deleted: basic_plugins/super_cmd/super_task_switch.py modified: basic_plugins/super_cmd/update_friend_group_info.py modified: basic_plugins/super_help/__init__.py modified: basic_plugins/update_info.py modified: configs/config.py modified: configs/utils/__init__.py modified: models/bag_user.py modified: models/ban_user.py modified: models/chat_history.py modified: models/friend_user.py modified: models/goods_info.py modified: models/group_info.py modified: models/group_member_info.py modified: models/level_user.py modified: models/sign_group_user.py modified: models/user_shop_gold_log.py modified: plugins/aconfig/__init__.py modified: plugins/ai/__init__.py modified: plugins/ai/data_source.py modified: plugins/bilibili_sub/__init__.py modified: plugins/bilibili_sub/data_source.py modified: plugins/bilibili_sub/model.py modified: plugins/black_word/__init__.py modified: plugins/black_word/model.py modified: plugins/black_word/utils.py modified: plugins/bt/data_source.py modified: plugins/genshin/almanac/__init__.py modified: plugins/genshin/material_remind/__init__.py modified: plugins/genshin/query_user/_models/__init__.py modified: plugins/genshin/query_user/_utils/__init__.py modified: plugins/genshin/query_user/bind/__init__.py modified: plugins/genshin/query_user/genshin_sign/__init__.py modified: plugins/genshin/query_user/genshin_sign/data_source.py modified: plugins/genshin/query_user/genshin_sign/init_task.py modified: plugins/genshin/query_user/mihoyobbs_sign/__init__.py modified: plugins/genshin/query_user/query_memo/__init__.py modified: plugins/genshin/query_user/query_memo/data_source.py modified: plugins/genshin/query_user/query_role/__init__.py modified: plugins/genshin/query_user/query_role/data_source.py modified: plugins/genshin/query_user/reset_today_query_user_data/__init__.py modified: plugins/genshin/query_user/resin_remind/__init__.py modified: plugins/genshin/query_user/resin_remind/init_task.py modified: plugins/gold_redbag/model.py modified: plugins/image_management/send_image/__init__.py modified: plugins/my_info/__init__.py modified: plugins/open_cases/models/buff_prices.py modified: plugins/open_cases/models/open_cases_user.py modified: plugins/open_cases/open_cases_c.py modified: plugins/open_cases/utils.py modified: plugins/parse_bilibili_json.py modified: plugins/pid_search.py modified: plugins/pix_gallery/__init__.py modified: plugins/pix_gallery/_data_source.py modified: plugins/pix_gallery/_model/omega_pixiv_illusts.py modified: plugins/pix_gallery/_model/pixiv.py modified: plugins/pix_gallery/_model/pixiv_keyword_user.py modified: plugins/pix_gallery/pix_add_keyword.py modified: plugins/pix_gallery/pix_pass_del_keyword.py modified: plugins/pix_gallery/pix_show_info.py modified: plugins/pix_gallery/pix_update.py modified: plugins/pixiv_rank_search/data_source.py modified: plugins/poke/__init__.py modified: plugins/russian/__init__.py modified: plugins/russian/data_source.py modified: plugins/russian/model.py modified: plugins/send_dinggong_voice/__init__.py modified: plugins/send_setu_/_model.py modified: plugins/send_setu_/send_setu/__init__.py modified: plugins/send_setu_/send_setu/data_source.py modified: plugins/send_setu_/update_setu/data_source.py modified: plugins/sign_in/goods_register.py modified: plugins/sign_in/group_user_checkin.py modified: plugins/sign_in/random_event.py modified: plugins/sign_in/utils.py modified: plugins/statistics/_model.py modified: plugins/statistics/statistics_handle.py modified: plugins/statistics/statistics_hook.py modified: plugins/update_picture.py modified: plugins/web_ui/api/request.py modified: plugins/word_bank/_model.py deleted: plugins/word_bank/_old_model.py modified: plugins/word_bank/_rule.py modified: plugins/word_bank/word_handle.py modified: plugins/word_clouds/data_source.py modified: resources/image/sign/sign_res/bar.png modified: resources/image/sign/sign_res/bar_white.png modified: services/db_context.py modified: services/log.py modified: utils/browser.py modified: utils/data_utils.py modified: utils/depends/__init__.py modified: utils/http_utils.py modified: utils/image_utils.py modified: utils/manager/admin_manager.py modified: utils/message_builder.py modified: utils/utils.py
279 lines
9.5 KiB
Python
Executable File
279 lines
9.5 KiB
Python
Executable File
import asyncio
|
||
import os
|
||
import random
|
||
import re
|
||
from asyncio.exceptions import TimeoutError
|
||
from typing import List, Optional, Tuple, Union
|
||
|
||
from asyncpg.exceptions import UniqueViolationError
|
||
from nonebot.adapters.onebot.v11 import Message, MessageSegment
|
||
|
||
from configs.config import NICKNAME, Config
|
||
from configs.path_config import IMAGE_PATH, TEMP_PATH
|
||
from services.log import logger
|
||
from utils.http_utils import AsyncHttpx
|
||
from utils.image_utils import compressed_image, get_img_hash
|
||
from utils.message_builder import image
|
||
from utils.utils import change_img_md5
|
||
|
||
from .._model import Setu
|
||
|
||
try:
|
||
import ujson as json
|
||
except ModuleNotFoundError:
|
||
import json
|
||
|
||
|
||
url = "https://api.lolicon.app/setu/v2"
|
||
path = "_setu"
|
||
r18_path = "_r18"
|
||
host_pattern = re.compile(r"https?://([^/]+)")
|
||
|
||
|
||
# 获取url
|
||
async def get_setu_urls(
|
||
tags: List[str], num: int = 1, r18: bool = False, command: str = ""
|
||
) -> Tuple[List[str], List[str], List[tuple], int]:
|
||
tags = tags[:3] if len(tags) > 3 else tags
|
||
params = {
|
||
"r18": 1 if r18 else 0, # 添加r18参数 0为否,1为是,2为混合
|
||
"tag": tags, # 若指定tag
|
||
"num": 20, # 一次返回的结果数量
|
||
"size": ["original"],
|
||
}
|
||
for count in range(3):
|
||
logger.info(f"get_setu_url: count --> {count}")
|
||
try:
|
||
response = await AsyncHttpx.get(
|
||
url, timeout=Config.get_config("send_setu", "TIMEOUT"), params=params
|
||
)
|
||
if response.status_code == 200:
|
||
data = response.json()
|
||
if not data["error"]:
|
||
data = data["data"]
|
||
(
|
||
urls,
|
||
text_list,
|
||
add_databases_list,
|
||
) = await asyncio.get_event_loop().run_in_executor(
|
||
None, _setu_data_process, data, command
|
||
)
|
||
num = num if num < len(data) else len(data)
|
||
random_idx = random.sample(range(len(data)), num)
|
||
x_urls = []
|
||
x_text_lst = []
|
||
for x in random_idx:
|
||
x_urls.append(urls[x])
|
||
x_text_lst.append(text_list[x])
|
||
if not x_urls:
|
||
return ["没找到符合条件的色图..."], [], [], 401
|
||
return x_urls, x_text_lst, add_databases_list, 200
|
||
else:
|
||
return ["没找到符合条件的色图..."], [], [], 401
|
||
except TimeoutError:
|
||
pass
|
||
except Exception as e:
|
||
logger.error(f"send_setu 访问页面错误 {type(e)}:{e}")
|
||
return ["我网线被人拔了..QAQ"], [], [], 999
|
||
|
||
|
||
headers = {
|
||
"User-Agent": "Mozilla/5.0 (Macintosh; Intel Mac OS X 10.6;"
|
||
" rv:2.0.1) Gecko/20100101 Firefox/4.0.1",
|
||
"Referer": "https://www.pixiv.net",
|
||
}
|
||
|
||
|
||
async def search_online_setu(
|
||
url_: str, id_: Optional[int] = None, path_: Optional[str] = None
|
||
) -> Tuple[Union[MessageSegment, str], int]:
|
||
"""
|
||
下载色图
|
||
:param url_: 色图url
|
||
:param id_: 本地id
|
||
:param path_: 存储路径
|
||
"""
|
||
ws_url = Config.get_config("pixiv", "PIXIV_NGINX_URL")
|
||
if ws_url:
|
||
host_match = re.match(host_pattern, url_)
|
||
host = host_match.group(1)
|
||
url_ = url_.replace(host, ws_url)
|
||
index = random.randint(1, 100000) if id_ is None else id_
|
||
path_ = IMAGE_PATH / path_ if path_ else TEMP_PATH
|
||
file_name = f"{index}_temp_setu.jpg" if path_ == TEMP_PATH else f"{index}.jpg"
|
||
path_.mkdir(parents=True, exist_ok=True)
|
||
for i in range(3):
|
||
logger.info(f"search_online_setu --> {i}")
|
||
try:
|
||
if not await AsyncHttpx.download_file(
|
||
url_,
|
||
path_ / file_name,
|
||
timeout=Config.get_config("send_setu", "TIMEOUT"),
|
||
):
|
||
continue
|
||
if id_ is not None:
|
||
if os.path.getsize(path_ / f"{index}.jpg") > 1024 * 1024 * 1.5:
|
||
compressed_image(
|
||
path_ / f"{index}.jpg",
|
||
)
|
||
logger.info(f"下载 lolicon 图片 {url_} 成功, id:{index}")
|
||
change_img_md5(path_ / file_name)
|
||
return image(path_ / file_name), index
|
||
except TimeoutError:
|
||
pass
|
||
except Exception as e:
|
||
logger.error(f"send_setu 下载图片错误 {type(e)}:{e}")
|
||
return "图片被小怪兽恰掉啦..!QAQ", -1
|
||
|
||
|
||
# 检测本地是否有id涩图,无的话则下载
|
||
async def check_local_exists_or_download(
|
||
setu_image: Setu,
|
||
) -> Tuple[MessageSegment, int]:
|
||
path_ = None
|
||
id_ = None
|
||
if Config.get_config("send_setu", "DOWNLOAD_SETU"):
|
||
id_ = setu_image.local_id
|
||
path_ = r18_path if setu_image.is_r18 else path
|
||
file = IMAGE_PATH / path_ / f"{setu_image.local_id}.jpg"
|
||
if file.exists():
|
||
change_img_md5(file)
|
||
return image(f"{setu_image.local_id}.jpg", path_), 200
|
||
return await search_online_setu(setu_image.img_url, id_, path_)
|
||
|
||
|
||
# 添加涩图数据到数据库
|
||
async def add_data_to_database(lst: List[tuple]):
|
||
tmp = []
|
||
for x in lst:
|
||
if x not in tmp:
|
||
tmp.append(x)
|
||
if tmp:
|
||
for x in tmp:
|
||
try:
|
||
idx = await Setu.filter(is_r18="R-18" in x[5]).count()
|
||
if not await Setu.exists(pid=x[2], img_url=x[4]):
|
||
await Setu.create(
|
||
local_id=idx,
|
||
title=x[0],
|
||
author=x[1],
|
||
pid=x[2],
|
||
img_hash=x[3],
|
||
img_url=x[4],
|
||
tags=x[5],
|
||
is_r18="R-18" in x[5],
|
||
)
|
||
except UniqueViolationError:
|
||
pass
|
||
|
||
|
||
# 拿到本地色图列表
|
||
async def get_setu_list(
|
||
index: Optional[int] = None, tags: Optional[List[str]] = None, r18: bool = False
|
||
) -> Tuple[list, int]:
|
||
if index:
|
||
image_count = await Setu.filter(is_r18=r18).count() - 1
|
||
if index < 0 or index > image_count:
|
||
return [f"超过当前上下限!({image_count})"], 999
|
||
image_list = [await Setu.query_image(index, r18=r18)]
|
||
elif tags:
|
||
image_list = await Setu.query_image(tags=tags, r18=r18)
|
||
else:
|
||
image_list = await Setu.query_image(r18=r18)
|
||
if not image_list:
|
||
return ["没找到符合条件的色图..."], 998
|
||
return image_list, 200
|
||
|
||
|
||
# 初始化消息
|
||
def gen_message(
|
||
setu_image: Setu, img_msg: bool = False
|
||
) -> Union[Message, MessageSegment]:
|
||
local_id = setu_image.local_id
|
||
title = setu_image.title
|
||
author = setu_image.author
|
||
pid = setu_image.pid
|
||
path_ = r18_path if setu_image.is_r18 else path
|
||
image_path = IMAGE_PATH / path_ / f"{local_id}.jpg"
|
||
if Config.get_config("send_setu", "SHOW_INFO"):
|
||
return Message(
|
||
f"id:{local_id}\n"
|
||
f"title:{title}\n"
|
||
f"author:{author}\n"
|
||
f"PID:{pid}\n" + image(image_path)
|
||
)
|
||
return image(image_path)
|
||
|
||
|
||
# 罗翔老师!
|
||
def get_luoxiang(impression):
|
||
probability = (
|
||
float(impression)
|
||
+ Config.get_config("send_setu", "INITIAL_SETU_PROBABILITY") * 100
|
||
)
|
||
if probability < random.randint(1, 101):
|
||
return (
|
||
"我为什么要给你发这个?"
|
||
+ image(
|
||
IMAGE_PATH
|
||
/ "luoxiang"
|
||
/ random.choice(os.listdir(IMAGE_PATH / "luoxiang"))
|
||
)
|
||
+ f"\n(快向{NICKNAME}签到提升好感度吧!)"
|
||
)
|
||
return None
|
||
|
||
|
||
async def find_img_index(img_url, user_id):
|
||
if not await AsyncHttpx.download_file(
|
||
img_url,
|
||
TEMP_PATH / f"{user_id}_find_setu_index.jpg",
|
||
timeout=Config.get_config("send_setu", "TIMEOUT"),
|
||
):
|
||
return "检索图片下载上失败..."
|
||
img_hash = str(get_img_hash(TEMP_PATH / f"{user_id}_find_setu_index.jpg"))
|
||
if setu_img := await Setu.get_or_none(img_hash=img_hash):
|
||
return (
|
||
f"id:{setu_img.local_id}\n"
|
||
f"title:{setu_img.title}\n"
|
||
f"author:{setu_img.author}\n"
|
||
f"PID:{setu_img.pid}"
|
||
)
|
||
return "该图不在色图库中或色图库未更新!"
|
||
|
||
|
||
# 处理色图数据
|
||
def _setu_data_process(data: dict, command: str) -> "list, list, list":
|
||
urls = []
|
||
text_list = []
|
||
add_databases_list = []
|
||
for i in range(len(data)):
|
||
img_url = data[i]["urls"]["original"]
|
||
img_url = (
|
||
img_url.replace("i.pixiv.cat", "i.pximg.net")
|
||
if "i.pixiv.cat" in img_url
|
||
else img_url
|
||
)
|
||
title = data[i]["title"]
|
||
author = data[i]["author"]
|
||
pid = data[i]["pid"]
|
||
urls.append(img_url)
|
||
text_list.append(f"title:{title}\nauthor:{author}\nPID:{pid}")
|
||
tags = []
|
||
for j in range(len(data[i]["tags"])):
|
||
tags.append(data[i]["tags"][j])
|
||
if command != "色图r":
|
||
if "R-18" in tags:
|
||
tags.remove("R-18")
|
||
add_databases_list.append(
|
||
(
|
||
title,
|
||
author,
|
||
pid,
|
||
"",
|
||
img_url,
|
||
",".join(tags),
|
||
)
|
||
)
|
||
return urls, text_list, add_databases_list
|