mirror of
https://github.com/zhenxun-org/zhenxun_bot.git
synced 2025-12-15 06:12:53 +08:00
* ✨ feat(core): 更新群组信息、Markdown 样式与 Pydantic 兼容层 - 【group】添加更新所有群组信息指令,并同步群组控制台数据 - 【markdown】支持合并 Markdown 的 CSS 来源 - 【pydantic-compat】提供 model_validate 兼容函数 * ✨ feat(core): 增强定时任务与群组标签管理,重构调度核心 ✨ 新功能 * **标签 (tags)**: 引入群组标签服务。 * 支持静态标签和动态标签 (基于 Alconna 规则自动匹配群信息)。 * 支持黑名单模式及 `@all` 特殊标签。 * 提供 `tag_manage` 超级用户插件 (list, create, edit, delete 等)。 * 群成员变动时自动失效动态标签缓存。 * **调度 (scheduler)**: 增强定时任务。 * 重构 `ScheduledJob` 模型,支持 `TAG`, `ALL_GROUPS` 等多种目标类型。 * 新增任务别名 (`name`)、创建者、权限、来源等字段。 * 支持一次性任务 (`schedule_once`) 和 Alconna 命令行参数 (`--params-cli`)。 * 新增执行选项 (`jitter`, `spread`) 和并发策略 (`ALLOW`, `SKIP`, `QUEUE`)。 * 支持批量获取任务状态。 ♻️ 重构优化 * **调度器核心**: * 拆分 `service.py` 为 `manager.py` (API) 和 `types.py` (模型)。 * 合并 `adapter.py` / `job.py` 至 `engine.py` (统一调度引擎)。 * 引入 `targeting.py` 模块管理任务目标解析。 * **调度器插件 (scheduler_admin)**: * 迁移命令参数校验逻辑至 `ArparmaBehavior`。 * 引入 `dependencies.py` 和 `data_source.py` 解耦业务逻辑与依赖注入。 * 适配新的任务目标类型展示。 * 🐛 fix(tag): 修复黑名单标签解析逻辑并优化标签详情展示 * ✨ feat(scheduler): 为多目标定时任务添加固定间隔串行执行选项 * ✨ feat(schedulerAdmin): 允许定时任务删除、暂停、恢复命令支持多ID操作 * 🚨 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>
118 lines
3.5 KiB
Python
118 lines
3.5 KiB
Python
from datetime import date, datetime
|
|
import re
|
|
|
|
import pytz
|
|
|
|
|
|
class TimeUtils:
|
|
DEFAULT_TIMEZONE = pytz.timezone("Asia/Shanghai")
|
|
|
|
@classmethod
|
|
def get_day_start(cls, target_date: date | datetime | None = None) -> datetime:
|
|
"""获取某天的0点时间
|
|
|
|
返回:
|
|
datetime: 今天某天的0点时间
|
|
"""
|
|
if not target_date:
|
|
target_date = datetime.now(cls.DEFAULT_TIMEZONE)
|
|
|
|
if isinstance(target_date, datetime) and target_date.tzinfo is None:
|
|
target_date = cls.DEFAULT_TIMEZONE.localize(target_date)
|
|
|
|
return (
|
|
target_date.replace(hour=0, minute=0, second=0, microsecond=0)
|
|
if isinstance(target_date, datetime)
|
|
else datetime.combine(
|
|
target_date, datetime.min.time(), tzinfo=cls.DEFAULT_TIMEZONE
|
|
)
|
|
)
|
|
|
|
@classmethod
|
|
def is_valid_date(cls, date_text: str, separator: str = "-") -> bool:
|
|
"""日期是否合法
|
|
|
|
参数:
|
|
date_text: 日期
|
|
separator: 分隔符
|
|
|
|
返回:
|
|
bool: 日期是否合法
|
|
"""
|
|
try:
|
|
datetime.strptime(date_text, f"%Y{separator}%m{separator}%d")
|
|
return True
|
|
except ValueError:
|
|
return False
|
|
|
|
@classmethod
|
|
def parse_time_string(cls, time_str: str) -> int:
|
|
"""
|
|
将带有单位的时间字符串 (e.g., "10s", "5m", "1h", "1d") 解析为总秒数。
|
|
"""
|
|
time_str = time_str.lower().strip()
|
|
match = re.match(r"^(\d+)([smhd])$", time_str)
|
|
if not match:
|
|
raise ValueError(
|
|
f"无效的时间格式: '{time_str}'。请使用如 '30s', '10m', '2h', '1d'的格式"
|
|
)
|
|
|
|
value, unit = int(match.group(1)), match.group(2)
|
|
|
|
if unit == "s":
|
|
return value
|
|
if unit == "m":
|
|
return value * 60
|
|
if unit == "h":
|
|
return value * 3600
|
|
if unit == "d":
|
|
return value * 86400
|
|
return 0
|
|
|
|
@classmethod
|
|
def parse_interval_to_dict(cls, interval_str: str) -> dict:
|
|
"""
|
|
将时间间隔字符串解析为 APScheduler 的 interval 触发器所需的字典。
|
|
"""
|
|
time_str_lower = interval_str.lower().strip()
|
|
match = re.match(r"^(\d+)([smhd])$", time_str_lower)
|
|
if not match:
|
|
raise ValueError(
|
|
"时间间隔格式错误, 请使用如 '30m', '2h', '1d', '10s' 的格式。"
|
|
)
|
|
|
|
value, unit = int(match.group(1)), match.group(2)
|
|
|
|
if unit == "s":
|
|
return {"seconds": value}
|
|
if unit == "m":
|
|
return {"minutes": value}
|
|
if unit == "h":
|
|
return {"hours": value}
|
|
if unit == "d":
|
|
return {"days": value}
|
|
return {}
|
|
|
|
@classmethod
|
|
def format_duration(cls, seconds: float) -> str:
|
|
"""
|
|
将秒数格式化为易于阅读的字符串 (例如 "1小时5分钟", "30.5秒")
|
|
"""
|
|
seconds = round(seconds, 1)
|
|
if seconds < 0.1:
|
|
return "不到1秒"
|
|
if seconds < 60:
|
|
return f"{seconds}秒"
|
|
|
|
minutes, sec_remainder = divmod(int(seconds), 60)
|
|
|
|
if minutes < 60:
|
|
if sec_remainder == 0:
|
|
return f"{minutes}分钟"
|
|
return f"{minutes}分钟{sec_remainder}秒"
|
|
|
|
hours, rem_minutes = divmod(minutes, 60)
|
|
if rem_minutes == 0:
|
|
return f"{hours}小时"
|
|
return f"{hours}小时{rem_minutes}分钟"
|