添加了 `revise_task` 路由动作类型,允许用户修改、纠正或重新执行最新活动任务结果。 实现了工具失败指导原则,防止相同类别工具重复失败。 为任务规划器添加了超时处理机制,避免长时间等待。 BREAKING CHANGE: 任务路由逻辑已更新,新增 `revise_task` 动作类型。 fix(api): 修复任务详情API返回完整流程投影 修复了任务详情API端点,现在会包含过滤后的流程运行、事件和工件信息, 并确保时间戳字段正确序列化。 refactor(engine): 优化任务技能解析器摘要节点处理 改进了任务技能解析器对摘要节点的处理逻辑,对于仅依赖文本生成功能的摘要节 点不再分配具体技能,直接使用依赖项输出进行汇总。 test: 增加任务修订和超时处理测试用例 添加了测试用例验证任务修订输入记录反馈、超时回退到单模式以及 摘要节点技能解析等新功能。
159 lines
5.9 KiB
Python
159 lines
5.9 KiB
Python
from __future__ import annotations
|
|
|
|
from pathlib import Path
|
|
|
|
from fastapi.testclient import TestClient
|
|
|
|
from beaver.interfaces.web.app import create_app
|
|
from beaver.services.agent_service import AgentService
|
|
|
|
|
|
def test_active_task_api_returns_open_task_and_hides_closed(tmp_path: Path) -> None:
|
|
service = AgentService(workspace=tmp_path)
|
|
loaded = service.create_loop().boot()
|
|
task = loaded.task_service.create_task( # type: ignore[union-attr]
|
|
session_id="web:active",
|
|
description="实现任务连续性",
|
|
metadata={"short_title": "任务连续性"},
|
|
)
|
|
app = create_app(service=service, manage_service_lifecycle=False)
|
|
|
|
with TestClient(app) as client:
|
|
active = client.get("/api/sessions/web:active/active-task")
|
|
listed = client.get("/api/tasks")
|
|
loaded.task_service.close_task(task.task_id, reason="done") # type: ignore[union-attr]
|
|
inactive = client.get("/api/sessions/web:active/active-task")
|
|
|
|
assert active.status_code == 200
|
|
assert active.json()["task_id"] == task.task_id
|
|
assert active.json()["short_title"] == "任务连续性"
|
|
assert listed.json()[0]["short_title"] == "任务连续性"
|
|
assert inactive.status_code == 200
|
|
assert inactive.json() is None
|
|
|
|
|
|
def test_active_task_api_hides_unengaged_cron_task(tmp_path: Path) -> None:
|
|
service = AgentService(workspace=tmp_path)
|
|
loaded = service.create_loop().boot()
|
|
hidden = loaded.task_service.create_task( # type: ignore[union-attr]
|
|
session_id="web:cron",
|
|
description="提醒用户喝水",
|
|
creator="cron",
|
|
metadata={"source": "scheduled_cron", "user_engaged": False},
|
|
)
|
|
visible = loaded.task_service.create_task( # type: ignore[union-attr]
|
|
session_id="web:engaged",
|
|
description="修改新闻总结",
|
|
creator="cron",
|
|
metadata={"source": "scheduled_run", "user_engaged": True},
|
|
)
|
|
app = create_app(service=service, manage_service_lifecycle=False)
|
|
|
|
with TestClient(app) as client:
|
|
hidden_response = client.get("/api/sessions/web:cron/active-task")
|
|
visible_response = client.get("/api/sessions/web:engaged/active-task")
|
|
|
|
assert hidden_response.status_code == 200
|
|
assert hidden_response.json() is None
|
|
assert visible_response.status_code == 200
|
|
assert visible_response.json()["task_id"] == visible.task_id
|
|
assert hidden.task_id != visible.task_id
|
|
|
|
|
|
def test_task_delete_api_removes_backend_task(tmp_path: Path) -> None:
|
|
service = AgentService(workspace=tmp_path)
|
|
loaded = service.create_loop().boot()
|
|
task = loaded.task_service.create_task( # type: ignore[union-attr]
|
|
session_id="web:delete",
|
|
description="删除这个任务",
|
|
)
|
|
app = create_app(service=service, manage_service_lifecycle=False)
|
|
|
|
with TestClient(app) as client:
|
|
deleted = client.delete(f"/api/tasks/{task.task_id}")
|
|
listed = client.get("/api/tasks")
|
|
missing = client.get(f"/api/tasks/{task.task_id}")
|
|
|
|
assert deleted.status_code == 200
|
|
assert deleted.json()["task_id"] == task.task_id
|
|
assert all(item["task_id"] != task.task_id for item in listed.json())
|
|
assert missing.status_code == 404
|
|
|
|
|
|
def test_task_detail_api_includes_filtered_process_projection(tmp_path: Path) -> None:
|
|
service = AgentService(workspace=tmp_path)
|
|
loaded = service.create_loop().boot()
|
|
task = loaded.task_service.create_task( # type: ignore[union-attr]
|
|
session_id="web:detail",
|
|
description="补充赛事数据",
|
|
)
|
|
other_task = loaded.task_service.create_task( # type: ignore[union-attr]
|
|
session_id="web:detail",
|
|
description="不相关任务",
|
|
)
|
|
loaded.session_manager.append_message(
|
|
"web:detail",
|
|
role="system",
|
|
event_type="task_execution_planned",
|
|
event_payload={
|
|
"task_id": task.task_id,
|
|
"attempt_index": 2,
|
|
"plan_mode": "team",
|
|
"strategy": "parallel",
|
|
"node_ids": ["search_match_result", "search_match_stats"],
|
|
"reason": "needs separate evidence gathering",
|
|
},
|
|
context_visible=False,
|
|
)
|
|
loaded.session_manager.append_message(
|
|
"web:detail",
|
|
role="system",
|
|
event_type="task_team_run_failed",
|
|
event_payload={
|
|
"task_id": task.task_id,
|
|
"attempt_index": 2,
|
|
"plan_mode": "team",
|
|
"strategy": "parallel",
|
|
"team_success": False,
|
|
"team_run_ids": ["sub-run"],
|
|
"node_results": [
|
|
{
|
|
"node_id": "search_match_stats",
|
|
"success": False,
|
|
"output_text": "",
|
|
"run_id": "sub-run",
|
|
"finish_reason": "max_tool_iterations",
|
|
"error": "max_tool_iterations",
|
|
}
|
|
],
|
|
"error": "one or more team nodes failed",
|
|
},
|
|
context_visible=False,
|
|
)
|
|
loaded.session_manager.append_message(
|
|
"web:detail",
|
|
role="system",
|
|
event_type="task_execution_planned",
|
|
event_payload={
|
|
"task_id": other_task.task_id,
|
|
"attempt_index": 1,
|
|
"plan_mode": "single",
|
|
"strategy": None,
|
|
"node_ids": [],
|
|
},
|
|
context_visible=False,
|
|
)
|
|
app = create_app(service=service, manage_service_lifecycle=False)
|
|
|
|
with TestClient(app) as client:
|
|
response = client.get(f"/api/tasks/{task.task_id}")
|
|
|
|
assert response.status_code == 200
|
|
payload = response.json()
|
|
assert [run["run_id"] for run in payload["process_runs"]] == [
|
|
f"task:{task.task_id}:attempt:2",
|
|
"sub-run",
|
|
]
|
|
assert {event["actor_name"] for event in payload["process_events"]} == {"Task Planner", "Task Team", "search_match_stats"}
|
|
assert all(event["metadata"]["task_id"] == task.task_id for event in payload["process_events"])
|