Rework the Feishu frontend so each user turn renders as a single
collapsible task card that patches itself in place, replacing the
dq-based streaming path that produced many fragmented messages.
- One _TaskCard per turn; hook reacts to summary / exit_reason events
from the agent loop and patches the same card.
- Each step is a foldable panel: header shows the summary, expanding
reveals three sections (auto-hidden when empty):
* Thinking - from response.thinking (separate field, not content)
* Tool Calls - tool name + truncated JSON args
* Output - response.content, with protocol tags stripped so
the header summary is not duplicated inside
- Final reply rendered as a schema 2.0 markdown card for consistency.
- Code-review pass per code_review_principles.md:
* _TaskCard owns only stateful card lifecycle (start/step/done/fail)
* Pure formatting extracted to module-level _build_step_detail and
_fmt_tool_call (no more reaching into card._private from the hook)
* Hook is a ~10-line dispatcher
* Flattened a 4-level nested lambda into a named function
633 lines
24 KiB
Python
633 lines
24 KiB
Python
import glob, json, os, queue as Q, re, sys, threading, time
|
||
|
||
PROJECT_ROOT = os.path.dirname(os.path.dirname(os.path.abspath(__file__)))
|
||
sys.path.insert(0, PROJECT_ROOT)
|
||
os.chdir(PROJECT_ROOT)
|
||
from agentmain import GeneraticAgent
|
||
from frontends.chatapp_common import format_restore
|
||
from frontends.continue_cmd import handle_frontend_command as handle_continue_frontend, reset_conversation
|
||
from llmcore import mykeys
|
||
|
||
import traceback
|
||
import lark_oapi as lark
|
||
from lark_oapi.api.im.v1 import *
|
||
|
||
_TAG_PATS = [r"<" + t + r">.*?</" + t + r">" for t in ("thinking", "summary", "tool_use", "file_content")]
|
||
_IMAGE_EXTS = {".png", ".jpg", ".jpeg", ".gif", ".bmp", ".webp", ".ico", ".tiff", ".tif"}
|
||
_AUDIO_EXTS = {".opus", ".mp3", ".wav", ".m4a", ".aac"}
|
||
_VIDEO_EXTS = {".mp4", ".mov", ".avi", ".mkv", ".webm"}
|
||
_FILE_TYPE_MAP = {
|
||
".opus": "opus",
|
||
".mp4": "mp4",
|
||
".pdf": "pdf",
|
||
".doc": "doc",
|
||
".docx": "doc",
|
||
".xls": "xls",
|
||
".xlsx": "xls",
|
||
".ppt": "ppt",
|
||
".pptx": "ppt",
|
||
}
|
||
_MSG_TYPE_MAP = {"image": "[image]", "audio": "[audio]", "file": "[file]", "media": "[media]", "sticker": "[sticker]"}
|
||
|
||
TEMP_DIR = os.path.join(PROJECT_ROOT, "temp")
|
||
MEDIA_DIR = os.path.join(TEMP_DIR, "feishu_media")
|
||
os.makedirs(MEDIA_DIR, exist_ok=True)
|
||
|
||
|
||
def _clean(text):
|
||
for pat in _TAG_PATS:
|
||
text = re.sub(pat, "", text, flags=re.DOTALL)
|
||
return re.sub(r"\n{3,}", "\n\n", text).strip() or "..."
|
||
|
||
|
||
def _extract_files(text):
|
||
return re.findall(r"\[FILE:([^\]]+)\]", text or "")
|
||
|
||
|
||
def _strip_files(text):
|
||
return re.sub(r"\[FILE:[^\]]+\]", "", text or "").strip()
|
||
|
||
|
||
def _display_text(text):
|
||
return _strip_files(_clean(text)) or "..."
|
||
|
||
|
||
def _to_allowed_set(value):
|
||
if value is None:
|
||
return set()
|
||
if isinstance(value, str):
|
||
value = [value]
|
||
return {str(x).strip() for x in value if str(x).strip()}
|
||
|
||
|
||
def _parse_json(raw):
|
||
if not raw:
|
||
return {}
|
||
try:
|
||
return json.loads(raw)
|
||
except Exception:
|
||
return {}
|
||
|
||
|
||
def _extract_share_card_content(content_json, msg_type):
|
||
parts = []
|
||
if msg_type == "share_chat":
|
||
parts.append(f"[shared chat: {content_json.get('chat_id', '')}]")
|
||
elif msg_type == "share_user":
|
||
parts.append(f"[shared user: {content_json.get('user_id', '')}]")
|
||
elif msg_type == "interactive":
|
||
parts.extend(_extract_interactive_content(content_json))
|
||
elif msg_type == "share_calendar_event":
|
||
parts.append(f"[shared calendar event: {content_json.get('event_key', '')}]")
|
||
elif msg_type == "system":
|
||
parts.append("[system message]")
|
||
elif msg_type == "merge_forward":
|
||
parts.append("[merged forward messages]")
|
||
return "\n".join([p for p in parts if p]).strip() or f"[{msg_type}]"
|
||
|
||
|
||
def _extract_interactive_content(content):
|
||
parts = []
|
||
if isinstance(content, str):
|
||
try:
|
||
content = json.loads(content)
|
||
except Exception:
|
||
return [content] if content.strip() else []
|
||
if not isinstance(content, dict):
|
||
return parts
|
||
title = content.get("title")
|
||
if isinstance(title, dict):
|
||
title_text = title.get("content", "") or title.get("text", "")
|
||
if title_text:
|
||
parts.append(f"title: {title_text}")
|
||
elif isinstance(title, str) and title:
|
||
parts.append(f"title: {title}")
|
||
elements = content.get("elements", [])
|
||
if isinstance(elements, list):
|
||
for row in elements:
|
||
if isinstance(row, dict):
|
||
parts.extend(_extract_element_content(row))
|
||
elif isinstance(row, list):
|
||
for el in row:
|
||
parts.extend(_extract_element_content(el))
|
||
card = content.get("card", {})
|
||
if card:
|
||
parts.extend(_extract_interactive_content(card))
|
||
header = content.get("header", {})
|
||
if isinstance(header, dict):
|
||
header_title = header.get("title", {})
|
||
if isinstance(header_title, dict):
|
||
header_text = header_title.get("content", "") or header_title.get("text", "")
|
||
if header_text:
|
||
parts.append(f"title: {header_text}")
|
||
return [p for p in parts if p]
|
||
|
||
|
||
def _extract_element_content(element):
|
||
parts = []
|
||
if not isinstance(element, dict):
|
||
return parts
|
||
tag = element.get("tag", "")
|
||
if tag in ("markdown", "lark_md"):
|
||
content = element.get("content", "")
|
||
if content:
|
||
parts.append(content)
|
||
elif tag == "div":
|
||
text = element.get("text", {})
|
||
if isinstance(text, dict):
|
||
text_content = text.get("content", "") or text.get("text", "")
|
||
if text_content:
|
||
parts.append(text_content)
|
||
elif isinstance(text, str) and text:
|
||
parts.append(text)
|
||
for field in element.get("fields", []) or []:
|
||
if isinstance(field, dict):
|
||
field_text = field.get("text", {})
|
||
if isinstance(field_text, dict):
|
||
content = field_text.get("content", "") or field_text.get("text", "")
|
||
if content:
|
||
parts.append(content)
|
||
elif tag == "a":
|
||
href = element.get("href", "")
|
||
text = element.get("text", "")
|
||
if href:
|
||
parts.append(f"link: {href}")
|
||
if text:
|
||
parts.append(text)
|
||
elif tag == "button":
|
||
text = element.get("text", {})
|
||
if isinstance(text, dict):
|
||
content = text.get("content", "") or text.get("text", "")
|
||
if content:
|
||
parts.append(content)
|
||
url = element.get("url", "") or (element.get("multi_url", {}) or {}).get("url", "")
|
||
if url:
|
||
parts.append(f"link: {url}")
|
||
elif tag == "img":
|
||
alt = element.get("alt", {})
|
||
if isinstance(alt, dict):
|
||
parts.append(alt.get("content", "[image]") or "[image]")
|
||
else:
|
||
parts.append("[image]")
|
||
for child in element.get("elements", []) or []:
|
||
parts.extend(_extract_element_content(child))
|
||
for col in element.get("columns", []) or []:
|
||
for child in (col.get("elements", []) if isinstance(col, dict) else []):
|
||
parts.extend(_extract_element_content(child))
|
||
return parts
|
||
|
||
|
||
def _extract_post_content(content_json):
|
||
def _parse_block(block):
|
||
if not isinstance(block, dict) or not isinstance(block.get("content"), list):
|
||
return None, []
|
||
texts, images = [], []
|
||
if block.get("title"):
|
||
texts.append(block.get("title"))
|
||
for row in block["content"]:
|
||
if not isinstance(row, list):
|
||
continue
|
||
for el in row:
|
||
if not isinstance(el, dict):
|
||
continue
|
||
tag = el.get("tag")
|
||
if tag in ("text", "a"):
|
||
texts.append(el.get("text", ""))
|
||
elif tag == "at":
|
||
texts.append(f"@{el.get('user_name', 'user')}")
|
||
elif tag == "img" and el.get("image_key"):
|
||
images.append(el["image_key"])
|
||
text = " ".join([t for t in texts if t]).strip()
|
||
return text or None, images
|
||
|
||
root = content_json
|
||
if isinstance(root, dict) and isinstance(root.get("post"), dict):
|
||
root = root["post"]
|
||
if not isinstance(root, dict):
|
||
return "", []
|
||
if "content" in root:
|
||
text, imgs = _parse_block(root)
|
||
if text or imgs:
|
||
return text or "", imgs
|
||
for key in ("zh_cn", "en_us", "ja_jp"):
|
||
if key in root:
|
||
text, imgs = _parse_block(root[key])
|
||
if text or imgs:
|
||
return text or "", imgs
|
||
for val in root.values():
|
||
if isinstance(val, dict):
|
||
text, imgs = _parse_block(val)
|
||
if text or imgs:
|
||
return text or "", imgs
|
||
return "", []
|
||
|
||
|
||
APP_ID = str(mykeys.get("fs_app_id", "") or "").strip()
|
||
APP_SECRET = str(mykeys.get("fs_app_secret", "") or "").strip()
|
||
ALLOWED_USERS = _to_allowed_set(mykeys.get("fs_allowed_users", []))
|
||
PUBLIC_ACCESS = not ALLOWED_USERS or "*" in ALLOWED_USERS
|
||
AGENT_TIMEOUT_SEC = 900
|
||
|
||
agent = GeneraticAgent()
|
||
threading.Thread(target=agent.run, daemon=True).start()
|
||
client, user_tasks = None, {}
|
||
|
||
|
||
def create_client():
|
||
return lark.Client.builder().app_id(APP_ID).app_secret(APP_SECRET).log_level(lark.LogLevel.INFO).build()
|
||
|
||
|
||
def _card_raw(elements):
|
||
return json.dumps({
|
||
"schema": "2.0",
|
||
"config": {"streaming_mode": False, "width_mode": "fill"},
|
||
"body": {"elements": elements},
|
||
}, ensure_ascii=False)
|
||
|
||
|
||
def _card(text):
|
||
return _card_raw([{"tag": "markdown", "content": text}])
|
||
|
||
|
||
def _send_raw(receive_id, payload, msg_type, rtype):
|
||
body = CreateMessageRequest.builder().receive_id_type(rtype).request_body(
|
||
CreateMessageRequestBody.builder().receive_id(receive_id).msg_type(msg_type).content(payload).build()
|
||
).build()
|
||
r = client.im.v1.message.create(body)
|
||
if r.success():
|
||
return r.data.message_id if r.data else None
|
||
print(f"发送失败: {r.code}, {r.msg}")
|
||
return None
|
||
|
||
|
||
def _patch_card(message_id, card_json):
|
||
body = PatchMessageRequest.builder().message_id(message_id).request_body(
|
||
PatchMessageRequestBody.builder().content(card_json).build()
|
||
).build()
|
||
r = client.im.v1.message.patch(body)
|
||
if not r.success():
|
||
print(f"[ERROR] patch_card 失败: {r.code}, {r.msg}")
|
||
return r.success()
|
||
|
||
|
||
def send_message(receive_id, content, msg_type="text", use_card=False, receive_id_type="open_id"):
|
||
if use_card:
|
||
return _send_raw(receive_id, _card(content), "interactive", receive_id_type)
|
||
if msg_type == "text":
|
||
return _send_raw(receive_id, json.dumps({"text": content}, ensure_ascii=False), "text", receive_id_type)
|
||
return _send_raw(receive_id, content, msg_type, receive_id_type)
|
||
|
||
|
||
def update_message(message_id, content):
|
||
return _patch_card(message_id, _card(content))
|
||
|
||
|
||
def _upload_image_sync(file_path):
|
||
try:
|
||
with open(file_path, "rb") as f:
|
||
request = CreateImageRequest.builder().request_body(
|
||
CreateImageRequestBody.builder().image_type("message").image(f).build()
|
||
).build()
|
||
response = client.im.v1.image.create(request)
|
||
if response.success():
|
||
return response.data.image_key
|
||
print(f"[ERROR] upload image failed: {response.code}, {response.msg}")
|
||
except Exception as e:
|
||
print(f"[ERROR] upload image failed {file_path}: {e}")
|
||
return None
|
||
|
||
|
||
def _upload_file_sync(file_path):
|
||
ext = os.path.splitext(file_path)[1].lower()
|
||
file_type = _FILE_TYPE_MAP.get(ext, "stream")
|
||
file_name = os.path.basename(file_path)
|
||
try:
|
||
with open(file_path, "rb") as f:
|
||
request = CreateFileRequest.builder().request_body(
|
||
CreateFileRequestBody.builder().file_type(file_type).file_name(file_name).file(f).build()
|
||
).build()
|
||
response = client.im.v1.file.create(request)
|
||
if response.success():
|
||
return response.data.file_key
|
||
print(f"[ERROR] upload file failed: {response.code}, {response.msg}")
|
||
except Exception as e:
|
||
print(f"[ERROR] upload file failed {file_path}: {e}")
|
||
return None
|
||
|
||
|
||
def _download_image_sync(message_id, image_key):
|
||
try:
|
||
request = GetMessageResourceRequest.builder().message_id(message_id).file_key(image_key).type("image").build()
|
||
response = client.im.v1.message_resource.get(request)
|
||
if response.success():
|
||
data = response.file.read() if hasattr(response.file, "read") else response.file
|
||
return data, response.file_name
|
||
print(f"[ERROR] download image failed: {response.code}, {response.msg}")
|
||
except Exception as e:
|
||
print(f"[ERROR] download image failed {image_key}: {e}")
|
||
return None, None
|
||
|
||
|
||
def _download_file_sync(message_id, file_key, resource_type="file"):
|
||
if resource_type == "audio":
|
||
resource_type = "file"
|
||
try:
|
||
request = GetMessageResourceRequest.builder().message_id(message_id).file_key(file_key).type(resource_type).build()
|
||
response = client.im.v1.message_resource.get(request)
|
||
if response.success():
|
||
data = response.file.read() if hasattr(response.file, "read") else response.file
|
||
return data, response.file_name
|
||
print(f"[ERROR] download {resource_type} failed: {response.code}, {response.msg}")
|
||
except Exception as e:
|
||
print(f"[ERROR] download {resource_type} failed {file_key}: {e}")
|
||
return None, None
|
||
|
||
|
||
def _download_and_save_media(msg_type, content_json, message_id):
|
||
data, filename = None, None
|
||
if msg_type == "image":
|
||
image_key = content_json.get("image_key")
|
||
if image_key and message_id:
|
||
data, filename = _download_image_sync(message_id, image_key)
|
||
if not filename:
|
||
filename = f"{image_key[:16]}.jpg"
|
||
elif msg_type in ("audio", "file", "media"):
|
||
file_key = content_json.get("file_key")
|
||
if file_key and message_id:
|
||
data, filename = _download_file_sync(message_id, file_key, msg_type)
|
||
if not filename:
|
||
filename = file_key[:16]
|
||
if msg_type == "audio" and filename and not filename.endswith(".opus"):
|
||
filename = f"{filename}.opus"
|
||
if data and filename:
|
||
file_path = os.path.join(MEDIA_DIR, os.path.basename(filename))
|
||
with open(file_path, "wb") as f:
|
||
f.write(data)
|
||
return file_path, filename
|
||
return None, None
|
||
|
||
|
||
def _describe_media(msg_type, file_path, filename):
|
||
if msg_type == "image":
|
||
return f"[image: {filename}]\n[Image: source: {file_path}]"
|
||
if msg_type == "audio":
|
||
return f"[audio: {filename}]\n[File: source: {file_path}]"
|
||
if msg_type in ("file", "media"):
|
||
return f"[{msg_type}: {filename}]\n[File: source: {file_path}]"
|
||
return f"[{msg_type}]\n[File: source: {file_path}]"
|
||
|
||
|
||
def _send_local_file(receive_id, file_path, receive_id_type="open_id"):
|
||
if not os.path.isfile(file_path):
|
||
send_message(receive_id, f"⚠️ 文件不存在: {file_path}", receive_id_type=receive_id_type)
|
||
return False
|
||
ext = os.path.splitext(file_path)[1].lower()
|
||
if ext in _IMAGE_EXTS:
|
||
image_key = _upload_image_sync(file_path)
|
||
if image_key:
|
||
send_message(receive_id, json.dumps({"image_key": image_key}, ensure_ascii=False), msg_type="image", receive_id_type=receive_id_type)
|
||
return True
|
||
else:
|
||
file_key = _upload_file_sync(file_path)
|
||
if file_key:
|
||
msg_type = "media" if ext in _AUDIO_EXTS or ext in _VIDEO_EXTS else "file"
|
||
send_message(receive_id, json.dumps({"file_key": file_key}, ensure_ascii=False), msg_type=msg_type, receive_id_type=receive_id_type)
|
||
return True
|
||
send_message(receive_id, f"⚠️ 文件发送失败: {os.path.basename(file_path)}", receive_id_type=receive_id_type)
|
||
return False
|
||
|
||
|
||
def _send_generated_files(receive_id, raw_text, receive_id_type="open_id"):
|
||
for file_path in _extract_files(raw_text):
|
||
_send_local_file(receive_id, file_path, receive_id_type)
|
||
|
||
|
||
def _build_user_message(message):
|
||
msg_type = message.message_type
|
||
message_id = message.message_id
|
||
content_json = _parse_json(message.content)
|
||
parts, image_paths = [], []
|
||
if msg_type == "text":
|
||
text = str(content_json.get("text", "") or "").strip()
|
||
if text:
|
||
parts.append(text)
|
||
elif msg_type == "post":
|
||
text, image_keys = _extract_post_content(content_json)
|
||
if text:
|
||
parts.append(text)
|
||
for image_key in image_keys:
|
||
file_path, filename = _download_and_save_media("image", {"image_key": image_key}, message_id)
|
||
if file_path and filename:
|
||
parts.append(_describe_media("image", file_path, filename))
|
||
image_paths.append(file_path)
|
||
else:
|
||
parts.append("[image: download failed]")
|
||
elif msg_type in ("image", "audio", "file", "media"):
|
||
file_path, filename = _download_and_save_media(msg_type, content_json, message_id)
|
||
if file_path and filename:
|
||
parts.append(_describe_media(msg_type, file_path, filename))
|
||
if msg_type == "image":
|
||
image_paths.append(file_path)
|
||
else:
|
||
parts.append(f"[{msg_type}: download failed]")
|
||
elif msg_type in ("share_chat", "share_user", "interactive", "share_calendar_event", "system", "merge_forward"):
|
||
parts.append(_extract_share_card_content(content_json, msg_type))
|
||
else:
|
||
parts.append(_MSG_TYPE_MAP.get(msg_type, f"[{msg_type}]"))
|
||
return "\n".join([p for p in parts if p]).strip(), image_paths
|
||
|
||
|
||
def _fmt_tool_call(tc):
|
||
name = tc.get('tool_name', '?')
|
||
args = {k: v for k, v in (tc.get('args') or {}).items() if not k.startswith('_')}
|
||
return f"- `{name}`({json.dumps(args, ensure_ascii=False)[:200]})"
|
||
|
||
|
||
def _build_step_detail(resp, tool_calls):
|
||
"""从 LLM response + tool_calls 组装单步展开详情(纯函数)。"""
|
||
parts = []
|
||
thinking = (getattr(resp, 'thinking', '') or '').strip() if resp else ''
|
||
if thinking:
|
||
parts.append(f"### 💭 Thinking\n{thinking}")
|
||
if tool_calls:
|
||
parts.append("### 🛠 Tool Calls\n" + "\n".join(_fmt_tool_call(tc) for tc in tool_calls))
|
||
content = _display_text((getattr(resp, 'content', '') or '')).strip() if resp else ''
|
||
if content and content != '...':
|
||
parts.append(f"### 📝 Output\n{content}")
|
||
return "\n\n".join(parts)
|
||
|
||
|
||
class _TaskCard:
|
||
"""飞书任务卡片:单卡片持续 patch;每步一个独立折叠面板(header 显示 summary,展开看详情)。"""
|
||
_DETAIL_LIMIT = 8000
|
||
|
||
def __init__(self, receive_id, rid_type):
|
||
self.rid, self.rtype = receive_id, rid_type
|
||
self.steps = [] # [(summary, detail), ...]
|
||
self.status = "🤔 思考中..."
|
||
self.final = None
|
||
self.msg_id = None
|
||
|
||
def _step_panel(self, idx, summary, detail):
|
||
detail = detail or "_(无输出)_"
|
||
if len(detail) > self._DETAIL_LIMIT:
|
||
detail = detail[:self._DETAIL_LIMIT] + f"\n\n…(已截断,共 {len(detail)} 字符)"
|
||
return {
|
||
"tag": "collapsible_panel", "expanded": False,
|
||
"header": {"title": {"tag": "plain_text", "content": f"Turn {idx} · {summary}"}},
|
||
"elements": [{"tag": "markdown", "content": detail}],
|
||
}
|
||
|
||
def _build(self):
|
||
els = [{"tag": "markdown", "content": f"**{self.status}**"}]
|
||
for i, (s, d) in enumerate(self.steps, 1):
|
||
els.append(self._step_panel(i, s, d))
|
||
if self.final:
|
||
els += [{"tag": "hr"}, {"tag": "markdown", "content": self.final}]
|
||
return _card_raw(els)
|
||
|
||
def _push(self):
|
||
card = self._build()
|
||
if self.msg_id:
|
||
_patch_card(self.msg_id, card)
|
||
else:
|
||
self.msg_id = _send_raw(self.rid, card, "interactive", self.rtype)
|
||
|
||
# ── 公开接口 ──
|
||
|
||
def start(self):
|
||
self._push()
|
||
|
||
def step(self, summary, detail=""):
|
||
self.steps.append((summary, detail))
|
||
self.status = f"⏳ 工作中 · Turn {len(self.steps)}"
|
||
self._push()
|
||
|
||
def done(self, text):
|
||
self.status = "✅ 已完成"
|
||
self.final = text or "_(无文本输出)_"
|
||
self._push()
|
||
|
||
def fail(self, msg):
|
||
self.status = f"❌ {msg}"
|
||
self._push()
|
||
|
||
|
||
def _make_task_hook(card, done_event, on_final):
|
||
"""飞书任务 hook:每轮 patch 卡片状态;结束触发 on_final(raw) 处理附件。"""
|
||
def hook(ctx):
|
||
try:
|
||
if ctx.get('exit_reason'):
|
||
resp = ctx.get('response')
|
||
raw = resp.content if hasattr(resp, 'content') else str(resp)
|
||
card.done(_display_text(raw))
|
||
on_final(raw)
|
||
done_event.set()
|
||
elif ctx.get('summary'):
|
||
detail = _build_step_detail(ctx.get('response'), ctx.get('tool_calls') or [])
|
||
card.step(ctx['summary'], detail)
|
||
except Exception as e:
|
||
print(f"[fs hook] error: {e}")
|
||
return hook
|
||
|
||
|
||
def handle_message(data):
|
||
event, message, sender = data.event, data.event.message, data.event.sender
|
||
open_id = sender.sender_id.open_id
|
||
chat_id = message.chat_id
|
||
if not PUBLIC_ACCESS and open_id not in ALLOWED_USERS:
|
||
print(f"未授权用户: {open_id}")
|
||
return
|
||
user_input, image_paths = _build_user_message(message)
|
||
if not user_input:
|
||
if chat_id:
|
||
send_message(chat_id, f"⚠️ 暂不支持处理此类飞书消息:{message.message_type}", receive_id_type="chat_id")
|
||
else:
|
||
send_message(open_id, f"⚠️ 暂不支持处理此类飞书消息:{message.message_type}")
|
||
return
|
||
print(f"收到消息 [{open_id}] ({message.message_type}, {len(image_paths)} images): {user_input[:200]}")
|
||
if message.message_type == "text" and user_input.startswith("/"):
|
||
return handle_command(open_id, user_input, chat_id)
|
||
|
||
def run_agent():
|
||
user_tasks[open_id] = {"running": True}
|
||
receive_id = chat_id or open_id
|
||
rid_type = "chat_id" if chat_id else "open_id"
|
||
done_event = threading.Event()
|
||
hook_key = f"fs_{open_id}"
|
||
card = _TaskCard(receive_id, rid_type)
|
||
card.start()
|
||
on_final = lambda raw: _send_generated_files(receive_id, raw, receive_id_type=rid_type)
|
||
if not hasattr(agent, '_turn_end_hooks'): agent._turn_end_hooks = {}
|
||
agent._turn_end_hooks[hook_key] = _make_task_hook(card, done_event, on_final)
|
||
try:
|
||
agent.put_task(user_input, source="feishu", images=image_paths)
|
||
start = time.time()
|
||
while not done_event.wait(timeout=3):
|
||
if not user_tasks.get(open_id, {}).get("running", True):
|
||
agent.abort()
|
||
card.fail("已停止")
|
||
break
|
||
if time.time() - start > AGENT_TIMEOUT_SEC:
|
||
agent.abort()
|
||
card.fail("任务超时")
|
||
break
|
||
except Exception as e:
|
||
traceback.print_exc()
|
||
card.fail(f"错误: {e}")
|
||
finally:
|
||
agent._turn_end_hooks.pop(hook_key, None)
|
||
user_tasks.pop(open_id, None)
|
||
|
||
threading.Thread(target=run_agent, daemon=True).start()
|
||
|
||
|
||
def handle_command(open_id, cmd, chat_id=None):
|
||
def _send_cmd_response(content):
|
||
if chat_id:
|
||
send_message(chat_id, content, receive_id_type="chat_id")
|
||
else:
|
||
send_message(open_id, content)
|
||
if cmd == "/stop":
|
||
if open_id in user_tasks:
|
||
user_tasks[open_id]["running"] = False
|
||
agent.abort()
|
||
_send_cmd_response("正在停止...")
|
||
elif cmd == "/new":
|
||
_send_cmd_response(reset_conversation(agent))
|
||
elif cmd == "/help":
|
||
_send_cmd_response("命令列表:\n/stop - 停止当前任务\n/status - 查看状态\n/restore - 恢复上次对话历史\n/continue - 列出可恢复会话\n/continue [n] - 恢复第 n 个会话\n/new - 开启新对话并清空当前上下文\n/help - 显示帮助")
|
||
elif cmd == "/status":
|
||
_send_cmd_response(f"状态: {'空闲' if not agent.is_running else '运行中'}")
|
||
elif cmd == "/restore":
|
||
try:
|
||
restored_info, err = format_restore()
|
||
if err:
|
||
return _send_cmd_response(err.replace("❌ ", ""))
|
||
restored, fname, count = restored_info
|
||
agent.history.extend(restored)
|
||
agent.abort()
|
||
_send_cmd_response(f"已恢复 {count} 轮对话\n来源: {fname}\n(仅恢复上下文,请输入新问题继续)")
|
||
except Exception as e:
|
||
_send_cmd_response(f"恢复失败: {e}")
|
||
elif cmd.startswith("/continue"):
|
||
_send_cmd_response(handle_continue_frontend(agent, cmd))
|
||
else:
|
||
_send_cmd_response(f"未知命令: {cmd}")
|
||
|
||
|
||
def main():
|
||
global client
|
||
if not APP_ID or not APP_SECRET:
|
||
print("错误: 请在 mykey.py 或 mykey.json 中配置 fs_app_id 和 fs_app_secret")
|
||
sys.exit(1)
|
||
client = create_client()
|
||
handler = lark.EventDispatcherHandler.builder("", "").register_p2_im_message_receive_v1(handle_message).build()
|
||
cli = lark.ws.Client(APP_ID, APP_SECRET, event_handler=handler, log_level=lark.LogLevel.INFO)
|
||
print("=" * 50 + "\n飞书 Agent 已启动(长连接模式)\n" + f"App ID: {APP_ID}\n等待消息...\n" + "=" * 50)
|
||
cli.start()
|
||
|
||
|
||
if __name__ == "__main__":
|
||
main()
|