2022-05-18 20:31:14 +08:00
|
|
|
|
import re
|
|
|
|
|
|
from datetime import datetime, timedelta
|
|
|
|
|
|
from typing import Tuple, Union
|
2023-02-26 22:17:26 +08:00
|
|
|
|
|
2022-05-19 15:14:20 +08:00
|
|
|
|
import pytz
|
2023-02-26 22:17:26 +08:00
|
|
|
|
from nonebot import get_driver, on_command
|
2022-05-18 20:31:14 +08:00
|
|
|
|
from nonebot.adapters.onebot.v11 import Message, MessageSegment
|
|
|
|
|
|
from nonebot.adapters.onebot.v11.event import GroupMessageEvent
|
|
|
|
|
|
from nonebot.matcher import Matcher
|
|
|
|
|
|
from nonebot.params import Arg, Command, CommandArg, Depends
|
|
|
|
|
|
from nonebot.typing import T_State
|
2023-02-26 22:17:26 +08:00
|
|
|
|
|
2022-05-18 20:31:14 +08:00
|
|
|
|
from configs.config import Config
|
|
|
|
|
|
|
2023-02-26 22:17:26 +08:00
|
|
|
|
from .data_source import draw_word_cloud, get_list_msg
|
|
|
|
|
|
|
2022-05-18 20:31:14 +08:00
|
|
|
|
__zx_plugin_name__ = "词云"
|
|
|
|
|
|
|
|
|
|
|
|
__plugin_usage__ = """
|
|
|
|
|
|
usage:
|
|
|
|
|
|
词云
|
|
|
|
|
|
指令:
|
2022-05-21 13:15:53 +08:00
|
|
|
|
今日词云:获取今天的词云
|
|
|
|
|
|
昨日词云:获取昨天的词云
|
|
|
|
|
|
本周词云:获取本周词云
|
|
|
|
|
|
本月词云:获取本月词云
|
|
|
|
|
|
年度词云:获取年度词云
|
2022-05-22 12:51:53 +08:00
|
|
|
|
|
2022-05-18 20:31:14 +08:00
|
|
|
|
历史词云(支持 ISO8601 格式的日期与时间,如 2022-02-22T22:22:22)
|
|
|
|
|
|
获取某日的词云
|
|
|
|
|
|
历史词云 2022-01-01
|
|
|
|
|
|
获取指定时间段的词云
|
|
|
|
|
|
历史词云
|
2022-05-21 13:15:53 +08:00
|
|
|
|
示例:历史词云 2022-01-01~2022-02-22
|
|
|
|
|
|
示例:历史词云 2022-02-22T11:11:11~2022-02-22T22:22:22
|
2022-05-22 12:51:53 +08:00
|
|
|
|
|
2022-05-18 20:31:14 +08:00
|
|
|
|
如果想要获取自己的发言,可在命令前添加 我的
|
2022-05-21 13:15:53 +08:00
|
|
|
|
示例:我的今日词云
|
2022-05-18 20:31:14 +08:00
|
|
|
|
""".strip()
|
|
|
|
|
|
__plugin_des__ = "词云"
|
|
|
|
|
|
__plugin_cmd__ = ["今日词云", "昨日词云", "本周词云"]
|
|
|
|
|
|
__plugin_version__ = 0.1
|
|
|
|
|
|
__plugin_author__ = "yajiwa"
|
|
|
|
|
|
__plugin_settings__ = {
|
|
|
|
|
|
"level": 5,
|
|
|
|
|
|
"default_status": True,
|
|
|
|
|
|
"limit_superuser": False,
|
|
|
|
|
|
"cmd": __plugin_cmd__,
|
|
|
|
|
|
}
|
|
|
|
|
|
wordcloud_cmd = on_command(
|
|
|
|
|
|
"wordcloud",
|
|
|
|
|
|
aliases={
|
|
|
|
|
|
"词云",
|
|
|
|
|
|
"今日词云",
|
|
|
|
|
|
"昨日词云",
|
|
|
|
|
|
"本周词云",
|
|
|
|
|
|
"本月词云",
|
|
|
|
|
|
"年度词云",
|
|
|
|
|
|
"历史词云",
|
|
|
|
|
|
"我的今日词云",
|
|
|
|
|
|
"我的昨日词云",
|
|
|
|
|
|
"我的本周词云",
|
|
|
|
|
|
"我的本月词云",
|
|
|
|
|
|
"我的年度词云",
|
|
|
|
|
|
"我的历史词云",
|
2023-02-26 22:17:26 +08:00
|
|
|
|
},
|
|
|
|
|
|
block=True,
|
|
|
|
|
|
priority=5,
|
2022-05-18 20:31:14 +08:00
|
|
|
|
)
|
|
|
|
|
|
Config.add_plugin_config(
|
|
|
|
|
|
"word_clouds",
|
|
|
|
|
|
"WORD_CLOUDS_TEMPLATE",
|
|
|
|
|
|
1,
|
2023-02-26 22:17:26 +08:00
|
|
|
|
help_="词云模板 参1:图片生成,默认使用真寻图片,可在项目路径resources/image/wordcloud下配置图片,多张则随机 | 参2/其他:黑底图片",
|
|
|
|
|
|
type=int,
|
2022-05-18 20:31:14 +08:00
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
def parse_datetime(key: str):
|
|
|
|
|
|
"""解析数字,并将结果存入 state 中"""
|
|
|
|
|
|
|
|
|
|
|
|
async def _key_parser(
|
2023-02-26 22:17:26 +08:00
|
|
|
|
matcher: Matcher,
|
|
|
|
|
|
state: T_State,
|
|
|
|
|
|
input_: Union[datetime, Message] = Arg(key),
|
2022-05-18 20:31:14 +08:00
|
|
|
|
):
|
2022-05-21 13:15:53 +08:00
|
|
|
|
if isinstance(input_, datetime):
|
2022-05-18 20:31:14 +08:00
|
|
|
|
return
|
|
|
|
|
|
|
2022-05-21 13:15:53 +08:00
|
|
|
|
plaintext = input_.extract_plain_text()
|
2022-05-18 20:31:14 +08:00
|
|
|
|
try:
|
|
|
|
|
|
state[key] = get_datetime_fromisoformat_with_timezone(plaintext)
|
|
|
|
|
|
except ValueError:
|
|
|
|
|
|
await matcher.reject_arg(key, "请输入正确的日期,不然我没法理解呢!")
|
|
|
|
|
|
|
|
|
|
|
|
return _key_parser
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
def get_datetime_now_with_timezone() -> datetime:
|
|
|
|
|
|
"""获取当前时间,并包含时区信息"""
|
|
|
|
|
|
return datetime.now().astimezone()
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
def get_datetime_fromisoformat_with_timezone(date_string: str) -> datetime:
|
|
|
|
|
|
"""从 iso8601 格式字符串中获取时间,并包含时区信息"""
|
|
|
|
|
|
return datetime.fromisoformat(date_string).astimezone()
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
@wordcloud_cmd.handle()
|
|
|
|
|
|
async def handle_first_receive(
|
2023-02-26 22:17:26 +08:00
|
|
|
|
event: GroupMessageEvent,
|
|
|
|
|
|
state: T_State,
|
|
|
|
|
|
commands: Tuple[str, ...] = Command(),
|
|
|
|
|
|
args: Message = CommandArg(),
|
2022-05-18 20:31:14 +08:00
|
|
|
|
):
|
|
|
|
|
|
command = commands[0]
|
|
|
|
|
|
|
|
|
|
|
|
if command.startswith("我的"):
|
|
|
|
|
|
state["my"] = True
|
|
|
|
|
|
command = command[2:]
|
|
|
|
|
|
else:
|
|
|
|
|
|
state["my"] = False
|
|
|
|
|
|
|
|
|
|
|
|
if command == "今日词云":
|
|
|
|
|
|
dt = get_datetime_now_with_timezone()
|
|
|
|
|
|
state["start"] = dt.replace(hour=0, minute=0, second=0, microsecond=0)
|
|
|
|
|
|
state["stop"] = dt
|
|
|
|
|
|
elif command == "昨日词云":
|
|
|
|
|
|
dt = get_datetime_now_with_timezone()
|
|
|
|
|
|
state["stop"] = dt.replace(hour=0, minute=0, second=0, microsecond=0)
|
|
|
|
|
|
state["start"] = state["stop"] - timedelta(days=1)
|
|
|
|
|
|
elif command == "本周词云":
|
|
|
|
|
|
dt = get_datetime_now_with_timezone()
|
|
|
|
|
|
state["start"] = dt.replace(
|
|
|
|
|
|
hour=0, minute=0, second=0, microsecond=0
|
|
|
|
|
|
) - timedelta(days=dt.weekday())
|
|
|
|
|
|
state["stop"] = dt
|
|
|
|
|
|
elif command == "本月词云":
|
|
|
|
|
|
dt = get_datetime_now_with_timezone()
|
|
|
|
|
|
state["start"] = dt.replace(day=1, hour=0, minute=0, second=0, microsecond=0)
|
|
|
|
|
|
state["stop"] = dt
|
|
|
|
|
|
elif command == "年度词云":
|
|
|
|
|
|
dt = get_datetime_now_with_timezone()
|
|
|
|
|
|
state["start"] = dt.replace(
|
|
|
|
|
|
month=1, day=1, hour=0, minute=0, second=0, microsecond=0
|
|
|
|
|
|
)
|
|
|
|
|
|
state["stop"] = dt
|
|
|
|
|
|
elif command == "历史词云":
|
|
|
|
|
|
plaintext = args.extract_plain_text().strip()
|
|
|
|
|
|
match = re.match(r"^(.+?)(?:~(.+))?$", plaintext)
|
|
|
|
|
|
if match:
|
|
|
|
|
|
start = match.group(1)
|
|
|
|
|
|
stop = match.group(2)
|
|
|
|
|
|
try:
|
|
|
|
|
|
state["start"] = get_datetime_fromisoformat_with_timezone(start)
|
|
|
|
|
|
if stop:
|
|
|
|
|
|
state["stop"] = get_datetime_fromisoformat_with_timezone(stop)
|
|
|
|
|
|
else:
|
|
|
|
|
|
# 如果没有指定结束日期,则认为是指查询这一天的词云
|
|
|
|
|
|
state["start"] = state["start"].replace(
|
|
|
|
|
|
hour=0, minute=0, second=0, microsecond=0
|
|
|
|
|
|
)
|
|
|
|
|
|
state["stop"] = state["start"] + timedelta(days=1)
|
|
|
|
|
|
except ValueError:
|
|
|
|
|
|
await wordcloud_cmd.finish("请输入正确的日期,不然我没法理解呢!")
|
|
|
|
|
|
else:
|
|
|
|
|
|
await wordcloud_cmd.finish()
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
@wordcloud_cmd.got(
|
|
|
|
|
|
"start",
|
|
|
|
|
|
prompt="请输入你要查询的起始日期(如 2022-01-01)",
|
|
|
|
|
|
parameterless=[Depends(parse_datetime("start"))],
|
|
|
|
|
|
)
|
|
|
|
|
|
@wordcloud_cmd.got(
|
|
|
|
|
|
"stop",
|
|
|
|
|
|
prompt="请输入你要查询的结束日期(如 2022-02-22)",
|
|
|
|
|
|
parameterless=[Depends(parse_datetime("stop"))],
|
|
|
|
|
|
)
|
|
|
|
|
|
async def handle_message(
|
2023-02-26 22:17:26 +08:00
|
|
|
|
event: GroupMessageEvent,
|
|
|
|
|
|
start: datetime = Arg(),
|
|
|
|
|
|
stop: datetime = Arg(),
|
|
|
|
|
|
my: bool = Arg(),
|
2022-05-18 20:31:14 +08:00
|
|
|
|
):
|
|
|
|
|
|
# 是否只查询自己的记录
|
|
|
|
|
|
if my:
|
|
|
|
|
|
user_id = int(event.user_id)
|
|
|
|
|
|
else:
|
|
|
|
|
|
user_id = None
|
2022-05-19 15:14:20 +08:00
|
|
|
|
# 将时间转换到 东八 时区
|
2023-02-26 22:17:26 +08:00
|
|
|
|
messages = await get_list_msg(
|
|
|
|
|
|
user_id,
|
|
|
|
|
|
int(event.group_id),
|
|
|
|
|
|
days=(
|
|
|
|
|
|
start.astimezone(pytz.timezone("Asia/Shanghai")),
|
|
|
|
|
|
stop.astimezone(pytz.timezone("Asia/Shanghai")),
|
|
|
|
|
|
),
|
|
|
|
|
|
)
|
2022-05-18 20:31:14 +08:00
|
|
|
|
if messages:
|
|
|
|
|
|
image_bytes = await draw_word_cloud(messages, get_driver().config)
|
|
|
|
|
|
if image_bytes:
|
|
|
|
|
|
await wordcloud_cmd.finish(MessageSegment.image(image_bytes), at_sender=my)
|
|
|
|
|
|
else:
|
|
|
|
|
|
await wordcloud_cmd.finish("生成词云失败", at_sender=my)
|
|
|
|
|
|
else:
|
|
|
|
|
|
await wordcloud_cmd.finish("没有获取到词云数据", at_sender=my)
|