From 6902919cda32de24bc4b60e15b119834f03b9cf7 Mon Sep 17 00:00:00 2001 From: liuwei Date: Wed, 19 Mar 2025 17:54:21 +0800 Subject: [PATCH] =?UTF-8?q?=E7=AD=BE=E5=88=B0=E5=8A=9F=E8=83=BD=E8=BF=81?= =?UTF-8?q?=E7=A7=BB=E5=88=B0=E6=8F=92=E4=BB=B6=E6=A8=A1=E5=9D=97=E4=B8=8B?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- message_sign/main.py | 205 --------------- message_sign/readme.md | 2 - plugins/message_sign/__init__.py | 7 + .../message_sign}/config.toml | 0 plugins/message_sign/main.py | 237 ++++++++++++++++++ robot.py | 8 - 6 files changed, 244 insertions(+), 215 deletions(-) delete mode 100644 message_sign/main.py delete mode 100644 message_sign/readme.md create mode 100644 plugins/message_sign/__init__.py rename {message_sign => plugins/message_sign}/config.toml (100%) create mode 100644 plugins/message_sign/main.py diff --git a/message_sign/main.py b/message_sign/main.py deleted file mode 100644 index a011c9a..0000000 --- a/message_sign/main.py +++ /dev/null @@ -1,205 +0,0 @@ -from datetime import datetime, timedelta -import logging -import mysql.connector.pooling -import tomllib -import pytz -import redis -from typing import Optional - -from wcferry import Wcf, WxMsg - -from message_util import MessageUtil -from robot_cmd.robot_command import GroupBotManager, Feature, PermissionStatus -from db.connection import DBConnectionManager -from db.sign_in import SignInDB -from db.sign_in_redis import SignInRedisDB - -# 创建表的SQL语句保留在这里,用于初始化表 -CREATE_TABLE_SQL = """ -CREATE TABLE IF NOT EXISTS t_sign_record ( - id BIGINT AUTO_INCREMENT PRIMARY KEY, - wx_id VARCHAR(100) NOT NULL, - group_id VARCHAR(100) NOT NULL, - wx_nick_name VARCHAR(100) NOT NULL, - points INT DEFAULT 0, - sign_stat DATETIME, - signin_streak INT DEFAULT 0, - create_time DATETIME DEFAULT CURRENT_TIMESTAMP, - update_time DATETIME DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP, - UNIQUE KEY unique_sign (wx_id, group_id) -) -""" - - -class SignInSystem: - def __init__(self, wcf: Wcf, gbm: GroupBotManager, all_contacts: dict, - db_pool: mysql.connector.pooling.MySQLConnectionPool, redis_pool: redis.ConnectionPool, - message_util: MessageUtil): - # 读取配置文件 - with open('message_sign/config.toml', 'rb') as f: - self.config = tomllib.load(f)['SignIn'] - - self.LOG = logging.getLogger(__name__) - - if not self.config['enable']: - raise Exception("签到功能未启用") - - self.wcf = wcf - self.gbm = gbm - self.message_util = message_util - self.all_contacts = all_contacts - - # 初始化数据库连接管理器 - self.db_manager = DBConnectionManager.get_instance() - # 初始化数据库操作类 - self.sign_in_db = SignInDB(self.db_manager) - self.sign_in_redis = SignInRedisDB(self.db_manager) - - self.command = self.config['command'] - self.min_point = self.config['min-point'] - self.max_point = self.config['max-point'] - self.streak_cycle = self.config['streak-cycle'] - self.max_streak_point = self.config['max-streak-point'] - # 时区设置 - self.timezone = 'Asia/Shanghai' - - # 从 Redis 初始化签到数据 - self.today_signin_count = self.sign_in_redis.load_signin_count() - last_reset_date = self.sign_in_redis.get_last_reset_date() - if last_reset_date: - self.last_reset_date = last_reset_date - else: - self.last_reset_date = datetime.now(tz=pytz.timezone(self.timezone)).date() - self.sign_in_redis.save_last_reset_date(self.last_reset_date) - - self.LOG.info(f"[签到] 组件初始化完成 {self.command_format}") - - @property - def command_format(self): - return ','.join(self.command) - - @property - def enable(self): - return self.config['enable'] - - def initialize_table(self): - """初始化数据库表""" - self.sign_in_db.execute_update(CREATE_TABLE_SQL) - - def reset_today_count_if_needed(self): - """检查并重置每日签到计数""" - current_date = datetime.now(tz=pytz.timezone(self.timezone)).date() - if current_date != self.last_reset_date: - self.today_signin_count.clear() - self.sign_in_redis.reset_daily_counts() - self.last_reset_date = current_date - self.sign_in_redis.save_last_reset_date(self.last_reset_date) - self.LOG.info(f"[签到] 已重置每日签到计数,日期更新为 {current_date}") - - def get_today_signin_count(self, group_id: str) -> int: - """获取群内今日签到人数(使用缓存)""" - self.reset_today_count_if_needed() - return self.today_signin_count.get(group_id, 0) - - def get_user_record(self, wx_id: str, group_id: str) -> Optional[dict]: - """获取用户签到记录""" - return self.sign_in_db.get_user_record(wx_id, group_id) - - def calculate_points(self, streak: int) -> int: - """根据连续签到天数计算积分""" - base_points = self.min_point - extra_points = min(streak // self.streak_cycle, self.max_streak_point) - total_points = base_points + extra_points - return min(total_points, self.max_point) - - def member_sign_in(self, message: WxMsg): - """会员签到功能""" - if not self.enable: - return - - content = str(message.content).strip() - command = content.split(" ") - if not len(command) or command[0] not in self.command: - return - - if self.gbm.get_group_permission(message.roomid, Feature.SIGNIN) == PermissionStatus.DISABLED: - return - - # 获取当前时间,带有时区信息 - current_time = datetime.now(tz=pytz.timezone(self.timezone)) - - # 获取当天零点的时间 - today_start = current_time.replace(hour=0, minute=0, second=0, microsecond=0) - - # 获取昨天的时间 - yesterday = today_start - timedelta(days=1) - - # 获取用户的签到记录 - user_record = self.get_user_record(message.sender, message.roomid) - wx_nick_name = self.all_contacts.get(message.sender, message.sender) - - # 判断用户是否已经签到过 - if user_record and user_record.get('sign_stat'): - sign_stat = user_record['sign_stat'] - - # 确保 sign_stat 和 today_start 是同一时区对象 - if isinstance(sign_stat, datetime) and sign_stat.tzinfo is None: - sign_stat = pytz.timezone(self.timezone).localize(sign_stat) - - # 如果 sign_stat 已经大于或等于今天的零点,则认为用户已经签到过了 - if sign_stat >= today_start: - self.message_util.send_text_msg(f"您今天已经签到过了!当前积分:{user_record['points']}", message.roomid, - message.sender) - return - - streak = 0 - streak_broken = False - old_streak = 1 - if user_record and user_record['sign_stat']: - last_sign_date = user_record['sign_stat'].replace(hour=0, minute=0, second=0, microsecond=0) - # 确保 sign_stat 和 today_start 是同一时区对象 - if isinstance(last_sign_date, datetime) and last_sign_date.tzinfo is None: - last_sign_date = pytz.timezone(self.timezone).localize(last_sign_date) - self.LOG.info( - f"last_sign_date: {last_sign_date}, yesterday: {yesterday}, user_streak: {user_record['signin_streak']}") - if last_sign_date == yesterday: - streak = user_record['signin_streak'] + 1 - old_streak = streak - streak_broken = False - else: - streak = 1 - streak_broken = True - else: - streak = 1 - - today_signin_rank = self.get_today_signin_count(message.roomid) + 1 - self.today_signin_count[message.roomid] = today_signin_rank - self.sign_in_redis.save_signin_count(message.roomid, today_signin_rank) - - points_to_add = self.calculate_points(streak) - - # 使用数据库操作类更新或创建签到记录 - if user_record: - self.sign_in_db.update_sign_record( - message.sender, message.roomid, wx_nick_name, - points_to_add, current_time, streak - ) - else: - self.sign_in_db.create_sign_record( - message.sender, message.roomid, wx_nick_name, - points_to_add, current_time, streak - ) - - output = f"签到成功,加[{points_to_add}]分,第[{today_signin_rank}]个!" - - if streak_broken and old_streak > 0: # 只有在真的断签且之前有签到记录时才显示 - output += f"断开了 {old_streak} 天连签!" - elif streak > 1: - output += f"连签 {streak} 天!" - - self.message_util.send_text_msg(output, message.roomid, - message.sender) - - def __del__(self): - """连接池由外部管理,不需要手动关闭""" - pass diff --git a/message_sign/readme.md b/message_sign/readme.md deleted file mode 100644 index c9aaf2b..0000000 --- a/message_sign/readme.md +++ /dev/null @@ -1,2 +0,0 @@ -# 签到功能 -## 用户每日签到,可以汇报签到次数 diff --git a/plugins/message_sign/__init__.py b/plugins/message_sign/__init__.py new file mode 100644 index 0000000..845dd69 --- /dev/null +++ b/plugins/message_sign/__init__.py @@ -0,0 +1,7 @@ +# 从当前包的main模块导入MessageSignPlugin类 +from .main import MessageSignPlugin + +# 提供get_plugin函数,返回插件实例 +def get_plugin(): + """获取插件实例""" + return MessageSignPlugin() \ No newline at end of file diff --git a/message_sign/config.toml b/plugins/message_sign/config.toml similarity index 100% rename from message_sign/config.toml rename to plugins/message_sign/config.toml diff --git a/plugins/message_sign/main.py b/plugins/message_sign/main.py new file mode 100644 index 0000000..f12a7a9 --- /dev/null +++ b/plugins/message_sign/main.py @@ -0,0 +1,237 @@ +from datetime import datetime, timedelta +import logging +import pytz +from typing import Dict, Any, List, Optional, Tuple + +from wcferry import Wcf, WxMsg + +from plugin_common.message_plugin_interface import MessagePluginInterface +from plugin_common.plugin_interface import PluginStatus +from plugins.stats_collector.decorators import plugin_stats_decorator +from robot_cmd.robot_command import Feature, PermissionStatus, GroupBotManager +from db.sign_in import SignInDB +from db.sign_in_redis import SignInRedisDB + + +class MessageSignPlugin(MessagePluginInterface): + """签到插件""" + + @property + def name(self) -> str: + return "签到系统" + + @property + def version(self) -> str: + return "1.0.0" + + @property + def description(self) -> str: + return "提供群聊签到功能,支持连续签到奖励和积分系统" + + @property + def author(self) -> str: + return "WeChatRobot Team" + + @property + def command_prefix(self) -> Optional[str]: + return "" # 不需要前缀,直接匹配命令 + + @property + def commands(self) -> List[str]: + return self._commands + + def __init__(self): + super().__init__() + self.today_signin_count = {} + self.last_reset_date = None + self.timezone = 'Asia/Shanghai' + self.sign_in_db = None + self.sign_in_redis = None + + def initialize(self, context: Dict[str, Any]) -> bool: + """初始化插件""" + self.LOG = logging.getLogger(f"Plugin.{self.name}") + self.LOG.info(f"正在初始化 {self.name} 插件...") + + # 保存上下文对象 + self.wcf = context.get("wcf") + self.event_system = context.get("event_system") + self.message_util = context.get("message_util") + self.gbm = context.get("gbm") + self.all_contacts = context.get("all_contacts", {}) + self.db_manager = context.get("db_manager") + + if not self.db_manager: + self.LOG.error("数据库连接管理器未初始化,插件无法正常工作") + return False + + # 初始化数据库操作类 + self.sign_in_db = SignInDB(self.db_manager) + self.sign_in_redis = SignInRedisDB(self.db_manager) + + # 从配置中获取参数 + sign_in_config = self._config.get("SignIn", {}) + self._commands = sign_in_config.get("command", ["签到", "每日签到", "qd", "Qd", "QD", "上班", "牛马"]) + self.min_point = sign_in_config.get("min-point", 3) + self.max_point = sign_in_config.get("max-point", 50) + self.streak_cycle = sign_in_config.get("streak-cycle", 1) + self.max_streak_point = sign_in_config.get("max-streak-point", 50) + self.enable = sign_in_config.get("enable", True) + + # 从 Redis 初始化签到数据 + self.today_signin_count = self.sign_in_redis.load_signin_count() + last_reset_date = self.sign_in_redis.get_last_reset_date() + if last_reset_date: + self.last_reset_date = last_reset_date + else: + self.last_reset_date = datetime.now(tz=pytz.timezone(self.timezone)).date() + self.sign_in_redis.save_last_reset_date(self.last_reset_date) + + self.LOG.info(f"[{self.name}] 插件初始化完成,指令:{self._commands}") + return True + + def start(self) -> bool: + """启动插件""" + self.LOG.info(f"[{self.name}] 插件已启动") + self.status = PluginStatus.RUNNING + return True + + def stop(self) -> bool: + """停止插件""" + self.LOG.info(f"[{self.name}] 插件已停止") + self.status = PluginStatus.STOPPED + return True + + def can_process(self, message: Dict[str, Any]) -> bool: + """检查是否可以处理该消息""" + if not self.enable: + return False + + content = str(message.get("content", "")).strip() + command = content.split(" ")[0] + + return command in self._commands + + @plugin_stats_decorator(plugin_name="签到系统") + def process_message(self, message: Dict[str, Any]) -> Tuple[bool, Optional[str]]: + """处理消息""" + content = str(message.get("content", "")).strip() + self.LOG.info(f"插件执行: {self.name}:{content}") + command = content.split(" ")[0] + sender = message.get("sender") + roomid = message.get("roomid", "") + wcf: Wcf = message.get("wcf") + gbm: GroupBotManager = message.get("gbm") + all_contacts = message.get("all_contacts", {}) + + # 检查权限 + if roomid and gbm.get_group_permission(roomid, Feature.SIGNIN) == PermissionStatus.DISABLED: + return False, "没有权限" + + try: + # 获取当前时间,带有时区信息 + current_time = datetime.now(tz=pytz.timezone(self.timezone)) + + # 获取当天零点的时间 + today_start = current_time.replace(hour=0, minute=0, second=0, microsecond=0) + + # 获取昨天的时间 + yesterday = today_start - timedelta(days=1) + + # 获取用户的签到记录 + user_record = self.get_user_record(sender, roomid) + wx_nick_name = all_contacts.get(sender, sender) + + # 判断用户是否已经签到过 + if user_record and user_record.get('sign_stat'): + sign_stat = user_record['sign_stat'] + + # 确保 sign_stat 和 today_start 是同一时区对象 + if isinstance(sign_stat, datetime) and sign_stat.tzinfo is None: + sign_stat = pytz.timezone(self.timezone).localize(sign_stat) + + # 如果 sign_stat 已经大于或等于今天的零点,则认为用户已经签到过了 + if sign_stat >= today_start: + wcf.send_text(f"您今天已经签到过了!当前积分:{user_record['points']}", + (roomid if roomid else sender), sender) + return True, "已签到" + + streak = 0 + streak_broken = False + old_streak = 1 + if user_record and user_record['sign_stat']: + last_sign_date = user_record['sign_stat'].replace(hour=0, minute=0, second=0, microsecond=0) + # 确保 sign_stat 和 today_start 是同一时区对象 + if isinstance(last_sign_date, datetime) and last_sign_date.tzinfo is None: + last_sign_date = pytz.timezone(self.timezone).localize(last_sign_date) + self.LOG.info( + f"last_sign_date: {last_sign_date}, yesterday: {yesterday}, user_streak: {user_record['signin_streak']}") + if last_sign_date == yesterday: + streak = user_record['signin_streak'] + 1 + old_streak = streak + streak_broken = False + else: + streak = 1 + streak_broken = True + else: + streak = 1 + + today_signin_rank = self.get_today_signin_count(roomid) + 1 + self.today_signin_count[roomid] = today_signin_rank + self.sign_in_redis.save_signin_count(roomid, today_signin_rank) + + points_to_add = self.calculate_points(streak) + + # 使用数据库操作类更新或创建签到记录 + if user_record: + self.sign_in_db.update_sign_record( + sender, roomid, wx_nick_name, + points_to_add, current_time, streak + ) + else: + self.sign_in_db.create_sign_record( + sender, roomid, wx_nick_name, + points_to_add, current_time, streak + ) + + output = f"签到成功,加[{points_to_add}]分,第[{today_signin_rank}]个!" + + if streak_broken and old_streak > 0: # 只有在真的断签且之前有签到记录时才显示 + output += f"断开了 {old_streak} 天连签!" + elif streak > 1: + output += f"连签 {streak} 天!" + + wcf.send_text(output, (roomid if roomid else sender), sender) + return True, "签到成功" + + except Exception as e: + self.LOG.error(f"处理签到请求出错: {e}") + wcf.send_text(f"签到出错:{e}", + (roomid if roomid else sender), sender) + return True, f"处理出错: {e}" + + def reset_today_count_if_needed(self): + """检查并重置每日签到计数""" + current_date = datetime.now(tz=pytz.timezone(self.timezone)).date() + if current_date != self.last_reset_date: + self.today_signin_count.clear() + self.sign_in_redis.reset_daily_counts() + self.last_reset_date = current_date + self.sign_in_redis.save_last_reset_date(self.last_reset_date) + self.LOG.info(f"[签到] 已重置每日签到计数,日期更新为 {current_date}") + + def get_today_signin_count(self, group_id: str) -> int: + """获取群内今日签到人数(使用缓存)""" + self.reset_today_count_if_needed() + return self.today_signin_count.get(group_id, 0) + + def get_user_record(self, wx_id: str, group_id: str) -> Optional[dict]: + """获取用户签到记录""" + return self.sign_in_db.get_user_record(wx_id, group_id) + + def calculate_points(self, streak: int) -> int: + """根据连续签到天数计算积分""" + base_points = self.min_point + extra_points = min(streak // self.streak_cycle, self.max_streak_point) + total_points = base_points + extra_points + return min(total_points, self.max_point) \ No newline at end of file diff --git a/robot.py b/robot.py index 0e3cde2..83a6bc6 100644 --- a/robot.py +++ b/robot.py @@ -26,7 +26,6 @@ from game_task.game_task_encyclopedia import game_process_message, get_group_ids from group_add.main import GroupAdd from group_auto.group_auto_invite import get_first_group_id, process_command from group_auto.group_member_change import GroupMemberChange -from message_sign.main import SignInSystem from message_storage.message_to_db import MessageStorage from plugin_common.event_system import EventType, EventSystem from plugin_common.message_plugin_interface import MessagePluginInterface @@ -114,8 +113,6 @@ class Robot(Job): self.message_storage = MessageStorage() # 群成员变更模块加载 self.gmc = GroupMemberChange(wcf, self.redis_pool) - # 签到模块加载 - self.signin = SignInSystem(wcf, self.gbm, self.allContacts, self.db_pool, self.redis_pool, self.message_util) # 加群测试 self.group_add = GroupAdd(wcf, self.gbm) @@ -317,11 +314,6 @@ class Robot(Job): except Exception as e: self.LOG.error(f"group_member_change error: {e}") - try: - self.signin.member_sign_in(message=msg) - except Exception as e: - self.LOG.error(f"member_sign_in error: {e}") - if msg.is_at(self.wxid): # 被@ self.toAt(msg) return # 处理完群聊信息,后面就不需要处理了