mirror of
https://github.com/langbot-app/LangBot.git
synced 2026-06-18 11:44:18 +00:00
89064a9d5b
* feat: add support for username * fix: lint * fix: migerations * fix: change to version 21 * fix: remove duplicate dbm021 migration and rename dbm022 * feat: add user_id and user_name display with copy functionality in BotSessionMonitor --------- Co-authored-by: wangcham <wangcham@gmail.com> Co-authored-by: Junyan Qin <rockchinq@gmail.com>
436 lines
17 KiB
Python
436 lines
17 KiB
Python
from quart import request
|
|
from ..wecom_api.WXBizMsgCrypt3 import WXBizMsgCrypt
|
|
import base64
|
|
import binascii
|
|
import httpx
|
|
import traceback
|
|
from quart import Quart
|
|
import xml.etree.ElementTree as ET
|
|
from typing import Callable
|
|
from .wecomcsevent import WecomCSEvent
|
|
import langbot_plugin.api.entities.builtin.platform.message as platform_message
|
|
import aiofiles
|
|
import time
|
|
|
|
|
|
class WecomCSClient:
|
|
def __init__(
|
|
self,
|
|
corpid: str,
|
|
secret: str,
|
|
token: str,
|
|
EncodingAESKey: str,
|
|
logger: None,
|
|
unified_mode: bool = False,
|
|
api_base_url: str = 'https://qyapi.weixin.qq.com/cgi-bin',
|
|
):
|
|
self.corpid = corpid
|
|
self.secret = secret
|
|
self.access_token_for_contacts = ''
|
|
self.token = token
|
|
self.aes = EncodingAESKey
|
|
self.base_url = api_base_url
|
|
self.access_token = ''
|
|
self.logger = logger
|
|
self.unified_mode = unified_mode
|
|
self.app = Quart(__name__)
|
|
|
|
# Customer info cache: {external_userid: (info_dict, timestamp)}
|
|
self._customer_cache: dict[str, tuple[dict, float]] = {}
|
|
self._cache_ttl = 60 # Cache TTL in seconds (1 minute)
|
|
|
|
# 只有在非统一模式下才注册独立路由
|
|
if not self.unified_mode:
|
|
self.app.add_url_rule(
|
|
'/callback/command', 'handle_callback', self.handle_callback_request, methods=['GET', 'POST']
|
|
)
|
|
|
|
self._message_handlers = {
|
|
'example': [],
|
|
}
|
|
|
|
async def get_pic_url(self, media_id: str):
|
|
if not await self.check_access_token():
|
|
self.access_token = await self.get_access_token(self.secret)
|
|
|
|
url = f'{self.base_url}/media/get?access_token={self.access_token}&media_id={media_id}'
|
|
|
|
async with httpx.AsyncClient() as client:
|
|
response = await client.get(url)
|
|
if response.headers.get('Content-Type', '').startswith('application/json'):
|
|
data = response.json()
|
|
if data.get('errcode') in [40014, 42001]:
|
|
self.access_token = await self.get_access_token(self.secret)
|
|
return await self.get_pic_url(media_id)
|
|
else:
|
|
raise Exception('Failed to get image: ' + str(data))
|
|
|
|
# 否则是图片,转成 base64
|
|
image_bytes = response.content
|
|
content_type = response.headers.get('Content-Type', '')
|
|
base64_str = base64.b64encode(image_bytes).decode('utf-8')
|
|
base64_str = f'data:{content_type};base64,{base64_str}'
|
|
return base64_str
|
|
|
|
# access——token操作
|
|
async def check_access_token(self):
|
|
return bool(self.access_token and self.access_token.strip())
|
|
|
|
async def check_access_token_for_contacts(self):
|
|
return bool(self.access_token_for_contacts and self.access_token_for_contacts.strip())
|
|
|
|
async def get_access_token(self, secret):
|
|
url = f'{self.base_url}/gettoken?corpid={self.corpid}&corpsecret={secret}'
|
|
async with httpx.AsyncClient() as client:
|
|
response = await client.get(url)
|
|
data = response.json()
|
|
if 'access_token' in data:
|
|
return data['access_token']
|
|
else:
|
|
raise Exception(f'未获取access token: {data}')
|
|
|
|
async def get_detailed_message_list(self, xml_msg: str):
|
|
# 在本方法中解析消息,并且获得消息的具体内容
|
|
if isinstance(xml_msg, bytes):
|
|
xml_msg = xml_msg.decode('utf-8')
|
|
root = ET.fromstring(xml_msg)
|
|
token = root.find('Token').text
|
|
open_kfid = root.find('OpenKfId').text
|
|
|
|
# if open_kfid in self.openkfid_list:
|
|
# return None
|
|
# else:
|
|
# self.openkfid_list.append(open_kfid)
|
|
|
|
if not await self.check_access_token():
|
|
self.access_token = await self.get_access_token(self.secret)
|
|
|
|
url = self.base_url + '/kf/sync_msg?access_token=' + self.access_token
|
|
async with httpx.AsyncClient() as client:
|
|
params = {
|
|
'token': token,
|
|
'voice_format': 0,
|
|
'open_kfid': open_kfid,
|
|
}
|
|
response = await client.post(url, json=params)
|
|
data = response.json()
|
|
if data['errcode'] == 40014 or data['errcode'] == 42001:
|
|
self.access_token = await self.get_access_token(self.secret)
|
|
return await self.get_detailed_message_list(xml_msg)
|
|
if data['errcode'] != 0:
|
|
raise Exception('Failed to get message')
|
|
|
|
last_msg_data = data['msg_list'][-1]
|
|
open_kfid = last_msg_data.get('open_kfid')
|
|
# 进行获取图片操作
|
|
if last_msg_data.get('msgtype') == 'image':
|
|
media_id = last_msg_data.get('image').get('media_id')
|
|
picurl = await self.get_pic_url(media_id)
|
|
last_msg_data['picurl'] = picurl
|
|
# await self.change_service_status(userid=external_userid,openkfid=open_kfid,servicer=servicer)
|
|
return last_msg_data
|
|
|
|
async def change_service_status(self, userid: str, openkfid: str, servicer: str):
|
|
if not await self.check_access_token():
|
|
self.access_token = await self.get_access_token(self.secret)
|
|
url = self.base_url + '/kf/service_state/get?access_token=' + self.access_token
|
|
async with httpx.AsyncClient() as client:
|
|
params = {
|
|
'open_kfid': openkfid,
|
|
'external_userid': userid,
|
|
'service_state': 1,
|
|
'servicer_userid': servicer,
|
|
}
|
|
response = await client.post(url, json=params)
|
|
data = response.json()
|
|
if data['errcode'] == 40014 or data['errcode'] == 42001:
|
|
self.access_token = await self.get_access_token(self.secret)
|
|
return await self.change_service_status(userid, openkfid)
|
|
if data['errcode'] != 0:
|
|
raise Exception('Failed to change service status: ' + str(data))
|
|
|
|
async def send_image(self, user_id: str, agent_id: int, media_id: str):
|
|
if not await self.check_access_token():
|
|
self.access_token = await self.get_access_token(self.secret)
|
|
url = self.base_url + '/media/upload?access_token=' + self.access_token
|
|
async with httpx.AsyncClient() as client:
|
|
params = {
|
|
'touser': user_id,
|
|
'toparty': '',
|
|
'totag': '',
|
|
'agentid': agent_id,
|
|
'msgtype': 'image',
|
|
'image': {
|
|
'media_id': media_id,
|
|
},
|
|
'safe': 0,
|
|
'enable_id_trans': 0,
|
|
'enable_duplicate_check': 0,
|
|
'duplicate_check_interval': 1800,
|
|
}
|
|
try:
|
|
response = await client.post(url, json=params)
|
|
data = response.json()
|
|
except Exception as e:
|
|
raise Exception('Failed to send image: ' + str(e))
|
|
|
|
# 企业微信错误码40014和42001,代表accesstoken问题
|
|
if data['errcode'] == 40014 or data['errcode'] == 42001:
|
|
self.access_token = await self.get_access_token(self.secret)
|
|
return await self.send_image(user_id, agent_id, media_id)
|
|
|
|
if data['errcode'] != 0:
|
|
raise Exception('Failed to send image: ' + str(data))
|
|
|
|
async def send_text_msg(self, open_kfid: str, external_userid: str, msgid: str, content: str):
|
|
if not await self.check_access_token():
|
|
self.access_token = await self.get_access_token(self.secret)
|
|
|
|
url = f'{self.base_url}/kf/send_msg?access_token={self.access_token}'
|
|
|
|
payload = {
|
|
'touser': external_userid,
|
|
'open_kfid': open_kfid,
|
|
'msgid': msgid,
|
|
'msgtype': 'text',
|
|
'text': {
|
|
'content': content,
|
|
},
|
|
}
|
|
|
|
async with httpx.AsyncClient() as client:
|
|
response = await client.post(url, json=payload)
|
|
|
|
data = response.json()
|
|
if data['errcode'] == 40014 or data['errcode'] == 42001:
|
|
self.access_token = await self.get_access_token(self.secret)
|
|
return await self.send_text_msg(open_kfid, external_userid, msgid, content)
|
|
if data['errcode'] != 0:
|
|
await self.logger.error(f'发送消息失败:{data}')
|
|
raise Exception('Failed to send message')
|
|
return data
|
|
|
|
async def handle_callback_request(self):
|
|
"""处理回调请求(独立端口模式,使用全局 request)。"""
|
|
return await self._handle_callback_internal(request)
|
|
|
|
async def handle_unified_webhook(self, req):
|
|
"""处理回调请求(统一 webhook 模式,显式传递 request)。
|
|
|
|
Args:
|
|
req: Quart Request 对象
|
|
|
|
Returns:
|
|
响应数据
|
|
"""
|
|
return await self._handle_callback_internal(req)
|
|
|
|
async def _handle_callback_internal(self, req):
|
|
"""
|
|
处理回调请求的内部实现,包括 GET 验证和 POST 消息接收。
|
|
|
|
Args:
|
|
req: Quart Request 对象
|
|
"""
|
|
try:
|
|
msg_signature = req.args.get('msg_signature')
|
|
timestamp = req.args.get('timestamp')
|
|
nonce = req.args.get('nonce')
|
|
try:
|
|
wxcpt = WXBizMsgCrypt(self.token, self.aes, self.corpid)
|
|
except Exception as e:
|
|
raise Exception(f'初始化失败,错误码: {e}')
|
|
|
|
if req.method == 'GET':
|
|
echostr = req.args.get('echostr')
|
|
ret, reply_echo_str = wxcpt.VerifyURL(msg_signature, timestamp, nonce, echostr)
|
|
if ret != 0:
|
|
raise Exception(f'验证失败,错误码: {ret}')
|
|
return reply_echo_str
|
|
|
|
elif req.method == 'POST':
|
|
encrypt_msg = await req.data
|
|
ret, xml_msg = wxcpt.DecryptMsg(encrypt_msg, msg_signature, timestamp, nonce)
|
|
if ret != 0:
|
|
raise Exception(f'消息解密失败,错误码: {ret}')
|
|
|
|
# 解析消息并处理
|
|
message_data = await self.get_detailed_message_list(xml_msg)
|
|
if message_data is not None:
|
|
event = WecomCSEvent.from_payload(message_data)
|
|
if event:
|
|
await self._handle_message(event)
|
|
|
|
return 'success'
|
|
except Exception as e:
|
|
if self.logger:
|
|
await self.logger.error(f'Error in handle_callback_request: {traceback.format_exc()}')
|
|
else:
|
|
traceback.print_exc()
|
|
return f'Error processing request: {str(e)}', 400
|
|
|
|
async def run_task(self, host: str, port: int, *args, **kwargs):
|
|
"""
|
|
启动 Quart 应用。
|
|
"""
|
|
await self.app.run_task(host=host, port=port, *args, **kwargs)
|
|
|
|
def on_message(self, msg_type: str):
|
|
"""
|
|
注册消息类型处理器。
|
|
"""
|
|
|
|
def decorator(func: Callable[[WecomCSEvent], None]):
|
|
if msg_type not in self._message_handlers:
|
|
self._message_handlers[msg_type] = []
|
|
self._message_handlers[msg_type].append(func)
|
|
return func
|
|
|
|
return decorator
|
|
|
|
async def _handle_message(self, event: WecomCSEvent):
|
|
"""
|
|
处理消息事件。
|
|
"""
|
|
msg_type = event.type
|
|
if msg_type in self._message_handlers:
|
|
for handler in self._message_handlers[msg_type]:
|
|
await handler(event)
|
|
|
|
@staticmethod
|
|
async def get_image_type(image_bytes: bytes) -> str:
|
|
"""
|
|
通过图片的magic numbers判断图片类型
|
|
"""
|
|
magic_numbers = {
|
|
b'\xff\xd8\xff': 'jpg',
|
|
b'\x89\x50\x4e\x47': 'png',
|
|
b'\x47\x49\x46': 'gif',
|
|
b'\x42\x4d': 'bmp',
|
|
b'\x00\x00\x01\x00': 'ico',
|
|
}
|
|
|
|
for magic, ext in magic_numbers.items():
|
|
if image_bytes.startswith(magic):
|
|
return ext
|
|
return 'jpg' # 默认返回jpg
|
|
|
|
async def upload_to_work(self, image: platform_message.Image):
|
|
"""
|
|
获取 media_id
|
|
"""
|
|
if not await self.check_access_token():
|
|
self.access_token = await self.get_access_token(self.secret)
|
|
|
|
url = self.base_url + '/media/upload?access_token=' + self.access_token + '&type=file'
|
|
file_bytes = None
|
|
file_name = 'uploaded_file.txt'
|
|
|
|
# 获取文件的二进制数据
|
|
if image.path:
|
|
async with aiofiles.open(image.path, 'rb') as f:
|
|
file_bytes = await f.read()
|
|
file_name = image.path.split('/')[-1]
|
|
elif image.url:
|
|
file_bytes = await self.download_image_to_bytes(image.url)
|
|
file_name = image.url.split('/')[-1]
|
|
elif image.base64:
|
|
try:
|
|
base64_data = image.base64
|
|
if ',' in base64_data:
|
|
base64_data = base64_data.split(',', 1)[1]
|
|
padding = 4 - (len(base64_data) % 4) if len(base64_data) % 4 else 0
|
|
padded_base64 = base64_data + '=' * padding
|
|
file_bytes = base64.b64decode(padded_base64)
|
|
except binascii.Error as e:
|
|
raise ValueError(f'Invalid base64 string: {str(e)}')
|
|
else:
|
|
raise ValueError('image对象出错')
|
|
|
|
# 设置 multipart/form-data 格式的文件
|
|
boundary = '-------------------------acebdf13572468'
|
|
headers = {'Content-Type': f'multipart/form-data; boundary={boundary}'}
|
|
body = (
|
|
(
|
|
f'--{boundary}\r\n'
|
|
f'Content-Disposition: form-data; name="media"; filename="{file_name}"; filelength={len(file_bytes)}\r\n'
|
|
f'Content-Type: application/octet-stream\r\n\r\n'
|
|
).encode('utf-8')
|
|
+ file_bytes
|
|
+ f'\r\n--{boundary}--\r\n'.encode('utf-8')
|
|
)
|
|
|
|
# 上传文件
|
|
async with httpx.AsyncClient() as client:
|
|
response = await client.post(url, headers=headers, content=body)
|
|
data = response.json()
|
|
if data['errcode'] == 40014 or data['errcode'] == 42001:
|
|
self.access_token = await self.get_access_token(self.secret)
|
|
media_id = await self.upload_to_work(image)
|
|
if data.get('errcode', 0) != 0:
|
|
raise Exception('failed to upload file')
|
|
|
|
media_id = data.get('media_id')
|
|
return media_id
|
|
|
|
async def download_image_to_bytes(self, url: str) -> bytes:
|
|
async with httpx.AsyncClient() as client:
|
|
response = await client.get(url)
|
|
response.raise_for_status()
|
|
return response.content
|
|
|
|
# 进行media_id的获取
|
|
async def get_media_id(self, image: platform_message.Image):
|
|
media_id = await self.upload_to_work(image=image)
|
|
return media_id
|
|
|
|
async def get_customer_info(self, external_userid: str) -> dict | None:
|
|
"""
|
|
Get customer information by external_userid with caching.
|
|
|
|
Uses a 1-minute cache to avoid repeated API calls for the same user.
|
|
|
|
Args:
|
|
external_userid: The external user ID of the customer.
|
|
|
|
Returns:
|
|
Customer info dict with 'nickname', 'avatar', etc., or None if not found.
|
|
"""
|
|
# Check cache first
|
|
current_time = time.time()
|
|
if external_userid in self._customer_cache:
|
|
cached_info, cached_time = self._customer_cache[external_userid]
|
|
if current_time - cached_time < self._cache_ttl:
|
|
return cached_info
|
|
|
|
# Cache miss or expired, fetch from API
|
|
if not await self.check_access_token():
|
|
self.access_token = await self.get_access_token(self.secret)
|
|
|
|
url = f'{self.base_url}/kf/customer/batchget?access_token={self.access_token}'
|
|
|
|
payload = {
|
|
'external_userid_list': [external_userid],
|
|
}
|
|
|
|
async with httpx.AsyncClient() as client:
|
|
response = await client.post(url, json=payload)
|
|
data = response.json()
|
|
|
|
if data.get('errcode') in [40014, 42001]:
|
|
self.access_token = await self.get_access_token(self.secret)
|
|
return await self.get_customer_info(external_userid)
|
|
|
|
if data.get('errcode', 0) != 0:
|
|
if self.logger:
|
|
await self.logger.warning(f'Failed to get customer info: {data}')
|
|
return None
|
|
|
|
customer_list = data.get('customer_list', [])
|
|
if customer_list:
|
|
customer_info = customer_list[0]
|
|
# Store in cache
|
|
self._customer_cache[external_userid] = (customer_info, current_time)
|
|
return customer_info
|
|
return None
|