zhenxun_bot/zhenxun/configs/utils/__init__.py

520 lines
14 KiB
Python
Raw Normal View History

from collections import defaultdict
from collections.abc import Callable
2022-12-04 16:46:33 +08:00
import copy
from datetime import datetime
2021-11-04 16:11:50 +08:00
from pathlib import Path
from typing import Any, Literal
modified: basic_plugins/admin_bot_manage/_data_source.py 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
2023-02-18 18:46:54 +08:00
import cattrs
from nonebot.compat import model_dump
from pydantic import BaseModel
from ruamel.yaml import YAML
2022-05-26 22:49:48 +08:00
from ruamel.yaml.scanner import ScannerError
2021-11-04 16:11:50 +08:00
from zhenxun.configs.path_config import DATA_PATH
from zhenxun.services.log import logger
from zhenxun.utils.enum import BlockType, LimitWatchType, PluginLimitType, PluginType
2024-02-04 04:18:54 +08:00
_yaml = YAML(pure=True)
_yaml.indent = 2
_yaml.allow_unicode = True
2024-02-04 04:18:54 +08:00
class Example(BaseModel):
"""
示例
"""
exec: str
"""执行命令"""
description: str = ""
"""命令描述"""
class Command(BaseModel):
"""
具体参数说明
"""
command: str
"""命令"""
params: list[str] = []
"""参数"""
description: str = ""
"""描述"""
examples: list[Example] = []
"""示例列表"""
2024-02-04 04:18:54 +08:00
class RegisterConfig(BaseModel):
"""
注册配置项
"""
key: str
"""配置项键"""
value: Any
"""配置项值"""
module: str | None = None
"""模块名"""
help: str | None
"""配置注解"""
default_value: Any | None = None
"""默认值"""
type: Any = None
"""参数类型"""
arg_parser: Callable | None = None
"""参数解析"""
class ConfigModel(BaseModel):
"""
配置项
"""
value: Any
"""配置项值"""
2024-02-04 04:18:54 +08:00
help: str | None
"""配置注解"""
2024-02-04 04:18:54 +08:00
default_value: Any | None = None
"""默认值"""
type: Any = None
"""参数类型"""
2024-02-04 04:18:54 +08:00
arg_parser: Callable | None = None
"""参数解析"""
def to_dict(self, **kwargs):
return model_dump(self, **kwargs)
class ConfigGroup(BaseModel):
"""
配置组
"""
module: str
"""模块名"""
2024-02-04 04:18:54 +08:00
name: str | None = None
"""插件名"""
configs: dict[str, ConfigModel] = defaultdict()
"""配置项列表"""
2024-02-04 04:18:54 +08:00
def get(self, c: str, default: Any = None) -> Any:
cfg = self.configs.get(c.upper())
2024-02-04 04:18:54 +08:00
if cfg is not None:
2024-07-21 19:06:50 +08:00
if cfg.value is not None:
return cfg.value
if cfg.default_value is not None:
return cfg.default_value
2024-02-04 04:18:54 +08:00
return default
def to_dict(self, **kwargs):
return model_dump(self, **kwargs)
2024-02-04 04:18:54 +08:00
class BaseBlock(BaseModel):
"""
插件阻断基本类插件阻断限制
"""
status: bool = True
"""限制状态"""
check_type: BlockType = BlockType.ALL
"""检查类型"""
watch_type: LimitWatchType = LimitWatchType.USER
"""监听对象"""
result: str | None = None
"""阻断时回复内容"""
_type: PluginLimitType = PluginLimitType.BLOCK
"""类型"""
def to_dict(self, **kwargs):
return model_dump(self, **kwargs)
2024-02-04 04:18:54 +08:00
class PluginCdBlock(BaseBlock):
"""
插件cd限制
"""
cd: int = 5
"""cd"""
_type: PluginLimitType = PluginLimitType.CD
"""类型"""
class PluginCountBlock(BaseBlock):
"""
插件次数限制
"""
max_count: int
"""最大调用次数"""
_type: PluginLimitType = PluginLimitType.COUNT
"""类型"""
class PluginSetting(BaseModel):
"""
插件基本配置
"""
level: int = 5
"""群权限等级"""
default_status: bool = True
"""进群默认开关状态"""
limit_superuser: bool = False
"""是否限制超级用户"""
cost_gold: int = 0
"""调用插件花费金币"""
class SchedulerModel(BaseModel):
trigger: Literal["date", "interval", "cron"]
"""trigger"""
day: int | None = None
"""天数"""
hour: int | None = None
"""小时"""
minute: int | None = None
"""分钟"""
second: int | None = None
""""""
run_date: datetime | None = None
"""运行日期"""
id: str | None = None
"""id"""
max_instances: int | None = None
"""最大运行实例"""
args: list | None = None
"""参数"""
kwargs: dict | None = None
"""参数"""
2024-02-25 03:18:34 +08:00
class Task(BaseBlock):
module: str
"""被动技能模块名"""
name: str
"""被动技能名称"""
status: bool = True
"""全局开关状态"""
2024-08-22 19:10:47 +08:00
create_status: bool = False
"""初次加载默认开关状态"""
default_status: bool = True
"""进群时默认状态"""
scheduler: SchedulerModel | None = None
"""定时任务配置"""
run_func: Callable | None = None
"""运行函数"""
check: Callable | None = None
"""检查函数"""
check_args: list = []
"""检查函数参数"""
2024-02-25 03:18:34 +08:00
2024-02-04 04:18:54 +08:00
class PluginExtraData(BaseModel):
"""
插件扩展信息
"""
author: str | None = None
"""作者"""
version: str | None = None
"""版本"""
2024-02-25 03:18:34 +08:00
plugin_type: PluginType = PluginType.NORMAL
2024-02-04 04:18:54 +08:00
"""插件类型"""
menu_type: str = "功能"
"""菜单类型"""
admin_level: int | None = None
"""管理员插件所需权限等级"""
2024-02-25 03:18:34 +08:00
configs: list[RegisterConfig] | None = None
2024-02-04 04:18:54 +08:00
"""插件配置"""
setting: PluginSetting | None = None
"""插件基本配置"""
2024-02-25 03:18:34 +08:00
limits: list[BaseBlock | PluginCdBlock | PluginCountBlock] | None = None
2024-02-04 04:18:54 +08:00
"""插件限制"""
commands: list[Command] = []
"""命令列表,用于说明帮助"""
2024-02-25 03:18:34 +08:00
tasks: list[Task] | None = None
"""技能被动"""
2024-05-04 13:48:12 +08:00
superuser_help: str | None = None
"""超级用户帮助"""
aliases: set[str] = set()
2024-05-23 13:58:53 +08:00
"""额外名称"""
sql_list: list[str] | None = None
"""常用sql"""
is_show: bool = True
"""是否显示在菜单中"""
2024-02-04 04:18:54 +08:00
def to_dict(self, **kwargs):
return model_dump(self, **kwargs)
class NoSuchConfig(Exception):
pass
2021-11-04 16:11:50 +08:00
class ConfigsManager:
"""
插件配置 资源 管理器
"""
def __init__(self, file: Path):
self._data: dict[str, ConfigGroup] = {}
2022-04-10 22:19:50 +08:00
self._simple_data: dict = {}
2024-02-04 04:18:54 +08:00
self._simple_file = DATA_PATH / "config.yaml"
_yaml = YAML()
2021-11-04 16:11:50 +08:00
if file:
file.parent.mkdir(exist_ok=True, parents=True)
self.file = file
self.load_data()
if self._simple_file.exists():
try:
with self._simple_file.open(encoding="utf8") as f:
self._simple_data = _yaml.load(f)
except ScannerError as e:
raise ScannerError(
f"{e}\n**********************************************\n"
f"****** 可能为config.yaml配置文件填写不规范 ******\n"
f"**********************************************"
) from e
2021-11-04 16:11:50 +08:00
2024-05-15 23:24:35 +08:00
def set_name(self, module: str, name: str):
"""设置插件配置中文名出
参数:
module: 模块名
name: 中文名称
异常:
ValueError: module不能为为空
"""
if not module:
raise ValueError("set_name: module不能为为空")
if data := self._data.get(module):
data.name = name
2021-11-04 16:11:50 +08:00
def add_plugin_config(
self,
module: str,
key: str,
2024-02-04 04:18:54 +08:00
value: Any,
2021-11-04 16:11:50 +08:00
*,
2024-02-04 04:18:54 +08:00
help: str | None = None,
default_value: Any = None,
type: type | None = None,
2024-02-04 04:18:54 +08:00
arg_parser: Callable | None = None,
2021-11-29 13:09:47 +08:00
_override: bool = False,
2021-11-04 16:11:50 +08:00
):
2024-02-04 04:18:54 +08:00
"""为插件添加一个配置,不会被覆盖,只有第一个生效
参数:
module: 模块
key:
value:
help: 配置注解.
default_value: 默认值.
type: 值类型.
arg_parser: 值解析器一般与webui配合使用.
_override: 强制覆盖值.
异常:
2024-05-15 23:24:35 +08:00
ValueError: module和key不能为为空
ValueError: 填写错误
2021-11-04 16:11:50 +08:00
"""
2024-02-04 04:18:54 +08:00
if not module or not key:
raise ValueError("add_plugin_config: module和key不能为为空")
if module in self._data and (config := self._data[module].configs.get(key)):
2024-02-04 04:18:54 +08:00
config.help = help
config.arg_parser = arg_parser
config.type = type
if _override:
config.value = value
config.default_value = default_value
else:
2021-11-04 16:11:50 +08:00
key = key.upper()
if not self._data.get(module):
self._data[module] = ConfigGroup(module=module)
2024-02-04 04:18:54 +08:00
self._data[module].configs[key] = ConfigModel(
value=value,
2024-02-04 04:18:54 +08:00
help=help,
default_value=default_value,
type=type,
)
2021-11-04 16:11:50 +08:00
def set_config(
self,
module: str,
key: str,
value: Any,
auto_save: bool = False,
):
2024-02-04 04:18:54 +08:00
"""设置配置值
参数:
module: 模块名
key: 配置名称
value:
auto_save: 自动保存.
2021-11-04 16:11:50 +08:00
"""
2024-10-29 08:26:41 +08:00
key = key.upper()
if module in self._data:
2024-10-29 08:26:41 +08:00
if self._data[module].configs.get(key):
self._data[module].configs[key].value = value
2024-10-29 08:26:41 +08:00
else:
self.add_plugin_config(module, key, value)
self._simple_data[module][key] = value
if auto_save:
2024-10-29 08:26:41 +08:00
self.save(save_simple_data=True)
2021-11-04 16:11:50 +08:00
2024-02-04 04:18:54 +08:00
def get_config(self, module: str, key: str, default: Any = None) -> Any:
"""获取指定配置值
参数:
module: 模块名
key: 配置键
default: 没有key值内容的默认返回值.
异常:
NoSuchConfig: 未查询到配置
返回:
Any: 配置值
2021-11-04 16:11:50 +08:00
"""
logger.debug(
f"尝试获取配置MODULE: [<u><y>{module}</y></u>] | KEY: [<u><y>{key}</y></u>]"
)
2021-11-04 16:11:50 +08:00
key = key.upper()
value = None
if module in self._data.keys():
config = self._data[module].configs.get(key) or self._data[
module
].configs.get(key)
if not config:
2024-02-25 03:18:34 +08:00
raise NoSuchConfig(
f"未查询到配置项 MODULE: [ {module} ] | KEY: [ {key} ]"
)
try:
if config.arg_parser:
value = config.arg_parser(value or config.default_value)
elif config.value is not None:
# try:
value = (
cattrs.structure(config.value, config.type)
if config.type
else config.value
)
elif config.default_value is not None:
value = (
cattrs.structure(config.default_value, config.type)
if config.type
else config.default_value
)
except Exception as e:
logger.warning(
f"配置项类型转换 MODULE: [<u><y>{module}</y></u>]"
" | KEY: [<u><y>{key}</y></u>]",
e=e,
)
value = config.value or config.default_value
2023-03-02 18:17:07 +08:00
if value is None:
value = default
logger.debug(
f"获取配置 MODULE: [<u><y>{module}</y></u>] | "
2024-08-30 23:50:45 +08:00
f" KEY: [<u><y>{key}</y></u>] -> [<u><c>{value}</c></u>]"
)
return value
2021-11-04 16:11:50 +08:00
2024-02-04 04:18:54 +08:00
def get(self, key: str) -> ConfigGroup:
"""获取插件配置数据
2021-11-29 13:09:47 +08:00
2024-02-04 04:18:54 +08:00
参数:
key: 一般为模块名
2021-11-04 16:11:50 +08:00
2024-02-04 04:18:54 +08:00
返回:
ConfigGroup: ConfigGroup
2021-11-04 16:11:50 +08:00
"""
2024-02-04 04:18:54 +08:00
return self._data.get(key) or ConfigGroup(module="")
def save(self, path: str | Path | None = None, save_simple_data: bool = False):
"""保存数据
参数:
path: 路径.
save_simple_data: 同时保存至config.yaml.
2021-11-04 16:11:50 +08:00
"""
2022-04-10 22:19:50 +08:00
if save_simple_data:
with open(self._simple_file, "w", encoding="utf8") as f:
2024-02-04 04:18:54 +08:00
_yaml.dump(self._simple_data, f)
path = path or self.file
data = {}
for module in self._data:
data[module] = {}
for config in self._data[module].configs:
value = self._data[module].configs[config].dict()
del value["type"]
2024-02-04 04:18:54 +08:00
del value["arg_parser"]
data[module][config] = value
2021-11-04 16:11:50 +08:00
with open(path, "w", encoding="utf8") as f:
2024-02-04 04:18:54 +08:00
_yaml.dump(data, f)
2021-11-04 16:11:50 +08:00
2022-02-19 18:20:19 +08:00
def reload(self):
2024-02-04 04:18:54 +08:00
"""重新加载配置文件"""
if self._simple_file.exists():
with open(self._simple_file, encoding="utf8") as f:
self._simple_data = _yaml.load(f)
for key in self._simple_data.keys():
for k in self._simple_data[key].keys():
self._data[key].configs[k].value = self._simple_data[key][k]
2022-02-19 18:20:19 +08:00
self.save()
def load_data(self):
2024-02-04 04:18:54 +08:00
"""加载数据
2024-02-04 04:18:54 +08:00
异常:
ValueError: 配置文件为空
"""
if not self.file.exists():
return
with open(self.file, encoding="utf8") as f:
temp_data = _yaml.load(f)
if not temp_data:
self.file.unlink()
raise ValueError(
"配置文件为空!\n"
"***********************************************************\n"
"****** 配置文件 plugins2config.yaml 为空,已删除,请重启 ******\n"
"***********************************************************"
)
count = 0
for module in temp_data:
config_group = ConfigGroup(module=module)
for config in temp_data[module]:
config_group.configs[config] = ConfigModel(**temp_data[module][config])
count += 1
self._data[module] = config_group
logger.info(
f"加载配置完成,共加载 <u><y>{len(temp_data)}</y></u> 个配置组及对应"
f" <u><y>{count}</y></u> 个配置项"
)
def get_data(self) -> dict[str, ConfigGroup]:
2022-12-04 16:46:33 +08:00
return copy.deepcopy(self._data)
2021-11-04 16:11:50 +08:00
def is_empty(self) -> bool:
return not bool(self._data)
def keys(self):
return self._data.keys()
def __str__(self):
return str(self._data)
def __setitem__(self, key, value):
self._data[key] = value
def __getitem__(self, key):
return self._data[key]