2020-12-02 19:52:45 +08:00
|
|
|
import re
|
2021-05-04 11:08:31 +08:00
|
|
|
from io import BytesIO
|
|
|
|
from pathlib import Path
|
|
|
|
from base64 import b64encode
|
2021-06-17 01:07:19 +08:00
|
|
|
from typing import Type, Union, Tuple, Mapping, Iterable, Optional
|
2020-12-02 19:52:45 +08:00
|
|
|
|
2020-12-06 02:30:19 +08:00
|
|
|
from nonebot.typing import overrides
|
2020-12-07 00:06:09 +08:00
|
|
|
from nonebot.adapters import Message as BaseMessage, MessageSegment as BaseMessageSegment
|
2020-12-03 16:04:14 +08:00
|
|
|
|
2020-12-02 19:52:45 +08:00
|
|
|
from .utils import log, escape, unescape, _b2s
|
|
|
|
|
|
|
|
|
2021-06-14 19:52:35 +08:00
|
|
|
class MessageSegment(BaseMessageSegment["Message"]):
|
2020-12-02 19:52:45 +08:00
|
|
|
"""
|
|
|
|
CQHTTP 协议 MessageSegment 适配。具体方法参考协议消息段类型或源码。
|
|
|
|
"""
|
|
|
|
|
2021-06-14 19:52:35 +08:00
|
|
|
@classmethod
|
2021-06-17 01:07:19 +08:00
|
|
|
def get_message_class(cls) -> Type["Message"]:
|
2021-06-14 19:52:35 +08:00
|
|
|
return Message
|
2020-12-02 19:52:45 +08:00
|
|
|
|
|
|
|
@overrides(BaseMessageSegment)
|
2020-12-10 02:13:25 +08:00
|
|
|
def __str__(self) -> str:
|
2020-12-02 19:52:45 +08:00
|
|
|
type_ = self.type
|
|
|
|
data = self.data.copy()
|
|
|
|
|
|
|
|
# process special types
|
|
|
|
if type_ == "text":
|
|
|
|
return escape(
|
|
|
|
data.get("text", ""), # type: ignore
|
|
|
|
escape_comma=False)
|
|
|
|
|
|
|
|
params = ",".join(
|
|
|
|
[f"{k}={escape(str(v))}" for k, v in data.items() if v is not None])
|
|
|
|
return f"[CQ:{type_}{',' if params else ''}{params}]"
|
|
|
|
|
|
|
|
@overrides(BaseMessageSegment)
|
|
|
|
def __add__(self, other) -> "Message":
|
2021-06-14 19:52:35 +08:00
|
|
|
return Message(self) + (MessageSegment.text(other) if isinstance(
|
|
|
|
other, str) else other)
|
2020-12-02 19:52:45 +08:00
|
|
|
|
2020-12-28 17:39:33 +08:00
|
|
|
@overrides(BaseMessageSegment)
|
|
|
|
def __radd__(self, other) -> "Message":
|
2021-03-30 16:21:56 +08:00
|
|
|
return (MessageSegment.text(other)
|
|
|
|
if isinstance(other, str) else Message(other)) + self
|
2020-12-28 17:39:33 +08:00
|
|
|
|
2020-12-10 02:13:25 +08:00
|
|
|
@overrides(BaseMessageSegment)
|
|
|
|
def is_text(self) -> bool:
|
|
|
|
return self.type == "text"
|
|
|
|
|
2020-12-02 19:52:45 +08:00
|
|
|
@staticmethod
|
|
|
|
def anonymous(ignore_failure: Optional[bool] = None) -> "MessageSegment":
|
|
|
|
return MessageSegment("anonymous", {"ignore": _b2s(ignore_failure)})
|
|
|
|
|
|
|
|
@staticmethod
|
|
|
|
def at(user_id: Union[int, str]) -> "MessageSegment":
|
|
|
|
return MessageSegment("at", {"qq": str(user_id)})
|
|
|
|
|
2020-12-23 17:12:53 +08:00
|
|
|
@staticmethod
|
|
|
|
def contact(type_: str, id: int) -> "MessageSegment":
|
|
|
|
return MessageSegment("contact", {"type": type_, "id": str(id)})
|
|
|
|
|
2020-12-02 19:52:45 +08:00
|
|
|
@staticmethod
|
|
|
|
def contact_group(group_id: int) -> "MessageSegment":
|
|
|
|
return MessageSegment("contact", {"type": "group", "id": str(group_id)})
|
|
|
|
|
|
|
|
@staticmethod
|
|
|
|
def contact_user(user_id: int) -> "MessageSegment":
|
|
|
|
return MessageSegment("contact", {"type": "qq", "id": str(user_id)})
|
|
|
|
|
|
|
|
@staticmethod
|
|
|
|
def dice() -> "MessageSegment":
|
|
|
|
return MessageSegment("dice", {})
|
|
|
|
|
|
|
|
@staticmethod
|
|
|
|
def face(id_: int) -> "MessageSegment":
|
|
|
|
return MessageSegment("face", {"id": str(id_)})
|
|
|
|
|
|
|
|
@staticmethod
|
|
|
|
def forward(id_: str) -> "MessageSegment":
|
|
|
|
log("WARNING", "Forward Message only can be received!")
|
|
|
|
return MessageSegment("forward", {"id": id_})
|
|
|
|
|
|
|
|
@staticmethod
|
2021-05-04 11:08:31 +08:00
|
|
|
def image(file: Union[str, bytes, BytesIO, Path],
|
2020-12-02 19:52:45 +08:00
|
|
|
type_: Optional[str] = None,
|
|
|
|
cache: bool = True,
|
|
|
|
proxy: bool = True,
|
|
|
|
timeout: Optional[int] = None) -> "MessageSegment":
|
2021-05-04 11:08:31 +08:00
|
|
|
if isinstance(file, BytesIO):
|
|
|
|
file = file.read()
|
|
|
|
if isinstance(file, bytes):
|
|
|
|
file = f"base64://{b64encode(file).decode()}"
|
|
|
|
elif isinstance(file, Path):
|
|
|
|
file = f"file:///{file.resolve()}"
|
2020-12-02 19:52:45 +08:00
|
|
|
return MessageSegment(
|
|
|
|
"image", {
|
|
|
|
"file": file,
|
|
|
|
"type": type_,
|
2021-05-02 14:24:12 +08:00
|
|
|
"cache": _b2s(cache),
|
|
|
|
"proxy": _b2s(proxy),
|
2020-12-02 19:52:45 +08:00
|
|
|
"timeout": timeout
|
|
|
|
})
|
|
|
|
|
|
|
|
@staticmethod
|
|
|
|
def json(data: str) -> "MessageSegment":
|
|
|
|
return MessageSegment("json", {"data": data})
|
|
|
|
|
|
|
|
@staticmethod
|
|
|
|
def location(latitude: float,
|
|
|
|
longitude: float,
|
|
|
|
title: Optional[str] = None,
|
|
|
|
content: Optional[str] = None) -> "MessageSegment":
|
|
|
|
return MessageSegment(
|
|
|
|
"location", {
|
|
|
|
"lat": str(latitude),
|
|
|
|
"lon": str(longitude),
|
|
|
|
"title": title,
|
|
|
|
"content": content
|
|
|
|
})
|
|
|
|
|
|
|
|
@staticmethod
|
|
|
|
def music(type_: str, id_: int) -> "MessageSegment":
|
|
|
|
return MessageSegment("music", {"type": type_, "id": id_})
|
|
|
|
|
|
|
|
@staticmethod
|
|
|
|
def music_custom(url: str,
|
|
|
|
audio: str,
|
|
|
|
title: str,
|
|
|
|
content: Optional[str] = None,
|
|
|
|
img_url: Optional[str] = None) -> "MessageSegment":
|
|
|
|
return MessageSegment(
|
|
|
|
"music", {
|
|
|
|
"type": "custom",
|
|
|
|
"url": url,
|
|
|
|
"audio": audio,
|
|
|
|
"title": title,
|
|
|
|
"content": content,
|
|
|
|
"image": img_url
|
|
|
|
})
|
|
|
|
|
|
|
|
@staticmethod
|
|
|
|
def node(id_: int) -> "MessageSegment":
|
|
|
|
return MessageSegment("node", {"id": str(id_)})
|
|
|
|
|
|
|
|
@staticmethod
|
|
|
|
def node_custom(user_id: int, nickname: str,
|
|
|
|
content: Union[str, "Message"]) -> "MessageSegment":
|
|
|
|
return MessageSegment("node", {
|
|
|
|
"user_id": str(user_id),
|
|
|
|
"nickname": nickname,
|
|
|
|
"content": content
|
|
|
|
})
|
|
|
|
|
|
|
|
@staticmethod
|
|
|
|
def poke(type_: str, id_: str) -> "MessageSegment":
|
|
|
|
return MessageSegment("poke", {"type": type_, "id": id_})
|
|
|
|
|
|
|
|
@staticmethod
|
2021-05-04 11:08:31 +08:00
|
|
|
def record(file: Union[str, bytes, BytesIO, Path],
|
2020-12-02 19:52:45 +08:00
|
|
|
magic: Optional[bool] = None,
|
|
|
|
cache: Optional[bool] = None,
|
|
|
|
proxy: Optional[bool] = None,
|
|
|
|
timeout: Optional[int] = None) -> "MessageSegment":
|
2021-05-04 11:08:31 +08:00
|
|
|
if isinstance(file, BytesIO):
|
|
|
|
file = file.read()
|
|
|
|
if isinstance(file, bytes):
|
|
|
|
file = f"base64://{b64encode(file).decode()}"
|
|
|
|
elif isinstance(file, Path):
|
|
|
|
file = f"file:///{file.resolve()}"
|
2020-12-23 17:12:53 +08:00
|
|
|
return MessageSegment(
|
|
|
|
"record", {
|
|
|
|
"file": file,
|
|
|
|
"magic": _b2s(magic),
|
2021-05-02 14:24:12 +08:00
|
|
|
"cache": _b2s(cache),
|
|
|
|
"proxy": _b2s(proxy),
|
2020-12-23 17:12:53 +08:00
|
|
|
"timeout": timeout
|
|
|
|
})
|
2020-12-02 19:52:45 +08:00
|
|
|
|
|
|
|
@staticmethod
|
|
|
|
def reply(id_: int) -> "MessageSegment":
|
|
|
|
return MessageSegment("reply", {"id": str(id_)})
|
|
|
|
|
|
|
|
@staticmethod
|
|
|
|
def rps() -> "MessageSegment":
|
|
|
|
return MessageSegment("rps", {})
|
|
|
|
|
|
|
|
@staticmethod
|
|
|
|
def shake() -> "MessageSegment":
|
|
|
|
return MessageSegment("shake", {})
|
|
|
|
|
|
|
|
@staticmethod
|
|
|
|
def share(url: str = "",
|
|
|
|
title: str = "",
|
|
|
|
content: Optional[str] = None,
|
2021-01-30 10:56:58 +08:00
|
|
|
image: Optional[str] = None) -> "MessageSegment":
|
2020-12-02 19:52:45 +08:00
|
|
|
return MessageSegment("share", {
|
|
|
|
"url": url,
|
|
|
|
"title": title,
|
|
|
|
"content": content,
|
2021-01-30 10:56:58 +08:00
|
|
|
"image": image
|
2020-12-02 19:52:45 +08:00
|
|
|
})
|
|
|
|
|
|
|
|
@staticmethod
|
|
|
|
def text(text: str) -> "MessageSegment":
|
|
|
|
return MessageSegment("text", {"text": text})
|
|
|
|
|
|
|
|
@staticmethod
|
2021-05-04 11:08:31 +08:00
|
|
|
def video(file: Union[str, bytes, BytesIO, Path],
|
2020-12-02 19:52:45 +08:00
|
|
|
cache: Optional[bool] = None,
|
|
|
|
proxy: Optional[bool] = None,
|
|
|
|
timeout: Optional[int] = None) -> "MessageSegment":
|
2021-05-04 11:08:31 +08:00
|
|
|
if isinstance(file, BytesIO):
|
|
|
|
file = file.read()
|
|
|
|
if isinstance(file, bytes):
|
|
|
|
file = f"base64://{b64encode(file).decode()}"
|
|
|
|
elif isinstance(file, Path):
|
|
|
|
file = f"file:///{file.resolve()}"
|
2021-05-02 14:24:12 +08:00
|
|
|
return MessageSegment(
|
|
|
|
"video", {
|
|
|
|
"file": file,
|
|
|
|
"cache": _b2s(cache),
|
|
|
|
"proxy": _b2s(proxy),
|
|
|
|
"timeout": timeout
|
|
|
|
})
|
2020-12-02 19:52:45 +08:00
|
|
|
|
|
|
|
@staticmethod
|
|
|
|
def xml(data: str) -> "MessageSegment":
|
|
|
|
return MessageSegment("xml", {"data": data})
|
|
|
|
|
|
|
|
|
2021-05-10 00:54:15 +08:00
|
|
|
class Message(BaseMessage[MessageSegment]):
|
2020-12-02 19:52:45 +08:00
|
|
|
"""
|
|
|
|
CQHTTP 协议 Message 适配。
|
|
|
|
"""
|
|
|
|
|
2021-06-14 19:52:35 +08:00
|
|
|
@classmethod
|
2021-06-17 01:07:19 +08:00
|
|
|
def get_segment_class(cls) -> Type[MessageSegment]:
|
2021-06-14 19:52:35 +08:00
|
|
|
return MessageSegment
|
|
|
|
|
|
|
|
@overrides(BaseMessage)
|
2021-06-17 01:07:19 +08:00
|
|
|
def __add__(self, other: Union[str, Mapping,
|
|
|
|
Iterable[Mapping]]) -> "Message":
|
2021-06-14 19:52:35 +08:00
|
|
|
return super(Message, self).__add__(
|
|
|
|
MessageSegment.text(other) if isinstance(other, str) else other)
|
|
|
|
|
|
|
|
@overrides(BaseMessage)
|
2021-06-17 01:07:19 +08:00
|
|
|
def __radd__(self, other: Union[str, Mapping,
|
|
|
|
Iterable[Mapping]]) -> "Message":
|
2021-06-14 19:52:35 +08:00
|
|
|
return super(Message, self).__radd__(
|
|
|
|
MessageSegment.text(other) if isinstance(other, str) else other)
|
2021-03-30 16:21:56 +08:00
|
|
|
|
2020-12-02 19:52:45 +08:00
|
|
|
@staticmethod
|
|
|
|
@overrides(BaseMessage)
|
2020-12-31 14:00:59 +08:00
|
|
|
def _construct(
|
|
|
|
msg: Union[str, Mapping,
|
|
|
|
Iterable[Mapping]]) -> Iterable[MessageSegment]:
|
|
|
|
if isinstance(msg, Mapping):
|
2020-12-02 19:52:45 +08:00
|
|
|
yield MessageSegment(msg["type"], msg.get("data") or {})
|
|
|
|
return
|
2020-12-31 14:00:59 +08:00
|
|
|
elif isinstance(msg, Iterable) and not isinstance(msg, str):
|
2020-12-02 19:52:45 +08:00
|
|
|
for seg in msg:
|
|
|
|
yield MessageSegment(seg["type"], seg.get("data") or {})
|
|
|
|
return
|
2021-01-04 22:24:43 +08:00
|
|
|
elif isinstance(msg, str):
|
2021-01-08 18:14:18 +08:00
|
|
|
|
2021-01-04 22:24:43 +08:00
|
|
|
def _iter_message(msg: str) -> Iterable[Tuple[str, str]]:
|
|
|
|
text_begin = 0
|
|
|
|
for cqcode in re.finditer(
|
|
|
|
r"\[CQ:(?P<type>[a-zA-Z0-9-_.]+)"
|
|
|
|
r"(?P<params>"
|
|
|
|
r"(?:,[a-zA-Z0-9-_.]+=[^,\]]+)*"
|
|
|
|
r"),?\]", msg):
|
|
|
|
yield "text", msg[text_begin:cqcode.pos + cqcode.start()]
|
|
|
|
text_begin = cqcode.pos + cqcode.end()
|
2021-01-08 18:14:18 +08:00
|
|
|
yield cqcode.group("type"), cqcode.group("params").lstrip(
|
|
|
|
",")
|
2021-01-04 22:24:43 +08:00
|
|
|
yield "text", msg[text_begin:]
|
|
|
|
|
|
|
|
for type_, data in _iter_message(msg):
|
|
|
|
if type_ == "text":
|
|
|
|
if data:
|
|
|
|
# only yield non-empty text segment
|
|
|
|
yield MessageSegment(type_, {"text": unescape(data)})
|
|
|
|
else:
|
|
|
|
data = {
|
|
|
|
k: unescape(v) for k, v in map(
|
|
|
|
lambda x: x.split("=", maxsplit=1),
|
|
|
|
filter(lambda x: x, (
|
|
|
|
x.lstrip() for x in data.split(","))))
|
|
|
|
}
|
|
|
|
yield MessageSegment(type_, data)
|
2021-01-08 18:14:18 +08:00
|
|
|
|
2021-06-14 19:52:35 +08:00
|
|
|
@overrides(BaseMessage)
|
2021-01-08 18:14:18 +08:00
|
|
|
def extract_plain_text(self) -> str:
|
2021-06-14 19:52:35 +08:00
|
|
|
return "".join(seg.data["text"] for seg in self if seg.is_text())
|