mirror of
https://github.com/jxxghp/MoviePilot.git
synced 2026-06-05 07:29:56 +08:00
feat: add standalone wechat clawbot notifications
This commit is contained in:
@@ -31,7 +31,7 @@ class WechatModule(_ModuleBase, _MessageBase[WeChat]):
|
||||
|
||||
@staticmethod
|
||||
def get_name() -> str:
|
||||
return "微信"
|
||||
return "企业微信"
|
||||
|
||||
@staticmethod
|
||||
def get_type() -> ModuleType:
|
||||
|
||||
263
app/modules/wechatclawbot/__init__.py
Normal file
263
app/modules/wechatclawbot/__init__.py
Normal file
@@ -0,0 +1,263 @@
|
||||
import json
|
||||
from typing import Any, Dict, List, Optional, Tuple, Union
|
||||
|
||||
from app.core.context import Context, MediaInfo
|
||||
from app.log import logger
|
||||
from app.modules import _MessageBase, _ModuleBase
|
||||
from app.modules.wechatclawbot.wechatclawbot import WechatClawBot
|
||||
from app.schemas import CommingMessage, Notification
|
||||
from app.schemas.types import MessageChannel, ModuleType
|
||||
|
||||
|
||||
class WechatClawBotModule(_ModuleBase, _MessageBase[WechatClawBot]):
|
||||
def init_module(self) -> None:
|
||||
"""初始化模块。"""
|
||||
self.stop()
|
||||
super().init_service(
|
||||
service_name=WechatClawBot.__name__.lower(), service_type=WechatClawBot
|
||||
)
|
||||
self._channel = MessageChannel.WechatClawBot
|
||||
|
||||
@staticmethod
|
||||
def get_name() -> str:
|
||||
return "微信 ClawBot"
|
||||
|
||||
@staticmethod
|
||||
def get_type() -> ModuleType:
|
||||
"""获取模块类型。"""
|
||||
return ModuleType.Notification
|
||||
|
||||
@staticmethod
|
||||
def get_subtype() -> MessageChannel:
|
||||
"""获取模块子类型。"""
|
||||
return MessageChannel.WechatClawBot
|
||||
|
||||
@staticmethod
|
||||
def get_priority() -> int:
|
||||
"""获取模块优先级。"""
|
||||
return 2
|
||||
|
||||
def stop(self):
|
||||
"""停止模块。"""
|
||||
for client in self.get_instances().values():
|
||||
if hasattr(client, "stop"):
|
||||
try:
|
||||
client.stop()
|
||||
except Exception as err:
|
||||
logger.error(f"停止微信 ClawBot 模块实例失败:{err}")
|
||||
|
||||
def test(self) -> Optional[Tuple[bool, str]]:
|
||||
"""测试模块连接性。"""
|
||||
if not self.get_instances():
|
||||
return None
|
||||
for name, client in self.get_instances().items():
|
||||
state, message = client.test_connection()
|
||||
if not state:
|
||||
return False, f"微信 ClawBot {name} 未就绪:{message}"
|
||||
return True, ""
|
||||
|
||||
def init_setting(self) -> Tuple[str, Union[str, bool]]:
|
||||
pass
|
||||
|
||||
@staticmethod
|
||||
def _load_json(body: Any) -> Optional[dict]:
|
||||
if isinstance(body, dict):
|
||||
payload = body
|
||||
elif isinstance(body, bytes):
|
||||
payload = json.loads(body.decode("utf-8", errors="ignore"))
|
||||
else:
|
||||
payload = json.loads(body)
|
||||
while isinstance(payload, str):
|
||||
payload = json.loads(payload)
|
||||
return payload if isinstance(payload, dict) else None
|
||||
|
||||
@staticmethod
|
||||
def _normalize_audio_refs(audio_refs: Any) -> Optional[List[str]]:
|
||||
if not audio_refs:
|
||||
return None
|
||||
if not isinstance(audio_refs, list):
|
||||
audio_refs = [audio_refs]
|
||||
normalized = [str(item).strip() for item in audio_refs if str(item).strip()]
|
||||
return normalized or None
|
||||
|
||||
@staticmethod
|
||||
def _normalize_files(files: Any) -> Optional[List[CommingMessage.MessageAttachment]]:
|
||||
if not files:
|
||||
return None
|
||||
if not isinstance(files, list):
|
||||
files = [files]
|
||||
normalized = []
|
||||
for item in files:
|
||||
if not isinstance(item, dict):
|
||||
continue
|
||||
ref = item.get("ref") or item.get("url") or item.get("file_url")
|
||||
if not ref:
|
||||
continue
|
||||
size = item.get("size")
|
||||
try:
|
||||
size = int(size) if size is not None else None
|
||||
except (TypeError, ValueError):
|
||||
size = None
|
||||
normalized.append(
|
||||
CommingMessage.MessageAttachment(
|
||||
ref=ref,
|
||||
name=item.get("name") or item.get("filename"),
|
||||
mime_type=item.get("mime_type") or item.get("content_type"),
|
||||
size=size,
|
||||
)
|
||||
)
|
||||
return normalized or None
|
||||
|
||||
def message_parser(
|
||||
self, source: str, body: Any, form: Any, args: Any
|
||||
) -> Optional[CommingMessage]:
|
||||
"""解析微信 ClawBot 转发到消息入口的 JSON 报文。"""
|
||||
client_config = self.get_config(source)
|
||||
if not client_config:
|
||||
return None
|
||||
try:
|
||||
message = self._load_json(body)
|
||||
except Exception as err:
|
||||
logger.debug(f"解析微信 ClawBot 消息失败:{err}")
|
||||
return None
|
||||
|
||||
if not message:
|
||||
return None
|
||||
channel_name = (message.get("__channel__") or "").strip().lower()
|
||||
if channel_name and channel_name != "wechatclawbot":
|
||||
return None
|
||||
|
||||
user_id = str(message.get("userid") or "").strip()
|
||||
if not user_id:
|
||||
return None
|
||||
|
||||
text = str(message.get("text") or "").strip()
|
||||
username = str(message.get("username") or user_id).strip() or user_id
|
||||
images = CommingMessage.MessageImage.normalize_list(message.get("images"))
|
||||
audio_refs = self._normalize_audio_refs(message.get("audio_refs"))
|
||||
files = self._normalize_files(message.get("files"))
|
||||
if not text and not images and not audio_refs and not files:
|
||||
return None
|
||||
|
||||
admins = [
|
||||
admin.strip()
|
||||
for admin in str(client_config.config.get("WECHATCLAWBOT_ADMINS") or "").split(",")
|
||||
if admin.strip()
|
||||
]
|
||||
if text.startswith("/") and admins and user_id not in admins:
|
||||
client = self.get_instance(client_config.name)
|
||||
if client:
|
||||
client.send_msg(title="只有管理员才有权限执行此命令", userid=user_id)
|
||||
return None
|
||||
|
||||
logger.info(
|
||||
f"收到来自 {client_config.name} 的微信 ClawBot 消息:"
|
||||
f"userid={user_id}, text={text}, images={len(images) if images else 0}, "
|
||||
f"audios={len(audio_refs) if audio_refs else 0}, files={len(files) if files else 0}"
|
||||
)
|
||||
return CommingMessage(
|
||||
channel=MessageChannel.WechatClawBot,
|
||||
source=client_config.name,
|
||||
userid=user_id,
|
||||
username=username,
|
||||
text=text,
|
||||
message_id=message.get("message_id"),
|
||||
chat_id=str(message.get("chat_id") or "") or None,
|
||||
images=images,
|
||||
audio_refs=audio_refs,
|
||||
files=files,
|
||||
)
|
||||
|
||||
def post_message(self, message: Notification, **kwargs) -> None:
|
||||
"""发送消息。"""
|
||||
for conf in self.get_configs().values():
|
||||
if not self.check_message(message, conf.name):
|
||||
continue
|
||||
targets = message.targets
|
||||
userid = message.userid
|
||||
if not userid and targets is not None:
|
||||
userid = targets.get("wechatclawbot_userid")
|
||||
if not userid:
|
||||
logger.warning("用户没有指定 微信 ClawBot 用户ID,消息无法发送")
|
||||
return
|
||||
client: WechatClawBot = self.get_instance(conf.name)
|
||||
if not client:
|
||||
continue
|
||||
if message.file_path:
|
||||
client.send_file(
|
||||
file_path=message.file_path,
|
||||
file_name=message.file_name,
|
||||
title=message.title,
|
||||
text=message.text,
|
||||
userid=userid,
|
||||
)
|
||||
elif message.voice_path:
|
||||
client.send_file(
|
||||
file_path=message.voice_path,
|
||||
title=message.voice_caption or message.title,
|
||||
text=message.text,
|
||||
userid=userid,
|
||||
)
|
||||
else:
|
||||
client.send_msg(
|
||||
title=message.title or "",
|
||||
text=message.text,
|
||||
image=message.image,
|
||||
userid=userid,
|
||||
link=message.link,
|
||||
)
|
||||
|
||||
def download_wechat_image_to_data_url(
|
||||
self, image_ref: str, source: str
|
||||
) -> Optional[str]:
|
||||
"""下载微信 ClawBot 图片并转换为 data URL。"""
|
||||
if not image_ref or not image_ref.startswith("wxclaw://image/"):
|
||||
return None
|
||||
client_config = self.get_config(source)
|
||||
if not client_config:
|
||||
return None
|
||||
client = self.get_instance(client_config.name)
|
||||
if not client:
|
||||
return None
|
||||
return client.download_image_to_data_url(image_ref)
|
||||
|
||||
def download_wechat_media_bytes(
|
||||
self, media_ref: str, source: str
|
||||
) -> Optional[bytes]:
|
||||
"""下载微信 ClawBot 语音或文件附件。"""
|
||||
if not media_ref or not media_ref.startswith(("wxclaw://file/", "wxclaw://voice/")):
|
||||
return None
|
||||
client_config = self.get_config(source)
|
||||
if not client_config:
|
||||
return None
|
||||
client = self.get_instance(client_config.name)
|
||||
if not client:
|
||||
return None
|
||||
return client.download_media_bytes(media_ref)
|
||||
|
||||
def post_medias_message(self, message: Notification, medias: List[MediaInfo]) -> None:
|
||||
"""发送媒体选择列表。"""
|
||||
for conf in self.get_configs().values():
|
||||
if not self.check_message(message, conf.name):
|
||||
continue
|
||||
client: WechatClawBot = self.get_instance(conf.name)
|
||||
if client:
|
||||
client.send_medias_msg(medias=medias, userid=message.userid)
|
||||
|
||||
def post_torrents_message(self, message: Notification, torrents: List[Context]) -> None:
|
||||
"""发送种子选择列表。"""
|
||||
for conf in self.get_configs().values():
|
||||
if not self.check_message(message, conf.name):
|
||||
continue
|
||||
client: WechatClawBot = self.get_instance(conf.name)
|
||||
if client:
|
||||
client.send_torrents_msg(
|
||||
torrents=torrents,
|
||||
userid=message.userid,
|
||||
title=message.title,
|
||||
link=message.link,
|
||||
)
|
||||
|
||||
def register_commands(self, commands: Dict[str, dict]):
|
||||
"""微信 ClawBot 不支持原生菜单命令,统一走文本交互。"""
|
||||
logger.debug("微信 ClawBot 不支持原生菜单命令,跳过命令注册")
|
||||
2033
app/modules/wechatclawbot/wechatclawbot.py
Normal file
2033
app/modules/wechatclawbot/wechatclawbot.py
Normal file
File diff suppressed because it is too large
Load Diff
Reference in New Issue
Block a user