mirror of
https://github.com/langbot-app/LangBot.git
synced 2026-06-04 04:54:36 +00:00
1818 lines
79 KiB
Python
1818 lines
79 KiB
Python
from __future__ import annotations
|
||
|
||
import lark_oapi
|
||
from lark_oapi.api.im.v1 import CreateImageRequest, CreateImageRequestBody, CreateFileRequest, CreateFileRequestBody
|
||
import traceback
|
||
import typing
|
||
import asyncio
|
||
import re
|
||
import base64
|
||
import uuid
|
||
import json
|
||
import time
|
||
import datetime
|
||
import hashlib
|
||
from Crypto.Cipher import AES
|
||
import tempfile
|
||
import os
|
||
import mimetypes
|
||
|
||
from langbot.pkg.utils import httpclient
|
||
import lark_oapi.ws.exception
|
||
import quart
|
||
from lark_oapi.api.im.v1 import *
|
||
import pydantic
|
||
from lark_oapi.api.cardkit.v1 import *
|
||
from lark_oapi.api.auth.v3 import *
|
||
from lark_oapi.core.model import *
|
||
|
||
import langbot_plugin.api.definition.abstract.platform.adapter as abstract_platform_adapter
|
||
import langbot_plugin.api.entities.builtin.platform.message as platform_message
|
||
import langbot_plugin.api.entities.builtin.platform.events as platform_events
|
||
import langbot_plugin.api.entities.builtin.platform.entities as platform_entities
|
||
import langbot_plugin.api.definition.abstract.platform.event_logger as abstract_platform_logger
|
||
|
||
|
||
class AESCipher(object):
|
||
def __init__(self, key):
|
||
self.bs = AES.block_size
|
||
self.key = hashlib.sha256(AESCipher.str_to_bytes(key)).digest()
|
||
|
||
@staticmethod
|
||
def str_to_bytes(data):
|
||
u_type = type(b''.decode('utf8'))
|
||
if isinstance(data, u_type):
|
||
return data.encode('utf8')
|
||
return data
|
||
|
||
@staticmethod
|
||
def _unpad(s):
|
||
return s[: -ord(s[len(s) - 1 :])]
|
||
|
||
def decrypt(self, enc):
|
||
iv = enc[: AES.block_size]
|
||
cipher = AES.new(self.key, AES.MODE_CBC, iv)
|
||
return self._unpad(cipher.decrypt(enc[AES.block_size :]))
|
||
|
||
def decrypt_string(self, enc):
|
||
enc = base64.b64decode(enc)
|
||
return self.decrypt(enc).decode('utf8')
|
||
|
||
|
||
class LarkMessageConverter(abstract_platform_adapter.AbstractMessageConverter):
|
||
@staticmethod
|
||
async def upload_image_to_lark(msg: platform_message.Image, api_client: lark_oapi.Client) -> typing.Optional[str]:
|
||
"""Upload an image to Lark and return the image_key, or None if upload fails."""
|
||
image_bytes = None
|
||
|
||
if msg.base64:
|
||
try:
|
||
# Remove data URL prefix if present
|
||
base64_data = msg.base64
|
||
if base64_data.startswith('data:'):
|
||
base64_data = base64_data.split(',', 1)[1]
|
||
image_bytes = base64.b64decode(base64_data)
|
||
except Exception as e:
|
||
print(f'Failed to decode base64 image: {e}')
|
||
traceback.print_exc()
|
||
return None
|
||
elif msg.url:
|
||
try:
|
||
session = httpclient.get_session()
|
||
async with session.get(msg.url) as response:
|
||
if response.status == 200:
|
||
image_bytes = await response.read()
|
||
else:
|
||
print(f'Failed to download image from {msg.url}: HTTP {response.status}')
|
||
return None
|
||
except Exception as e:
|
||
print(f'Failed to download image from {msg.url}: {e}')
|
||
traceback.print_exc()
|
||
return None
|
||
elif msg.path:
|
||
try:
|
||
with open(msg.path, 'rb') as f:
|
||
image_bytes = f.read()
|
||
except Exception as e:
|
||
print(f'Failed to read image from path {msg.path}: {e}')
|
||
traceback.print_exc()
|
||
return None
|
||
|
||
if image_bytes is None:
|
||
print(
|
||
f'No image data available for Image message (url={msg.url}, base64={bool(msg.base64)}, path={msg.path})'
|
||
)
|
||
return None
|
||
|
||
try:
|
||
# Create a temporary file to store the image bytes
|
||
import tempfile
|
||
import os
|
||
|
||
with tempfile.NamedTemporaryFile(delete=False) as temp_file:
|
||
temp_file.write(image_bytes)
|
||
temp_file.flush()
|
||
temp_file_path = temp_file.name
|
||
|
||
try:
|
||
# Create image request using the temporary file
|
||
request = (
|
||
CreateImageRequest.builder()
|
||
.request_body(
|
||
CreateImageRequestBody.builder().image_type('message').image(open(temp_file_path, 'rb')).build()
|
||
)
|
||
.build()
|
||
)
|
||
|
||
response = await api_client.im.v1.image.acreate(request)
|
||
|
||
if not response.success():
|
||
print(
|
||
f'client.im.v1.image.create failed, code: {response.code}, msg: {response.msg}, log_id: {response.get_log_id()}'
|
||
)
|
||
return None
|
||
|
||
return response.data.image_key
|
||
finally:
|
||
# Clean up the temporary file
|
||
os.unlink(temp_file_path)
|
||
except Exception as e:
|
||
print(f'Failed to upload image to Lark: {e}')
|
||
traceback.print_exc()
|
||
return None
|
||
|
||
@staticmethod
|
||
async def upload_file_to_lark(
|
||
file_bytes: bytes,
|
||
api_client: lark_oapi.Client,
|
||
file_type: str,
|
||
file_name: str = 'file',
|
||
duration: typing.Optional[int] = None,
|
||
) -> typing.Optional[str]:
|
||
"""Upload a file to Lark and return the file_key, or None if upload fails.
|
||
|
||
Args:
|
||
file_bytes: Raw file bytes.
|
||
api_client: Lark API client.
|
||
file_type: Lark file type, e.g. 'opus', 'mp4', 'pdf', 'doc', etc.
|
||
file_name: Display name for the file.
|
||
duration: Duration in milliseconds (for audio files).
|
||
"""
|
||
try:
|
||
with tempfile.NamedTemporaryFile(delete=False) as temp_file:
|
||
temp_file.write(file_bytes)
|
||
temp_file_path = temp_file.name
|
||
|
||
try:
|
||
body_builder = (
|
||
CreateFileRequestBody.builder()
|
||
.file_type(file_type)
|
||
.file_name(file_name)
|
||
.file(open(temp_file_path, 'rb'))
|
||
)
|
||
if duration is not None:
|
||
body_builder = body_builder.duration(duration)
|
||
|
||
request = CreateFileRequest.builder().request_body(body_builder.build()).build()
|
||
|
||
response = await api_client.im.v1.file.acreate(request)
|
||
|
||
if not response.success():
|
||
print(
|
||
f'client.im.v1.file.create failed, code: {response.code}, msg: {response.msg}, log_id: {response.get_log_id()}'
|
||
)
|
||
return None
|
||
|
||
return response.data.file_key
|
||
finally:
|
||
os.unlink(temp_file_path)
|
||
except Exception as e:
|
||
print(f'Failed to upload file to Lark: {e}')
|
||
traceback.print_exc()
|
||
return None
|
||
|
||
@staticmethod
|
||
async def _get_media_bytes(
|
||
msg: typing.Union[platform_message.Voice, platform_message.File],
|
||
) -> typing.Optional[bytes]:
|
||
"""Get bytes from a Voice or File message (base64, url, or path)."""
|
||
data = None
|
||
|
||
if msg.base64:
|
||
try:
|
||
base64_str = msg.base64
|
||
if ',' in base64_str:
|
||
base64_str = base64_str.split(',', 1)[1]
|
||
data = base64.b64decode(base64_str)
|
||
except Exception:
|
||
pass
|
||
elif msg.url:
|
||
try:
|
||
session = httpclient.get_session()
|
||
async with session.get(msg.url) as resp:
|
||
if resp.status == 200:
|
||
data = await resp.read()
|
||
except Exception:
|
||
pass
|
||
elif msg.path:
|
||
try:
|
||
with open(msg.path, 'rb') as f:
|
||
data = f.read()
|
||
except Exception:
|
||
pass
|
||
|
||
return data
|
||
|
||
@staticmethod
|
||
async def yiri2target(
|
||
message_chain: platform_message.MessageChain, api_client: lark_oapi.Client
|
||
) -> typing.Tuple[list, list]:
|
||
"""Convert message chain to Lark format.
|
||
|
||
Returns:
|
||
Tuple of (text_elements, image_keys):
|
||
- text_elements: List of paragraphs for post message format
|
||
- media_items: List of dicts with 'msg_type' and 'content' for separate media messages
|
||
"""
|
||
message_elements = []
|
||
media_items = []
|
||
pending_paragraph = []
|
||
|
||
# Regex pattern to match Markdown image syntax: 
|
||
markdown_image_pattern = re.compile(r'!\[([^\]]*)\]\(([^)]+)\)')
|
||
|
||
async def process_text_with_images(text: str) -> typing.Tuple[str, list]:
|
||
"""Extract Markdown images from text and return cleaned text + image URLs."""
|
||
extracted_urls = []
|
||
|
||
# Find all Markdown images
|
||
matches = list(markdown_image_pattern.finditer(text))
|
||
if not matches:
|
||
return text, []
|
||
|
||
# Extract URLs and remove image syntax from text
|
||
cleaned_text = text
|
||
for match in reversed(matches): # Reverse to maintain correct positions
|
||
url = match.group(2)
|
||
extracted_urls.insert(0, url) # Insert at beginning since we're going in reverse
|
||
# Replace image syntax with empty string or a placeholder
|
||
cleaned_text = cleaned_text[: match.start()] + cleaned_text[match.end() :]
|
||
|
||
# Clean up multiple consecutive newlines that might result from removing images
|
||
cleaned_text = re.sub(r'\n{3,}', '\n\n', cleaned_text)
|
||
cleaned_text = cleaned_text.strip()
|
||
|
||
return cleaned_text, extracted_urls
|
||
|
||
for msg in message_chain:
|
||
if isinstance(msg, platform_message.Plain):
|
||
# Ensure text is valid UTF-8
|
||
try:
|
||
text = msg.text.encode('utf-8').decode('utf-8')
|
||
except UnicodeError:
|
||
try:
|
||
text = msg.text.encode('latin1').decode('utf-8')
|
||
except UnicodeError:
|
||
text = msg.text.encode('utf-8', errors='replace').decode('utf-8')
|
||
|
||
# Check for and extract Markdown images from text
|
||
cleaned_text, extracted_urls = await process_text_with_images(text)
|
||
|
||
# Split by blank lines to create separate paragraphs for Lark post format.
|
||
# Lark truncates md elements at the first \n\n, so we must use the
|
||
# post format's native paragraph structure instead.
|
||
if cleaned_text:
|
||
segments = re.split(r'\n\s*\n', cleaned_text)
|
||
for i, segment in enumerate(segments):
|
||
segment = segment.strip()
|
||
if not segment:
|
||
continue
|
||
if i > 0 and pending_paragraph:
|
||
message_elements.append(pending_paragraph)
|
||
pending_paragraph = []
|
||
pending_paragraph.append({'tag': 'md', 'text': segment})
|
||
|
||
# Process extracted image URLs
|
||
for url in extracted_urls:
|
||
temp_image = platform_message.Image(url=url)
|
||
image_key = await LarkMessageConverter.upload_image_to_lark(temp_image, api_client)
|
||
if image_key:
|
||
media_items.append({'msg_type': 'image', 'content': {'image_key': image_key}})
|
||
|
||
elif isinstance(msg, platform_message.At):
|
||
pending_paragraph.append({'tag': 'at', 'user_id': msg.target, 'style': []})
|
||
elif isinstance(msg, platform_message.AtAll):
|
||
pending_paragraph.append({'tag': 'at', 'user_id': 'all', 'style': []})
|
||
elif isinstance(msg, platform_message.Image):
|
||
image_key = await LarkMessageConverter.upload_image_to_lark(msg, api_client)
|
||
if image_key:
|
||
media_items.append({'msg_type': 'image', 'content': {'image_key': image_key}})
|
||
elif isinstance(msg, platform_message.Voice):
|
||
data = await LarkMessageConverter._get_media_bytes(msg)
|
||
if data:
|
||
duration = int(msg.length * 1000) if msg.length else None
|
||
file_key = await LarkMessageConverter.upload_file_to_lark(
|
||
data, api_client, file_type='opus', file_name='voice.opus', duration=duration
|
||
)
|
||
if file_key:
|
||
media_items.append({'msg_type': 'audio', 'content': {'file_key': file_key}})
|
||
elif isinstance(msg, platform_message.File):
|
||
data = await LarkMessageConverter._get_media_bytes(msg)
|
||
if data:
|
||
file_name = msg.name or 'file'
|
||
# Guess file_type from extension
|
||
ext = os.path.splitext(file_name)[1].lstrip('.').lower() if file_name else ''
|
||
file_type_map = {
|
||
'opus': 'opus',
|
||
'mp4': 'mp4',
|
||
'pdf': 'pdf',
|
||
'doc': 'doc',
|
||
'docx': 'doc',
|
||
'xls': 'xls',
|
||
'xlsx': 'xls',
|
||
'ppt': 'ppt',
|
||
'pptx': 'ppt',
|
||
}
|
||
file_type = file_type_map.get(ext, 'stream')
|
||
file_key = await LarkMessageConverter.upload_file_to_lark(
|
||
data, api_client, file_type=file_type, file_name=file_name
|
||
)
|
||
if file_key:
|
||
media_items.append({'msg_type': 'file', 'content': {'file_key': file_key}})
|
||
elif isinstance(msg, platform_message.Forward):
|
||
for node in msg.node_list:
|
||
sub_elements, sub_media = await LarkMessageConverter.yiri2target(node.message_chain, api_client)
|
||
message_elements.extend(sub_elements)
|
||
media_items.extend(sub_media)
|
||
|
||
if pending_paragraph:
|
||
message_elements.append(pending_paragraph)
|
||
|
||
return message_elements, media_items
|
||
|
||
@staticmethod
|
||
async def target2yiri(
|
||
message: lark_oapi.api.im.v1.model.event_message.EventMessage,
|
||
api_client: lark_oapi.Client,
|
||
) -> platform_message.MessageChain:
|
||
message_content = json.loads(message.content)
|
||
|
||
lb_msg_list = []
|
||
|
||
msg_create_time = datetime.datetime.fromtimestamp(int(message.create_time) / 1000)
|
||
|
||
lb_msg_list.append(platform_message.Source(id=message.message_id, time=msg_create_time))
|
||
|
||
if message.message_type == 'text':
|
||
element_list = []
|
||
|
||
def text_element_recur(text_ele: dict) -> list[dict]:
|
||
if text_ele['text'] == '':
|
||
return []
|
||
|
||
at_pattern = re.compile(r'@_user_[\d]+')
|
||
at_matches = at_pattern.findall(text_ele['text'])
|
||
|
||
name_mapping = {}
|
||
for mathc in at_matches:
|
||
for mention in message.mentions:
|
||
if mention.key == mathc:
|
||
name_mapping[mathc] = mention.name
|
||
break
|
||
|
||
if len(name_mapping.keys()) == 0:
|
||
return [text_ele]
|
||
|
||
# 只处理第一个,剩下的递归处理
|
||
text_split = text_ele['text'].split(list(name_mapping.keys())[0])
|
||
|
||
new_list = []
|
||
|
||
left_text = text_split[0]
|
||
right_text = text_split[1]
|
||
|
||
new_list.extend(text_element_recur({'tag': 'text', 'text': left_text, 'style': []}))
|
||
|
||
new_list.append(
|
||
{
|
||
'tag': 'at',
|
||
'user_id': list(name_mapping.keys())[0],
|
||
'user_name': name_mapping[list(name_mapping.keys())[0]],
|
||
'style': [],
|
||
}
|
||
)
|
||
|
||
new_list.extend(text_element_recur({'tag': 'text', 'text': right_text, 'style': []}))
|
||
|
||
return new_list
|
||
|
||
element_list = text_element_recur({'tag': 'text', 'text': message_content['text'], 'style': []})
|
||
|
||
message_content = {'title': '', 'content': element_list}
|
||
|
||
elif message.message_type == 'post':
|
||
new_list = []
|
||
|
||
for ele in message_content['content']:
|
||
if type(ele) is dict:
|
||
new_list.append(ele)
|
||
elif type(ele) is list:
|
||
new_list.extend(ele)
|
||
|
||
message_content['content'] = new_list
|
||
elif message.message_type == 'image':
|
||
message_content['content'] = [{'tag': 'img', 'image_key': message_content['image_key'], 'style': []}]
|
||
elif message.message_type == 'file':
|
||
message_content['content'] = [
|
||
{'tag': 'file', 'file_key': message_content['file_key'], 'file_name': message_content['file_name']}
|
||
]
|
||
elif message.message_type == 'audio':
|
||
message_content['content'] = [
|
||
{
|
||
'tag': 'audio',
|
||
'file_key': message_content['file_key'],
|
||
'duration': message_content.get('duration', 0),
|
||
}
|
||
]
|
||
|
||
for ele in message_content['content']:
|
||
if ele['tag'] == 'text':
|
||
lb_msg_list.append(platform_message.Plain(text=ele['text']))
|
||
elif ele['tag'] == 'at':
|
||
lb_msg_list.append(platform_message.At(target=ele['user_name']))
|
||
elif ele['tag'] == 'img':
|
||
image_key = ele['image_key']
|
||
|
||
request: GetMessageResourceRequest = (
|
||
GetMessageResourceRequest.builder()
|
||
.message_id(message.message_id)
|
||
.file_key(image_key)
|
||
.type('image')
|
||
.build()
|
||
)
|
||
|
||
response: GetMessageResourceResponse = await api_client.im.v1.message_resource.aget(request)
|
||
|
||
if not response.success():
|
||
raise Exception(
|
||
f'client.im.v1.message_resource.get failed, code: {response.code}, msg: {response.msg}, log_id: {response.get_log_id()}, resp: \n{json.dumps(json.loads(response.raw.content), indent=4, ensure_ascii=False)}'
|
||
)
|
||
|
||
image_bytes = response.file.read()
|
||
image_base64 = base64.b64encode(image_bytes).decode()
|
||
|
||
image_format = response.raw.headers['content-type']
|
||
|
||
lb_msg_list.append(platform_message.Image(base64=f'data:{image_format};base64,{image_base64}'))
|
||
elif ele['tag'] == 'audio':
|
||
file_key = ele['file_key']
|
||
duration = ele['duration']
|
||
|
||
# Download audio file
|
||
request: GetMessageResourceRequest = (
|
||
GetMessageResourceRequest.builder()
|
||
.message_id(message.message_id)
|
||
.file_key(file_key)
|
||
.type('file')
|
||
.build()
|
||
)
|
||
|
||
try:
|
||
response: GetMessageResourceResponse = await api_client.im.v1.message_resource.aget(request)
|
||
|
||
if not response.success():
|
||
print(f'Failed to download audio: code: {response.code}, msg: {response.msg}')
|
||
lb_msg_list.append(platform_message.Plain(text='[Audio file download failed]'))
|
||
return platform_message.MessageChain(lb_msg_list)
|
||
|
||
# Read audio bytes
|
||
audio_bytes = response.file.read()
|
||
audio_base64 = base64.b64encode(audio_bytes).decode()
|
||
|
||
# Get content type from response headers
|
||
content_type = response.raw.headers.get('content-type', 'audio/mpeg')
|
||
|
||
mime_main = content_type.split(';')[0].strip()
|
||
ext = mimetypes.guess_extension(mime_main) or '.bin'
|
||
temp_dir = tempfile.gettempdir()
|
||
temp_file_path = os.path.join(temp_dir, f'lark_audio_{file_key}{ext}')
|
||
|
||
with open(temp_file_path, 'wb') as f:
|
||
f.write(audio_bytes)
|
||
|
||
# Create Voice message: prefer path/url + length, include base64 as optional data URI
|
||
lb_msg_list.append(
|
||
platform_message.Voice(
|
||
voice_id=file_key,
|
||
url=f'file://{temp_file_path}',
|
||
path=temp_file_path,
|
||
base64=f'data:{content_type};base64,{audio_base64}',
|
||
length=(duration // 1000) if duration else None,
|
||
)
|
||
)
|
||
except Exception as e:
|
||
print(f'Error downloading audio: {e}')
|
||
traceback.print_exc()
|
||
lb_msg_list.append(platform_message.Plain(text='[Audio file download error]'))
|
||
|
||
elif ele['tag'] == 'file':
|
||
file_key = ele['file_key']
|
||
file_name = ele['file_name']
|
||
|
||
request: GetMessageResourceRequest = (
|
||
GetMessageResourceRequest.builder()
|
||
.message_id(message.message_id)
|
||
.file_key(file_key)
|
||
.type('file')
|
||
.build()
|
||
)
|
||
|
||
response: GetMessageResourceResponse = await api_client.im.v1.message_resource.aget(request)
|
||
|
||
if not response.success():
|
||
raise Exception(
|
||
f'client.im.v1.message_resource.get failed, code: {response.code}, msg: {response.msg}, log_id: {response.get_log_id()}, resp: \n{json.dumps(json.loads(response.raw.content), indent=4, ensure_ascii=False)}'
|
||
)
|
||
|
||
file_bytes = response.file.read()
|
||
file_base64 = base64.b64encode(file_bytes).decode()
|
||
|
||
file_format = response.raw.headers['content-type']
|
||
|
||
file_size = len(file_bytes)
|
||
|
||
# Determine extension from content-type if possible
|
||
content_type = response.raw.headers.get('content-type', '')
|
||
mime_main = content_type.split(';')[0].strip() if content_type else ''
|
||
ext = mimetypes.guess_extension(mime_main) or ''
|
||
|
||
# Ensure a safe filename (avoid path components)
|
||
safe_name = os.path.basename(file_name).replace('/', '_').replace('\\', '_')
|
||
if ext and not safe_name.lower().endswith(ext.lower()):
|
||
filename_with_ext = f'{safe_name}{ext}'
|
||
else:
|
||
filename_with_ext = safe_name
|
||
|
||
temp_dir = tempfile.gettempdir()
|
||
temp_file_path = os.path.join(temp_dir, f'lark_{file_key}_{filename_with_ext}')
|
||
|
||
with open(temp_file_path, 'wb') as f:
|
||
f.write(file_bytes)
|
||
|
||
# Create File message with local path and file:// URL
|
||
lb_msg_list.append(
|
||
platform_message.File(
|
||
id=file_key,
|
||
name=file_name,
|
||
size=file_size,
|
||
url=f'file://{temp_file_path}',
|
||
path=temp_file_path,
|
||
base64=f'data:{file_format};base64,{file_base64}', # not including base64 by default to save memory; can be added if needed
|
||
)
|
||
)
|
||
|
||
return platform_message.MessageChain(lb_msg_list)
|
||
|
||
|
||
class LarkEventConverter(abstract_platform_adapter.AbstractEventConverter):
|
||
_processed_thread_quote_cache: typing.ClassVar[dict[str, float]] = {}
|
||
_processed_thread_quote_cache_max_size: typing.ClassVar[int] = 4096
|
||
_processed_thread_quote_cache_ttl_seconds: typing.ClassVar[int] = 86400
|
||
|
||
@classmethod
|
||
def _prune_processed_thread_quote_cache(cls, now: typing.Optional[float] = None) -> None:
|
||
if now is None:
|
||
now = time.time()
|
||
|
||
expire_before = now - cls._processed_thread_quote_cache_ttl_seconds
|
||
while cls._processed_thread_quote_cache:
|
||
oldest_key, oldest_ts = next(iter(cls._processed_thread_quote_cache.items()))
|
||
if oldest_ts >= expire_before:
|
||
break
|
||
cls._processed_thread_quote_cache.pop(oldest_key, None)
|
||
|
||
while len(cls._processed_thread_quote_cache) > cls._processed_thread_quote_cache_max_size:
|
||
oldest_key = next(iter(cls._processed_thread_quote_cache))
|
||
cls._processed_thread_quote_cache.pop(oldest_key, None)
|
||
|
||
@classmethod
|
||
def _mark_thread_quote_processed(cls, thread_id: str) -> None:
|
||
now = time.time()
|
||
cls._prune_processed_thread_quote_cache(now)
|
||
cls._processed_thread_quote_cache[thread_id] = now
|
||
|
||
@classmethod
|
||
def _extract_quote_message_id(cls, message: EventMessage) -> typing.Optional[str]:
|
||
"""
|
||
Extract the message ID to quote from the given message.
|
||
|
||
Rules:
|
||
- First thread reply in a topic: return parent_id and mark topic as processed
|
||
- Follow-up thread replies in the same topic: return None
|
||
- Non-thread message: return parent_id if valid (non-empty, different from message_id)
|
||
|
||
Thread reply state is kept in a bounded TTL cache to avoid unbounded memory growth.
|
||
"""
|
||
parent_id = getattr(message, 'parent_id', None)
|
||
if not parent_id:
|
||
return None
|
||
|
||
message_id = getattr(message, 'message_id', None)
|
||
if parent_id == message_id:
|
||
return None
|
||
|
||
thread_id = getattr(message, 'thread_id', None)
|
||
if thread_id:
|
||
cls._prune_processed_thread_quote_cache()
|
||
if thread_id in cls._processed_thread_quote_cache:
|
||
return None
|
||
cls._mark_thread_quote_processed(thread_id)
|
||
|
||
return parent_id
|
||
|
||
@staticmethod
|
||
def _build_event_message_from_message_item(message_item: Message) -> typing.Optional[EventMessage]:
|
||
"""
|
||
Build EventMessage from SDK typed Message item.
|
||
|
||
Returns None if body or content is missing.
|
||
"""
|
||
body = getattr(message_item, 'body', None)
|
||
if not body:
|
||
return None
|
||
|
||
content = getattr(body, 'content', None)
|
||
if not content:
|
||
return None
|
||
|
||
event_data = {
|
||
'message_id': message_item.message_id,
|
||
'message_type': message_item.msg_type,
|
||
'content': content,
|
||
'create_time': message_item.create_time,
|
||
'mentions': getattr(message_item, 'mentions', []) or [],
|
||
}
|
||
|
||
# Preserve thread-related fields
|
||
if hasattr(message_item, 'parent_id') and message_item.parent_id:
|
||
event_data['parent_id'] = message_item.parent_id
|
||
if hasattr(message_item, 'root_id') and message_item.root_id:
|
||
event_data['root_id'] = message_item.root_id
|
||
if hasattr(message_item, 'thread_id') and message_item.thread_id:
|
||
event_data['thread_id'] = message_item.thread_id
|
||
if hasattr(message_item, 'chat_id') and message_item.chat_id:
|
||
event_data['chat_id'] = message_item.chat_id
|
||
|
||
return EventMessage(event_data)
|
||
|
||
@staticmethod
|
||
async def _fetch_quoted_message(
|
||
quote_message_id: str,
|
||
api_client: lark_oapi.Client,
|
||
) -> typing.Optional[platform_message.MessageChain]:
|
||
"""
|
||
Fetch the quoted message and convert to MessageChain.
|
||
|
||
Returns None if:
|
||
- API call fails
|
||
- Response items is empty
|
||
- Message item normalization fails
|
||
"""
|
||
request = GetMessageRequest.builder().message_id(quote_message_id).build()
|
||
response = await api_client.im.v1.message.aget(request)
|
||
|
||
if not response.success():
|
||
return None
|
||
|
||
items = getattr(response.data, 'items', None)
|
||
if not items:
|
||
return None
|
||
|
||
message_item = items[0]
|
||
event_message = LarkEventConverter._build_event_message_from_message_item(message_item)
|
||
if event_message is None:
|
||
return None
|
||
|
||
quote_chain = await LarkMessageConverter.target2yiri(event_message, api_client)
|
||
return quote_chain
|
||
|
||
@staticmethod
|
||
async def yiri2target(
|
||
event: platform_events.MessageEvent,
|
||
) -> lark_oapi.im.v1.P2ImMessageReceiveV1:
|
||
pass
|
||
|
||
@staticmethod
|
||
async def target2yiri(
|
||
event: lark_oapi.im.v1.P2ImMessageReceiveV1, api_client: lark_oapi.Client
|
||
) -> platform_events.Event:
|
||
message_chain = await LarkMessageConverter.target2yiri(event.event.message, api_client)
|
||
|
||
# Check for quote/reply message
|
||
# Extract files/images/voice from quote and add them as top-level components
|
||
# so that plugins like FileReader can process them the same way as direct messages
|
||
quote_message_id = LarkEventConverter._extract_quote_message_id(event.event.message)
|
||
if quote_message_id:
|
||
quote_chain = await LarkEventConverter._fetch_quoted_message(quote_message_id, api_client)
|
||
if quote_chain:
|
||
# Filter out Source component from quoted chain, keep only content
|
||
quote_components = [comp for comp in quote_chain if not isinstance(comp, platform_message.Source)]
|
||
|
||
# Add quoted content as top-level components instead of wrapping in Quote
|
||
for comp in quote_components:
|
||
if isinstance(comp, platform_message.File):
|
||
# Add file as top-level component (same as direct message)
|
||
message_chain.append(comp)
|
||
elif isinstance(comp, platform_message.Image):
|
||
# Add image as top-level component
|
||
message_chain.append(comp)
|
||
elif isinstance(comp, platform_message.Voice):
|
||
# Add voice as top-level component
|
||
message_chain.append(comp)
|
||
elif isinstance(comp, platform_message.Plain):
|
||
# Add text with context prefix
|
||
message_chain.append(platform_message.Plain(text=f'[引用消息] {comp.text}'))
|
||
|
||
if event.event.message.chat_type == 'p2p':
|
||
return platform_events.FriendMessage(
|
||
sender=platform_entities.Friend(
|
||
id=event.event.sender.sender_id.open_id,
|
||
nickname=event.event.sender.sender_id.union_id,
|
||
remark='',
|
||
),
|
||
message_chain=message_chain,
|
||
time=event.event.message.create_time,
|
||
source_platform_object=event,
|
||
)
|
||
elif event.event.message.chat_type == 'group':
|
||
return platform_events.GroupMessage(
|
||
sender=platform_entities.GroupMember(
|
||
id=event.event.sender.sender_id.open_id,
|
||
member_name=event.event.sender.sender_id.union_id,
|
||
permission=platform_entities.Permission.Member,
|
||
group=platform_entities.Group(
|
||
id=event.event.message.chat_id,
|
||
name='',
|
||
permission=platform_entities.Permission.Member,
|
||
),
|
||
special_title='',
|
||
),
|
||
message_chain=message_chain,
|
||
time=event.event.message.create_time,
|
||
source_platform_object=event,
|
||
)
|
||
|
||
|
||
CARD_ID_CACHE_SIZE = 500
|
||
CARD_ID_CACHE_MAX_LIFETIME = 20 * 60 # 20分钟
|
||
|
||
|
||
class LarkAdapter(abstract_platform_adapter.AbstractMessagePlatformAdapter):
|
||
bot: lark_oapi.ws.Client = pydantic.Field(exclude=True)
|
||
api_client: lark_oapi.Client = pydantic.Field(exclude=True)
|
||
|
||
bot_account_id: str # 用于在流水线中识别at是否是本bot,直接以bot_name作为标识
|
||
lark_tenant_key: str = pydantic.Field(exclude=True, default='') # 飞书企业key
|
||
|
||
message_converter: LarkMessageConverter = LarkMessageConverter()
|
||
event_converter: LarkEventConverter = LarkEventConverter()
|
||
cipher: AESCipher
|
||
|
||
listeners: typing.Dict[
|
||
typing.Type[platform_events.Event],
|
||
typing.Callable[[platform_events.Event, abstract_platform_adapter.AbstractMessagePlatformAdapter], None],
|
||
]
|
||
|
||
quart_app: quart.Quart = pydantic.Field(exclude=True)
|
||
|
||
card_id_dict: dict[str, str] # 消息id到卡片id的映射,便于创建卡片后的发送消息到指定卡片
|
||
|
||
# Monitoring message ID mapping for feedback correlation
|
||
# Temp: user Lark message ID → monitoring_message_id (populated by on_monitoring_message_created, consumed by create_message_card)
|
||
pending_monitoring_msg: dict[str, str]
|
||
# Final: reply Lark message ID → (monitoring_message_id, timestamp) (used by feedback callbacks)
|
||
reply_to_monitoring_msg: dict[str, tuple[str, float]]
|
||
_MONITORING_MAPPING_TTL = 600 # 10 minutes
|
||
|
||
seq: int # 用于在发送卡片消息中识别消息顺序,直接以seq作为标识
|
||
bot_uuid: str = None # 机器人UUID
|
||
app_ticket: str = None # 商店应用用到
|
||
app_access_token: str = None # 商店应用用到
|
||
app_access_token_expire_at: int = None
|
||
tenant_access_tokens: dict[str, dict[str, str]] = {} # 租户access_token映射
|
||
|
||
def __init__(self, config: dict, logger: abstract_platform_logger.AbstractEventLogger, **kwargs):
|
||
quart_app = quart.Quart(__name__)
|
||
|
||
async def on_message(event: lark_oapi.im.v1.P2ImMessageReceiveV1):
|
||
lb_event = await self.event_converter.target2yiri(event, self.api_client)
|
||
|
||
await self.listeners[type(lb_event)](lb_event, self)
|
||
|
||
def sync_on_message(event: lark_oapi.im.v1.P2ImMessageReceiveV1):
|
||
asyncio.create_task(on_message(event))
|
||
|
||
def sync_on_card_action(event):
|
||
try:
|
||
action_value_obj = getattr(getattr(event.event, 'action', None), 'value', {})
|
||
action_value = action_value_obj.get('feedback', '') if isinstance(action_value_obj, dict) else ''
|
||
|
||
if action_value == '有帮助':
|
||
feedback_type = 1
|
||
elif action_value == '无帮助':
|
||
feedback_type = 2
|
||
else:
|
||
from lark_oapi.event.callback.model.p2_card_action_trigger import P2CardActionTriggerResponse
|
||
|
||
return P2CardActionTriggerResponse({'toast': {'type': 'success', 'content': '操作成功'}})
|
||
|
||
operator = getattr(event.event, 'operator', None)
|
||
context = getattr(event.event, 'context', None)
|
||
|
||
user_id = getattr(operator, 'open_id', None) or getattr(operator, 'user_id', None)
|
||
open_chat_id = getattr(context, 'open_chat_id', None)
|
||
open_message_id = getattr(context, 'open_message_id', None)
|
||
|
||
if open_chat_id:
|
||
session_id = f'group_{open_chat_id}'
|
||
elif user_id:
|
||
session_id = f'person_{user_id}'
|
||
else:
|
||
session_id = None
|
||
|
||
# Resolve monitoring message ID from reply message mapping
|
||
monitoring_msg_id = None
|
||
if open_message_id and open_message_id in self.reply_to_monitoring_msg:
|
||
monitoring_msg_id = self.reply_to_monitoring_msg[open_message_id][0]
|
||
|
||
feedback_event = platform_events.FeedbackEvent(
|
||
feedback_id=getattr(event.header, 'event_id', str(uuid.uuid4())),
|
||
feedback_type=feedback_type,
|
||
feedback_content=action_value,
|
||
user_id=user_id,
|
||
session_id=session_id,
|
||
message_id=open_message_id,
|
||
stream_id=monitoring_msg_id,
|
||
source_platform_object=event,
|
||
)
|
||
|
||
if platform_events.FeedbackEvent in self.listeners:
|
||
loop = asyncio.get_event_loop()
|
||
if loop.is_running():
|
||
asyncio.create_task(self.listeners[platform_events.FeedbackEvent](feedback_event, self))
|
||
else:
|
||
loop.run_until_complete(self.listeners[platform_events.FeedbackEvent](feedback_event, self))
|
||
|
||
from lark_oapi.event.callback.model.p2_card_action_trigger import P2CardActionTriggerResponse
|
||
|
||
return P2CardActionTriggerResponse({'toast': {'type': 'success', 'content': '感谢您的反馈'}})
|
||
except Exception:
|
||
asyncio.create_task(self.logger.error(f'Error in lark card action callback: {traceback.format_exc()}'))
|
||
from lark_oapi.event.callback.model.p2_card_action_trigger import P2CardActionTriggerResponse
|
||
|
||
return P2CardActionTriggerResponse({'toast': {'type': 'error', 'content': '反馈处理失败'}})
|
||
|
||
event_handler = (
|
||
lark_oapi.EventDispatcherHandler.builder('', '')
|
||
.register_p2_im_message_receive_v1(sync_on_message)
|
||
.register_p2_card_action_trigger(sync_on_card_action)
|
||
.build()
|
||
)
|
||
|
||
bot_account_id = config['bot_name']
|
||
|
||
domain = self._resolve_domain(config)
|
||
bot = lark_oapi.ws.Client(config['app_id'], config['app_secret'], event_handler=event_handler, domain=domain)
|
||
api_client = self.build_api_client(config)
|
||
cipher = AESCipher(config.get('encrypt-key', ''))
|
||
self.request_app_ticket(api_client, config)
|
||
|
||
super().__init__(
|
||
config=config,
|
||
logger=logger,
|
||
lark_tenant_key=config.get('lark_tenant_key', ''),
|
||
card_id_dict={},
|
||
pending_monitoring_msg={},
|
||
reply_to_monitoring_msg={},
|
||
seq=1,
|
||
listeners={},
|
||
quart_app=quart_app,
|
||
bot=bot,
|
||
api_client=api_client,
|
||
bot_account_id=bot_account_id,
|
||
cipher=cipher,
|
||
**kwargs,
|
||
)
|
||
|
||
def request_app_ticket(self, api_client, config):
|
||
app_id = config['app_id']
|
||
app_secret = config['app_secret']
|
||
print(f'Requesting app ticket for app_id: {app_id[:3]}***{app_id[-3:]}')
|
||
if 'isv' == config.get('app_type', 'self'):
|
||
request: ResendAppTicketRequest = (
|
||
ResendAppTicketRequest.builder()
|
||
.request_body(ResendAppTicketRequestBody.builder().app_id(app_id).app_secret(app_secret).build())
|
||
.build()
|
||
)
|
||
response: ResendAppTicketResponse = api_client.auth.v3.app_ticket.resend(request)
|
||
if not response.success():
|
||
raise Exception(
|
||
f'client.auth.v3.auth.app_ticket_resend failed, code: {response.code}, msg: {response.msg}, log_id: {response.get_log_id()}, resp: \n{json.dumps(json.loads(response.raw.content), indent=4, ensure_ascii=False)}'
|
||
)
|
||
|
||
def request_app_access_token(self):
|
||
app_id = self.config['app_id']
|
||
app_secret = self.config['app_secret']
|
||
if 'isv' == self.config.get('app_type', 'self'):
|
||
request: CreateAppAccessTokenRequest = (
|
||
CreateAppAccessTokenRequest.builder()
|
||
.request_body(
|
||
CreateAppAccessTokenRequestBody.builder()
|
||
.app_id(app_id)
|
||
.app_secret(app_secret)
|
||
.app_ticket(self.app_ticket)
|
||
.build()
|
||
)
|
||
.build()
|
||
)
|
||
response: CreateAppAccessTokenResponse = self.api_client.auth.v3.app_access_token.create(request)
|
||
if not response.success():
|
||
raise Exception(
|
||
f'client.auth.v3.auth.app_access_token failed, code: {response.code}, msg: {response.msg}, log_id: {response.get_log_id()}, resp: \n{json.dumps(json.loads(response.raw.content), indent=4, ensure_ascii=False)}'
|
||
)
|
||
content = json.loads(response.raw.content)
|
||
self.app_access_token = content['app_access_token']
|
||
self.app_access_token_expire_at = int(time.time()) + content['expire'] - 300
|
||
|
||
def get_app_access_token(self):
|
||
if 'isv' != self.config.get('app_type', 'self'):
|
||
return None
|
||
if (
|
||
self.app_access_token is None
|
||
or self.app_access_token_expire_at is None
|
||
or int(time.time()) >= self.app_access_token_expire_at
|
||
):
|
||
self.request_app_access_token()
|
||
return self.app_access_token
|
||
|
||
def request_tenant_access_token(self, tenant_key: str):
|
||
app_access_token = self.get_app_access_token()
|
||
if 'isv' == self.config.get('app_type', 'self'):
|
||
request: CreateTenantAccessTokenRequest = (
|
||
CreateTenantAccessTokenRequest.builder()
|
||
.request_body(
|
||
CreateTenantAccessTokenRequestBody.builder()
|
||
.app_access_token(app_access_token)
|
||
.tenant_key(tenant_key)
|
||
.build()
|
||
)
|
||
.build()
|
||
)
|
||
response: CreateTenantAccessTokenResponse = self.api_client.auth.v3.tenant_access_token.create(request)
|
||
if not response.success():
|
||
raise Exception(
|
||
f'client.auth.v3.auth.tenant_access_token failed, code: {response.code}, msg: {response.msg}, log_id: {response.get_log_id()}, resp: \n{json.dumps(json.loads(response.raw.content), indent=4, ensure_ascii=False)}'
|
||
)
|
||
content = json.loads(response.raw.content)
|
||
tenant_access_token = content['tenant_access_token']
|
||
expire = content['expire']
|
||
self.tenant_access_tokens[tenant_key] = {
|
||
'token': tenant_access_token,
|
||
'expire_at': int(time.time()) + expire - 300,
|
||
}
|
||
|
||
def get_tenant_access_token(self, tenant_key: str):
|
||
if tenant_key is None or 'isv' != self.config.get('app_type', 'self'):
|
||
return None
|
||
tenant_access_token = self.tenant_access_tokens.get(tenant_key)
|
||
if tenant_access_token is None or int(time.time()) >= tenant_access_token['expire_at']:
|
||
self.request_tenant_access_token(tenant_key)
|
||
return self.tenant_access_tokens.get(tenant_key)['token'] if self.tenant_access_tokens.get(tenant_key) else None
|
||
|
||
def get_launcher_id(self, event: platform_events.MessageEvent) -> str | None:
|
||
"""
|
||
Get topic-scoped launcher_id for thread-aware session isolation.
|
||
|
||
For group thread messages, returns "{group_id}_{thread_id}"
|
||
to ensure conversation context stays stable per topic.
|
||
|
||
Returns None for non-thread messages or P2P messages.
|
||
"""
|
||
source_event = getattr(event.source_platform_object, 'event', None)
|
||
if not source_event:
|
||
return None
|
||
|
||
message = getattr(source_event, 'message', None)
|
||
if not message:
|
||
return None
|
||
|
||
thread_id = getattr(message, 'thread_id', None)
|
||
if not thread_id:
|
||
return None
|
||
|
||
if isinstance(event, platform_events.GroupMessage):
|
||
return f'{event.group.id}_{thread_id}'
|
||
|
||
return None
|
||
|
||
@staticmethod
|
||
def _resolve_domain(config) -> str:
|
||
domain = config.get('domain', lark_oapi.FEISHU_DOMAIN)
|
||
if domain == 'custom':
|
||
domain = config.get('custom_domain', '')
|
||
if not domain:
|
||
raise ValueError('Custom domain is required when domain is set to "custom"')
|
||
return domain.rstrip('/')
|
||
|
||
def build_api_client(self, config):
|
||
app_id = config['app_id']
|
||
app_secret = config['app_secret']
|
||
domain = self._resolve_domain(config)
|
||
api_client = lark_oapi.Client.builder().app_id(app_id).app_secret(app_secret).domain(domain).build()
|
||
if 'isv' == config.get('app_type', 'self'):
|
||
api_client = (
|
||
lark_oapi.Client.builder()
|
||
.app_id(app_id)
|
||
.app_secret(app_secret)
|
||
.app_type(lark_oapi.AppType.ISV)
|
||
.domain(domain)
|
||
.build()
|
||
)
|
||
return api_client
|
||
|
||
async def send_message(self, target_type: str, target_id: str, message: platform_message.MessageChain):
|
||
text_elements, media_items = await self.message_converter.yiri2target(message, self.api_client)
|
||
|
||
# Map standard target_type to Feishu receive_id_type
|
||
if target_type == 'person':
|
||
receive_id_type = 'open_id'
|
||
elif target_type == 'group':
|
||
receive_id_type = 'chat_id'
|
||
else:
|
||
receive_id_type = target_type
|
||
|
||
# Send text message if there are text elements
|
||
if text_elements:
|
||
needs_post = any(ele['tag'] == 'at' for paragraph in text_elements for ele in paragraph)
|
||
|
||
if needs_post:
|
||
msg_type = 'post'
|
||
final_content = json.dumps(
|
||
{
|
||
'zh_Hans': {
|
||
'title': '',
|
||
'content': text_elements,
|
||
},
|
||
}
|
||
)
|
||
else:
|
||
msg_type = 'text'
|
||
parts = []
|
||
for paragraph in text_elements:
|
||
para_text = ''.join(ele.get('text', '') for ele in paragraph)
|
||
if para_text:
|
||
parts.append(para_text)
|
||
final_content = json.dumps({'text': '\n\n'.join(parts)})
|
||
|
||
request: CreateMessageRequest = (
|
||
CreateMessageRequest.builder()
|
||
.receive_id_type(receive_id_type)
|
||
.request_body(
|
||
CreateMessageRequestBody.builder()
|
||
.receive_id(target_id)
|
||
.content(final_content)
|
||
.msg_type(msg_type)
|
||
.uuid(str(uuid.uuid4()))
|
||
.build()
|
||
)
|
||
.build()
|
||
)
|
||
|
||
app_access_token = self.get_app_access_token()
|
||
req_opt: RequestOption = (
|
||
RequestOption.builder().app_ticket(self.app_ticket).app_access_token(app_access_token).build()
|
||
)
|
||
response: CreateMessageResponse = self.api_client.im.v1.message.create(request, req_opt)
|
||
|
||
if not response.success():
|
||
raise Exception(
|
||
f'client.im.v1.message.create failed, code: {response.code}, msg: {response.msg}, log_id: {response.get_log_id()}, resp: \n{json.dumps(json.loads(response.raw.content), indent=4, ensure_ascii=False)}'
|
||
)
|
||
|
||
# Send media messages separately (image, audio, file, etc.)
|
||
for media in media_items:
|
||
request: CreateMessageRequest = (
|
||
CreateMessageRequest.builder()
|
||
.receive_id_type(receive_id_type)
|
||
.request_body(
|
||
CreateMessageRequestBody.builder()
|
||
.receive_id(target_id)
|
||
.content(json.dumps(media['content']))
|
||
.msg_type(media['msg_type'])
|
||
.uuid(str(uuid.uuid4()))
|
||
.build()
|
||
)
|
||
.build()
|
||
)
|
||
|
||
app_access_token = self.get_app_access_token()
|
||
req_opt: RequestOption = (
|
||
RequestOption.builder().app_ticket(self.app_ticket).app_access_token(app_access_token).build()
|
||
)
|
||
response: CreateMessageResponse = self.api_client.im.v1.message.create(request, req_opt)
|
||
|
||
if not response.success():
|
||
raise Exception(
|
||
f'client.im.v1.message.create ({media["msg_type"]}) failed, code: {response.code}, msg: {response.msg}, log_id: {response.get_log_id()}, resp: \n{json.dumps(json.loads(response.raw.content), indent=4, ensure_ascii=False)}'
|
||
)
|
||
|
||
async def is_stream_output_supported(self) -> bool:
|
||
is_stream = False
|
||
if self.config.get('enable-stream-reply', None):
|
||
is_stream = True
|
||
return is_stream
|
||
|
||
async def on_monitoring_message_created(self, query, monitoring_message_id: str):
|
||
"""Called by pipeline after monitoring message is created, to map user message ID to monitoring message ID."""
|
||
try:
|
||
user_msg_id = query.message_event.message_chain.message_id
|
||
if user_msg_id:
|
||
self.pending_monitoring_msg[user_msg_id] = monitoring_message_id
|
||
except Exception as e:
|
||
await self.logger.debug(f'Failed to map message to monitoring message: {e}')
|
||
|
||
def _cleanup_monitoring_mapping(self):
|
||
"""Remove entries older than TTL from the reply-to-monitoring mapping."""
|
||
now = time.time()
|
||
expired = [k for k, (_, ts) in self.reply_to_monitoring_msg.items() if now - ts > self._MONITORING_MAPPING_TTL]
|
||
for k in expired:
|
||
del self.reply_to_monitoring_msg[k]
|
||
|
||
async def create_card_id(self, message_id):
|
||
try:
|
||
# self.logger.debug('飞书支持stream输出,创建卡片......')
|
||
|
||
card_data = {
|
||
'schema': '2.0',
|
||
'config': {
|
||
'update_multi': True,
|
||
'streaming_mode': True,
|
||
'streaming_config': {
|
||
'print_step': {'default': 1},
|
||
'print_frequency_ms': {'default': 70},
|
||
'print_strategy': 'fast',
|
||
},
|
||
},
|
||
'body': {
|
||
'direction': 'vertical',
|
||
'padding': '12px 12px 12px 12px',
|
||
'elements': [
|
||
{
|
||
'tag': 'div',
|
||
'text': {
|
||
'tag': 'plain_text',
|
||
'content': 'LangBot',
|
||
'text_size': 'normal',
|
||
'text_align': 'left',
|
||
'text_color': 'default',
|
||
},
|
||
'icon': {
|
||
'tag': 'custom_icon',
|
||
'img_key': 'img_v3_02p3_05c65d5d-9bad-440a-a2fb-c89571bfd5bg',
|
||
},
|
||
},
|
||
{
|
||
'tag': 'markdown',
|
||
'content': '',
|
||
'text_align': 'left',
|
||
'text_size': 'normal',
|
||
'margin': '0px 0px 0px 0px',
|
||
'element_id': 'streaming_txt',
|
||
},
|
||
{
|
||
'tag': 'markdown',
|
||
'content': '',
|
||
'text_align': 'left',
|
||
'text_size': 'normal',
|
||
'margin': '0px 0px 0px 0px',
|
||
},
|
||
{
|
||
'tag': 'column_set',
|
||
'horizontal_spacing': '8px',
|
||
'horizontal_align': 'left',
|
||
'columns': [
|
||
{
|
||
'tag': 'column',
|
||
'width': 'weighted',
|
||
'elements': [
|
||
{
|
||
'tag': 'markdown',
|
||
'content': '',
|
||
'text_align': 'left',
|
||
'text_size': 'normal',
|
||
'margin': '0px 0px 0px 0px',
|
||
},
|
||
{
|
||
'tag': 'markdown',
|
||
'content': '',
|
||
'text_align': 'left',
|
||
'text_size': 'normal',
|
||
'margin': '0px 0px 0px 0px',
|
||
},
|
||
{
|
||
'tag': 'markdown',
|
||
'content': '',
|
||
'text_align': 'left',
|
||
'text_size': 'normal',
|
||
'margin': '0px 0px 0px 0px',
|
||
},
|
||
],
|
||
'padding': '0px 0px 0px 0px',
|
||
'direction': 'vertical',
|
||
'horizontal_spacing': '8px',
|
||
'vertical_spacing': '2px',
|
||
'horizontal_align': 'left',
|
||
'vertical_align': 'top',
|
||
'margin': '0px 0px 0px 0px',
|
||
'weight': 1,
|
||
}
|
||
],
|
||
'margin': '0px 0px 0px 0px',
|
||
},
|
||
{'tag': 'hr', 'margin': '0px 0px 0px 0px'},
|
||
{
|
||
'tag': 'column_set',
|
||
'horizontal_spacing': '12px',
|
||
'horizontal_align': 'right',
|
||
'columns': [
|
||
{
|
||
'tag': 'column',
|
||
'width': 'weighted',
|
||
'elements': [
|
||
{
|
||
'tag': 'markdown',
|
||
'content': '<font color="grey-600">以上内容由 AI 生成,仅供参考。更多详细、准确信息可点击引用链接查看</font>',
|
||
'text_align': 'left',
|
||
'text_size': 'notation',
|
||
'margin': '4px 0px 0px 0px',
|
||
'icon': {
|
||
'tag': 'standard_icon',
|
||
'token': 'robot_outlined',
|
||
'color': 'grey',
|
||
},
|
||
}
|
||
],
|
||
'padding': '0px 0px 0px 0px',
|
||
'direction': 'vertical',
|
||
'horizontal_spacing': '8px',
|
||
'vertical_spacing': '8px',
|
||
'horizontal_align': 'left',
|
||
'vertical_align': 'top',
|
||
'margin': '0px 0px 0px 0px',
|
||
'weight': 1,
|
||
},
|
||
{
|
||
'tag': 'column',
|
||
'width': '20px',
|
||
'elements': [
|
||
{
|
||
'tag': 'button',
|
||
'text': {'tag': 'plain_text', 'content': ''},
|
||
'type': 'text',
|
||
'width': 'fill',
|
||
'size': 'medium',
|
||
'icon': {'tag': 'standard_icon', 'token': 'thumbsup_outlined'},
|
||
'hover_tips': {'tag': 'plain_text', 'content': '有帮助'},
|
||
'behaviors': [{'type': 'callback', 'value': {'feedback': '有帮助'}}],
|
||
'margin': '0px 0px 0px 0px',
|
||
}
|
||
],
|
||
'padding': '0px 0px 0px 0px',
|
||
'direction': 'vertical',
|
||
'horizontal_spacing': '8px',
|
||
'vertical_spacing': '8px',
|
||
'horizontal_align': 'left',
|
||
'vertical_align': 'top',
|
||
'margin': '0px 0px 0px 0px',
|
||
},
|
||
{
|
||
'tag': 'column',
|
||
'width': '30px',
|
||
'elements': [
|
||
{
|
||
'tag': 'button',
|
||
'text': {'tag': 'plain_text', 'content': ''},
|
||
'type': 'text',
|
||
'width': 'default',
|
||
'size': 'medium',
|
||
'icon': {'tag': 'standard_icon', 'token': 'thumbdown_outlined'},
|
||
'hover_tips': {'tag': 'plain_text', 'content': '无帮助'},
|
||
'behaviors': [{'type': 'callback', 'value': {'feedback': '无帮助'}}],
|
||
'margin': '0px 0px 0px 0px',
|
||
}
|
||
],
|
||
'padding': '0px 0px 0px 0px',
|
||
'vertical_spacing': '8px',
|
||
'horizontal_align': 'left',
|
||
'vertical_align': 'top',
|
||
'margin': '0px 0px 0px 0px',
|
||
},
|
||
],
|
||
'margin': '0px 0px 4px 0px',
|
||
},
|
||
],
|
||
},
|
||
}
|
||
# delay / fast 创建卡片模板,delay 延迟打印,fast 实时打印,可以自定义更好看的消息模板
|
||
|
||
request: CreateCardRequest = (
|
||
CreateCardRequest.builder()
|
||
.request_body(CreateCardRequestBody.builder().type('card_json').data(json.dumps(card_data)).build())
|
||
.build()
|
||
)
|
||
|
||
# 发起请求
|
||
response: CreateCardResponse = self.api_client.cardkit.v1.card.create(request)
|
||
|
||
# 处理失败返回
|
||
if not response.success():
|
||
raise Exception(
|
||
f'client.cardkit.v1.card.create failed, code: {response.code}, msg: {response.msg}, log_id: {response.get_log_id()}, resp: \n{json.dumps(json.loads(response.raw.content), indent=4, ensure_ascii=False)}'
|
||
)
|
||
|
||
self.card_id_dict[message_id] = response.data.card_id
|
||
|
||
card_id = response.data.card_id
|
||
return card_id
|
||
|
||
except Exception as e:
|
||
raise e
|
||
|
||
async def create_message_card(self, message_id, event) -> str:
|
||
"""
|
||
创建卡片消息。
|
||
使用卡片消息是因为普通消息更新次数有限制,而大模型流式返回结果可能很多而超过限制,而飞书卡片没有这个限制(api免费次数有限)
|
||
"""
|
||
# message_id = event.message_chain.message_id
|
||
|
||
card_id = await self.create_card_id(message_id)
|
||
content = {
|
||
'type': 'card',
|
||
'data': {'card_id': card_id, 'template_variable': {'content': 'Thinking...'}},
|
||
} # 当收到消息时发送消息模板,可添加模板变量,详情查看飞书中接口文档
|
||
request: ReplyMessageRequest = (
|
||
ReplyMessageRequest.builder()
|
||
.message_id(event.message_chain.message_id)
|
||
.request_body(
|
||
ReplyMessageRequestBody.builder().content(json.dumps(content)).msg_type('interactive').build()
|
||
)
|
||
.build()
|
||
)
|
||
tenant_key = event.source_platform_object.header.tenant_key if event.source_platform_object else None
|
||
app_access_token = self.get_app_access_token()
|
||
tenant_access_token = self.get_tenant_access_token(tenant_key)
|
||
req_opt: RequestOption = (
|
||
RequestOption.builder()
|
||
.app_ticket(self.app_ticket)
|
||
.tenant_key(tenant_key)
|
||
.app_access_token(app_access_token)
|
||
.tenant_access_token(tenant_access_token)
|
||
.build()
|
||
)
|
||
# 发起请求
|
||
response: ReplyMessageResponse = await self.api_client.im.v1.message.areply(request, req_opt)
|
||
|
||
# 处理失败返回
|
||
if not response.success():
|
||
raise Exception(
|
||
f'client.im.v1.message.reply failed, code: {response.code}, msg: {response.msg}, log_id: {response.get_log_id()}, resp: \n{json.dumps(json.loads(response.raw.content), indent=4, ensure_ascii=False)}'
|
||
)
|
||
|
||
# Transfer monitoring message mapping: user msg ID → reply msg ID
|
||
try:
|
||
user_msg_id = event.message_chain.message_id
|
||
reply_msg_id = getattr(response.data, 'message_id', None)
|
||
monitoring_msg_id = self.pending_monitoring_msg.pop(user_msg_id, None)
|
||
if reply_msg_id and monitoring_msg_id:
|
||
self.reply_to_monitoring_msg[reply_msg_id] = (monitoring_msg_id, time.time())
|
||
self._cleanup_monitoring_mapping()
|
||
except Exception as e:
|
||
asyncio.create_task(self.logger.debug(f'Failed to transfer monitoring mapping in create_message_card: {e}'))
|
||
|
||
return True
|
||
|
||
async def reply_message(
|
||
self,
|
||
message_source: platform_events.MessageEvent,
|
||
message: platform_message.MessageChain,
|
||
quote_origin: bool = False,
|
||
):
|
||
# 不再需要了,因为message_id已经被包含到message_chain中
|
||
# lark_event = await self.event_converter.yiri2target(message_source)
|
||
text_elements, media_items = await self.message_converter.yiri2target(message, self.api_client)
|
||
|
||
# Send text message if there are text elements
|
||
if text_elements:
|
||
# Determine msg_type based on content: use 'post' if at mentions
|
||
# are present (requires post paragraph structure), otherwise 'text'
|
||
needs_post = any(ele['tag'] == 'at' for paragraph in text_elements for ele in paragraph)
|
||
|
||
if needs_post:
|
||
msg_type = 'post'
|
||
final_content = json.dumps(
|
||
{
|
||
'zh_Hans': {
|
||
'title': '',
|
||
'content': text_elements,
|
||
},
|
||
}
|
||
)
|
||
else:
|
||
msg_type = 'text'
|
||
parts = []
|
||
for paragraph in text_elements:
|
||
para_text = ''.join(ele.get('text', '') for ele in paragraph)
|
||
if para_text:
|
||
parts.append(para_text)
|
||
final_content = json.dumps({'text': '\n\n'.join(parts)})
|
||
|
||
request: ReplyMessageRequest = (
|
||
ReplyMessageRequest.builder()
|
||
.message_id(message_source.message_chain.message_id)
|
||
.request_body(
|
||
ReplyMessageRequestBody.builder()
|
||
.content(final_content)
|
||
.msg_type(msg_type)
|
||
.reply_in_thread(False)
|
||
.uuid(str(uuid.uuid4()))
|
||
.build()
|
||
)
|
||
.build()
|
||
)
|
||
|
||
tenant_key = (
|
||
message_source.source_platform_object.header.tenant_key
|
||
if message_source.source_platform_object
|
||
else None
|
||
)
|
||
app_access_token = self.get_app_access_token()
|
||
tenant_access_token = self.get_tenant_access_token(tenant_key)
|
||
req_opt: RequestOption = (
|
||
RequestOption.builder()
|
||
.app_ticket(self.app_ticket)
|
||
.tenant_key(tenant_key)
|
||
.app_access_token(app_access_token)
|
||
.tenant_access_token(tenant_access_token)
|
||
.build()
|
||
)
|
||
response: ReplyMessageResponse = await self.api_client.im.v1.message.areply(request, req_opt)
|
||
|
||
if not response.success():
|
||
raise Exception(
|
||
f'client.im.v1.message.reply failed, code: {response.code}, msg: {response.msg}, log_id: {response.get_log_id()}, resp: \n{json.dumps(json.loads(response.raw.content), indent=4, ensure_ascii=False)}'
|
||
)
|
||
|
||
# Send media messages separately (image, audio, file, etc.)
|
||
for media in media_items:
|
||
request: ReplyMessageRequest = (
|
||
ReplyMessageRequest.builder()
|
||
.message_id(message_source.message_chain.message_id)
|
||
.request_body(
|
||
ReplyMessageRequestBody.builder()
|
||
.content(json.dumps(media['content']))
|
||
.msg_type(media['msg_type'])
|
||
.reply_in_thread(False)
|
||
.uuid(str(uuid.uuid4()))
|
||
.build()
|
||
)
|
||
.build()
|
||
)
|
||
|
||
tenant_key = (
|
||
message_source.source_platform_object.header.tenant_key
|
||
if message_source.source_platform_object
|
||
else None
|
||
)
|
||
app_access_token = self.get_app_access_token()
|
||
tenant_access_token = self.get_tenant_access_token(tenant_key)
|
||
req_opt: RequestOption = (
|
||
RequestOption.builder()
|
||
.app_ticket(self.app_ticket)
|
||
.tenant_key(tenant_key)
|
||
.app_access_token(app_access_token)
|
||
.tenant_access_token(tenant_access_token)
|
||
.build()
|
||
)
|
||
response: ReplyMessageResponse = await self.api_client.im.v1.message.areply(request, req_opt)
|
||
|
||
if not response.success():
|
||
raise Exception(
|
||
f'client.im.v1.message.reply ({media["msg_type"]}) failed, code: {response.code}, msg: {response.msg}, log_id: {response.get_log_id()}, resp: \n{json.dumps(json.loads(response.raw.content), indent=4, ensure_ascii=False)}'
|
||
)
|
||
|
||
async def reply_message_chunk(
|
||
self,
|
||
message_source: platform_events.MessageEvent,
|
||
bot_message,
|
||
message: platform_message.MessageChain,
|
||
quote_origin: bool = False,
|
||
is_final: bool = False,
|
||
):
|
||
"""
|
||
回复消息变成更新卡片消息
|
||
"""
|
||
# self.seq += 1
|
||
message_id = bot_message.resp_message_id
|
||
msg_seq = bot_message.msg_sequence
|
||
if msg_seq % 8 == 0 or is_final:
|
||
text_elements, media_items = await self.message_converter.yiri2target(message, self.api_client)
|
||
|
||
text_message = ''
|
||
if text_elements:
|
||
parts = []
|
||
for paragraph in text_elements:
|
||
para_text = ''.join(ele['text'] for ele in paragraph if ele['tag'] in ('text', 'md'))
|
||
if para_text:
|
||
parts.append(para_text)
|
||
text_message = '\n\n'.join(parts)
|
||
|
||
# content = {
|
||
# 'type': 'card_json',
|
||
# 'data': {'card_id': self.card_id_dict[message_id], 'elements': {'content': text_message}},
|
||
# }
|
||
|
||
request: ContentCardElementRequest = (
|
||
ContentCardElementRequest.builder()
|
||
.card_id(self.card_id_dict[message_id])
|
||
.element_id('streaming_txt')
|
||
.request_body(
|
||
ContentCardElementRequestBody.builder()
|
||
# .uuid("a0d69e20-1dd1-458b-k525-dfeca4015204")
|
||
.content(text_message)
|
||
.sequence(msg_seq)
|
||
.build()
|
||
)
|
||
.build()
|
||
)
|
||
|
||
if is_final and bot_message.tool_calls is None:
|
||
# self.seq = 1 # 消息回复结束之后重置seq
|
||
self.card_id_dict.pop(message_id) # 清理已经使用过的卡片
|
||
|
||
tenant_key = (
|
||
message_source.source_platform_object.header.tenant_key
|
||
if message_source.source_platform_object
|
||
else None
|
||
)
|
||
app_access_token = self.get_app_access_token()
|
||
tenant_access_token = self.get_tenant_access_token(tenant_key)
|
||
req_opt: RequestOption = (
|
||
RequestOption.builder()
|
||
.app_ticket(self.app_ticket)
|
||
.tenant_key(tenant_key)
|
||
.app_access_token(app_access_token)
|
||
.tenant_access_token(tenant_access_token)
|
||
.build()
|
||
)
|
||
# 发起请求
|
||
response: ContentCardElementResponse = self.api_client.cardkit.v1.card_element.content(request, req_opt)
|
||
|
||
# 处理失败返回
|
||
if not response.success():
|
||
raise Exception(
|
||
f'client.im.v1.message.patch failed, code: {response.code}, msg: {response.msg}, log_id: {response.get_log_id()}, resp: \n{json.dumps(json.loads(response.raw.content), indent=4, ensure_ascii=False)}'
|
||
)
|
||
return
|
||
|
||
# Send media messages when streaming is done
|
||
if is_final and media_items:
|
||
for media in media_items:
|
||
media_request: ReplyMessageRequest = (
|
||
ReplyMessageRequest.builder()
|
||
.message_id(message_source.message_chain.message_id)
|
||
.request_body(
|
||
ReplyMessageRequestBody.builder()
|
||
.content(json.dumps(media['content']))
|
||
.msg_type(media['msg_type'])
|
||
.reply_in_thread(False)
|
||
.uuid(str(uuid.uuid4()))
|
||
.build()
|
||
)
|
||
.build()
|
||
)
|
||
media_response: ReplyMessageResponse = await self.api_client.im.v1.message.areply(
|
||
media_request, req_opt
|
||
)
|
||
if not media_response.success():
|
||
raise Exception(
|
||
f'client.im.v1.message.reply ({media["msg_type"]}) failed, code: {media_response.code}, msg: {media_response.msg}, log_id: {media_response.get_log_id()}'
|
||
)
|
||
|
||
async def is_muted(self, group_id: int) -> bool:
|
||
return False
|
||
|
||
def register_listener(
|
||
self,
|
||
event_type: typing.Type[platform_events.Event],
|
||
callback: typing.Callable[
|
||
[platform_events.Event, abstract_platform_adapter.AbstractMessagePlatformAdapter], None
|
||
],
|
||
):
|
||
self.listeners[event_type] = callback
|
||
|
||
def unregister_listener(
|
||
self,
|
||
event_type: typing.Type[platform_events.Event],
|
||
callback: typing.Callable[
|
||
[platform_events.Event, abstract_platform_adapter.AbstractMessagePlatformAdapter], None
|
||
],
|
||
):
|
||
self.listeners.pop(event_type)
|
||
|
||
def set_bot_uuid(self, bot_uuid: str):
|
||
"""设置 bot UUID(用于生成 webhook URL)"""
|
||
self.bot_uuid = bot_uuid
|
||
|
||
def get_event_type(self, data):
|
||
schema = '1.0'
|
||
if 'schema' in data:
|
||
schema = data['schema']
|
||
if '2.0' == schema:
|
||
return data['header']['event_type']
|
||
elif 'event' in data:
|
||
return data['event']['type']
|
||
else:
|
||
return data['type']
|
||
|
||
async def handle_unified_webhook(self, bot_uuid: str, path: str, request):
|
||
"""处理统一 webhook 请求。
|
||
Args:
|
||
bot_uuid: Bot 的 UUID
|
||
path: 子路径(如果有的话)
|
||
request: Quart Request 对象
|
||
Returns:
|
||
响应数据
|
||
"""
|
||
try:
|
||
data = await request.json
|
||
|
||
if 'encrypt' in data:
|
||
data = self.cipher.decrypt_string(data['encrypt'])
|
||
data = json.loads(data)
|
||
type = self.get_event_type(data)
|
||
context = EventContext(data)
|
||
if 'url_verification' == type:
|
||
# todo 验证verification token
|
||
return {'challenge': data.get('challenge')}
|
||
elif 'app_ticket' == type:
|
||
self.app_ticket = context.event['app_ticket']
|
||
elif 'im.message.receive_v1' == type:
|
||
try:
|
||
p2v1 = P2ImMessageReceiveV1()
|
||
p2v1.header = context.header
|
||
event = P2ImMessageReceiveV1Data()
|
||
event.message = EventMessage(context.event['message'])
|
||
event.sender = EventSender(context.event['sender'])
|
||
p2v1.event = event
|
||
p2v1.schema = context.schema
|
||
event = await self.event_converter.target2yiri(p2v1, self.api_client)
|
||
except Exception:
|
||
await self.logger.error(f'Error in lark callback: {traceback.format_exc()}')
|
||
|
||
if event.__class__ in self.listeners:
|
||
await self.listeners[event.__class__](event, self)
|
||
elif 'card.action.trigger' == type:
|
||
try:
|
||
event_data = data.get('event', {})
|
||
operator = event_data.get('operator', {})
|
||
action = event_data.get('action', {})
|
||
context_data = event_data.get('context', {})
|
||
|
||
action_value_obj = action.get('value', {})
|
||
action_value = action_value_obj.get('feedback', '') if isinstance(action_value_obj, dict) else ''
|
||
|
||
if action_value == '有帮助':
|
||
feedback_type = 1
|
||
elif action_value == '无帮助':
|
||
feedback_type = 2
|
||
else:
|
||
return {'toast': {'type': 'success', 'content': '操作成功'}}
|
||
|
||
user_id = operator.get('open_id') or operator.get('user_id')
|
||
open_chat_id = context_data.get('open_chat_id')
|
||
open_message_id = context_data.get('open_message_id')
|
||
|
||
if open_chat_id:
|
||
session_id = f'group_{open_chat_id}'
|
||
elif user_id:
|
||
session_id = f'person_{user_id}'
|
||
else:
|
||
session_id = None
|
||
|
||
# Resolve monitoring message ID from reply message mapping
|
||
monitoring_msg_id = None
|
||
if open_message_id and open_message_id in self.reply_to_monitoring_msg:
|
||
monitoring_msg_id = self.reply_to_monitoring_msg[open_message_id][0]
|
||
|
||
feedback_event = platform_events.FeedbackEvent(
|
||
feedback_id=data.get('header', {}).get('event_id', str(uuid.uuid4())),
|
||
feedback_type=feedback_type,
|
||
feedback_content=action_value,
|
||
user_id=user_id,
|
||
session_id=session_id,
|
||
message_id=open_message_id,
|
||
stream_id=monitoring_msg_id,
|
||
source_platform_object=data,
|
||
)
|
||
|
||
if platform_events.FeedbackEvent in self.listeners:
|
||
await self.listeners[platform_events.FeedbackEvent](feedback_event, self)
|
||
|
||
return {'toast': {'type': 'success', 'content': '感谢您的反馈'}}
|
||
except Exception:
|
||
await self.logger.error(f'Error in lark card action callback: {traceback.format_exc()}')
|
||
return {'toast': {'type': 'error', 'content': '反馈处理失败'}}
|
||
|
||
elif 'im.chat.member.bot.added_v1' == type:
|
||
try:
|
||
bot_added_welcome_msg = self.config.get('bot_added_welcome', '')
|
||
if bot_added_welcome_msg:
|
||
final_content = {
|
||
'zh_Hans': {
|
||
'title': '',
|
||
'content': [[{'tag': 'md', 'text': bot_added_welcome_msg}]],
|
||
},
|
||
}
|
||
chat_id = context.event['chat_id']
|
||
request: CreateMessageRequest = (
|
||
CreateMessageRequest.builder()
|
||
.receive_id_type('chat_id')
|
||
.request_body(
|
||
CreateMessageRequestBody.builder()
|
||
.receive_id(chat_id)
|
||
.content(json.dumps(final_content))
|
||
.msg_type('post')
|
||
.uuid(str(uuid.uuid4()))
|
||
.build()
|
||
)
|
||
.build()
|
||
)
|
||
tenant_key = context.header.tenant_key if context.header else None
|
||
app_access_token = self.get_app_access_token()
|
||
tenant_access_token = self.get_tenant_access_token(tenant_key)
|
||
req_opt: RequestOption = (
|
||
RequestOption.builder()
|
||
.app_ticket(self.app_ticket)
|
||
.tenant_key(tenant_key)
|
||
.app_access_token(app_access_token)
|
||
.tenant_access_token(tenant_access_token)
|
||
.build()
|
||
)
|
||
response: CreateMessageResponse = self.api_client.im.v1.message.create(request, req_opt)
|
||
|
||
if not response.success():
|
||
raise Exception(
|
||
f'client.im.v1.message.create failed, code: {response.code}, msg: {response.msg}, log_id: {response.get_log_id()}, resp: \n{json.dumps(json.loads(response.raw.content), indent=4, ensure_ascii=False)}'
|
||
)
|
||
except Exception as e:
|
||
print(f'im.chat.member.bot.added_v1: {e}')
|
||
await self.logger.error(f'Error in lark callback: {traceback.format_exc()}')
|
||
|
||
return {'code': 200, 'message': 'ok'}
|
||
except Exception as e:
|
||
print(f'Error in lark callback: {e}')
|
||
await self.logger.error(f'Error in lark callback: {traceback.format_exc()}')
|
||
return {'code': 500, 'message': 'error'}
|
||
|
||
async def run_async(self):
|
||
enable_webhook = self.config['enable-webhook']
|
||
|
||
if not enable_webhook:
|
||
try:
|
||
await self.bot._connect()
|
||
except lark_oapi.ws.exception.ClientException as e:
|
||
raise e
|
||
except Exception as e:
|
||
await self.bot._disconnect()
|
||
if self.bot._auto_reconnect:
|
||
await self.bot._reconnect()
|
||
else:
|
||
raise e
|
||
else:
|
||
# 统一 webhook 模式下,不启动独立的 Quart 应用
|
||
# 保持运行但不启动独立端口
|
||
|
||
async def keep_alive():
|
||
while True:
|
||
await asyncio.sleep(1)
|
||
|
||
await keep_alive()
|
||
|
||
async def kill(self) -> bool:
|
||
# 需要断开连接,不然旧的连接会继续运行,导致飞书消息来时会随机选择一个连接
|
||
# 断开时lark.ws.Client的_receive_message_loop会打印error日志: receive message loop exit。然后进行重连,
|
||
# 所以要设置_auto_reconnect=False,让其不重连。
|
||
self.bot._auto_reconnect = False
|
||
await self.bot._disconnect()
|
||
return False
|