From 58ec4578b950bbeecb976563bf19e86bac51dc14 Mon Sep 17 00:00:00 2001
From: th-kim0823
Date: Thu, 7 May 2026 14:44:09 +0900
Subject: [PATCH] =?UTF-8?q?=F0=9F=93=9D=20docs(spec):=20p9-fb-30=20MCP=20s?=
=?UTF-8?q?erver=20(stdio)=20=EC=84=A4=EA=B3=84=20=EB=AC=B8=EC=84=9C?=
MIME-Version: 1.0
Content-Type: text/plain; charset=UTF-8
Content-Transfer-Encoding: 8bit
`kebab mcp` 신규 subcommand + new crate `kebab-mcp` 도입을 위한
brainstorm 산출물. agent integration "MVP" 완성 (Claude Code / Cursor /
OpenAI Agents 등 host-agnostic 사용 가능).
핵심 결정:
- `kebab mcp` subcommand (kebab-cli 내, 신규 binary 아님)
- 4 read-only tool (`search` / `ask` / `schema` / `doctor`) — ingest /
fetch / list_docs / inspect_chunk 는 fb-31 / fb-35 / 후속에서 추가
- Resources / Prompts 모두 skip (tools only)
- Rust MCP SDK 사용 — rmcp 채택 우선, plan 단계 verify
- stdio 단일 transport — fb-29 deferral 따라 HTTP-SSE P+
- error mapping: tool dispatch 실패만 isError=true + error.v1 content,
refusal / no-hit / unhealthy 는 정상 응답 (semantic flag 으로 분기)
- classify 모듈 이전: kebab-cli::error_classify → kebab-app::error_wire
(kebab-cli + kebab-mcp 둘 다 동일 모듈 사용, facade 룰 준수)
- capability flag `mcp_server` false → true
릴리스: 0.3.0 → 0.4.0 minor — 신규 surface + new crate + 디자인 §10.1
변경 + capability flip 모두 trigger.
Co-Authored-By: Claude Opus 4.7 (1M context)
---
.../2026-05-07-p9-fb-30-mcp-server-design.md | 222 ++++++++++++++++++
1 file changed, 222 insertions(+)
create mode 100644 docs/superpowers/specs/2026-05-07-p9-fb-30-mcp-server-design.md
diff --git a/docs/superpowers/specs/2026-05-07-p9-fb-30-mcp-server-design.md b/docs/superpowers/specs/2026-05-07-p9-fb-30-mcp-server-design.md
new file mode 100644
index 0000000..dd85bf5
--- /dev/null
+++ b/docs/superpowers/specs/2026-05-07-p9-fb-30-mcp-server-design.md
@@ -0,0 +1,222 @@
+---
+title: "p9-fb-30 — MCP server (stdio) — agent host 무관 protocol surface"
+date: 2026-05-07
+status: design (brainstorm 완료, plan 단계 대기)
+target_version: 0.4.0
+task_spec: ../../../tasks/p9/p9-fb-30-mcp-server.md
+contract_source: ../specs/2026-04-27-kebab-final-form-design.md
+contract_sections: [§7 RAG, §10 UX]
+depends_on: [p9-fb-27]
+unblocks: []
+---
+
+# MCP server (stdio) — 설계
+
+## 동기
+
+현재 외부 AI 통합은 `integrations/claude-code/kebab/` skill 한 종류 — Claude Code subprocess wrapper. Cursor / OpenAI Agents / Copilot CLI 등 다른 host 는 별도 wrapper 작성 필요.
+
+MCP (Model Context Protocol) 가 표준 — 한 번 server 구현하면 MCP-aware host 모두 지원. 본 task 는 stdio MCP server 도입. fb-29 HTTP daemon 은 deferred (single-user local-first 환경에서 daemon 복잡도 비대 — fb-30 stdio 가 동일 사용자 가치 제공).
+
+fb-27 (introspection + error wire) 의 capability matrix + error.v1 wire 가 본 task 의 prerequisite ✅.
+
+## 결정 요약
+
+| 결정 | 선택 |
+|------|------|
+| Dispatch | `kebab mcp` subcommand (kebab-cli 내) |
+| Tool surface (v1) | `search` / `ask` / `schema` / `doctor` (read-only, 4 개) |
+| Resources / Prompts | 모두 skip (tools only) |
+| 구현 | Rust MCP SDK (`rmcp` 또는 plan 단계 채택) |
+| Transport | stdio 단일 (HTTP-SSE 는 fb-29 deferral 따라 P+) |
+| Output | 모든 tool 이 wire schema v1 JSON 을 text content 로 반환 |
+| Multi-turn `ask` | optional `session_id` (kebab-app 의 `ask_with_session_with_config` 활용) |
+
+## Surface 1 — `kebab mcp` 신규 subcommand
+
+### CLI
+
+```
+kebab mcp # stdio JSON-RPC server 시작
+kebab mcp --config # config 명시 (P3-5 / P4-3 패턴)
+```
+
+`--config` 외 추가 flag 없음. agent host 가 spawn 명령에서 환경 변수로 추가 설정 주입.
+
+### Crate boundary
+
+새 crate `crates/kebab-mcp/` (lib only). `kebab-cli` 의 `Cmd::Mcp` arm 이 한 줄 entry — `kebab_mcp::serve_stdio(cfg)?`.
+
+```
+kebab-cli ──► kebab-mcp ──► kebab-app ──► kebab-store-* / kebab-llm-* / kebab-parse-*
+ │ │
+ └─ rmcp ─────┴─ kebab-config / kebab-core
+```
+
+CLAUDE.md facade 룰 준수:
+- `kebab-mcp` 는 `kebab-app` facade + `kebab-config` + `kebab-core` 만 import. 구현 crate 직접 금지.
+- `kebab-cli` 는 `kebab-mcp` 만 알고 MCP 내부 미인지 — 다른 UI crate (TUI / desktop) 가 mcp surface 필요해지면 동일하게 import.
+- `rmcp` (또는 채택 SDK) 는 `kebab-mcp` 의 `[dependencies]` 만 — kebab-cli 는 transitive.
+
+CLAUDE.md 의 "UI crates" 카테고리에 `kebab-mcp` 추가 (의존 경계 절).
+
+## Surface 2 — Tool catalog (4 tools)
+
+`tools/list` response 가 4 tool 을 노출. 각 tool 은 inputSchema (JSON Schema) 가 inline.
+
+### `search`
+
+| 항목 | 값 |
+|------|-----|
+| description | "Lexical / vector / hybrid retrieval over indexed corpus." |
+| input | `{ query: string (required), mode?: "lexical" \| "vector" \| "hybrid" (default "hybrid"), k?: integer (default 10, range 1-100) }` |
+| facade | `kebab_app::search_with_config(&cfg, query, mode, k)` |
+| output | text content = `serde_json::to_string(wire::wire_search_hits(&hits))` |
+
+빈 결과 = 정상 응답 (empty `search_hit.v1` array). NoHitSignal 의 exit-code 분기는 stdio 무관.
+
+### `ask`
+
+| 항목 | 값 |
+|------|-----|
+| description | "Grounded RAG answer with citations. Returns answer.v1 with grounded=false when KB lacks context." |
+| input | `{ query: string (required), session_id?: string }` |
+| facade | `session_id` 있으면 `ask_with_session_with_config`, 없으면 `ask_with_config` |
+| output | text content = `serde_json::to_string(wire::wire_answer(&answer))` |
+
+Refusal (`grounded: false`) = 정상 응답. agent 가 wire payload 의 `grounded` flag 로 분기. `refusal_reason` 도 답변에 포함.
+
+### `schema`
+
+| 항목 | 값 |
+|------|-----|
+| description | "Introspection — wire schemas, capabilities, model versions, index stats." |
+| input | `{}` (no args) |
+| facade | `schema_with_config(&cfg)` |
+| output | text content = `serde_json::to_string(wire::wire_schema(&schema))` |
+
+`capabilities.mcp_server` 가 `true` (본 PR 에서 `capabilities_snapshot()` 갱신).
+
+### `doctor`
+
+| 항목 | 값 |
+|------|-----|
+| description | "Health check — config / data dir / Ollama reachability." |
+| input | `{}` (no args) |
+| facade | `doctor_with_config_path(cli.config.as_deref())` (or equivalent) |
+| output | text content = `serde_json::to_string(wire::wire_doctor(&report))` |
+
+DoctorUnhealthy = 정상 응답 (doctor.v1 with `ok: false`). agent 가 검사.
+
+## Surface 3 — Lifecycle / capabilities / error mapping
+
+### Initialize handshake
+
+server `initialize` 응답:
+
+```jsonc
+{
+ "protocolVersion": "",
+ "capabilities": {
+ "tools": { "listChanged": false }
+ },
+ "serverInfo": {
+ "name": "kebab",
+ "version": ""
+ }
+}
+```
+
+resources / prompts / sampling / notifications — 모두 미선언.
+
+### Tool error envelope
+
+| 시나리오 | MCP 응답 | content |
+|----------|----------|---------|
+| facade `Err(e)` | `{ isError: true, content: [{ type: "text", text: }] }` | `error_wire::classify(&e, false)` 결과 |
+| facade `Ok(...)` | `{ isError: false, content: [{ type: "text", text: }] }` | search_hit.v1 / answer.v1 / schema.v1 / doctor.v1 |
+
+protocol-level error (invalid method / malformed params / panic) 는 SDK 가 JSON-RPC error envelope 으로 자동 처리.
+
+### Refusal / no-hit / unhealthy 가 isError 아님
+
+CLI 의 exit code 1 (refusal/no-hit) / 3 (doctor unhealthy) 는 stdio 환경에 의미 없음. 모두 `isError: false` 정상 응답으로 반환 — agent 가 wire payload 의 semantic flag (`grounded` / 빈 array / `ok: false`) 로 분기. 이게 MCP 표준 패턴 — error envelope 은 protocol 실패 전용.
+
+### classify 모듈 이전 (load-bearing 구조 변경)
+
+`kebab-cli::error_classify` (fb-27 도입) 를 `kebab-app::error_wire` 로 promotion. 본 PR 에 포함:
+
+- `crates/kebab-app/src/error_wire.rs` 신규 — `ErrorV1` struct + `classify(&anyhow::Error, verbose: bool) -> ErrorV1` 함수 + `classify_llm` helper. fb-27 commit `c91228e` 의 `error_classify.rs` 코드 그대로 이전.
+- `crates/kebab-app/src/lib.rs` `pub mod error_wire;` + re-export.
+- `crates/kebab-cli/src/error_classify.rs` 삭제 — `kebab-cli::main` 의 import 가 `kebab_app::error_wire::*` 로 변경.
+- 기존 7 unit test 도 함께 이전 (`kebab-app/src/error_wire.rs::tests`).
+- `kebab-cli::wire::wire_error_v1` 의 `&crate::error_classify::ErrorV1` → `&kebab_app::ErrorV1` 1 줄 변경.
+- kebab-cli 의 reqwest dev-dep 는 유지 (`llm_unreachable_classifies` 가 함께 이동) — 또는 reqwest dev-dep 도 kebab-app 으로 이전.
+
+근거: kebab-cli + kebab-mcp 둘 다 동일 classify 사용. UI crate (kebab-cli) 가 다른 UI crate import 는 facade 룰 위반. kebab-app 으로 promotion 이 정공법.
+
+### Concurrency
+
+stdio JSON-RPC 는 클라이언트 측 순차 호출 default 지만 MCP spec 은 동시 호출 허용. tokio runtime (kebab-app 의 `rt-multi-thread` feature 활용):
+
+- 각 `tools/call` request 가 독립 task — `tokio::task::spawn`.
+- facade 가 sync API (현재 대부분) → `tokio::task::spawn_blocking` wrap.
+- 한 process 안에 SQLite / Lance / fastembed connection 공유 — 한 번 init 후 모든 tool call 이 hot. `kebab-app` 의 facade 가 매 호출 마다 `Config` load + store open 시 cold-start 절감 효과 약화 — plan 단계에서 server-scope `App` 인스턴스 (혹은 connection pool) 도입 검토.
+
+세부 async pattern + connection lifetime 은 plan 단계 결정 (rmcp SDK 의 dispatch 모델에 의존).
+
+## Out of scope (defer)
+
+- **HTTP-SSE transport** — fb-29 P+ 와 묶어 진행. 본 task 는 stdio 단일.
+- **Resources** — `kebab://chunk/` / `kebab://doc/` URI scheme. fb-35 verbatim fetch 와 함께 v2.
+- **Prompts** — reusable prompt template. RAG 자체가 prompt template 내장 — 사용자 가치 약함, defer.
+- **Streaming `ask`** — fb-33 streaming ask 와 함께 ndjson delta tool 결과.
+- **`ingest_file` / `ingest_stdin` tools** — fb-31 single-file ingest 머지 시 추가.
+- **`fetch` (verbatim doc/chunk)** — fb-35 verbatim fetch 머지 시 추가.
+- **`list_docs` / `inspect_chunk` tools** — demand 발생 시.
+- **Server logging notifications** (`notifications/message`) — SDK 자동 처리만.
+- **Sampling capability** — 본 server 는 sampling 미수행.
+
+## Testing 전략
+
+| crate | type | 파일 | 검증 |
+|-------|------|------|------|
+| `kebab-app` | unit | `src/error_wire.rs::tests` | 기존 7 classify test (fb-27 에서 promotion) |
+| `kebab-mcp` | unit | `src/lib.rs::tests` | tool input schema parse + dispatch (mock or TempDir) |
+| `kebab-mcp` | integration | `tests/initialize.rs` | initialize handshake — protocolVersion / serverInfo / capabilities.tools 정확 |
+| `kebab-mcp` | integration | `tests/tools_list.rs` | `tools/list` 가 4 tool name + inputSchema 정확 반환 |
+| `kebab-mcp` | integration | `tests/tools_call_search.rs` | search tool call → text content = search_hit.v1 array, isError=false |
+| `kebab-mcp` | integration | `tests/tools_call_ask.rs` | ask tool call → answer.v1 (refusal 시 grounded=false 정상) |
+| `kebab-mcp` | integration | `tests/tools_call_schema.rs` | schema.v1 정확 + capabilities.mcp_server=true 검증 |
+| `kebab-mcp` | integration | `tests/tools_call_doctor.rs` | doctor.v1 정확 |
+| `kebab-mcp` | integration | `tests/error_mapping.rs` | bad config 로 호출 → tool error with error.v1 + isError=true |
+| `kebab-cli` | integration | `tests/cli_mcp_smoke.rs` | `target/debug/kebab mcp` spawn + 1 round-trip JSON-RPC |
+| `kebab-app` | unit | `tests/schema_report.rs` (기존) | `capabilities.mcp_server == true` assertion 1 줄 추가 |
+
+JSON-RPC client = rmcp 의 in-process test harness (지원 시) 또는 hand-roll line write/read 헬퍼.
+
+## Spec / doc sync (PR 같은 commit)
+
+1. **frozen design §10.1** — MCP transport 절 추가 (또는 §10.2 신설). stdio-only / 4 tool / capability flag flip 명시.
+2. **README.md** — 명령 표 에 `kebab mcp` row + MCP usage section (Claude Code `~/.claude/mcp.json` config 예시).
+3. **HANDOFF.md** — `2026-05-?? P9 post-도그푸딩 (p9-fb-30)` 한 줄.
+4. **CLAUDE.md** — facade 룰 절 에 `kebab-mcp` UI crate 카테고리 추가. 새 crate 카운트 갱신 (~20 → ~21).
+5. **integrations/claude-code/kebab/SKILL.md** — MCP 사용 권장 + Claude Code `~/.claude/mcp.json` 예시 한 블록 추가. 기존 subprocess wrapper 형태도 backwards-compat 유지 (일부 사용자가 MCP 미지원 host 에서 호출).
+6. **HOTFIXES.md** — `2026-05-?? — fb-30` entry. classify 모듈 이전 + capability flag flip + 기타 deviation 명시.
+7. **`tasks/p9/p9-fb-30-mcp-server.md`** — status `open` → `completed`, banner 갱신, depends_on 갱신 (이미 fb-29 제거됨 from 2026-05-07 commit).
+
+## Release trigger
+
+0.3.0 → **0.4.0** minor bump — fb-30 머지 = 신규 CLI surface (`kebab mcp`) + new crate (`kebab-mcp`) + capability flag flip (`mcp_server: true`) + design §10 변경 (3 trigger 모두 발동).
+
+agent integration "MVP" 완성 신호. release notes 에 강조: "MCP 표준 protocol 으로 Claude Code / Cursor / OpenAI Agents 등 host-agnostic 사용 가능."
+
+## Risks / notes
+
+- **rmcp version maturity**: plan 단계 verify 필요. 미존재 / 미성숙 시 hand-roll JSON-RPC 또는 hybrid (transport hand-roll + spec literal struct serde) fallback. rmcp 채택 가정 하 spec 작성됨 — 심각한 호환성 문제 발생 시 spec 갱신 + HOTFIXES.
+- **classify 이전의 회귀 위험**: kebab-cli 의 7 test + 1 wire test 가 import path 변경. mechanical 이지만 누락 시 컴파일 실패로 catch.
+- **`Config` resolution per call**: 매 tool call 마다 `Config::load(...)` + `App` open 하면 daemon 의 hot-cache 효과 미미. 첫 call 시 server-scope `App` 인스턴스 만들고 이후 재사용 — plan 단계 concrete 설계.
+- **MCP version evolution**: spec 가 진화 중. SDK pin 따르고 README 에 명시. major change 발생 시 별 task.
+- **ask 의 multi-turn session 의 정합**: kebab session 은 kebab 의 RAG history. agent host 도 자체 conversation 추적. 둘이 다른 식별자 — sync 필요 시 사용자가 명시적으로 `session_id` 매핑. 본 PR scope 밖 — agent 사용 가이드에 명시.
+- **Tool error 에 hint 손실 위험**: `error_wire::classify` 가 `hint: Option` 채움. MCP 응답에서 `hint` 가 보존되는지 — text content 의 JSON 이 `hint` field 그대로 가짐. agent 가 parse 하면 readable. OK.
+- **stdin/stdout 충돌**: kebab-mcp 가 stdin/stdout 으로 JSON-RPC 통신. `tracing` log 가 stdout 으로 쓰면 protocol 깨짐. 모든 log 는 stderr 또는 file (`~/.local/state/kebab/logs/`) — kebab-app 의 logging init 가 이미 stderr 기본. 명시 verify.