`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) <noreply@anthropic.com>
13 KiB
title, date, status, target_version, task_spec, contract_source, contract_sections, depends_on, unblocks
| title | date | status | target_version | task_spec | contract_source | contract_sections | depends_on | unblocks | |||
|---|---|---|---|---|---|---|---|---|---|---|---|
| p9-fb-30 — MCP server (stdio) — agent host 무관 protocol surface | 2026-05-07 | design (brainstorm 완료, plan 단계 대기) | 0.4.0 | ../../../tasks/p9/p9-fb-30-mcp-server.md | ../specs/2026-04-27-kebab-final-form-design.md |
|
|
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 <path> # 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-appfacade +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 응답:
{
"protocolVersion": "<rmcp 가 pin 하는 stable version, 예: 2025-03-26>",
"capabilities": {
"tools": { "listChanged": false }
},
"serverInfo": {
"name": "kebab",
"version": "<env!(\"CARGO_PKG_VERSION\")>"
}
}
resources / prompts / sampling / notifications — 모두 미선언.
Tool error envelope
| 시나리오 | MCP 응답 | content |
|---|---|---|
facade Err(e) |
{ isError: true, content: [{ type: "text", text: <error.v1 JSON> }] } |
error_wire::classify(&e, false) 결과 |
facade Ok(...) |
{ isError: false, content: [{ type: "text", text: <wire JSON> }] } |
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신규 —ErrorV1struct +classify(&anyhow::Error, verbose: bool) -> ErrorV1함수 +classify_llmhelper. fb-27 commitc91228e의error_classify.rs코드 그대로 이전.crates/kebab-app/src/lib.rspub 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::ErrorV11 줄 변경.- 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/callrequest 가 독립 task —tokio::task::spawn. - facade 가 sync API (현재 대부분) →
tokio::task::spawn_blockingwrap. - 한 process 안에 SQLite / Lance / fastembed connection 공유 — 한 번 init 후 모든 tool call 이 hot.
kebab-app의 facade 가 매 호출 마다Configload + store open 시 cold-start 절감 효과 약화 — plan 단계에서 server-scopeApp인스턴스 (혹은 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/<id>/kebab://doc/<id>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_stdintools — fb-31 single-file ingest 머지 시 추가.fetch(verbatim doc/chunk) — fb-35 verbatim fetch 머지 시 추가.list_docs/inspect_chunktools — 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)
- frozen design §10.1 — MCP transport 절 추가 (또는 §10.2 신설). stdio-only / 4 tool / capability flag flip 명시.
- README.md — 명령 표 에
kebab mcprow + MCP usage section (Claude Code~/.claude/mcp.jsonconfig 예시). - HANDOFF.md —
2026-05-?? P9 post-도그푸딩 (p9-fb-30)한 줄. - CLAUDE.md — facade 룰 절 에
kebab-mcpUI crate 카테고리 추가. 새 crate 카운트 갱신 (~20 → ~21). - integrations/claude-code/kebab/SKILL.md — MCP 사용 권장 + Claude Code
~/.claude/mcp.json예시 한 블록 추가. 기존 subprocess wrapper 형태도 backwards-compat 유지 (일부 사용자가 MCP 미지원 host 에서 호출). - HOTFIXES.md —
2026-05-?? — fb-30entry. classify 모듈 이전 + capability flag flip + 기타 deviation 명시. tasks/p9/p9-fb-30-mcp-server.md— statusopen→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.
Configresolution per call: 매 tool call 마다Config::load(...)+Appopen 하면 daemon 의 hot-cache 효과 미미. 첫 call 시 server-scopeApp인스턴스 만들고 이후 재사용 — 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<String>채움. MCP 응답에서hint가 보존되는지 — text content 의 JSON 이hintfield 그대로 가짐. agent 가 parse 하면 readable. OK. - stdin/stdout 충돌: kebab-mcp 가 stdin/stdout 으로 JSON-RPC 통신.
tracinglog 가 stdout 으로 쓰면 protocol 깨짐. 모든 log 는 stderr 또는 file (~/.local/state/kebab/logs/) — kebab-app 의 logging init 가 이미 stderr 기본. 명시 verify.