2024-12-10 19:49:11 +08:00
|
|
|
|
from collections.abc import Callable
|
2022-12-04 16:46:33 +08:00
|
|
|
|
import copy
|
2021-11-04 16:11:50 +08:00
|
|
|
|
from pathlib import Path
|
2025-08-04 23:36:12 +08:00
|
|
|
|
from typing import Any, TypeVar
|
2023-02-18 18:46:54 +08:00
|
|
|
|
|
2023-02-26 22:17:26 +08:00
|
|
|
|
import cattrs
|
2025-08-04 23:36:12 +08:00
|
|
|
|
from pydantic import BaseModel, Field
|
2024-12-10 19:49:11 +08:00
|
|
|
|
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
|
|
|
|
|
2024-08-29 22:01:34 +08:00
|
|
|
|
from zhenxun.configs.path_config import DATA_PATH
|
2024-12-10 19:49:11 +08:00
|
|
|
|
from zhenxun.services.log import logger
|
2025-08-04 23:36:12 +08:00
|
|
|
|
from zhenxun.utils.pydantic_compat import (
|
|
|
|
|
|
_dump_pydantic_obj,
|
|
|
|
|
|
_is_pydantic_type,
|
|
|
|
|
|
model_dump,
|
|
|
|
|
|
parse_as,
|
|
|
|
|
|
)
|
2025-06-21 16:33:21 +08:00
|
|
|
|
|
|
|
|
|
|
from .models import (
|
|
|
|
|
|
AICallableParam,
|
|
|
|
|
|
AICallableProperties,
|
|
|
|
|
|
AICallableTag,
|
|
|
|
|
|
BaseBlock,
|
|
|
|
|
|
Command,
|
|
|
|
|
|
ConfigModel,
|
|
|
|
|
|
Example,
|
|
|
|
|
|
PluginCdBlock,
|
|
|
|
|
|
PluginCountBlock,
|
|
|
|
|
|
PluginExtraData,
|
|
|
|
|
|
PluginSetting,
|
|
|
|
|
|
RegisterConfig,
|
|
|
|
|
|
Task,
|
|
|
|
|
|
)
|
2023-02-26 22:17:26 +08:00
|
|
|
|
|
2024-02-04 04:18:54 +08:00
|
|
|
|
_yaml = YAML(pure=True)
|
|
|
|
|
|
_yaml.indent = 2
|
|
|
|
|
|
_yaml.allow_unicode = True
|
2023-02-26 22:17:26 +08:00
|
|
|
|
|
2025-06-21 16:33:21 +08:00
|
|
|
|
T = TypeVar("T")
|
2024-02-04 04:18:54 +08:00
|
|
|
|
|
2024-09-27 16:59:41 +08:00
|
|
|
|
|
2025-06-21 16:33:21 +08:00
|
|
|
|
class NoSuchConfig(Exception):
|
|
|
|
|
|
pass
|
2024-09-27 16:59:41 +08:00
|
|
|
|
|
|
|
|
|
|
|
2023-02-26 22:17:26 +08:00
|
|
|
|
class ConfigGroup(BaseModel):
|
|
|
|
|
|
"""
|
|
|
|
|
|
配置组
|
|
|
|
|
|
"""
|
|
|
|
|
|
|
|
|
|
|
|
module: str
|
|
|
|
|
|
"""模块名"""
|
2024-02-04 04:18:54 +08:00
|
|
|
|
name: str | None = None
|
|
|
|
|
|
"""插件名"""
|
2025-02-04 02:15:21 +08:00
|
|
|
|
configs: dict[str, ConfigModel] = Field(default_factory=dict)
|
2023-02-26 22:17:26 +08:00
|
|
|
|
"""配置项列表"""
|
|
|
|
|
|
|
2025-06-21 16:33:21 +08:00
|
|
|
|
def get(self, c: str, default: Any = None, *, build_model: bool = True) -> Any:
|
|
|
|
|
|
"""
|
|
|
|
|
|
获取配置项的值。如果指定了类型,会自动构建实例。
|
|
|
|
|
|
"""
|
|
|
|
|
|
key = c.upper()
|
|
|
|
|
|
cfg = self.configs.get(key)
|
|
|
|
|
|
|
|
|
|
|
|
if cfg is None:
|
|
|
|
|
|
return default
|
|
|
|
|
|
|
|
|
|
|
|
value_to_process = cfg.value if cfg.value is not None else cfg.default_value
|
|
|
|
|
|
|
|
|
|
|
|
if value_to_process is None:
|
|
|
|
|
|
return default
|
|
|
|
|
|
|
2025-07-08 23:20:13 +08:00
|
|
|
|
if cfg.arg_parser:
|
2025-06-21 16:33:21 +08:00
|
|
|
|
try:
|
2025-07-08 23:20:13 +08:00
|
|
|
|
return cfg.arg_parser(value_to_process)
|
2025-06-21 16:33:21 +08:00
|
|
|
|
except Exception as e:
|
2025-07-08 23:20:13 +08:00
|
|
|
|
logger.debug(
|
|
|
|
|
|
f"配置项类型转换 MODULE: [<u><y>{self.module}</y></u>] | "
|
|
|
|
|
|
f"KEY: [<u><y>{key}</y></u>] 的自定义解析器失败,将使用原始值",
|
|
|
|
|
|
e=e,
|
|
|
|
|
|
)
|
|
|
|
|
|
return value_to_process
|
|
|
|
|
|
|
|
|
|
|
|
if not build_model or not cfg.type:
|
|
|
|
|
|
return value_to_process
|
2024-02-04 04:18:54 +08:00
|
|
|
|
|
2025-07-08 23:20:13 +08:00
|
|
|
|
try:
|
|
|
|
|
|
if _is_pydantic_type(cfg.type):
|
|
|
|
|
|
parsed_value = parse_as(cfg.type, value_to_process)
|
|
|
|
|
|
return parsed_value
|
|
|
|
|
|
else:
|
|
|
|
|
|
structured_value = cattrs.structure(value_to_process, cfg.type)
|
|
|
|
|
|
return structured_value
|
|
|
|
|
|
except Exception as e:
|
|
|
|
|
|
logger.error(
|
|
|
|
|
|
f"❌ 配置项 '{self.module}.{key}' 自动类型转换失败 "
|
|
|
|
|
|
f"(目标类型: {cfg.type}),将返回原始值。请检查配置文件格式。错误: {e}",
|
|
|
|
|
|
e=e,
|
|
|
|
|
|
)
|
|
|
|
|
|
return value_to_process
|
2024-02-04 04:18:54 +08:00
|
|
|
|
|
2025-01-07 14:20:30 +08:00
|
|
|
|
def to_dict(self, **kwargs):
|
|
|
|
|
|
return model_dump(self, **kwargs)
|
|
|
|
|
|
|
2023-02-26 22:17:26 +08:00
|
|
|
|
|
2021-11-04 16:11:50 +08:00
|
|
|
|
class ConfigsManager:
|
|
|
|
|
|
"""
|
|
|
|
|
|
插件配置 与 资源 管理器
|
|
|
|
|
|
"""
|
|
|
|
|
|
|
|
|
|
|
|
def __init__(self, file: Path):
|
2024-08-29 22:01:34 +08:00
|
|
|
|
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"
|
2025-01-16 09:13:28 +08:00
|
|
|
|
self.add_module = []
|
2023-02-26 22:17:26 +08:00
|
|
|
|
_yaml = YAML()
|
2021-11-04 16:11:50 +08:00
|
|
|
|
if file:
|
|
|
|
|
|
file.parent.mkdir(exist_ok=True, parents=True)
|
|
|
|
|
|
self.file = file
|
2023-02-26 22:17:26 +08:00
|
|
|
|
self.load_data()
|
|
|
|
|
|
if self._simple_file.exists():
|
|
|
|
|
|
try:
|
2024-08-29 22:01:34 +08:00
|
|
|
|
with self._simple_file.open(encoding="utf8") as f:
|
2023-02-26 22:17:26 +08:00
|
|
|
|
self._simple_data = _yaml.load(f)
|
|
|
|
|
|
except ScannerError as e:
|
|
|
|
|
|
raise ScannerError(
|
|
|
|
|
|
f"{e}\n**********************************************\n"
|
|
|
|
|
|
f"****** 可能为config.yaml配置文件填写不规范 ******\n"
|
|
|
|
|
|
f"**********************************************"
|
2024-08-29 22:01:34 +08:00
|
|
|
|
) 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
|
|
|
|
|
|
|
2025-07-11 10:10:33 +08:00
|
|
|
|
def _merge_dicts(self, new_data: dict, original_data: dict) -> dict:
|
|
|
|
|
|
"""合并两个字典,只进行key值的新增和删除操作,不修改原有key的值
|
|
|
|
|
|
|
|
|
|
|
|
递归处理嵌套字典,确保所有层级的key保持一致
|
|
|
|
|
|
|
|
|
|
|
|
参数:
|
|
|
|
|
|
new_data: 新数据字典
|
|
|
|
|
|
original_data: 原数据字典
|
|
|
|
|
|
|
|
|
|
|
|
返回:
|
|
|
|
|
|
合并后的字典
|
|
|
|
|
|
"""
|
|
|
|
|
|
result = dict(original_data)
|
|
|
|
|
|
|
|
|
|
|
|
for key, value in new_data.items():
|
|
|
|
|
|
if key not in original_data:
|
|
|
|
|
|
result[key] = value
|
|
|
|
|
|
elif isinstance(value, dict) and isinstance(original_data[key], dict):
|
|
|
|
|
|
result[key] = self._merge_dicts(value, original_data[key])
|
|
|
|
|
|
|
|
|
|
|
|
return result
|
|
|
|
|
|
|
|
|
|
|
|
def _normalize_config_data(self, value: Any, original_value: Any = None) -> Any:
|
|
|
|
|
|
"""标准化配置数据,处理BaseModel和字典的情况
|
|
|
|
|
|
|
|
|
|
|
|
参数:
|
|
|
|
|
|
value: 要标准化的值
|
|
|
|
|
|
original_value: 原始值,用于合并字典
|
|
|
|
|
|
|
|
|
|
|
|
返回:
|
|
|
|
|
|
标准化后的值
|
|
|
|
|
|
"""
|
|
|
|
|
|
processed_value = _dump_pydantic_obj(value)
|
|
|
|
|
|
|
|
|
|
|
|
if isinstance(processed_value, dict) and original_value is not None:
|
|
|
|
|
|
processed_original = _dump_pydantic_obj(original_value)
|
|
|
|
|
|
|
|
|
|
|
|
if isinstance(processed_original, dict):
|
|
|
|
|
|
return self._merge_dicts(processed_value, processed_original)
|
|
|
|
|
|
|
|
|
|
|
|
return processed_value
|
|
|
|
|
|
|
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,
|
2024-08-29 22:01:34 +08:00
|
|
|
|
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
|
|
|
|
"""
|
2025-07-11 10:10:33 +08:00
|
|
|
|
key = key.upper()
|
2023-02-26 22:17:26 +08:00
|
|
|
|
if not module or not key:
|
|
|
|
|
|
raise ValueError("add_plugin_config: module和key不能为为空")
|
2025-06-21 16:33:21 +08:00
|
|
|
|
|
2025-07-11 10:10:33 +08:00
|
|
|
|
existing_value = None
|
|
|
|
|
|
if module in self._data and (config := self._data[module].configs.get(key)):
|
|
|
|
|
|
existing_value = config.value
|
|
|
|
|
|
|
|
|
|
|
|
processed_value = self._normalize_config_data(value, existing_value)
|
|
|
|
|
|
processed_default_value = self._normalize_config_data(default_value)
|
2025-06-21 16:33:21 +08:00
|
|
|
|
|
2025-01-16 09:13:28 +08:00
|
|
|
|
self.add_module.append(f"{module}:{key}".lower())
|
2023-02-26 22:17:26 +08:00
|
|
|
|
if module in self._data and (config := self._data[module].configs.get(key)):
|
2024-02-04 04:18:54 +08:00
|
|
|
|
config.help = help
|
2023-02-26 22:17:26 +08:00
|
|
|
|
config.arg_parser = arg_parser
|
|
|
|
|
|
config.type = type
|
|
|
|
|
|
if _override:
|
2025-06-21 16:33:21 +08:00
|
|
|
|
config.value = processed_value
|
|
|
|
|
|
config.default_value = processed_default_value
|
2023-02-26 22:17:26 +08:00
|
|
|
|
else:
|
2021-11-04 16:11:50 +08:00
|
|
|
|
key = key.upper()
|
2023-02-26 22:17:26 +08:00
|
|
|
|
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(
|
2025-06-21 16:33:21 +08:00
|
|
|
|
value=processed_value,
|
2024-02-04 04:18:54 +08:00
|
|
|
|
help=help,
|
2025-06-21 16:33:21 +08:00
|
|
|
|
default_value=processed_default_value,
|
2023-02-26 22:17:26 +08:00
|
|
|
|
type=type,
|
2025-06-21 16:33:21 +08:00
|
|
|
|
arg_parser=arg_parser,
|
2023-02-26 22:17:26 +08:00
|
|
|
|
)
|
2021-11-04 16:11:50 +08:00
|
|
|
|
|
2023-02-26 22:17:26 +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()
|
2023-02-26 22:17:26 +08:00
|
|
|
|
if module in self._data:
|
2025-06-21 16:33:21 +08:00
|
|
|
|
if module not in self._simple_data:
|
|
|
|
|
|
self._simple_data[module] = {}
|
2024-10-29 08:26:41 +08:00
|
|
|
|
if self._data[module].configs.get(key):
|
2023-02-26 22:17:26 +08:00
|
|
|
|
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
|
2022-08-22 22:49:35 +08:00
|
|
|
|
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
|
|
|
|
|
2025-06-21 16:33:21 +08:00
|
|
|
|
def get_config(
|
|
|
|
|
|
self,
|
|
|
|
|
|
module: str,
|
|
|
|
|
|
key: str,
|
|
|
|
|
|
default: Any = None,
|
|
|
|
|
|
*,
|
|
|
|
|
|
build_model: bool = True,
|
|
|
|
|
|
) -> Any:
|
|
|
|
|
|
"""
|
|
|
|
|
|
获取指定配置值,自动构建Pydantic模型或其它类型实例。
|
|
|
|
|
|
- 兼容Pydantic V1/V2。
|
|
|
|
|
|
- 支持 list[BaseModel] 等泛型容器。
|
|
|
|
|
|
- 优先使用Pydantic原生方式解析,失败后回退到cattrs。
|
|
|
|
|
|
"""
|
|
|
|
|
|
key = key.upper()
|
|
|
|
|
|
config_group = self._data.get(module)
|
|
|
|
|
|
if not config_group:
|
|
|
|
|
|
return default
|
2024-02-04 04:18:54 +08:00
|
|
|
|
|
2025-06-21 16:33:21 +08:00
|
|
|
|
config = config_group.configs.get(key)
|
|
|
|
|
|
if not config:
|
|
|
|
|
|
return default
|
2024-02-04 04:18:54 +08:00
|
|
|
|
|
2025-06-21 16:33:21 +08:00
|
|
|
|
value_to_process = (
|
|
|
|
|
|
config.value if config.value is not None else config.default_value
|
2023-02-26 22:17:26 +08:00
|
|
|
|
)
|
2025-06-21 16:33:21 +08:00
|
|
|
|
if value_to_process is None:
|
|
|
|
|
|
return default
|
|
|
|
|
|
|
|
|
|
|
|
if config.arg_parser:
|
2024-08-29 22:01:34 +08:00
|
|
|
|
try:
|
2025-06-21 16:33:21 +08:00
|
|
|
|
return config.arg_parser(value_to_process)
|
2024-08-29 22:01:34 +08:00
|
|
|
|
except Exception as e:
|
2025-06-20 19:08:06 +08:00
|
|
|
|
logger.debug(
|
2024-08-29 22:01:34 +08:00
|
|
|
|
f"配置项类型转换 MODULE: [<u><y>{module}</y></u>]"
|
2025-06-20 19:08:06 +08:00
|
|
|
|
f" | KEY: [<u><y>{key}</y></u>] 将使用原始值",
|
2024-08-29 22:01:34 +08:00
|
|
|
|
e=e,
|
|
|
|
|
|
)
|
2025-06-21 16:33:21 +08:00
|
|
|
|
|
|
|
|
|
|
if config.type:
|
|
|
|
|
|
if _is_pydantic_type(config.type):
|
|
|
|
|
|
if build_model:
|
|
|
|
|
|
try:
|
|
|
|
|
|
return parse_as(config.type, value_to_process)
|
|
|
|
|
|
except Exception as e:
|
|
|
|
|
|
logger.warning(
|
|
|
|
|
|
f"pydantic类型转换失败 MODULE: [<u><y>{module}</y></u>] | "
|
|
|
|
|
|
f"KEY: [<u><y>{key}</y></u>].",
|
|
|
|
|
|
e=e,
|
|
|
|
|
|
)
|
|
|
|
|
|
else:
|
|
|
|
|
|
try:
|
|
|
|
|
|
return cattrs.structure(value_to_process, config.type)
|
|
|
|
|
|
except Exception as e:
|
|
|
|
|
|
logger.warning(
|
|
|
|
|
|
f"cattrs类型转换失败 MODULE: [<u><y>{module}</y></u>] | "
|
|
|
|
|
|
f"KEY: [<u><y>{key}</y></u>].",
|
|
|
|
|
|
e=e,
|
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
|
|
return value_to_process
|
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
|
|
|
|
"""
|
2025-10-01 18:42:19 +08:00
|
|
|
|
if key not in self._data:
|
|
|
|
|
|
self._data[key] = ConfigGroup(module=key)
|
|
|
|
|
|
return self._data[key]
|
2024-02-04 04:18:54 +08:00
|
|
|
|
|
|
|
|
|
|
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)
|
2023-02-26 22:17:26 +08:00
|
|
|
|
path = path or self.file
|
2025-07-11 10:10:33 +08:00
|
|
|
|
save_data = {
|
|
|
|
|
|
module: {
|
|
|
|
|
|
config_key: model_dump(config_model, exclude={"type", "arg_parser"})
|
|
|
|
|
|
for config_key, config_model in config_group.configs.items()
|
|
|
|
|
|
}
|
|
|
|
|
|
for module, config_group in self._data.items()
|
|
|
|
|
|
}
|
2021-11-04 16:11:50 +08:00
|
|
|
|
with open(path, "w", encoding="utf8") as f:
|
2025-06-21 16:33:21 +08:00
|
|
|
|
_yaml.dump(save_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
|
|
|
|
"""重新加载配置文件"""
|
2022-08-21 18:09:31 +08:00
|
|
|
|
if self._simple_file.exists():
|
2024-08-29 22:01:34 +08:00
|
|
|
|
with open(self._simple_file, encoding="utf8") as f:
|
2022-08-21 18:09:31 +08:00
|
|
|
|
self._simple_data = _yaml.load(f)
|
|
|
|
|
|
for key in self._simple_data.keys():
|
|
|
|
|
|
for k in self._simple_data[key].keys():
|
2023-02-26 22:17:26 +08:00
|
|
|
|
self._data[key].configs[k].value = self._simple_data[key][k]
|
2022-02-19 18:20:19 +08:00
|
|
|
|
self.save()
|
|
|
|
|
|
|
2023-02-26 22:17:26 +08:00
|
|
|
|
def load_data(self):
|
2024-02-04 04:18:54 +08:00
|
|
|
|
"""加载数据
|
2023-02-26 22:17:26 +08:00
|
|
|
|
|
2024-02-04 04:18:54 +08:00
|
|
|
|
异常:
|
|
|
|
|
|
ValueError: 配置文件为空!
|
2023-02-26 22:17:26 +08:00
|
|
|
|
"""
|
2024-08-29 22:01:34 +08:00
|
|
|
|
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"
|
|
|
|
|
|
"***********************************************************"
|
2023-02-26 22:17:26 +08:00
|
|
|
|
)
|
2024-08-29 22:01:34 +08:00
|
|
|
|
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> 个配置组及对应"
|
2024-09-10 20:17:09 +08:00
|
|
|
|
f" <u><y>{count}</y></u> 个配置项"
|
2024-08-29 22:01:34 +08:00
|
|
|
|
)
|
2023-02-26 22:17:26 +08:00
|
|
|
|
|
2024-08-29 22:01:34 +08:00
|
|
|
|
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]
|
2025-06-21 16:33:21 +08:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
__all__ = [
|
|
|
|
|
|
"AICallableParam",
|
|
|
|
|
|
"AICallableProperties",
|
|
|
|
|
|
"AICallableTag",
|
|
|
|
|
|
"BaseBlock",
|
|
|
|
|
|
"Command",
|
|
|
|
|
|
"ConfigGroup",
|
|
|
|
|
|
"ConfigModel",
|
|
|
|
|
|
"ConfigsManager",
|
|
|
|
|
|
"Example",
|
|
|
|
|
|
"NoSuchConfig",
|
|
|
|
|
|
"PluginCdBlock",
|
|
|
|
|
|
"PluginCountBlock",
|
|
|
|
|
|
"PluginExtraData",
|
|
|
|
|
|
"PluginSetting",
|
|
|
|
|
|
"RegisterConfig",
|
|
|
|
|
|
"Task",
|
|
|
|
|
|
]
|