use std::path::Path;
use serde_json::json;
use crate::errors::Result;
use super::{
backup_config_file, load_jsonc_file, load_jsonc_file_strict, safe_write_json_file,
AgentIntegration, DoctorCounters, HealthcheckContext, InstallContext,
};
pub struct CopilotIntegration;
impl AgentIntegration for CopilotIntegration {
fn name(&self) -> &'static str {
"GitHub Copilot (VS Code)"
}
fn id(&self) -> &'static str {
"copilot"
}
fn install(&self, ctx: &InstallContext) -> Result<()> {
let settings_path = super::vscode_data_dir(&ctx.home).join("User/settings.json");
if let Some(parent) = settings_path.parent() {
std::fs::create_dir_all(parent).ok();
}
let backup = backup_config_file(&settings_path)?;
let mut settings = match load_jsonc_file_strict(&settings_path) {
Ok(v) => v,
Err(e) => {
if let Some(ref b) = backup {
eprintln!(" Backup preserved at: {}", b.display());
}
return Err(e);
}
};
settings["mcp"]["servers"]["tokensave"] = json!({
"type": "stdio",
"command": ctx.tokensave_bin,
"args": ["serve"]
});
safe_write_json_file(&settings_path, &settings, backup.as_deref())?;
eprintln!(
"\x1b[32m✔\x1b[0m Added tokensave MCP server to {}",
settings_path.display()
);
eprintln!();
eprintln!("Setup complete. Next steps:");
eprintln!(" 1. cd into your project and run: tokensave sync");
eprintln!(" 2. Restart VS Code — tokensave tools are now available in GitHub Copilot");
Ok(())
}
fn uninstall(&self, ctx: &InstallContext) -> Result<()> {
let settings_path = super::vscode_data_dir(&ctx.home).join("User/settings.json");
uninstall_mcp_server(&settings_path);
eprintln!();
eprintln!("Uninstall complete. Tokensave has been removed from GitHub Copilot (VS Code).");
eprintln!("Restart VS Code for changes to take effect.");
Ok(())
}
fn healthcheck(&self, dc: &mut DoctorCounters, ctx: &HealthcheckContext) {
eprintln!("\n\x1b[1mGitHub Copilot (VS Code) integration\x1b[0m");
doctor_check_settings(dc, &ctx.home);
}
fn is_detected(&self, home: &Path) -> bool {
super::vscode_data_dir(home).join("User").is_dir()
}
fn has_tokensave(&self, home: &Path) -> bool {
let settings_path = super::vscode_data_dir(home).join("User/settings.json");
if !settings_path.exists() {
return false;
}
let json = load_jsonc_file(&settings_path);
json.get("mcp")
.and_then(|v| v.get("servers"))
.and_then(|v| v.get("tokensave"))
.is_some()
}
}
fn uninstall_mcp_server(settings_path: &Path) {
if !settings_path.exists() {
eprintln!(
" {} not found, skipping",
settings_path.display()
);
return;
}
let mut settings = load_jsonc_file(settings_path);
let removed = settings
.get_mut("mcp")
.and_then(|mcp| mcp.get_mut("servers"))
.and_then(|servers| servers.as_object_mut())
.and_then(|map| map.remove("tokensave"))
.is_some();
if !removed {
eprintln!(
" No tokensave MCP server in {}, skipping",
settings_path.display()
);
return;
}
if let Some(mcp) = settings.get_mut("mcp") {
let servers_empty = mcp
.get("servers")
.and_then(|v| v.as_object())
.is_some_and(|o| o.is_empty());
if servers_empty {
mcp.as_object_mut().map(|o| o.remove("servers"));
}
let mcp_empty = settings
.get("mcp")
.and_then(|v| v.as_object())
.is_some_and(|o| o.is_empty());
if mcp_empty {
settings.as_object_mut().map(|o| o.remove("mcp"));
}
}
let pretty = serde_json::to_string_pretty(&settings).unwrap_or_default();
std::fs::write(settings_path, format!("{pretty}\n")).ok();
eprintln!(
"\x1b[32m✔\x1b[0m Removed tokensave MCP server from {}",
settings_path.display()
);
}
fn doctor_check_settings(dc: &mut DoctorCounters, home: &Path) {
let settings_path = super::vscode_data_dir(home).join("User/settings.json");
if !settings_path.exists() {
dc.warn(&format!(
"{} not found — run `tokensave install --agent copilot` if you use GitHub Copilot",
settings_path.display()
));
return;
}
let settings = load_jsonc_file(&settings_path);
let server = settings
.get("mcp")
.and_then(|v| v.get("servers"))
.and_then(|v| v.get("tokensave"));
let Some(server) = server.and_then(|v| v.as_object()) else {
dc.fail(&format!(
"MCP server NOT registered in {} — run `tokensave install --agent copilot`",
settings_path.display()
));
return;
};
dc.pass(&format!(
"MCP server registered in {}",
settings_path.display()
));
let has_serve = server
.get("args")
.and_then(|v| v.as_array())
.is_some_and(|arr| arr.iter().any(|v| v.as_str() == Some("serve")));
if has_serve {
dc.pass("MCP server args include \"serve\"");
} else {
dc.fail("MCP server args missing \"serve\" — run `tokensave install --agent copilot`");
}
}