mcp_server.rs

 1use std::path::PathBuf;
 2use std::sync::Arc;
 3
 4use crate::claude::edit_tool::EditTool;
 5use crate::claude::permission_tool::PermissionTool;
 6use crate::claude::read_tool::ReadTool;
 7use crate::claude::write_tool::WriteTool;
 8use acp_thread::AcpThread;
 9#[cfg(not(test))]
10use anyhow::Context as _;
11use anyhow::Result;
12use collections::HashMap;
13use context_server::types::{
14    Implementation, InitializeParams, InitializeResponse, ProtocolVersion, ServerCapabilities,
15    ToolsCapabilities, requests,
16};
17use gpui::{App, AsyncApp, Task, WeakEntity};
18use project::Fs;
19use serde::Serialize;
20
21pub struct ClaudeZedMcpServer {
22    server: context_server::listener::McpServer,
23}
24
25pub const SERVER_NAME: &str = "zed";
26
27impl ClaudeZedMcpServer {
28    pub async fn new(
29        thread_rx: watch::Receiver<WeakEntity<AcpThread>>,
30        fs: Arc<dyn Fs>,
31        cx: &AsyncApp,
32    ) -> Result<Self> {
33        let mut mcp_server = context_server::listener::McpServer::new(cx).await?;
34        mcp_server.handle_request::<requests::Initialize>(Self::handle_initialize);
35
36        mcp_server.add_tool(PermissionTool::new(fs.clone(), thread_rx.clone()));
37        mcp_server.add_tool(ReadTool::new(thread_rx.clone()));
38        mcp_server.add_tool(EditTool::new(thread_rx.clone()));
39        mcp_server.add_tool(WriteTool::new(thread_rx.clone()));
40
41        Ok(Self { server: mcp_server })
42    }
43
44    pub fn server_config(&self) -> Result<McpServerConfig> {
45        #[cfg(not(test))]
46        let zed_path = std::env::current_exe()
47            .context("finding current executable path for use in mcp_server")?;
48
49        #[cfg(test)]
50        let zed_path = crate::e2e_tests::get_zed_path();
51
52        Ok(McpServerConfig {
53            command: zed_path,
54            args: vec![
55                "--nc".into(),
56                self.server.socket_path().display().to_string(),
57            ],
58            env: None,
59        })
60    }
61
62    fn handle_initialize(_: InitializeParams, cx: &App) -> Task<Result<InitializeResponse>> {
63        cx.foreground_executor().spawn(async move {
64            Ok(InitializeResponse {
65                protocol_version: ProtocolVersion("2025-06-18".into()),
66                capabilities: ServerCapabilities {
67                    experimental: None,
68                    logging: None,
69                    completions: None,
70                    prompts: None,
71                    resources: None,
72                    tools: Some(ToolsCapabilities {
73                        list_changed: Some(false),
74                    }),
75                },
76                server_info: Implementation {
77                    name: SERVER_NAME.into(),
78                    version: "0.1.0".into(),
79                },
80                meta: None,
81            })
82        })
83    }
84}
85
86#[derive(Serialize)]
87#[serde(rename_all = "camelCase")]
88pub struct McpConfig {
89    pub mcp_servers: HashMap<String, McpServerConfig>,
90}
91
92#[derive(Serialize, Clone)]
93#[serde(rename_all = "camelCase")]
94pub struct McpServerConfig {
95    pub command: PathBuf,
96    pub args: Vec<String>,
97    #[serde(skip_serializing_if = "Option::is_none")]
98    pub env: Option<HashMap<String, String>>,
99}