- 添加流式文本推送,支持 `message.delta` 消息类型 - 优化 Run 主流程,增加工具调用与流式数据发布逻辑 - 更新 `phpunit.xml` 环境变量,支持 Agent 配置项 - 扩展文档,完善工具调用与消息类型说明
171 lines
5.9 KiB
PHP
171 lines
5.9 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,
|
||
]);
|
||
}
|
||
|
||
/**
|
||
* 追加 Agent 流式文本增量(仅用于 SSE 推送,不落库)。
|
||
*
|
||
* message.delta 消息只用于实时流式推送,不需要持久化到数据库。
|
||
* 最终的完整回复会通过 appendAgentMessage() 落库。
|
||
*
|
||
* @param array<string, mixed> $meta
|
||
*/
|
||
public function appendAgentDelta(string $sessionId, string $runId, string $content, int $deltaIndex, array $meta = []): void
|
||
{
|
||
// 1. 创建临时 Message 对象(不保存到数据库)
|
||
$message = new Message([
|
||
'message_id' => (string) \Illuminate\Support\Str::uuid(),
|
||
'session_id' => $sessionId,
|
||
'role' => Message::ROLE_AGENT,
|
||
'type' => 'message.delta',
|
||
'content' => $content,
|
||
'payload' => array_merge($meta, [
|
||
'run_id' => $runId,
|
||
'delta_index' => $deltaIndex,
|
||
]),
|
||
'dedupe_key' => "run:{$runId}:agent:delta:{$deltaIndex}",
|
||
'seq' => 0, // delta 消息不需要真实的 seq
|
||
'created_at' => now(),
|
||
]);
|
||
|
||
// 2. 仅发布 Redis 事件,供 SSE 实时推送
|
||
$this->publishDeltaMessage($message);
|
||
}
|
||
|
||
/**
|
||
* 发布 delta 消息到 Redis(仅用于 SSE 推送)。
|
||
*
|
||
* 此方法不保存消息到数据库,只发布事件供 SSE 客户端接收。
|
||
*/
|
||
private function publishDeltaMessage(Message $message): void
|
||
{
|
||
$root = \Illuminate\Support\Facades\Redis::getFacadeRoot();
|
||
$isMocked = $root instanceof \Mockery\MockInterface;
|
||
|
||
// 如果 Redis 不可用(测试环境),直接返回
|
||
if (!class_exists(\Redis::class) && !$isMocked) {
|
||
return;
|
||
}
|
||
|
||
$channel = "session:{$message->session_id}:messages";
|
||
|
||
try {
|
||
\Illuminate\Support\Facades\Redis::publish(
|
||
$channel,
|
||
json_encode($message->toArray(), JSON_UNESCAPED_UNICODE | JSON_INVALID_UTF8_IGNORE)
|
||
);
|
||
} catch (\Throwable $e) {
|
||
logger()->warning('Redis publish failed for delta message', [
|
||
'session_id' => $message->session_id,
|
||
'run_id' => $message->payload['run_id'] ?? null,
|
||
'delta_index' => $message->payload['delta_index'] ?? null,
|
||
'error' => $e->getMessage(),
|
||
]);
|
||
}
|
||
}
|
||
|
||
/**
|
||
* @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",
|
||
]);
|
||
}
|
||
}
|