2025-02-24 09:33:06 +08:00
|
|
|
|
import asyncio
|
2024-12-10 19:49:11 +08:00
|
|
|
|
from collections.abc import Awaitable, Callable
|
2024-05-20 22:03:11 +08:00
|
|
|
|
import random
|
2025-06-16 09:18:58 +08:00
|
|
|
|
from typing import cast
|
2024-02-29 03:07:31 +08:00
|
|
|
|
|
2024-05-04 13:48:12 +08:00
|
|
|
|
import httpx
|
2024-02-29 03:07:31 +08:00
|
|
|
|
import nonebot
|
2024-02-25 03:18:34 +08:00
|
|
|
|
from nonebot.adapters import Bot
|
2024-12-10 19:49:11 +08:00
|
|
|
|
from nonebot.utils import is_coroutine_callable
|
|
|
|
|
|
from nonebot_plugin_alconna import SupportScope
|
|
|
|
|
|
from nonebot_plugin_alconna.uniseg import Receipt, Target, UniMessage
|
2024-12-25 12:03:49 +08:00
|
|
|
|
from nonebot_plugin_uninfo import SceneType, Uninfo, get_interface
|
|
|
|
|
|
from nonebot_plugin_uninfo.model import Member
|
2024-12-10 19:49:11 +08:00
|
|
|
|
from pydantic import BaseModel
|
2024-02-25 03:18:34 +08:00
|
|
|
|
|
2024-08-29 22:01:34 +08:00
|
|
|
|
from zhenxun.configs.config import BotConfig
|
|
|
|
|
|
from zhenxun.models.friend_user import FriendUser
|
2024-09-09 23:01:15 +08:00
|
|
|
|
from zhenxun.models.group_console import GroupConsole
|
2024-12-10 19:49:11 +08:00
|
|
|
|
from zhenxun.services.log import logger
|
|
|
|
|
|
from zhenxun.utils.exception import NotFindSuperuser
|
|
|
|
|
|
from zhenxun.utils.http_utils import AsyncHttpx
|
|
|
|
|
|
from zhenxun.utils.message import MessageUtils
|
2024-02-25 03:18:34 +08:00
|
|
|
|
|
2024-10-18 18:57:55 +08:00
|
|
|
|
driver = nonebot.get_driver()
|
|
|
|
|
|
|
2024-02-25 03:18:34 +08:00
|
|
|
|
|
2024-05-16 19:54:30 +08:00
|
|
|
|
class UserData(BaseModel):
|
|
|
|
|
|
name: str
|
|
|
|
|
|
"""昵称"""
|
|
|
|
|
|
card: str | None = None
|
|
|
|
|
|
"""名片/备注"""
|
|
|
|
|
|
user_id: str
|
|
|
|
|
|
"""用户id"""
|
|
|
|
|
|
group_id: str | None = None
|
|
|
|
|
|
"""群组id"""
|
2024-12-25 12:03:49 +08:00
|
|
|
|
channel_id: str | None = None
|
|
|
|
|
|
"""频道id"""
|
2024-05-16 19:54:30 +08:00
|
|
|
|
role: str | None = None
|
|
|
|
|
|
"""角色"""
|
|
|
|
|
|
avatar_url: str | None = None
|
|
|
|
|
|
"""头像url"""
|
|
|
|
|
|
join_time: int | None = None
|
|
|
|
|
|
"""加入时间"""
|
|
|
|
|
|
|
|
|
|
|
|
|
2024-03-27 20:09:30 +08:00
|
|
|
|
class PlatformUtils:
|
2024-10-18 18:57:55 +08:00
|
|
|
|
@classmethod
|
2024-10-18 23:13:16 +08:00
|
|
|
|
def is_qbot(cls, session: Uninfo | Bot) -> bool:
|
2024-10-18 18:57:55 +08:00
|
|
|
|
"""判断bot是否为qq官bot
|
|
|
|
|
|
|
|
|
|
|
|
参数:
|
|
|
|
|
|
session: Uninfo
|
|
|
|
|
|
|
|
|
|
|
|
返回:
|
|
|
|
|
|
bool: 是否为官bot
|
|
|
|
|
|
"""
|
2024-10-18 23:13:16 +08:00
|
|
|
|
if isinstance(session, Bot):
|
|
|
|
|
|
return bool(BotConfig.get_qbot_uid(session.self_id))
|
2025-02-03 21:23:14 +08:00
|
|
|
|
if BotConfig.get_qbot_uid(session.self_id):
|
|
|
|
|
|
return True
|
2024-10-18 18:57:55 +08:00
|
|
|
|
return session.scope == SupportScope.qq_api
|
|
|
|
|
|
|
2024-05-29 02:01:26 +08:00
|
|
|
|
@classmethod
|
|
|
|
|
|
async def ban_user(cls, bot: Bot, user_id: str, group_id: str, duration: int):
|
|
|
|
|
|
"""禁言
|
|
|
|
|
|
|
|
|
|
|
|
参数:
|
|
|
|
|
|
bot: Bot
|
|
|
|
|
|
user_id: 用户id
|
|
|
|
|
|
group_id: 群组id
|
|
|
|
|
|
duration: 禁言时长(分钟)
|
|
|
|
|
|
"""
|
2024-12-25 12:03:49 +08:00
|
|
|
|
if cls.get_platform(bot) == "qq":
|
2024-05-29 02:01:26 +08:00
|
|
|
|
await bot.set_group_ban(
|
|
|
|
|
|
group_id=int(group_id),
|
|
|
|
|
|
user_id=int(user_id),
|
|
|
|
|
|
duration=duration * 60,
|
|
|
|
|
|
)
|
|
|
|
|
|
|
2024-05-20 22:03:11 +08:00
|
|
|
|
@classmethod
|
|
|
|
|
|
async def send_superuser(
|
|
|
|
|
|
cls,
|
2025-07-11 10:07:23 +08:00
|
|
|
|
bot: Bot | None,
|
2024-08-19 23:07:43 +08:00
|
|
|
|
message: UniMessage | str,
|
2024-05-20 22:03:11 +08:00
|
|
|
|
superuser_id: str | None = None,
|
2025-06-09 14:39:28 +08:00
|
|
|
|
) -> list[tuple[str, Receipt]]:
|
2024-05-20 22:03:11 +08:00
|
|
|
|
"""发送消息给超级用户
|
|
|
|
|
|
|
|
|
|
|
|
参数:
|
2025-07-11 10:07:23 +08:00
|
|
|
|
bot: Bot,没有传入时使用get_bot随机获取
|
2024-05-20 22:03:11 +08:00
|
|
|
|
message: 消息
|
|
|
|
|
|
superuser_id: 指定超级用户id.
|
|
|
|
|
|
|
|
|
|
|
|
异常:
|
|
|
|
|
|
NotFindSuperuser: 未找到超级用户id
|
|
|
|
|
|
|
|
|
|
|
|
返回:
|
|
|
|
|
|
Receipt | None: Receipt
|
|
|
|
|
|
"""
|
2025-07-11 10:07:23 +08:00
|
|
|
|
if not bot:
|
|
|
|
|
|
bot = nonebot.get_bot()
|
2025-06-09 14:39:28 +08:00
|
|
|
|
superuser_ids = []
|
|
|
|
|
|
if superuser_id:
|
|
|
|
|
|
superuser_ids.append(superuser_id)
|
|
|
|
|
|
elif platform := cls.get_platform(bot):
|
|
|
|
|
|
if platform_superusers := BotConfig.get_superuser(platform):
|
|
|
|
|
|
superuser_ids = platform_superusers
|
|
|
|
|
|
else:
|
|
|
|
|
|
raise NotFindSuperuser()
|
2024-08-19 23:07:43 +08:00
|
|
|
|
if isinstance(message, str):
|
|
|
|
|
|
message = MessageUtils.build_message(message)
|
2025-06-09 14:39:28 +08:00
|
|
|
|
result = []
|
|
|
|
|
|
for superuser_id in superuser_ids:
|
|
|
|
|
|
try:
|
|
|
|
|
|
result.append(
|
|
|
|
|
|
(
|
|
|
|
|
|
superuser_id,
|
|
|
|
|
|
await cls.send_message(bot, superuser_id, None, message),
|
|
|
|
|
|
)
|
|
|
|
|
|
)
|
|
|
|
|
|
except Exception as e:
|
|
|
|
|
|
logger.error(
|
|
|
|
|
|
"发送消息给超级用户失败",
|
|
|
|
|
|
"PlatformUtils:send_superuser",
|
|
|
|
|
|
target=superuser_id,
|
|
|
|
|
|
e=e,
|
|
|
|
|
|
)
|
|
|
|
|
|
return result
|
2024-05-20 22:03:11 +08:00
|
|
|
|
|
2024-05-16 19:54:30 +08:00
|
|
|
|
@classmethod
|
|
|
|
|
|
async def get_group_member_list(cls, bot: Bot, group_id: str) -> list[UserData]:
|
|
|
|
|
|
"""获取群组/频道成员列表
|
|
|
|
|
|
|
|
|
|
|
|
参数:
|
|
|
|
|
|
bot: Bot
|
|
|
|
|
|
group_id: 群组/频道id
|
|
|
|
|
|
|
|
|
|
|
|
返回:
|
|
|
|
|
|
list[UserData]: 用户数据列表
|
|
|
|
|
|
"""
|
2024-12-25 12:03:49 +08:00
|
|
|
|
if interface := get_interface(bot):
|
|
|
|
|
|
members: list[Member] = await interface.get_members(
|
|
|
|
|
|
SceneType.GROUP, group_id
|
|
|
|
|
|
)
|
|
|
|
|
|
return [
|
|
|
|
|
|
UserData(
|
|
|
|
|
|
name=member.user.name or "",
|
|
|
|
|
|
card=member.nick,
|
|
|
|
|
|
user_id=member.user.id,
|
|
|
|
|
|
group_id=group_id,
|
|
|
|
|
|
role=member.role.id if member.role else "",
|
|
|
|
|
|
avatar_url=member.user.avatar,
|
|
|
|
|
|
join_time=int(member.joined_at.timestamp())
|
|
|
|
|
|
if member.joined_at
|
|
|
|
|
|
else None,
|
|
|
|
|
|
)
|
|
|
|
|
|
for member in members
|
|
|
|
|
|
]
|
2024-05-16 19:54:30 +08:00
|
|
|
|
return []
|
|
|
|
|
|
|
|
|
|
|
|
@classmethod
|
|
|
|
|
|
async def get_user(
|
2024-12-25 12:03:49 +08:00
|
|
|
|
cls,
|
|
|
|
|
|
bot: Bot,
|
|
|
|
|
|
user_id: str,
|
|
|
|
|
|
group_id: str | None = None,
|
|
|
|
|
|
channel_id: str | None = None,
|
2024-05-16 19:54:30 +08:00
|
|
|
|
) -> UserData | None:
|
|
|
|
|
|
"""获取用户信息
|
|
|
|
|
|
|
|
|
|
|
|
参数:
|
|
|
|
|
|
bot: Bot
|
|
|
|
|
|
user_id: 用户id
|
2024-12-25 12:03:49 +08:00
|
|
|
|
group_id: 群组id.
|
|
|
|
|
|
channel_id: 频道id.
|
2024-05-16 19:54:30 +08:00
|
|
|
|
|
|
|
|
|
|
返回:
|
|
|
|
|
|
UserData | None: 用户数据
|
|
|
|
|
|
"""
|
2025-02-24 09:33:06 +08:00
|
|
|
|
if not (interface := get_interface(bot)):
|
|
|
|
|
|
return None
|
|
|
|
|
|
member = None
|
|
|
|
|
|
user = None
|
|
|
|
|
|
if channel_id:
|
|
|
|
|
|
member = await interface.get_member(
|
|
|
|
|
|
SceneType.CHANNEL_TEXT, channel_id, user_id
|
|
|
|
|
|
)
|
2024-12-25 12:03:49 +08:00
|
|
|
|
if member:
|
2025-02-24 09:33:06 +08:00
|
|
|
|
user = member.user
|
|
|
|
|
|
elif group_id:
|
|
|
|
|
|
member = await interface.get_member(SceneType.GROUP, group_id, user_id)
|
|
|
|
|
|
if member:
|
|
|
|
|
|
user = member.user
|
|
|
|
|
|
else:
|
|
|
|
|
|
user = await interface.get_user(user_id)
|
|
|
|
|
|
if not user:
|
|
|
|
|
|
return None
|
|
|
|
|
|
return (
|
|
|
|
|
|
UserData(
|
|
|
|
|
|
name=user.name or "",
|
|
|
|
|
|
card=member.nick,
|
|
|
|
|
|
user_id=user.id,
|
|
|
|
|
|
group_id=group_id,
|
|
|
|
|
|
channel_id=channel_id,
|
|
|
|
|
|
role=member.role.id if member.role else None,
|
|
|
|
|
|
join_time=(
|
|
|
|
|
|
int(member.joined_at.timestamp()) if member.joined_at else None
|
|
|
|
|
|
),
|
|
|
|
|
|
)
|
|
|
|
|
|
if member
|
|
|
|
|
|
else UserData(
|
|
|
|
|
|
name=user.name or "",
|
|
|
|
|
|
user_id=user.id,
|
|
|
|
|
|
group_id=group_id,
|
|
|
|
|
|
channel_id=channel_id,
|
|
|
|
|
|
)
|
|
|
|
|
|
)
|
2024-05-16 19:54:30 +08:00
|
|
|
|
|
2024-05-04 13:48:12 +08:00
|
|
|
|
@classmethod
|
2024-10-18 18:57:55 +08:00
|
|
|
|
async def get_user_avatar(
|
|
|
|
|
|
cls, user_id: str, platform: str, appid: str | None = None
|
|
|
|
|
|
) -> bytes | None:
|
2024-05-04 13:48:12 +08:00
|
|
|
|
"""快捷获取用户头像
|
|
|
|
|
|
|
|
|
|
|
|
参数:
|
|
|
|
|
|
user_id: 用户id
|
|
|
|
|
|
platform: 平台
|
|
|
|
|
|
"""
|
2024-10-18 18:57:55 +08:00
|
|
|
|
url = None
|
2024-05-04 13:48:12 +08:00
|
|
|
|
if platform == "qq":
|
2024-10-18 18:57:55 +08:00
|
|
|
|
if user_id.isdigit():
|
2025-07-01 10:30:22 +08:00
|
|
|
|
url = f"http://q1.qlogo.cn/g?b=qq&nk={user_id}&s=640"
|
2024-10-18 18:57:55 +08:00
|
|
|
|
else:
|
2025-01-10 18:39:23 +08:00
|
|
|
|
url = f"https://q.qlogo.cn/qqapp/{appid}/{user_id}/640"
|
2024-10-18 18:57:55 +08:00
|
|
|
|
return await AsyncHttpx.get_content(url) if url else None
|
2024-05-04 13:48:12 +08:00
|
|
|
|
|
2024-09-14 05:23:55 +08:00
|
|
|
|
@classmethod
|
2024-10-18 18:57:55 +08:00
|
|
|
|
def get_user_avatar_url(
|
|
|
|
|
|
cls, user_id: str, platform: str, appid: str | None = None
|
|
|
|
|
|
) -> str | None:
|
2024-09-14 05:23:55 +08:00
|
|
|
|
"""快捷获取用户头像url
|
|
|
|
|
|
|
|
|
|
|
|
参数:
|
|
|
|
|
|
user_id: 用户id
|
|
|
|
|
|
platform: 平台
|
|
|
|
|
|
"""
|
2025-01-10 18:39:23 +08:00
|
|
|
|
if platform != "qq":
|
2024-10-18 18:57:55 +08:00
|
|
|
|
return None
|
2025-01-10 18:39:23 +08:00
|
|
|
|
if user_id.isdigit():
|
|
|
|
|
|
return f"http://q1.qlogo.cn/g?b=qq&nk={user_id}&s=160"
|
|
|
|
|
|
else:
|
|
|
|
|
|
return f"https://q.qlogo.cn/qqapp/{appid}/{user_id}/640"
|
2024-09-14 05:23:55 +08:00
|
|
|
|
|
2024-05-04 13:48:12 +08:00
|
|
|
|
@classmethod
|
|
|
|
|
|
async def get_group_avatar(cls, gid: str, platform: str) -> bytes | None:
|
|
|
|
|
|
"""快捷获取用群头像
|
|
|
|
|
|
|
|
|
|
|
|
参数:
|
|
|
|
|
|
gid: 群组id
|
|
|
|
|
|
platform: 平台
|
|
|
|
|
|
"""
|
|
|
|
|
|
if platform == "qq":
|
|
|
|
|
|
url = f"http://p.qlogo.cn/gh/{gid}/{gid}/640/"
|
|
|
|
|
|
async with httpx.AsyncClient() as client:
|
|
|
|
|
|
for _ in range(3):
|
|
|
|
|
|
try:
|
|
|
|
|
|
return (await client.get(url)).content
|
2024-08-29 22:01:34 +08:00
|
|
|
|
except Exception:
|
2024-05-04 13:48:12 +08:00
|
|
|
|
logger.error(
|
|
|
|
|
|
"获取群头像错误", "Util", target=gid, platform=platform
|
|
|
|
|
|
)
|
|
|
|
|
|
return None
|
|
|
|
|
|
|
2024-03-18 16:10:44 +08:00
|
|
|
|
@classmethod
|
|
|
|
|
|
async def send_message(
|
|
|
|
|
|
cls,
|
|
|
|
|
|
bot: Bot,
|
|
|
|
|
|
user_id: str | None,
|
|
|
|
|
|
group_id: str | None,
|
2024-08-11 15:57:33 +08:00
|
|
|
|
message: str | UniMessage,
|
2024-05-20 22:03:11 +08:00
|
|
|
|
) -> Receipt | None:
|
2024-03-18 16:10:44 +08:00
|
|
|
|
"""发送消息
|
|
|
|
|
|
|
|
|
|
|
|
参数:
|
|
|
|
|
|
bot: Bot
|
|
|
|
|
|
user_id: 用户id
|
|
|
|
|
|
group_id: 群组id或频道id
|
|
|
|
|
|
message: 消息文本
|
|
|
|
|
|
|
|
|
|
|
|
返回:
|
2024-05-20 22:03:11 +08:00
|
|
|
|
Receipt | None: 是否发送成功
|
2024-03-18 16:10:44 +08:00
|
|
|
|
"""
|
2024-12-25 12:03:49 +08:00
|
|
|
|
if target := cls.get_target(user_id=user_id, group_id=group_id):
|
2024-08-11 15:57:33 +08:00
|
|
|
|
send_message = (
|
|
|
|
|
|
MessageUtils.build_message(message)
|
|
|
|
|
|
if isinstance(message, str)
|
|
|
|
|
|
else message
|
|
|
|
|
|
)
|
|
|
|
|
|
return await send_message.send(target=target, bot=bot)
|
2024-05-20 22:03:11 +08:00
|
|
|
|
return None
|
2024-03-18 16:10:44 +08:00
|
|
|
|
|
2024-02-25 03:18:34 +08:00
|
|
|
|
@classmethod
|
2024-02-28 13:51:16 +08:00
|
|
|
|
async def update_group(cls, bot: Bot) -> int:
|
2024-02-25 03:18:34 +08:00
|
|
|
|
"""更新群组信息
|
|
|
|
|
|
|
|
|
|
|
|
参数:
|
|
|
|
|
|
bot: Bot
|
|
|
|
|
|
|
|
|
|
|
|
返回:
|
|
|
|
|
|
int: 更新个数
|
|
|
|
|
|
"""
|
|
|
|
|
|
create_list = []
|
2024-08-25 15:49:36 +08:00
|
|
|
|
update_list = []
|
2024-02-29 03:07:31 +08:00
|
|
|
|
group_list, platform = await cls.get_group_list(bot)
|
2024-02-28 13:51:16 +08:00
|
|
|
|
if group_list:
|
2024-08-25 15:49:36 +08:00
|
|
|
|
db_group = await GroupConsole.all()
|
2024-12-25 12:03:49 +08:00
|
|
|
|
db_group_id: list[tuple[str, str]] = [
|
|
|
|
|
|
(group.group_id, group.channel_id) for group in db_group
|
|
|
|
|
|
]
|
2024-02-25 03:18:34 +08:00
|
|
|
|
for group in group_list:
|
|
|
|
|
|
group.platform = platform
|
2024-08-25 15:49:36 +08:00
|
|
|
|
if (group.group_id, group.channel_id) not in db_group_id:
|
2024-02-25 03:18:34 +08:00
|
|
|
|
create_list.append(group)
|
|
|
|
|
|
logger.debug(
|
|
|
|
|
|
"群聊信息更新成功",
|
|
|
|
|
|
"更新群信息",
|
|
|
|
|
|
target=f"{group.group_id}:{group.channel_id}",
|
|
|
|
|
|
)
|
2024-08-25 15:49:36 +08:00
|
|
|
|
else:
|
2024-08-29 22:01:34 +08:00
|
|
|
|
_group = next(
|
2024-08-25 15:49:36 +08:00
|
|
|
|
g
|
|
|
|
|
|
for g in db_group
|
|
|
|
|
|
if g.group_id == group.group_id
|
|
|
|
|
|
and g.channel_id == group.channel_id
|
2024-08-29 22:01:34 +08:00
|
|
|
|
)
|
2024-08-25 15:49:36 +08:00
|
|
|
|
_group.group_name = group.group_name
|
|
|
|
|
|
_group.max_member_count = group.max_member_count
|
|
|
|
|
|
_group.member_count = group.member_count
|
|
|
|
|
|
update_list.append(_group)
|
2024-02-25 03:18:34 +08:00
|
|
|
|
if create_list:
|
|
|
|
|
|
await GroupConsole.bulk_create(create_list, 10)
|
2024-08-25 15:49:36 +08:00
|
|
|
|
if group_list:
|
|
|
|
|
|
await GroupConsole.bulk_update(
|
|
|
|
|
|
update_list, ["group_name", "max_member_count", "member_count"], 10
|
|
|
|
|
|
)
|
2024-02-25 03:18:34 +08:00
|
|
|
|
return len(create_list)
|
|
|
|
|
|
|
|
|
|
|
|
@classmethod
|
2024-10-18 18:57:55 +08:00
|
|
|
|
def get_platform(cls, t: Bot | Uninfo) -> str:
|
2024-02-29 03:07:31 +08:00
|
|
|
|
"""获取平台
|
|
|
|
|
|
|
|
|
|
|
|
参数:
|
|
|
|
|
|
bot: Bot
|
|
|
|
|
|
|
|
|
|
|
|
返回:
|
|
|
|
|
|
str | None: 平台
|
|
|
|
|
|
"""
|
2024-10-18 18:57:55 +08:00
|
|
|
|
if isinstance(t, Bot):
|
|
|
|
|
|
if interface := get_interface(t):
|
|
|
|
|
|
info = interface.basic_info()
|
|
|
|
|
|
platform = info["scope"].lower()
|
|
|
|
|
|
return "qq" if platform.startswith("qq") else platform
|
|
|
|
|
|
else:
|
|
|
|
|
|
platform = t.basic["scope"].lower()
|
2024-10-10 23:25:46 +08:00
|
|
|
|
return "qq" if platform.startswith("qq") else platform
|
|
|
|
|
|
return "unknown"
|
2024-02-29 03:07:31 +08:00
|
|
|
|
|
2025-02-24 09:33:06 +08:00
|
|
|
|
@classmethod
|
|
|
|
|
|
def is_forward_merge_supported(cls, t: Bot | Uninfo) -> bool:
|
|
|
|
|
|
"""是否支持转发消息
|
|
|
|
|
|
|
|
|
|
|
|
参数:
|
|
|
|
|
|
t: bot | Uninfo
|
|
|
|
|
|
|
|
|
|
|
|
返回:
|
|
|
|
|
|
bool: 是否支持转发消息
|
|
|
|
|
|
"""
|
|
|
|
|
|
if not isinstance(t, Bot):
|
|
|
|
|
|
return t.basic["scope"] == SupportScope.qq_client
|
|
|
|
|
|
if interface := get_interface(t):
|
|
|
|
|
|
info = interface.basic_info()
|
|
|
|
|
|
return info["scope"] == SupportScope.qq_client
|
|
|
|
|
|
return False
|
|
|
|
|
|
|
2024-02-29 03:07:31 +08:00
|
|
|
|
@classmethod
|
2024-12-25 12:03:49 +08:00
|
|
|
|
async def get_group_list(
|
|
|
|
|
|
cls, bot: Bot, only_group: bool = False
|
|
|
|
|
|
) -> tuple[list[GroupConsole], str]:
|
2024-02-25 03:18:34 +08:00
|
|
|
|
"""获取群组列表
|
|
|
|
|
|
|
|
|
|
|
|
参数:
|
|
|
|
|
|
bot: Bot
|
2024-12-25 12:03:49 +08:00
|
|
|
|
only_group: 是否只获取群组(不获取channel)
|
2024-02-25 03:18:34 +08:00
|
|
|
|
|
|
|
|
|
|
返回:
|
2024-05-04 13:48:12 +08:00
|
|
|
|
tuple[list[GroupConsole], str]: 群组列表, 平台
|
2024-02-25 03:18:34 +08:00
|
|
|
|
"""
|
2025-02-03 21:23:14 +08:00
|
|
|
|
if not (interface := get_interface(bot)):
|
|
|
|
|
|
return [], ""
|
|
|
|
|
|
platform = cls.get_platform(bot)
|
|
|
|
|
|
result_list = []
|
|
|
|
|
|
scenes = await interface.get_scenes(SceneType.GROUP)
|
|
|
|
|
|
for scene in scenes:
|
|
|
|
|
|
group_id = scene.id
|
|
|
|
|
|
result_list.append(
|
|
|
|
|
|
GroupConsole(
|
|
|
|
|
|
group_id=scene.id,
|
|
|
|
|
|
group_name=scene.name,
|
2024-12-25 12:03:49 +08:00
|
|
|
|
)
|
2025-02-03 21:23:14 +08:00
|
|
|
|
)
|
|
|
|
|
|
if not only_group and platform != "qq":
|
|
|
|
|
|
if channel_list := await interface.get_scenes(parent_scene_id=group_id):
|
|
|
|
|
|
result_list.extend(
|
|
|
|
|
|
GroupConsole(
|
|
|
|
|
|
group_id=scene.id,
|
|
|
|
|
|
group_name=channel.name,
|
|
|
|
|
|
channel_id=channel.id,
|
|
|
|
|
|
)
|
|
|
|
|
|
for channel in channel_list
|
|
|
|
|
|
)
|
|
|
|
|
|
return result_list, platform
|
2024-02-25 03:18:34 +08:00
|
|
|
|
|
|
|
|
|
|
@classmethod
|
2024-02-28 13:51:16 +08:00
|
|
|
|
async def update_friend(cls, bot: Bot) -> int:
|
2024-02-25 03:18:34 +08:00
|
|
|
|
"""更新好友信息
|
|
|
|
|
|
|
|
|
|
|
|
参数:
|
|
|
|
|
|
bot: Bot
|
|
|
|
|
|
|
|
|
|
|
|
返回:
|
|
|
|
|
|
int: 更新个数
|
|
|
|
|
|
"""
|
|
|
|
|
|
create_list = []
|
2024-02-29 03:07:31 +08:00
|
|
|
|
friend_list, platform = await cls.get_friend_list(bot)
|
2024-02-28 13:51:16 +08:00
|
|
|
|
if friend_list:
|
2024-02-25 03:18:34 +08:00
|
|
|
|
user_id_list = await FriendUser.all().values_list("user_id", flat=True)
|
|
|
|
|
|
for friend in friend_list:
|
|
|
|
|
|
friend.platform = platform
|
|
|
|
|
|
if friend.user_id not in user_id_list:
|
|
|
|
|
|
create_list.append(friend)
|
|
|
|
|
|
if create_list:
|
|
|
|
|
|
await FriendUser.bulk_create(create_list, 10)
|
|
|
|
|
|
return len(create_list)
|
|
|
|
|
|
|
|
|
|
|
|
@classmethod
|
2024-02-29 03:07:31 +08:00
|
|
|
|
async def get_friend_list(cls, bot: Bot) -> tuple[list[FriendUser], str]:
|
2024-02-25 03:18:34 +08:00
|
|
|
|
"""获取好友列表
|
|
|
|
|
|
|
|
|
|
|
|
参数:
|
|
|
|
|
|
bot: Bot
|
|
|
|
|
|
|
|
|
|
|
|
返回:
|
|
|
|
|
|
list[FriendUser]: 好友列表
|
|
|
|
|
|
"""
|
2024-12-25 12:03:49 +08:00
|
|
|
|
if interface := get_interface(bot):
|
|
|
|
|
|
user_list = await interface.get_users()
|
2024-02-25 03:18:34 +08:00
|
|
|
|
return [
|
2024-12-25 12:03:49 +08:00
|
|
|
|
FriendUser(user_id=u.id, user_name=u.name) for u in user_list
|
|
|
|
|
|
], cls.get_platform(bot)
|
2024-02-28 13:51:16 +08:00
|
|
|
|
return [], ""
|
2024-02-29 03:07:31 +08:00
|
|
|
|
|
|
|
|
|
|
@classmethod
|
2024-03-18 16:10:44 +08:00
|
|
|
|
def get_target(
|
|
|
|
|
|
cls,
|
2024-12-25 12:03:49 +08:00
|
|
|
|
*,
|
2024-03-18 16:10:44 +08:00
|
|
|
|
user_id: str | None = None,
|
|
|
|
|
|
group_id: str | None = None,
|
2024-08-11 15:57:33 +08:00
|
|
|
|
channel_id: str | None = None,
|
2024-03-18 16:10:44 +08:00
|
|
|
|
):
|
2024-02-29 03:07:31 +08:00
|
|
|
|
"""获取发生Target
|
|
|
|
|
|
|
|
|
|
|
|
参数:
|
|
|
|
|
|
bot: Bot
|
2024-05-04 13:48:12 +08:00
|
|
|
|
user_id: 用户id
|
|
|
|
|
|
group_id: 频道id或群组id
|
2024-08-11 15:57:33 +08:00
|
|
|
|
channel_id: 频道id
|
2024-02-29 03:07:31 +08:00
|
|
|
|
|
|
|
|
|
|
返回:
|
|
|
|
|
|
target: 对应平台Target
|
|
|
|
|
|
"""
|
|
|
|
|
|
target = None
|
2024-12-25 12:03:49 +08:00
|
|
|
|
if group_id and channel_id:
|
|
|
|
|
|
target = Target(channel_id, parent_id=group_id, channel=True)
|
|
|
|
|
|
elif group_id:
|
|
|
|
|
|
target = Target(group_id)
|
|
|
|
|
|
elif user_id:
|
|
|
|
|
|
target = Target(user_id, private=True)
|
2024-02-29 03:07:31 +08:00
|
|
|
|
return target
|
|
|
|
|
|
|
|
|
|
|
|
|
2025-06-16 09:18:58 +08:00
|
|
|
|
class BroadcastEngine:
|
|
|
|
|
|
def __init__(
|
|
|
|
|
|
self,
|
|
|
|
|
|
message: str | UniMessage,
|
|
|
|
|
|
bot: Bot | list[Bot] | None = None,
|
|
|
|
|
|
bot_id: str | set[str] | None = None,
|
|
|
|
|
|
ignore_group: list[str] | None = None,
|
|
|
|
|
|
check_func: Callable[[Bot, str], Awaitable] | None = None,
|
|
|
|
|
|
log_cmd: str | None = None,
|
|
|
|
|
|
platform: str | None = None,
|
|
|
|
|
|
):
|
|
|
|
|
|
"""广播引擎
|
|
|
|
|
|
|
|
|
|
|
|
参数:
|
|
|
|
|
|
message: 广播消息内容
|
|
|
|
|
|
bot: 指定bot对象.
|
|
|
|
|
|
bot_id: 指定bot id.
|
|
|
|
|
|
ignore_group: 忽略群聊列表.
|
|
|
|
|
|
check_func: 发送前对群聊检测方法,判断是否发送.
|
|
|
|
|
|
log_cmd: 日志标记.
|
|
|
|
|
|
platform: 指定平台.
|
|
|
|
|
|
|
|
|
|
|
|
异常:
|
|
|
|
|
|
ValueError: 没有可用的Bot对象
|
|
|
|
|
|
"""
|
|
|
|
|
|
if ignore_group is None:
|
|
|
|
|
|
ignore_group = []
|
|
|
|
|
|
self.message = MessageUtils.build_message(message)
|
|
|
|
|
|
self.ignore_group = ignore_group
|
|
|
|
|
|
self.check_func = check_func
|
|
|
|
|
|
self.log_cmd = log_cmd
|
|
|
|
|
|
self.platform = platform
|
|
|
|
|
|
self.bot_list = []
|
|
|
|
|
|
self.count = 0
|
|
|
|
|
|
if bot:
|
|
|
|
|
|
self.bot_list = [bot] if isinstance(bot, Bot) else bot
|
|
|
|
|
|
if isinstance(bot_id, str):
|
|
|
|
|
|
bot_id = set(bot_id)
|
|
|
|
|
|
if bot_id:
|
|
|
|
|
|
for i in bot_id:
|
|
|
|
|
|
try:
|
|
|
|
|
|
self.bot_list.append(nonebot.get_bot(i))
|
|
|
|
|
|
except KeyError:
|
2025-07-07 09:12:30 +08:00
|
|
|
|
logger.warning(f"Bot:{i} 对象未连接或不存在", log_cmd)
|
2025-06-16 09:18:58 +08:00
|
|
|
|
if not self.bot_list:
|
2025-07-07 09:12:30 +08:00
|
|
|
|
try:
|
|
|
|
|
|
bot = nonebot.get_bot()
|
|
|
|
|
|
self.bot_list.append(bot)
|
|
|
|
|
|
logger.warning(
|
|
|
|
|
|
f"广播任务未传入Bot对象,使用默认Bot {bot.self_id}", log_cmd
|
|
|
|
|
|
)
|
|
|
|
|
|
except Exception as e:
|
|
|
|
|
|
raise ValueError("当前没有可用的Bot对象...", log_cmd) from e
|
2025-06-16 09:18:58 +08:00
|
|
|
|
|
|
|
|
|
|
async def call_check(self, bot: Bot, group_id: str) -> bool:
|
|
|
|
|
|
"""运行发送检测函数
|
|
|
|
|
|
|
|
|
|
|
|
参数:
|
|
|
|
|
|
bot: Bot
|
|
|
|
|
|
group_id: 群组id
|
|
|
|
|
|
|
|
|
|
|
|
返回:
|
|
|
|
|
|
bool: 是否发送
|
|
|
|
|
|
"""
|
|
|
|
|
|
if not self.check_func:
|
|
|
|
|
|
return True
|
|
|
|
|
|
if is_coroutine_callable(self.check_func):
|
|
|
|
|
|
is_run = await self.check_func(bot, group_id)
|
|
|
|
|
|
else:
|
|
|
|
|
|
is_run = self.check_func(bot, group_id)
|
|
|
|
|
|
return cast(bool, is_run)
|
|
|
|
|
|
|
|
|
|
|
|
async def __send_message(self, bot: Bot, group: GroupConsole):
|
|
|
|
|
|
"""群组发送消息
|
|
|
|
|
|
|
|
|
|
|
|
参数:
|
|
|
|
|
|
bot: Bot
|
|
|
|
|
|
group: GroupConsole
|
|
|
|
|
|
"""
|
|
|
|
|
|
key = f"{group.group_id}:{group.channel_id}"
|
|
|
|
|
|
if not await self.call_check(bot, group.group_id):
|
|
|
|
|
|
logger.debug(
|
|
|
|
|
|
"广播方法检测运行方法为 False, 已跳过该群组...",
|
|
|
|
|
|
self.log_cmd,
|
|
|
|
|
|
group_id=group.group_id,
|
|
|
|
|
|
)
|
|
|
|
|
|
return
|
|
|
|
|
|
if target := PlatformUtils.get_target(
|
|
|
|
|
|
group_id=group.group_id,
|
|
|
|
|
|
channel_id=group.channel_id,
|
|
|
|
|
|
):
|
|
|
|
|
|
self.ignore_group.append(key)
|
|
|
|
|
|
await MessageUtils.build_message(self.message).send(target, bot)
|
|
|
|
|
|
logger.debug("广播消息发送成功...", self.log_cmd, target=key)
|
|
|
|
|
|
else:
|
|
|
|
|
|
logger.warning("广播消息获取Target失败...", self.log_cmd, target=key)
|
|
|
|
|
|
|
|
|
|
|
|
async def broadcast(self) -> int:
|
|
|
|
|
|
"""广播消息
|
|
|
|
|
|
|
|
|
|
|
|
返回:
|
|
|
|
|
|
int: 成功发送次数
|
|
|
|
|
|
"""
|
|
|
|
|
|
for bot in self.bot_list:
|
|
|
|
|
|
if self.platform and self.platform != PlatformUtils.get_platform(bot):
|
|
|
|
|
|
continue
|
|
|
|
|
|
group_list, _ = await PlatformUtils.get_group_list(bot)
|
|
|
|
|
|
if not group_list:
|
|
|
|
|
|
continue
|
|
|
|
|
|
for group in group_list:
|
|
|
|
|
|
if (
|
|
|
|
|
|
group.group_id in self.ignore_group
|
|
|
|
|
|
or group.channel_id in self.ignore_group
|
|
|
|
|
|
):
|
|
|
|
|
|
continue
|
|
|
|
|
|
try:
|
|
|
|
|
|
await self.__send_message(bot, group)
|
|
|
|
|
|
await asyncio.sleep(random.randint(1, 3))
|
|
|
|
|
|
self.count += 1
|
|
|
|
|
|
except Exception as e:
|
|
|
|
|
|
logger.warning(
|
|
|
|
|
|
"广播消息发送失败", self.log_cmd, target=group.group_id, e=e
|
|
|
|
|
|
)
|
|
|
|
|
|
return self.count
|
|
|
|
|
|
|
|
|
|
|
|
|
2024-02-29 03:07:31 +08:00
|
|
|
|
async def broadcast_group(
|
2024-08-11 15:57:33 +08:00
|
|
|
|
message: str | UniMessage,
|
2024-02-29 03:07:31 +08:00
|
|
|
|
bot: Bot | list[Bot] | None = None,
|
2024-08-29 22:01:34 +08:00
|
|
|
|
bot_id: str | set[str] | None = None,
|
2025-06-16 09:18:58 +08:00
|
|
|
|
ignore_group: list[str] = [],
|
2024-10-18 18:57:55 +08:00
|
|
|
|
check_func: Callable[[Bot, str], Awaitable] | None = None,
|
2024-02-29 03:07:31 +08:00
|
|
|
|
log_cmd: str | None = None,
|
2025-06-16 09:18:58 +08:00
|
|
|
|
platform: str | None = None,
|
|
|
|
|
|
) -> int:
|
2024-02-29 03:07:31 +08:00
|
|
|
|
"""获取所有Bot或指定Bot对象广播群聊
|
|
|
|
|
|
|
2024-03-29 23:27:25 +08:00
|
|
|
|
参数:
|
2024-02-29 03:07:31 +08:00
|
|
|
|
message: 广播消息内容
|
|
|
|
|
|
bot: 指定bot对象.
|
|
|
|
|
|
bot_id: 指定bot id.
|
|
|
|
|
|
ignore_group: 忽略群聊列表.
|
|
|
|
|
|
check_func: 发送前对群聊检测方法,判断是否发送.
|
|
|
|
|
|
log_cmd: 日志标记.
|
|
|
|
|
|
platform: 指定平台
|
2025-06-16 09:18:58 +08:00
|
|
|
|
|
|
|
|
|
|
返回:
|
|
|
|
|
|
int: 成功发送次数
|
2024-02-29 03:07:31 +08:00
|
|
|
|
"""
|
2025-06-16 09:18:58 +08:00
|
|
|
|
if not message.strip():
|
|
|
|
|
|
raise ValueError("群聊广播消息不能为空...")
|
|
|
|
|
|
return await BroadcastEngine(
|
|
|
|
|
|
message=message,
|
|
|
|
|
|
bot=bot,
|
|
|
|
|
|
bot_id=bot_id,
|
|
|
|
|
|
ignore_group=ignore_group,
|
|
|
|
|
|
check_func=check_func,
|
|
|
|
|
|
log_cmd=log_cmd,
|
|
|
|
|
|
platform=platform,
|
|
|
|
|
|
).broadcast()
|