mirror of
https://github.com/PaiGramTeam/PaiGram.git
synced 2025-01-15 00:35:58 +00:00
123 lines
5.3 KiB
Python
123 lines
5.3 KiB
Python
|
from functools import partial
|
|||
|
from typing import Dict, List, TYPE_CHECKING
|
|||
|
|
|||
|
from telegram import InlineKeyboardButton, InlineKeyboardMarkup
|
|||
|
|
|||
|
from core.plugin import Plugin, handler
|
|||
|
from gram_core.plugin.methods.migrate_data import IMigrateData, MigrateDataException
|
|||
|
from gram_core.services.players import PlayersService
|
|||
|
from utils.log import logger
|
|||
|
|
|||
|
if TYPE_CHECKING:
|
|||
|
from telegram import Update
|
|||
|
from telegram.ext import ContextTypes
|
|||
|
|
|||
|
|
|||
|
class MigrateAdmin(Plugin):
|
|||
|
def __init__(self, players_service: PlayersService):
|
|||
|
self.players_service = players_service
|
|||
|
self.cache_data: Dict[int, List[IMigrateData]] = {}
|
|||
|
self.wait_time = 60
|
|||
|
|
|||
|
async def _add_pop_cache_job(self, user_id: int) -> None:
|
|||
|
if user_id in self.cache_data:
|
|||
|
del self.cache_data[user_id]
|
|||
|
|
|||
|
def add_pop_cache_job(self, user_id: int) -> None:
|
|||
|
job_queue = self.application.job_queue
|
|||
|
if job_queue is None:
|
|||
|
raise RuntimeError
|
|||
|
job_queue.run_once(
|
|||
|
callback=partial(self._add_pop_cache_job, user_id=user_id), when=60, name=f"{user_id}|migrate_pop_cache"
|
|||
|
)
|
|||
|
|
|||
|
def cancel_pop_cache_job(self, user_id: int) -> None:
|
|||
|
job_queue = self.application.job_queue
|
|||
|
if job_queue is None:
|
|||
|
raise RuntimeError
|
|||
|
if job := job_queue.get_jobs_by_name(f"{user_id}|migrate_pop_cache"):
|
|||
|
job[0].schedule_removal()
|
|||
|
|
|||
|
@handler.command(command="migrate_admin", block=False, admin=True)
|
|||
|
async def migrate_admin_command(self, update: "Update", context: "ContextTypes.DEFAULT_TYPE"):
|
|||
|
message = update.effective_message
|
|||
|
args = self.get_args(context)
|
|||
|
logger.info("管理员 %s[%s] migrate_admin 命令请求", message.from_user.full_name, message.from_user.id)
|
|||
|
if (not args) or len(args) < 2:
|
|||
|
await message.reply_text("参数错误,请指定新旧用户 id !")
|
|||
|
return
|
|||
|
try:
|
|||
|
old_user_id, new_user_id = int(args[0]), int(args[1])
|
|||
|
except ValueError:
|
|||
|
await message.reply_text("参数错误,请指定新旧用户 id !")
|
|||
|
return
|
|||
|
if old_user_id in self.cache_data:
|
|||
|
await message.reply_text("该用户正在迁移数据中,请稍后再试!")
|
|||
|
return
|
|||
|
data = []
|
|||
|
players = await self.players_service.get_all_by_user_id(old_user_id)
|
|||
|
for _, instance in self.application.managers.plugins_map.items():
|
|||
|
if _data := await instance.get_migrate_data(old_user_id, new_user_id, players):
|
|||
|
data.append(_data)
|
|||
|
if not data:
|
|||
|
await message.reply_text("没有需要迁移的数据!")
|
|||
|
return
|
|||
|
text = "确定迁移以下数据?\n\n"
|
|||
|
for d in data:
|
|||
|
text += f"- {await d.migrate_data_msg()}\n"
|
|||
|
self.cache_data[old_user_id] = data
|
|||
|
buttons = [
|
|||
|
[
|
|||
|
InlineKeyboardButton(
|
|||
|
"确定迁移",
|
|||
|
callback_data=f"migrate_admin|{old_user_id}",
|
|||
|
)
|
|||
|
],
|
|||
|
]
|
|||
|
await message.reply_text(text, reply_markup=InlineKeyboardMarkup(buttons))
|
|||
|
self.add_pop_cache_job(old_user_id)
|
|||
|
|
|||
|
async def try_migrate_data(self, user_id: int) -> str:
|
|||
|
text = []
|
|||
|
for d in self.cache_data[user_id]:
|
|||
|
try:
|
|||
|
logger.info("开始迁移数据 class[%s]", d.__class__.__name__)
|
|||
|
await d.migrate_data()
|
|||
|
logger.info("迁移数据成功 class[%s]", d.__class__.__name__)
|
|||
|
except MigrateDataException as e:
|
|||
|
text.append(e.msg)
|
|||
|
except Exception as e:
|
|||
|
logger.exception("迁移数据失败,未知错误! class[%s]", d.__class__.__name__, exc_info=e)
|
|||
|
text.append("迁移部分数据出现未知错误,请联系管理员!")
|
|||
|
if text:
|
|||
|
return "- " + "\n- ".join(text)
|
|||
|
|
|||
|
@handler.callback_query(pattern=r"^migrate_admin\|", block=False)
|
|||
|
async def callback_query_migrate_admin(self, update: "Update", _: "ContextTypes.DEFAULT_TYPE") -> None:
|
|||
|
callback_query = update.callback_query
|
|||
|
user = callback_query.from_user
|
|||
|
message = callback_query.message
|
|||
|
logger.info("管理员 %s[%s] migrate_admin callback 请求", user.full_name, user.id)
|
|||
|
|
|||
|
async def get_migrate_admin_callback(callback_query_data: str) -> int:
|
|||
|
_data = callback_query_data.split("|")
|
|||
|
_old_user_id = int(_data[1])
|
|||
|
logger.debug("callback_query_data函数返回 old_user_id[%s]", _old_user_id)
|
|||
|
return _old_user_id
|
|||
|
|
|||
|
old_user_id = await get_migrate_admin_callback(callback_query.data)
|
|||
|
if old_user_id not in self.cache_data:
|
|||
|
await callback_query.answer("请求已过期,请重新发起请求!", show_alert=True)
|
|||
|
self.add_delete_message_job(message, delay=5)
|
|||
|
return
|
|||
|
self.cancel_pop_cache_job(old_user_id)
|
|||
|
await message.edit_text("正在迁移数据,请稍后...", reply_markup=None)
|
|||
|
try:
|
|||
|
text = await self.try_migrate_data(old_user_id)
|
|||
|
finally:
|
|||
|
await self._add_pop_cache_job(old_user_id)
|
|||
|
if text:
|
|||
|
await message.edit_text(f"迁移部分数据失败!\n\n{text}")
|
|||
|
return
|
|||
|
await message.edit_text("迁移数据成功!")
|