mirror of
				https://github.com/LiteyukiStudio/LiteyukiBot.git
				synced 2025-10-26 15:46:23 +00:00 
			
		
		
		
	✨ 新增 stat rank 功能
This commit is contained in:
		| @@ -31,7 +31,7 @@ if get_config("debug", False): | ||||
|         def on_modified(self, event): | ||||
|             if event.src_path.endswith(src_excludes_extensions) or event.is_directory or "__pycache__" in event.src_path: | ||||
|                 return | ||||
|             nonebot.logger.debug(f"{event.src_path} modified, reloading bot...") | ||||
|             nonebot.logger.info(f"{event.src_path} modified, reloading bot...") | ||||
|             Reloader.reload() | ||||
|  | ||||
|  | ||||
| @@ -40,7 +40,7 @@ if get_config("debug", False): | ||||
|         Handler for resource file changes | ||||
|         """ | ||||
|         def on_modified(self, event): | ||||
|             nonebot.logger.debug(f"{event.src_path} modified, reloading resource...") | ||||
|             nonebot.logger.info(f"{event.src_path} modified, reloading resource...") | ||||
|             load_resources() | ||||
|  | ||||
|  | ||||
|   | ||||
| @@ -1,12 +1,16 @@ | ||||
| import time | ||||
| from typing import Any | ||||
|  | ||||
| from collections import Counter | ||||
|  | ||||
| from nonebot import Bot | ||||
|  | ||||
| from liteyuki.utils.message.html_tool import template2image | ||||
| from .common import MessageEventModel, msg_db | ||||
| from liteyuki.utils.base.language import Language | ||||
| from liteyuki.utils.base.resource import get_path | ||||
| from liteyuki.utils.message.npl import convert_seconds_to_time | ||||
| from contextvars import ContextVar | ||||
| from liteyuki.utils.message.string_tool import convert_seconds_to_time | ||||
| from ...utils.external.logo import get_group_icon, get_user_icon | ||||
|  | ||||
|  | ||||
| async def count_msg_by_bot_id(bot_id: str) -> int: | ||||
| @@ -22,12 +26,18 @@ async def count_msg_by_bot_id(bot_id: str) -> int: | ||||
|     return len(msg_rows) | ||||
|  | ||||
|  | ||||
| async def get_stat_msg_image(duration: int, period: int, group_id: str = None, bot_id: str = None, user_id: str = None, | ||||
|                              ulang: Language = Language()) -> bytes: | ||||
| async def get_stat_msg_image( | ||||
|         duration: int, | ||||
|         period: int, | ||||
|         group_id: str = None, | ||||
|         bot_id: str = None, | ||||
|         user_id: str = None, | ||||
|         ulang: Language = Language() | ||||
| ) -> bytes: | ||||
|     """ | ||||
|     获取统计消息 | ||||
|     Args: | ||||
|         ctx: | ||||
|         user_id: | ||||
|         ulang: | ||||
|         bot_id: | ||||
|         group_id: | ||||
| @@ -78,10 +88,11 @@ async def get_stat_msg_image(duration: int, period: int, group_id: str = None, b | ||||
|     templates = { | ||||
|             "data": [ | ||||
|                     { | ||||
|                 "name": ulang.get("stat.message") | ||||
|                             "name"  : ulang.get("stat.message") | ||||
|                                       + f"    Period {convert_seconds_to_time(period)}" + f"    Duration {convert_seconds_to_time(duration)}" | ||||
|                         + (f"    Group {group_id}" if group_id else "") + (f"    Bot {bot_id}" if bot_id else "") + (f"    User {user_id}" if user_id else ""), | ||||
|                 "times": timestamps, | ||||
|                                       + (f"    Group {group_id}" if group_id else "") + (f"    Bot {bot_id}" if bot_id else "") + ( | ||||
|                                               f"    User {user_id}" if user_id else ""), | ||||
|                             "times" : timestamps, | ||||
|                             "counts": msg_count | ||||
|                     } | ||||
|             ] | ||||
| @@ -89,9 +100,73 @@ async def get_stat_msg_image(duration: int, period: int, group_id: str = None, b | ||||
|  | ||||
|     return await template2image(get_path("templates/stat_msg.html"), templates) | ||||
|  | ||||
|     # if not timestamps or period_start_time != timestamps[-1]: | ||||
|     #     timestamps.append(period_start_time) | ||||
|     #     msg_count.append(1) | ||||
|     # else: | ||||
|     #     msg_count[-1] += 1 | ||||
|     # | ||||
|  | ||||
| async def get_stat_rank_image( | ||||
|         rank_type: str, | ||||
|         limit: dict[str, Any], | ||||
|         ulang: Language = Language(), | ||||
|         bot: Bot = None, | ||||
| ) -> bytes: | ||||
|     if rank_type == "user": | ||||
|         condition = "user_id != ''" | ||||
|         condition_args = [] | ||||
|     else: | ||||
|         condition = "group_id != ''" | ||||
|         condition_args = [] | ||||
|  | ||||
|     for k, v in limit.items(): | ||||
|         match k: | ||||
|             case "user_id": | ||||
|                 condition += " AND user_id = ?" | ||||
|                 condition_args.append(v) | ||||
|             case "group_id": | ||||
|                 condition += " AND group_id = ?" | ||||
|                 condition_args.append(v) | ||||
|             case "bot_id": | ||||
|                 condition += " AND bot_id = ?" | ||||
|                 condition_args.append(v) | ||||
|             case "duration": | ||||
|                 condition += " AND time > ?" | ||||
|                 condition_args.append(v) | ||||
|  | ||||
|     msg_rows = msg_db.where_all( | ||||
|         MessageEventModel(), | ||||
|         condition, | ||||
|         *condition_args | ||||
|     ) | ||||
|  | ||||
|     """ | ||||
|         { | ||||
|             name: string,   # user name or group name | ||||
|             count: int,     # message count | ||||
|             icon: string    # icon url | ||||
|         } | ||||
|     """ | ||||
|  | ||||
|     if rank_type == "user": | ||||
|         ranking_counter = Counter([msg.user_id for msg in msg_rows]) | ||||
|     else: | ||||
|         ranking_counter = Counter([msg.group_id for msg in msg_rows]) | ||||
|     sorted_data = sorted(ranking_counter.items(), key=lambda x: x[1], reverse=True) | ||||
|  | ||||
|     ranking: list[dict[str, Any]] = [ | ||||
|             { | ||||
|                     "name" : _[0], | ||||
|                     "count": _[1], | ||||
|                     "icon" : await (get_group_icon(platform="qq", group_id=_[0]) if rank_type == "group" else get_user_icon( | ||||
|                         platform="qq", user_id=_[0] | ||||
|                     )) | ||||
|             } | ||||
|             for _ in sorted_data[0:min(len(sorted_data), limit["rank"])] | ||||
|     ] | ||||
|  | ||||
|     templates = { | ||||
|             "data": | ||||
|                 { | ||||
|                         "name"   : ulang.get("stat.rank") + f"    Type {rank_type}" + f"    Limit {limit}", | ||||
|                         "ranking": ranking | ||||
|                 } | ||||
|  | ||||
|     } | ||||
|  | ||||
|     return await template2image(get_path("templates/stat_rank.html"), templates, debug=True) | ||||
|   | ||||
| @@ -1,5 +1,5 @@ | ||||
| from nonebot import Bot, require | ||||
| from liteyuki.utils.message.npl import convert_duration, convert_time_to_seconds | ||||
| from liteyuki.utils.message.string_tool import convert_duration, convert_time_to_seconds | ||||
| from .stat_api import * | ||||
| from liteyuki.utils import event as event_utils | ||||
| from liteyuki.utils.base.language import Language | ||||
| @@ -7,7 +7,16 @@ from liteyuki.utils.base.ly_typing import T_MessageEvent | ||||
|  | ||||
| require("nonebot_plugin_alconna") | ||||
|  | ||||
| from nonebot_plugin_alconna import UniMessage, on_alconna, Alconna, Args, Subcommand, Arparma, Option | ||||
| from nonebot_plugin_alconna import ( | ||||
|     UniMessage, | ||||
|     on_alconna, | ||||
|     Alconna, | ||||
|     Args, | ||||
|     Subcommand, | ||||
|     Arparma, | ||||
|     Option, | ||||
|     MultiVar | ||||
| ) | ||||
|  | ||||
| stat_msg = on_alconna( | ||||
|     Alconna( | ||||
| @@ -42,6 +51,33 @@ stat_msg = on_alconna( | ||||
|             ), | ||||
|             alias={"msg", "m"}, | ||||
|             help_text="查看统计次数内的消息" | ||||
|         ), | ||||
|         Subcommand( | ||||
|             "rank", | ||||
|             Option( | ||||
|                 "-u|--user", | ||||
|                 help_text="以用户为指标", | ||||
|             ), | ||||
|             Option( | ||||
|                 "-g|--group", | ||||
|                 help_text="以群组为指标", | ||||
|             ), | ||||
|             Option( | ||||
|                 "-l|--limit", | ||||
|                 Args["limit", MultiVar(str)], | ||||
|                 help_text="限制参数,使用key=val格式", | ||||
|             ), | ||||
|             Option( | ||||
|                 "-d|--duration", | ||||
|                 Args["duration", str, "1d"], | ||||
|                 help_text="统计时间", | ||||
|             ), | ||||
|             Option( | ||||
|                 "-r|--rank", | ||||
|                 Args["rank", int, 20], | ||||
|                 help_text="指定排名", | ||||
|             ), | ||||
|             alias={"r"}, | ||||
|         ) | ||||
|     ), | ||||
|     aliases={"stat"} | ||||
| @@ -51,7 +87,6 @@ stat_msg = on_alconna( | ||||
| @stat_msg.assign("message") | ||||
| async def _(result: Arparma, event: T_MessageEvent, bot: Bot): | ||||
|     ulang = Language(event_utils.get_user_id(event)) | ||||
|  | ||||
|     try: | ||||
|         duration = convert_time_to_seconds(result.other_args.get("duration", "2d"))  # 秒数 | ||||
|         period = convert_time_to_seconds(result.other_args.get("period", "1m")) | ||||
| @@ -77,3 +112,24 @@ async def _(result: Arparma, event: T_MessageEvent, bot: Bot): | ||||
|  | ||||
|     img = await get_stat_msg_image(duration=duration, period=period, group_id=group_id, bot_id=bot_id, user_id=user_id, ulang=ulang) | ||||
|     await stat_msg.send(UniMessage.image(raw=img)) | ||||
|  | ||||
|  | ||||
| @stat_msg.assign("rank") | ||||
| async def _(result: Arparma, event: T_MessageEvent, bot: Bot): | ||||
|     ulang = Language(event_utils.get_user_id(event)) | ||||
|     rank_type = "user" | ||||
|     duration = convert_time_to_seconds(result.other_args.get("duration", "1d")) | ||||
|     print(result) | ||||
|     if result.subcommands.get("rank").options.get("user"): | ||||
|         rank_type = "user" | ||||
|     elif result.subcommands.get("rank").options.get("group"): | ||||
|         rank_type = "group" | ||||
|  | ||||
|     limit = result.other_args.get("limit", {}) | ||||
|     if limit: | ||||
|         limit = dict([i.split("=") for i in limit]) | ||||
|     limit["duration"] = time.time() - duration  # 起始时间戳 | ||||
|     limit["rank"] = result.other_args.get("rank", 20) | ||||
|  | ||||
|     img = await get_stat_rank_image(rank_type=rank_type, limit=limit, ulang=ulang) | ||||
|     await stat_msg.send(UniMessage.image(raw=img)) | ||||
|   | ||||
| @@ -1 +1,2 @@ | ||||
| stat.message=统计消息 | ||||
| stat.rank=发言排名 | ||||
| @@ -0,0 +1,25 @@ | ||||
| let data = JSON.parse(document.getElementById("data").innerText)    // object | ||||
|  | ||||
| const rowDiv = document.importNode(document.getElementById("row-template").content, true) | ||||
|  | ||||
| function randomHideChar(str) { | ||||
|     // 随机隐藏6位以上字符串的中间连续四位字符,用*代替 | ||||
|     if (str.length <= 6) { | ||||
|         return str | ||||
|     } | ||||
|     let start = Math.floor(str.length / 2) - 2 | ||||
|     return str.slice(0, start) + "****" + str.slice(start + 4) | ||||
| } | ||||
| data["ranking"].forEach((item) => { | ||||
|     let row = rowDiv.cloneNode(true) | ||||
|     let rowID = item["name"] | ||||
|     let rowIconSrc = item["icon"] | ||||
|     let rowCount = item["count"] | ||||
|  | ||||
|     row.querySelector(".row-name").innerText = randomHideChar(rowID) | ||||
|     row.querySelector(".row-icon").src = rowIconSrc | ||||
|     row.querySelector(".row-count").innerText = rowCount | ||||
|  | ||||
|     document.body.appendChild(row) | ||||
| }) | ||||
|  | ||||
| @@ -0,0 +1,54 @@ | ||||
| <!DOCTYPE html> | ||||
| <html lang="zh" xmlns="http://www.w3.org/1999/html"> | ||||
| <head> | ||||
|     <meta charset="UTF-8"> | ||||
|     <title>Liteyuki Stats Message</title> | ||||
|     <link rel="stylesheet" href="./css/card.css"> | ||||
|     <link rel="stylesheet" href="./css/fonts.css"> | ||||
|     <link rel="stylesheet" href="./css/stat_rank.css"> | ||||
|     <style> | ||||
|         .row { | ||||
|             height: 100px; | ||||
|             display: flex; | ||||
|             background-color: rgba(255, 255, 255, 0.9); | ||||
|             border-radius: 100px; | ||||
|             margin-bottom: 10px; | ||||
|             padding-right: 10px; | ||||
|         } | ||||
|          | ||||
|         .row-name { | ||||
|             font-size: 40px; | ||||
|             align-content: center; | ||||
|             width: 100px; | ||||
|             text-align: left; | ||||
|         } | ||||
|          | ||||
|         .row-icon { | ||||
|             border-radius: 50%; | ||||
|             margin-right: auto; | ||||
|         } | ||||
|          | ||||
|         .row-count { | ||||
|             align-content: center; | ||||
|             font-size: 40px; | ||||
|             /*    靠右*/ | ||||
|             margin-left: auto; | ||||
|         } | ||||
|     </style> | ||||
| </head> | ||||
|  | ||||
| <body> | ||||
|  | ||||
| <template id="row-template"> | ||||
|     <div class="row"> | ||||
|         <img src="./img/arrow-up.svg" alt="up" class="row-icon"> | ||||
|         <div class="row-name"></div> | ||||
|         <div class="row-count"></div> | ||||
|     </div> | ||||
| </template> | ||||
|  | ||||
| <div class="data-storage" id="data">{{ data | tojson }}</div> | ||||
| <script src="https://cdnjs.cloudflare.com/ajax/libs/echarts/5.5.0/echarts.min.js"></script> | ||||
| <script src="./js/stat_rank.js"></script> | ||||
| <script src="./js/card.js"></script> | ||||
| </body> | ||||
							
								
								
									
										0
									
								
								liteyuki/utils/external/__init__.py
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										0
									
								
								liteyuki/utils/external/__init__.py
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
								
								
									
										40
									
								
								liteyuki/utils/external/logo.py
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										40
									
								
								liteyuki/utils/external/logo.py
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,40 @@ | ||||
| async def get_user_icon(platform: str, user_id: str) -> str: | ||||
|     """ | ||||
|     获取用户头像 | ||||
|     Args: | ||||
|         platform: qq, telegram, discord... | ||||
|         user_id: 1234567890 | ||||
|  | ||||
|     Returns: | ||||
|         str: 头像链接 | ||||
|     """ | ||||
|     match platform: | ||||
|         case "qq": | ||||
|             return f"http://q1.qlogo.cn/g?b=qq&nk={user_id}&s=640" | ||||
|         case "telegram": | ||||
|             return f"https://t.me/i/userpic/320/{user_id}.jpg" | ||||
|         case "discord": | ||||
|             return f"https://cdn.discordapp.com/avatars/{user_id}/" | ||||
|         case _: | ||||
|             return "" | ||||
|  | ||||
|  | ||||
| async def get_group_icon(platform: str, group_id: str) -> str: | ||||
|     """ | ||||
|     获取群组头像 | ||||
|     Args: | ||||
|         platform: qq, telegram, discord... | ||||
|         group_id: 1234567890 | ||||
|  | ||||
|     Returns: | ||||
|         str: 头像链接 | ||||
|     """ | ||||
|     match platform: | ||||
|         case "qq": | ||||
|             return f"http://p.qlogo.cn/gh/{group_id}/{group_id}/640" | ||||
|         case "telegram": | ||||
|             return f"https://t.me/c/{group_id}/" | ||||
|         case "discord": | ||||
|             return f"https://cdn.discordapp.com/icons/{group_id}/" | ||||
|         case _: | ||||
|             return "" | ||||
							
								
								
									
										0
									
								
								liteyuki/utils/nb/__init__.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										0
									
								
								liteyuki/utils/nb/__init__.py
									
									
									
									
									
										Normal file
									
								
							
		Reference in New Issue
	
	Block a user