use std::io::Write;
use std::path::Path;
use crate::errors::{Result, TokenSaveError};
use super::{AgentIntegration, DoctorCounters, HealthcheckContext, InstallContext};
pub struct VibeIntegration;
fn vibe_home(home: &Path) -> std::path::PathBuf {
if let Ok(vibe) = std::env::var("VIBE_HOME") {
let vibe_path = std::path::PathBuf::from(&vibe);
if vibe_path.starts_with(home) {
return vibe_path;
}
}
home.join(".vibe")
}
fn vibe_config_path(home: &Path) -> std::path::PathBuf {
vibe_home(home).join("config.toml")
}
fn vibe_prompt_path(home: &Path) -> std::path::PathBuf {
vibe_home(home).join("prompts/cli.md")
}
const TOML_MARKER: &str = "name = \"tokensave\"";
impl AgentIntegration for VibeIntegration {
fn name(&self) -> &'static str {
"Mistral Vibe"
}
fn id(&self) -> &'static str {
"vibe"
}
fn install(&self, ctx: &InstallContext) -> Result<()> {
let vibe_dir = vibe_home(&ctx.home);
std::fs::create_dir_all(&vibe_dir).ok();
let config_path = vibe_config_path(&ctx.home);
install_mcp_server(&config_path, &ctx.tokensave_bin)?;
let prompt_dir = vibe_dir.join("prompts");
std::fs::create_dir_all(&prompt_dir).ok();
let prompt_path = vibe_prompt_path(&ctx.home);
install_prompt_rules(&prompt_path)?;
eprintln!();
eprintln!("Setup complete. Next steps:");
eprintln!(" 1. cd into your project and run: tokensave init");
eprintln!(" 2. Start a new Vibe session — tokensave tools are now available");
Ok(())
}
fn uninstall(&self, ctx: &InstallContext) -> Result<()> {
let config_path = vibe_config_path(&ctx.home);
uninstall_mcp_server(&config_path);
uninstall_prompt_rules(&vibe_prompt_path(&ctx.home));
eprintln!();
eprintln!("Uninstall complete. Tokensave has been removed from Mistral Vibe.");
eprintln!("Start a new Vibe session for changes to take effect.");
Ok(())
}
fn healthcheck(&self, dc: &mut DoctorCounters, ctx: &HealthcheckContext) {
eprintln!("\n\x1b[1mMistral Vibe integration\x1b[0m");
doctor_check_config(dc, &ctx.home);
doctor_check_prompt(dc, &ctx.home);
}
fn is_detected(&self, home: &Path) -> bool {
vibe_home(home).is_dir()
}
fn has_tokensave(&self, home: &Path) -> bool {
let config_path = vibe_config_path(home);
if !config_path.exists() {
return false;
}
let contents = std::fs::read_to_string(&config_path).unwrap_or_default();
contents.contains(TOML_MARKER)
}
}
fn install_mcp_server(config_path: &Path, tokensave_bin: &str) -> Result<()> {
let existing = if config_path.exists() {
std::fs::read_to_string(config_path).unwrap_or_default()
} else {
String::new()
};
if existing.contains(TOML_MARKER) {
eprintln!(
" tokensave MCP server already registered in {}, skipping",
config_path.display()
);
return Ok(());
}
let block = format!(
"\n[[mcp_servers]]\n\
name = \"tokensave\"\n\
transport = \"stdio\"\n\
command = \"{tokensave_bin}\"\n\
args = [\"serve\"]\n"
);
let mut f = std::fs::OpenOptions::new()
.create(true)
.append(true)
.open(config_path)
.map_err(|e| TokenSaveError::Config {
message: format!("failed to open {}: {e}", config_path.display()),
})?;
f.write_all(block.as_bytes())
.map_err(|e| TokenSaveError::Config {
message: format!("failed to write {}: {e}", config_path.display()),
})?;
eprintln!(
"\x1b[32m✔\x1b[0m Added tokensave MCP server to {}",
config_path.display()
);
Ok(())
}
fn install_prompt_rules(prompt_path: &Path) -> Result<()> {
let marker = "## Prefer tokensave MCP tools";
let existing = if prompt_path.exists() {
std::fs::read_to_string(prompt_path).unwrap_or_default()
} else {
String::new()
};
if existing.contains(marker) {
eprintln!(" Vibe prompt already contains tokensave rules, skipping");
return Ok(());
}
let mut f = std::fs::OpenOptions::new()
.create(true)
.append(true)
.open(prompt_path)
.map_err(|e| TokenSaveError::Config {
message: format!("failed to open {}: {e}", prompt_path.display()),
})?;
write!(
f,
"\n{marker}\n\n\
Before reading source files or scanning the codebase, use the tokensave MCP tools \
(`tokensave_context`, `tokensave_search`, `tokensave_callers`, `tokensave_callees`, \
`tokensave_impact`, `tokensave_node`, `tokensave_files`, `tokensave_affected`). \
They provide instant semantic results from a pre-built knowledge graph and are \
faster than file reads.\n\n\
If a code analysis question cannot be fully answered by tokensave MCP tools, \
try querying the SQLite database directly at `.tokensave/tokensave.db` \
(tables: `nodes`, `edges`, `files`). Use SQL to answer complex structural queries \
that go beyond what the built-in tools expose.\n\n\
When a tokensave tool result contains a `tokensave_metrics:` line, report the \
savings to the user (e.g. \"TokenSave'd ~N tokens\"). Never silently omit this.\n"
)
.map_err(|e| TokenSaveError::Config {
message: format!("failed to write Vibe prompt: {e}"),
})?;
eprintln!(
"\x1b[32m✔\x1b[0m Added tokensave rules to {}",
prompt_path.display()
);
Ok(())
}
fn uninstall_mcp_server(config_path: &Path) {
if !config_path.exists() {
eprintln!(" {} not found, skipping", config_path.display());
return;
}
let Ok(contents) = std::fs::read_to_string(config_path) else {
return;
};
if !contents.contains(TOML_MARKER) {
eprintln!(
" No tokensave MCP server in {}, skipping",
config_path.display()
);
return;
}
let lines: Vec<&str> = contents.lines().collect();
let mut result: Vec<&str> = Vec::new();
let mut skip = false;
for line in &lines {
if line.trim() == "[[mcp_servers]]" {
skip = false;
}
if skip {
let trimmed = line.trim();
if trimmed.starts_with("[[") || (trimmed.starts_with('[') && !trimmed.starts_with("[["))
{
skip = false;
} else {
continue;
}
}
if line.contains(TOML_MARKER) {
while let Some(last) = result.last() {
if last.trim() == "[[mcp_servers]]" {
result.pop();
break;
}
if last.trim().is_empty() {
result.pop();
} else {
break;
}
}
skip = true;
continue;
}
result.push(line);
}
while result.last().is_some_and(|l| l.trim().is_empty()) {
result.pop();
}
let new_contents = if result.is_empty() {
String::new()
} else {
format!("{}\n", result.join("\n"))
};
if new_contents.trim().is_empty() {
std::fs::remove_file(config_path).ok();
eprintln!(
"\x1b[32m✔\x1b[0m Removed {} (was empty)",
config_path.display()
);
} else {
std::fs::write(config_path, &new_contents).ok();
eprintln!(
"\x1b[32m✔\x1b[0m Removed tokensave MCP server from {}",
config_path.display()
);
}
}
fn uninstall_prompt_rules(prompt_path: &Path) {
if !prompt_path.exists() {
return;
}
let Ok(contents) = std::fs::read_to_string(prompt_path) else {
return;
};
if !contents.contains("tokensave") {
eprintln!(" Vibe prompt does not contain tokensave rules, skipping");
return;
}
let marker = "## Prefer tokensave MCP tools";
let Some(start) = contents.find(marker) else {
return;
};
let before = &contents[..start];
let trimmed = before.trim_end().to_string();
if trimmed.is_empty() {
std::fs::remove_file(prompt_path).ok();
eprintln!(
"\x1b[32m✔\x1b[0m Removed {} (was empty)",
prompt_path.display()
);
} else {
std::fs::write(prompt_path, format!("{trimmed}\n")).ok();
eprintln!(
"\x1b[32m✔\x1b[0m Removed tokensave rules from {}",
prompt_path.display()
);
}
}
fn doctor_check_config(dc: &mut DoctorCounters, home: &Path) {
let config_path = vibe_config_path(home);
if !config_path.exists() {
dc.warn(&format!(
"{} not found — run `tokensave install --agent vibe` if you use Mistral Vibe",
config_path.display()
));
return;
}
let contents = std::fs::read_to_string(&config_path).unwrap_or_default();
if contents.contains(TOML_MARKER) {
dc.pass(&format!(
"MCP server registered in {}",
config_path.display()
));
} else {
dc.fail(&format!(
"MCP server NOT registered in {} — run `tokensave install --agent vibe`",
config_path.display()
));
}
}
fn doctor_check_prompt(dc: &mut DoctorCounters, home: &Path) {
let prompt_path = vibe_prompt_path(home);
if prompt_path.exists() {
let has_rules = std::fs::read_to_string(&prompt_path)
.unwrap_or_default()
.contains("tokensave");
if has_rules {
dc.pass("Vibe prompt contains tokensave rules");
} else {
dc.fail("Vibe prompt missing tokensave rules — run `tokensave install --agent vibe`");
}
} else {
dc.warn("Vibe prompt does not exist");
}
}