mirror of
https://github.com/zhenxun-org/zhenxun_bot.git
synced 2025-12-14 13:42:56 +08:00
Some checks failed
检查bot是否运行正常 / bot check (push) Has been cancelled
CodeQL Code Security Analysis / Analyze (${{ matrix.language }}) (none, javascript-typescript) (push) Has been cancelled
CodeQL Code Security Analysis / Analyze (${{ matrix.language }}) (none, python) (push) Has been cancelled
Sequential Lint and Type Check / ruff-call (push) Has been cancelled
Release Drafter / Update Release Draft (push) Has been cancelled
Force Sync to Aliyun / sync (push) Has been cancelled
Update Version / update-version (push) Has been cancelled
Sequential Lint and Type Check / pyright-call (push) Has been cancelled
* ♻️ refactor: 统一图片渲染架构并引入通用UI组件系统 🎨 **渲染服务重构** - 统一图片渲染入口,引入主题系统支持 - 优化Jinja2环境管理,支持主题覆盖和插件命名空间 - 新增UI缓存机制和主题重载功能 ✨ **通用UI组件系统** - 新增 zhenxun.ui 模块,提供数据模型和构建器 - 引入BaseBuilder基类,支持链式调用 - 新增多种UI构建器:InfoCard, Markdown, Table, Chart, Layout等 - 新增通用组件:Divider, Badge, ProgressBar, UserInfoBlock 🔄 **插件迁移** - 迁移9个内置插件至新渲染系统 - 移除各插件中分散的图片生成工具 - 优化数据处理和渲染逻辑 💥 **Breaking Changes** - 移除旧的图片渲染接口和模板路径 - TEMPLATE_PATH 更名为 THEMES_PATH - 插件需适配新的RendererService和zhenxun.ui模块 * ✅ test(check): 更新自检插件测试中的渲染服务模拟 * ♻️ refactor(renderer): 将缓存文件名哈希算法切换到 SHA256 * ♻️ refactor(shop): 移除商店HTML图片生成模块 * 🚨 auto fix by pre-commit hooks --------- Co-authored-by: webjoin111 <455457521@qq.com> Co-authored-by: pre-commit-ci[bot] <66853113+pre-commit-ci[bot]@users.noreply.github.com>
203 lines
6.9 KiB
Python
203 lines
6.9 KiB
Python
from collections import namedtuple
|
||
from collections.abc import Callable
|
||
from pathlib import Path
|
||
import platform
|
||
from typing import cast
|
||
|
||
from nonebot.adapters.onebot.v11 import Bot
|
||
from nonebot.adapters.onebot.v11.event import GroupMessageEvent
|
||
from nonebug import App
|
||
import pytest
|
||
from pytest_mock import MockerFixture
|
||
|
||
from tests.config import BotId, GroupId, MessageId, UserId
|
||
from tests.utils import _v11_group_message_event
|
||
|
||
platform_uname = platform.uname_result(
|
||
system="Linux",
|
||
node="zhenxun",
|
||
release="5.15.0-1027-azure",
|
||
version="#1 SMP Debian 5.15.0-1027-azure",
|
||
machine="x86_64",
|
||
) # type: ignore
|
||
cpuinfo_get_cpu_info = {"brand_raw": "Intel(R) Core(TM) i7-10700K"}
|
||
|
||
|
||
def init_mocker(mocker: MockerFixture, tmp_path: Path):
|
||
mock_psutil = mocker.patch("zhenxun.builtin_plugins.check.data_source.psutil")
|
||
|
||
# Define namedtuples for complex return values
|
||
CpuFreqs = namedtuple("CpuFreqs", ["current"]) # noqa: PYI024
|
||
VirtualMemoryInfo = namedtuple("VirtualMemoryInfo", ["used", "total", "percent"]) # noqa: PYI024
|
||
SwapInfo = namedtuple("SwapInfo", ["used", "total", "percent"]) # noqa: PYI024
|
||
DiskUsage = namedtuple("DiskUsage", ["used", "total", "free", "percent"]) # noqa: PYI024
|
||
|
||
# Set specific return values for psutil methods
|
||
mock_psutil.cpu_percent.return_value = 1.0 # CPU 使用率
|
||
mock_psutil.cpu_freq.return_value = CpuFreqs(current=0.0) # CPU 频率
|
||
mock_psutil.cpu_count.return_value = 1 # CPU 核心数
|
||
|
||
# Memory Info
|
||
mock_psutil.virtual_memory.return_value = VirtualMemoryInfo(
|
||
used=1 * 1024**3, # 1 GB in bytes for used memory
|
||
total=1 * 1024**3, # 1 GB in bytes for total memory
|
||
percent=100.0, # 100% of memory used
|
||
)
|
||
|
||
# Swap Info
|
||
mock_psutil.swap_memory.return_value = SwapInfo(
|
||
used=1 * 1024**3, # 1 GB in bytes for used swap space
|
||
total=1 * 1024**3, # 1 GB in bytes for total swap space
|
||
percent=100.0, # 100% of swap space used
|
||
)
|
||
|
||
# Disk Usage
|
||
mock_psutil.disk_usage.return_value = DiskUsage(
|
||
used=1 * 1024**3, # 1 GB in bytes for used disk space
|
||
total=1 * 1024**3, # 1 GB in bytes for total disk space
|
||
free=0, # No free space
|
||
percent=100.0, # 100% of disk space used
|
||
)
|
||
|
||
mock_cpuinfo = mocker.patch("zhenxun.builtin_plugins.check.data_source.cpuinfo")
|
||
mock_cpuinfo.get_cpu_info.return_value = cpuinfo_get_cpu_info
|
||
|
||
mock_platform = mocker.patch("zhenxun.builtin_plugins.check.data_source.platform")
|
||
mock_platform.uname.return_value = platform_uname
|
||
|
||
mock_render_service = mocker.patch(
|
||
"zhenxun.builtin_plugins.check.renderer_service.render"
|
||
)
|
||
mock_render_service_return = mocker.AsyncMock()
|
||
mock_render_service.return_value = mock_render_service_return
|
||
|
||
mock_build_message = mocker.patch(
|
||
"zhenxun.builtin_plugins.check.MessageUtils.build_message"
|
||
)
|
||
mock_build_message_return = mocker.AsyncMock()
|
||
mock_build_message.return_value = mock_build_message_return
|
||
|
||
return (
|
||
mock_psutil,
|
||
mock_cpuinfo,
|
||
mock_platform,
|
||
mock_render_service,
|
||
mock_render_service_return,
|
||
mock_build_message,
|
||
mock_build_message_return,
|
||
)
|
||
|
||
|
||
@pytest.mark.xfail
|
||
async def test_check(
|
||
app: App,
|
||
mocker: MockerFixture,
|
||
create_bot: Callable,
|
||
tmp_path: Path,
|
||
) -> None:
|
||
"""
|
||
测试自检
|
||
"""
|
||
from zhenxun.builtin_plugins.check import _self_check_matcher
|
||
|
||
(
|
||
mock_psutil,
|
||
mock_cpuinfo,
|
||
mock_platform,
|
||
mock_render_service,
|
||
mock_render_service_return,
|
||
mock_build_message,
|
||
mock_build_message_return,
|
||
) = init_mocker(mocker, tmp_path)
|
||
async with app.test_matcher(_self_check_matcher) as ctx:
|
||
bot = create_bot(ctx)
|
||
bot: Bot = cast(Bot, bot)
|
||
raw_message = "自检"
|
||
event: GroupMessageEvent = _v11_group_message_event(
|
||
message=raw_message,
|
||
self_id=BotId.QQ_BOT,
|
||
user_id=UserId.SUPERUSER,
|
||
group_id=GroupId.GROUP_ID_LEVEL_5,
|
||
message_id=MessageId.MESSAGE_ID_3,
|
||
to_me=True,
|
||
)
|
||
ctx.receive_event(bot=bot, event=event)
|
||
ctx.should_ignore_rule(_self_check_matcher)
|
||
|
||
mock_render_service.assert_awaited_once()
|
||
mock_build_message.assert_called_once_with(mock_render_service_return)
|
||
mock_build_message_return.send.assert_awaited_once()
|
||
|
||
|
||
@pytest.mark.xfail
|
||
async def test_check_arm(
|
||
app: App,
|
||
mocker: MockerFixture,
|
||
create_bot: Callable,
|
||
tmp_path: Path,
|
||
) -> None:
|
||
"""
|
||
测试自检(arm)
|
||
"""
|
||
from zhenxun.builtin_plugins.check import _self_check_matcher
|
||
|
||
platform_uname_arm = platform.uname_result(
|
||
system="Linux",
|
||
node="zhenxun",
|
||
release="5.15.0-1017-oracle",
|
||
version="#22~20.04.1-Ubuntu SMP Wed Aug 24 11:13:15 UTC 2022",
|
||
machine="aarch64",
|
||
) # type: ignore
|
||
mock_subprocess_check_output = mocker.patch(
|
||
"zhenxun.builtin_plugins.check.data_source.subprocess.check_output"
|
||
)
|
||
mock_environ_copy = mocker.patch(
|
||
"zhenxun.builtin_plugins.check.data_source.os.environ.copy"
|
||
)
|
||
mock_environ_copy_return = mocker.MagicMock()
|
||
mock_environ_copy.return_value = mock_environ_copy_return
|
||
(
|
||
mock_psutil,
|
||
mock_cpuinfo,
|
||
mock_platform,
|
||
mock_render_service,
|
||
mock_render_service_return,
|
||
mock_build_message,
|
||
mock_build_message_return,
|
||
) = init_mocker(mocker, tmp_path)
|
||
|
||
mock_platform.uname.return_value = platform_uname_arm
|
||
mock_cpuinfo.get_cpu_info.return_value = {}
|
||
mock_psutil.cpu_freq.return_value = {}
|
||
|
||
async with app.test_matcher(_self_check_matcher) as ctx:
|
||
bot = create_bot(ctx)
|
||
bot: Bot = cast(Bot, bot)
|
||
raw_message = "自检"
|
||
event: GroupMessageEvent = _v11_group_message_event(
|
||
message=raw_message,
|
||
self_id=BotId.QQ_BOT,
|
||
user_id=UserId.SUPERUSER,
|
||
group_id=GroupId.GROUP_ID_LEVEL_5,
|
||
message_id=MessageId.MESSAGE_ID_3,
|
||
to_me=True,
|
||
)
|
||
ctx.receive_event(bot=bot, event=event)
|
||
ctx.should_ignore_rule(_self_check_matcher)
|
||
mock_subprocess_check_output.assert_has_calls(
|
||
[
|
||
mocker.call(["lscpu"], env=mock_environ_copy_return),
|
||
mocker.call().decode(),
|
||
mocker.call().decode().splitlines(),
|
||
mocker.call().decode().splitlines().__iter__(),
|
||
mocker.call(["dmidecode", "-s", "processor-frequency"]),
|
||
mocker.call().decode(),
|
||
mocker.call().decode().split(),
|
||
mocker.call().decode().split().__getitem__(0),
|
||
mocker.call().decode().split().__getitem__().__float__(),
|
||
] # type: ignore
|
||
)
|
||
mock_render_service.assert_awaited_once()
|
||
mock_build_message.assert_called_once_with(mock_render_service_return)
|
||
mock_build_message_return.send.assert_awaited_once()
|