AG-UI protocol adapter plugin for Spakky Agent
Project description
spakky-agui
spakky-agui는spakky-agent를 위한 공식 AG-UI (Agent User Interaction) protocol adapter plugin입니다. 선언형@Agent실행 스트림을 AG-UI 이벤트로 투영(project)하여 SSE (Server-Sent Events), HTTP streaming, WebSocket, CLI stdio로 노출하고, deferred-tool 방식의 HITL (Human-in-the-loop) 승인 흐름을 지원합니다.
언제 필요한가
애플리케이션이 선언형 @Agent workflow를 실행하고, 그 실행 이벤트(토큰, 도구 호출,
승인 요청, 종료)를 AG-UI 호환 프런트엔드에 실시간 스트리밍하려 할 때 사용합니다.
렌더링(프런트엔드 UI)은 본 plugin의 범위 밖입니다 — 본 plugin은 와이어 프로토콜(SSE
프레임, HTTP JSON-line chunk, WebSocket text message, 또는 stdout JSON-line payload)만
책임집니다.
설치
pip install spakky-agui
durable Agent 실행(state·signal·evidence repository)과 모델 어댑터(IAgentModel)는
별도 provider가 제공합니다. spakky-agui는 inbound SSE/HTTP streaming/WebSocket/stdio
프로토콜 어댑터만 제공합니다.
설정
설정은 SPAKKY_AGUI_ 환경변수 접두사를 사용하는 AgUiConfig로 읽습니다.
| 설정 | 기본값 | 목적 |
|---|---|---|
SPAKKY_AGUI_SSE_PATH |
/agui |
SSE endpoint가 마운트되는 경로 |
SPAKKY_AGUI_WEBSOCKET_PATH |
/agui/ws |
WebSocket endpoint가 마운트되는 경로 |
SPAKKY_AGUI_HTTP_STREAM_PATH |
/agui/stream |
HTTP streaming endpoint가 마운트되는 경로 |
SPAKKY_AGUI_EMIT_STATE_SNAPSHOT |
true |
중립 STATE_SNAPSHOT 이벤트를 AG-UI로 투영할지 여부 |
SPAKKY_AGUI_MESSAGES_SNAPSHOT_ENABLED |
false |
RUN_FINISHED 직전에 MESSAGES_SNAPSHOT 프레임을 1회 방출할지 여부 |
동작 구조
AgentRunner.run_events() → 중립 AgentEvent (런너가 native로 방출)
중립 AgentEvent ──(AgUiProjector)──▶ AG-UI BaseEvent
AG-UI BaseEvent ──(EventEncoder)──▶ "data: {...}\n\n" SSE 프레임
또는 "{...}\n" HTTP streaming chunk
또는 WebSocket text message
또는 stdout "{...}\n" stdio payload
AgentRunner.run_events(): 런너가 중립AgentEventtaxonomy를 native로 방출하는 스트림입니다. 메시지/추론 delta, 도구 호출start/args-delta/end/result생명주기, run/step 경계를 각각 별개 이벤트로 내보내므로, 어댑터는 별도 재구성 없이 1:1로 투영합니다.AgUiProjector: 중립 이벤트의 delta를 AG-UI의 START/CONTENT/END 프레이밍으로 투영하는 상태 기계입니다. 열린 message/reasoning/tool 프레임을 추적하고, 스트림이 중간에 끊겨도finish()가 열린 프레임을 닫아 와이어 형식을 보존합니다.AgUiRunDriver:run_events()를 projector·encoder에 연결하는 async generator입니다.StreamingResponse에 직접 전달됩니다. 승인 대기는RunPausedEvent로 들어오며, projector가 이를 deferred-tool 승인 프레임으로 직접 투영합니다(아래 HITL 참조).
이벤트 매핑 (중립 → AG-UI)
중립 AgentEvent |
AG-UI 이벤트 |
|---|---|
MESSAGE_DELTA |
TEXT_MESSAGE_START (id 변경 시) + TEXT_MESSAGE_CONTENT (빈 delta는 생략) |
REASONING_DELTA |
REASONING_START + REASONING_MESSAGE_START + REASONING_MESSAGE_CONTENT |
TOOL_CALL_START |
TOOL_CALL_START (parentMessageId는 열린 message로 fallback) |
TOOL_CALL_ARGS_DELTA |
TOOL_CALL_ARGS (빈 delta는 생략) |
TOOL_CALL_END |
TOOL_CALL_END |
TOOL_CALL_RESULT |
TOOL_CALL_RESULT (content는 result의 JSON 텍스트) |
RUN_STARTED |
RUN_STARTED (threadId=conversation, runId, parentRunId=neutral parent_run_id가 있을 때) |
RUN_PAUSED |
열린 프레임 닫기 → 승인 pause는 hitl_approval deferred tool TOOL_CALL_START/ARGS/END |
RUN_FINISHED |
열린 프레임 닫기 → RUN_FINISHED 또는 RUN_ERROR |
STEP_STARTED/STEP_FINISHED |
STEP_STARTED/STEP_FINISHED |
STATE_SNAPSHOT |
STATE_SNAPSHOT (emit_state_snapshot=true일 때만) |
STATE_DELTA |
STATE_DELTA (JSON Patch) |
ARTIFACT |
CUSTOM (name=artifact) — AG-UI에 native artifact 이벤트가 없음 |
사용법
@Agent 위에 @AGUICompatible를 쌓고 FastAPI host를 Pod으로 등록합니다. plugin
initialize()는 AgUiConfig, AgUiAgentRegistry, FastAPI mount post-processor를 등록하므로
애플리케이션 코드는 run_driver_factory나 add_agui_endpoint()를 호출하지 않습니다.
from fastapi import FastAPI
from spakky.agent import Agent, AgentExecutionSpec, IAgentModel
from spakky.core.application.application import SpakkyApplication
from spakky.core.application.application_context import ApplicationContext
from spakky.core.pod.annotations.pod import Pod
from spakky.plugins.agui import AGUICompatible
@Pod(name="fastapi_app")
def fastapi_app() -> FastAPI:
return FastAPI()
@AGUICompatible()
@Agent(spec=AgentExecutionSpec(name="assistant", objective="answer with tools"))
class Assistant:
def __init__(self, model: IAgentModel) -> None:
self._model = model
application = SpakkyApplication(ApplicationContext())
application.load_plugins().scan(my_app).start()
app = application.container.get(FastAPI)
여러 Agent를 AG-UI로 노출한다면 각 Agent가 서로 다른 path를 선언해야 합니다.
@AGUICompatible(
sse_path="/agents/researcher/agui",
http_stream_path="/agents/researcher/agui/stream",
websocket_path="/agents/researcher/agui/ws",
)
@Agent(spec=AgentExecutionSpec(name="researcher"))
class Researcher:
...
@AGUICompatible에는 MCP 서버명을 굽지 않습니다. 서비스나 사용자가 run마다 붙일 외부 MCP
server를 고르면 AG-UI forwardedProps.mcp가 core RunAgentInput.metadata["mcp"]로 변환되고,
spakky-mcp가 그 run에만 toolset을 합류시킵니다.
{
"threadId": "conv-1",
"runId": "run-1",
"messages": [{"id": "u1", "role": "user", "content": "check the issue"}],
"tools": [],
"context": [],
"forwardedProps": {
"modelSelection": {"provider": "openrouter", "model": "anthropic/claude-sonnet"},
"mcp": {"servers": ["github"]}
}
}
POST /agui로 AG-UI RunAgentInput을 보내면 text/event-stream 응답으로 실행
이벤트가 스트리밍됩니다. POST /agui/stream은 같은 AG-UI event payload를 SSE data:
프레이밍 없이 application/x-ndjson HTTP response chunks로 순차 전달합니다.
WebSocket 클라이언트는 /agui/ws에 연결한 뒤 같은 AG-UI
RunAgentInput JSON을 text/JSON message로 보내고, 실행 이벤트를 AG-UI encoded text
message로 순서대로 받습니다. 같은 연결에서 후속 RunAgentInput을 보내 승인 결정
(forwardedProps.approvalDecision 또는 deferred tool-result message)을 전달할 수 있습니다.
SSE/HTTP/WebSocket/stdio 입력 경계는 AG-UI threadId·runId·마지막 user text·approval
resume 여부를 core RunAgentInput으로 변환합니다. 중립 RunPausedEvent나 delegated child
event처럼 runner가 이미 parent_run_id를 가진 이벤트를 내보내면 projector는 이를 AG-UI
parentRunId로 그대로 투영합니다.
CLI stdio 경계는 아직 host command가 필요하므로 lower-level AgUiStdioCommand를 사용합니다.
입력은 RunAgentInput JSON 문서를 stdin 또는 문자열 인자로 받고 stdout에 AG-UI event payload를
한 줄에 하나씩 출력합니다. Typer 같은 CLI plugin은 이 callable을 command로 등록하면 됩니다.
stdio 전용 host는 AgUiStdioCommand에 lower-level RunDriverFactory를 전달합니다. 일반
FastAPI SSE/HTTP/WebSocket 노출은 위의 @AGUICompatible 선언만 사용합니다.
HITL — deferred-tool 승인 흐름
AG-UI에는 1급 승인 이벤트가 없으므로, 승인 요청은 deferred tool call로 표면화됩니다.
run_events()는 승인이 필요한 도구에서 dispatch 없이 멈출 때, 성공 RUN_FINISHED 대신
RunPausedEvent(reason=APPROVAL_REQUIRED)를 방출합니다. 이 이벤트는 승인 prompt,
approval id, tool call id, allowed decisions를 담고 있으므로 어댑터가 durable state를
재조회하지 않고 직접 표면화할 수 있습니다.
- 런너가 승인이 필요한 도구에서 멈추면,
run_events()는RunPausedEvent를 방출합니다.AgUiProjector는 이를hitl_approval도구의TOOL_CALL_START/ARGS/END프레임으로 투영합니다 — 결과(result) 프레임은 없습니다 (결과가 지연되었기 때문). - 클라이언트는 이 deferred tool을 렌더링하고 사람의 결정을 받습니다.
- 클라이언트는 다음
RunAgentInput에 그 결정을 담아 다시 POST합니다 — deferred call id를 향한 tool-result 메시지로, 또는forwardedProps.approvalDecision으로. 두 payload 모두{"request_id": "<approval id>", "decision": "approve|reject|modify|defer|cancel"}형태여야 하며,modified_payload와comment를 선택적으로 포함할 수 있습니다. ingest_decision이 그 결정을 디코딩하여 durable signal queue에APPROVAL_DECISIONsignal로 적재하면, 런너가run_events()를 다시 돌며 멈췄던 지점을 재개합니다 — APPROVE는 도구 결과와RUN_FINISHED로, REJECT는 terminal error가 있는 runner 종료를 거쳐 AG-UIRUN_ERROR로 투영됩니다.
매핑 충실도
도구·메시지·실행 이벤트 매핑은 run_events()를 통해 **완전 무손실(lossless)**입니다. 런너가
메시지/추론 delta, 도구 호출 start/args-delta/end/result 생명주기, run/step 경계를
각각 별개의 중립 AgentEvent로 native 방출하므로, 어댑터는 거친 yield를 재구성하지 않고
1:1로 투영합니다. run_events()는 reasoning을 지원하지 않는 모델에서는 REASONING_DELTA를
생략하며(graceful degrade), 현재 모델 루프가 생성하지 않는 STATE_SNAPSHOT/STATE_DELTA/
ARTIFACT는 live 런에서 방출되지 않습니다. projector는 taxonomy 완전성을 위해 이들 종류도
계속 처리합니다.
개발 검증
패키지 단위 검증은 해당 패키지 디렉토리에서 실행합니다.
uv run ruff format .
uv run ruff check .
uv run pyrefly check
uv run pytest
pytest는 각 패키지 pyproject.toml의 coverage 설정을 사용합니다.
라이선스
MIT License
Project details
Release history Release notifications | RSS feed
Download files
Download the file for your platform. If you're not sure which to choose, learn more about installing packages.
Source Distribution
Built Distribution
Filter files by name, interpreter, ABI, and platform.
If you're not sure about the file name format, learn more about wheel file names.
Copy a direct link to the current filters
File details
Details for the file spakky_agui-7.0.0.tar.gz.
File metadata
- Download URL: spakky_agui-7.0.0.tar.gz
- Upload date:
- Size: 21.4 kB
- Tags: Source
- Uploaded using Trusted Publishing? Yes
- Uploaded via: twine/6.1.0 CPython/3.13.12
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
450bb55aa9448bce571bae0d946ef3b532b024f3e6d51346433117c3e4d6d950
|
|
| MD5 |
a3684e1103fd2e3a65370f09481b5269
|
|
| BLAKE2b-256 |
e880f8d92c1dda22aa7f0e3d6522ee2873fe0d2d61f7b0b38e0940507fa20f42
|
Provenance
The following attestation bundles were made for spakky_agui-7.0.0.tar.gz:
Publisher:
release.yml on E5presso/spakky-framework
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
spakky_agui-7.0.0.tar.gz -
Subject digest:
450bb55aa9448bce571bae0d946ef3b532b024f3e6d51346433117c3e4d6d950 - Sigstore transparency entry: 1969786464
- Sigstore integration time:
-
Permalink:
E5presso/spakky-framework@7cc4185678359df49253b24f0e5ea0b4f0ed870b -
Branch / Tag:
refs/heads/main - Owner: https://github.com/E5presso
-
Access:
public
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
release.yml@7cc4185678359df49253b24f0e5ea0b4f0ed870b -
Trigger Event:
workflow_dispatch
-
Statement type:
File details
Details for the file spakky_agui-7.0.0-py3-none-any.whl.
File metadata
- Download URL: spakky_agui-7.0.0-py3-none-any.whl
- Upload date:
- Size: 29.7 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? Yes
- Uploaded via: twine/6.1.0 CPython/3.13.12
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
56c6c451e1c5f71129348e9a6fceb33aea1dc1fec60a10400c185627b95408db
|
|
| MD5 |
11363e019df357090b78b64e6cc99736
|
|
| BLAKE2b-256 |
54b3258bf84084cb566c7b14fc765f1da09852cda5861708a109b692fb49eece
|
Provenance
The following attestation bundles were made for spakky_agui-7.0.0-py3-none-any.whl:
Publisher:
release.yml on E5presso/spakky-framework
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
spakky_agui-7.0.0-py3-none-any.whl -
Subject digest:
56c6c451e1c5f71129348e9a6fceb33aea1dc1fec60a10400c185627b95408db - Sigstore transparency entry: 1969786677
- Sigstore integration time:
-
Permalink:
E5presso/spakky-framework@7cc4185678359df49253b24f0e5ea0b4f0ed870b -
Branch / Tag:
refs/heads/main - Owner: https://github.com/E5presso
-
Access:
public
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
release.yml@7cc4185678359df49253b24f0e5ea0b4f0ed870b -
Trigger Event:
workflow_dispatch
-
Statement type: