- 支持 tool.call 和 tool.result 消息类型处理 - 引入 Tool 调度与执行逻辑,支持超时与结果截断 - 增加 ToolRegistry 和 ToolExecutor 管理工具定义与执行 - 更新上下文构建与消息映射逻辑,适配工具闭环处理 - 扩展配置与环境变量,支持 Tool 调用相关选项 - 增强单元测试覆盖工具调用与执行情景 - 更新文档和 OpenAPI,新增工具相关说明与模型定义
129 lines
4.3 KiB
PHP
129 lines
4.3 KiB
PHP
<?php
|
|
|
|
namespace App\Services;
|
|
|
|
use App\Models\Message;
|
|
use App\Services\Tool\ToolCall;
|
|
use App\Services\Tool\ToolResult;
|
|
|
|
class OutputSink
|
|
{
|
|
public function __construct(private readonly ChatService $chatService)
|
|
{
|
|
}
|
|
|
|
/**
|
|
* @param array<string, mixed> $meta
|
|
*/
|
|
public function appendAgentMessage(string $sessionId, string $runId, string $content, array $meta = [], ?string $dedupeKey = null): Message
|
|
{
|
|
$dedupeKey = $dedupeKey ?? "run:{$runId}:agent:message";
|
|
|
|
return $this->chatService->appendMessage([
|
|
'session_id' => $sessionId,
|
|
'role' => Message::ROLE_AGENT,
|
|
'type' => 'agent.message',
|
|
'content' => $content,
|
|
'payload' => array_merge($meta, ['run_id' => $runId]),
|
|
'dedupe_key' => $dedupeKey,
|
|
]);
|
|
}
|
|
|
|
/**
|
|
* @param array<string, mixed> $meta
|
|
*/
|
|
public function appendAgentDelta(string $sessionId, string $runId, string $content, int $deltaIndex, array $meta = []): Message
|
|
{
|
|
$dedupeKey = "run:{$runId}:agent:delta:{$deltaIndex}";
|
|
|
|
return $this->chatService->appendMessage([
|
|
'session_id' => $sessionId,
|
|
'role' => Message::ROLE_AGENT,
|
|
'type' => 'message.delta',
|
|
'content' => $content,
|
|
'payload' => array_merge($meta, [
|
|
'run_id' => $runId,
|
|
'delta_index' => $deltaIndex,
|
|
]),
|
|
'dedupe_key' => $dedupeKey,
|
|
], $wasDeduped);
|
|
}
|
|
|
|
/**
|
|
* @param array<string, mixed> $meta
|
|
*/
|
|
public function appendRunStatus(string $sessionId, string $runId, string $status, array $meta = [], ?bool &$wasDeduped = null): Message
|
|
{
|
|
$dedupeKey = $meta['dedupe_key'] ?? null;
|
|
unset($meta['dedupe_key']);
|
|
|
|
return $this->chatService->appendMessage([
|
|
'session_id' => $sessionId,
|
|
'role' => Message::ROLE_SYSTEM,
|
|
'type' => 'run.status',
|
|
'content' => null,
|
|
'payload' => array_merge($meta, [
|
|
'run_id' => $runId,
|
|
'status' => $status,
|
|
]),
|
|
'dedupe_key' => $dedupeKey,
|
|
], $wasDeduped);
|
|
}
|
|
|
|
/**
|
|
* @param array<string, mixed> $meta
|
|
*/
|
|
public function appendError(string $sessionId, string $runId, string $code, string $message, array $meta = [], ?string $dedupeKey = null): Message
|
|
{
|
|
return $this->chatService->appendMessage([
|
|
'session_id' => $sessionId,
|
|
'role' => Message::ROLE_SYSTEM,
|
|
'type' => 'error',
|
|
'content' => $code,
|
|
'payload' => array_merge($meta, [
|
|
'run_id' => $runId,
|
|
'message' => $message,
|
|
]),
|
|
'dedupe_key' => $dedupeKey,
|
|
]);
|
|
}
|
|
|
|
public function appendToolCall(string $sessionId, ToolCall $toolCall): Message
|
|
{
|
|
return $this->chatService->appendMessage([
|
|
'session_id' => $sessionId,
|
|
'role' => Message::ROLE_AGENT,
|
|
'type' => 'tool.call',
|
|
'content' => $toolCall->rawArguments ?: json_encode($toolCall->arguments, JSON_UNESCAPED_UNICODE),
|
|
'payload' => [
|
|
'run_id' => $toolCall->parentRunId,
|
|
'tool_run_id' => $toolCall->runId,
|
|
'tool_call_id' => $toolCall->toolCallId,
|
|
'name' => $toolCall->name,
|
|
'arguments' => $toolCall->arguments,
|
|
],
|
|
'dedupe_key' => "run:{$toolCall->parentRunId}:tool_call:{$toolCall->toolCallId}",
|
|
]);
|
|
}
|
|
|
|
public function appendToolResult(string $sessionId, ToolResult $result): Message
|
|
{
|
|
return $this->chatService->appendMessage([
|
|
'session_id' => $sessionId,
|
|
'role' => Message::ROLE_TOOL,
|
|
'type' => 'tool.result',
|
|
'content' => $result->output,
|
|
'payload' => [
|
|
'run_id' => $result->runId,
|
|
'parent_run_id' => $result->parentRunId,
|
|
'tool_call_id' => $result->toolCallId,
|
|
'name' => $result->name,
|
|
'status' => $result->status,
|
|
'error' => $result->error,
|
|
'truncated' => $result->truncated,
|
|
],
|
|
'dedupe_key' => "run:{$result->runId}:tool_result",
|
|
]);
|
|
}
|
|
}
|