use std::path::{Path, PathBuf};
use crate::compiler::mcp::{HeaderValue, McpTransport};
use crate::error::{ConfigError, MarsError};
use crate::lock::ItemKind;
use crate::types::DestPath;
use super::{ConfigEntry, HookEntry, McpServerEntry, TargetAdapter, hook_command};
#[derive(Debug)]
pub struct ClaudeAdapter;
impl TargetAdapter for ClaudeAdapter {
fn name(&self) -> &str {
".claude"
}
fn skill_variant_key(&self) -> Option<&str> {
Some("claude")
}
fn default_dest_path(&self, kind: ItemKind, name: &str) -> Option<DestPath> {
match kind {
ItemKind::Skill => Some(DestPath::from(format!("skills/{name}").as_str())),
_ => None,
}
}
fn write_config_entries(
&self,
entries: &[ConfigEntry],
target_dir: &Path,
) -> Result<Vec<PathBuf>, MarsError> {
let mut written = Vec::new();
let mcp_servers: Vec<&McpServerEntry> = entries
.iter()
.filter_map(|e| {
if let ConfigEntry::McpServer(s) = e {
Some(s)
} else {
None
}
})
.collect();
let hooks: Vec<&HookEntry> = entries
.iter()
.filter_map(|e| {
if let ConfigEntry::Hook(h) = e {
Some(h)
} else {
None
}
})
.collect();
if !mcp_servers.is_empty() {
let path = write_mcp_json(target_dir, &mcp_servers)?;
written.push(path);
}
if !hooks.is_empty() {
let path = write_hooks_settings(target_dir, &hooks)?;
written.push(path);
}
Ok(written)
}
fn remove_config_entries(
&self,
entry_keys: &[String],
target_dir: &Path,
) -> Result<(), MarsError> {
remove_mcp_entries_by_key(entry_keys, target_dir)?;
remove_hook_entries_by_key(entry_keys, target_dir)?;
Ok(())
}
}
fn write_mcp_json(target_dir: &Path, servers: &[&McpServerEntry]) -> Result<PathBuf, MarsError> {
let path = target_dir.join(".mcp.json");
let mut root: serde_json::Value = if path.is_file() {
let raw = std::fs::read_to_string(&path).map_err(MarsError::from)?;
serde_json::from_str(&raw).unwrap_or_else(|_| serde_json::json!({}))
} else {
serde_json::json!({})
};
let mcp_obj = root
.as_object_mut()
.ok_or_else(|| {
MarsError::Config(crate::error::ConfigError::Invalid {
message: format!("{} is not a JSON object", path.display()),
})
})?
.entry("mcpServers")
.or_insert_with(|| serde_json::json!({}));
let mcp_map = mcp_obj.as_object_mut().ok_or_else(|| {
MarsError::Config(crate::error::ConfigError::Invalid {
message: format!("{}: mcpServers is not an object", path.display()),
})
})?;
for server in servers {
let mut entry = match server.transport {
McpTransport::Stdio => serde_json::json!({
"command": server.command,
"args": server.args,
}),
McpTransport::Http => {
let mut http_entry = serde_json::json!({
"type": "http",
"url": server.url,
});
if !server.headers.is_empty() {
let headers_obj: serde_json::Map<String, serde_json::Value> = server
.headers
.iter()
.map(|(k, v)| {
let value = match v {
HeaderValue::EnvRef(env_ref) => serde_json::Value::String(format!(
"${{{}}}",
env_ref.var_name()
)),
HeaderValue::Plain(plain) => {
serde_json::Value::String(plain.clone())
}
};
(k.clone(), value)
})
.collect();
http_entry["headers"] = serde_json::Value::Object(headers_obj);
}
http_entry
}
};
if !server.env.is_empty() {
let env_obj: serde_json::Map<String, serde_json::Value> = server
.env
.iter()
.map(|(k, v)| (k.clone(), serde_json::Value::String(format!("${{{v}}}"))))
.collect();
entry["env"] = serde_json::Value::Object(env_obj);
}
mcp_map.insert(server.name.clone(), entry);
}
let content = serde_json::to_string_pretty(&root).map_err(|e| {
MarsError::Config(crate::error::ConfigError::Invalid {
message: format!("failed to serialize {}: {e}", path.display()),
})
})?;
crate::fs::atomic_write(&path, content.as_bytes())?;
Ok(path)
}
fn remove_mcp_entries_by_key(entry_keys: &[String], target_dir: &Path) -> Result<(), MarsError> {
let path = target_dir.join(".mcp.json");
if !path.is_file() {
return Ok(());
}
let raw = std::fs::read_to_string(&path).map_err(MarsError::from)?;
let mut root: serde_json::Value =
serde_json::from_str(&raw).unwrap_or_else(|_| serde_json::json!({}));
if let Some(mcp_map) = root
.as_object_mut()
.and_then(|o| o.get_mut("mcpServers"))
.and_then(|v| v.as_object_mut())
{
for key in entry_keys {
if let Some(name) = key.strip_prefix("mcp:") {
mcp_map.remove(name);
}
}
}
let content = serde_json::to_string_pretty(&root).map_err(|e| {
MarsError::Config(crate::error::ConfigError::Invalid {
message: format!("failed to serialize {}: {e}", path.display()),
})
})?;
crate::fs::atomic_write(&path, content.as_bytes())?;
Ok(())
}
fn write_hooks_settings(target_dir: &Path, hooks: &[&HookEntry]) -> Result<PathBuf, MarsError> {
let path = target_dir.join("settings.json");
let mut root: serde_json::Value = if path.is_file() {
let raw = std::fs::read_to_string(&path).map_err(MarsError::from)?;
serde_json::from_str(&raw).unwrap_or_else(|_| serde_json::json!({}))
} else {
serde_json::json!({})
};
let hooks_section = root
.as_object_mut()
.ok_or_else(|| {
MarsError::Config(crate::error::ConfigError::Invalid {
message: format!("{} is not a JSON object", path.display()),
})
})?
.entry("hooks")
.or_insert_with(|| serde_json::json!({}));
let hooks_map = hooks_section.as_object_mut().ok_or_else(|| {
MarsError::Config(crate::error::ConfigError::Invalid {
message: format!("{}: hooks is not an object", path.display()),
})
})?;
for hook in hooks {
let native_event = &hook.native_event;
let command_entry = serde_json::json!({
"type": "command",
"command": hook_command(&hook.script_path),
});
let hook_binding = serde_json::json!({
"matcher": "",
"hooks": [command_entry],
});
let event_hooks = hooks_map
.entry(native_event.clone())
.or_insert_with(|| serde_json::json!([]))
.as_array_mut()
.ok_or_else(|| {
MarsError::Config(ConfigError::Invalid {
message: format!("{}: hooks.{native_event} is not an array", path.display()),
})
})?;
remove_managed_hook_bindings(event_hooks, &hook.name);
event_hooks.push(hook_binding);
}
let content = serde_json::to_string_pretty(&root).map_err(|e| {
MarsError::Config(crate::error::ConfigError::Invalid {
message: format!("failed to serialize {}: {e}", path.display()),
})
})?;
crate::fs::atomic_write(&path, content.as_bytes())?;
Ok(path)
}
fn remove_managed_hook_bindings(bindings: &mut Vec<serde_json::Value>, hook_name: &str) {
bindings.retain(|binding| {
let Some(inner_hooks) = binding.get("hooks").and_then(|h| h.as_array()) else {
return true;
};
!inner_hooks.iter().any(|h| {
h.get("command")
.and_then(|c| c.as_str())
.map(|cmd| is_managed_hook_command_for(cmd, hook_name))
.unwrap_or(false)
})
});
}
fn is_managed_hook_command_for(command: &str, hook_name: &str) -> bool {
let normalized = command.replace('\\', "/").replace("//", "/");
normalized.contains(&format!("/hooks/{hook_name}/"))
}
fn remove_hook_entries_by_key(entry_keys: &[String], target_dir: &Path) -> Result<(), MarsError> {
let path = target_dir.join("settings.json");
if !path.is_file() {
return Ok(());
}
let hook_keys: Vec<(String, &str)> = entry_keys
.iter()
.filter_map(|k| {
let rest = k.strip_prefix("hook:")?;
let (event, name) = rest.split_once(':')?;
Some((claude_hook_event(event)?.to_string(), name))
})
.collect();
if hook_keys.is_empty() {
return Ok(());
}
let raw = std::fs::read_to_string(&path).map_err(MarsError::from)?;
let mut root: serde_json::Value =
serde_json::from_str(&raw).unwrap_or_else(|_| serde_json::json!({}));
if let Some(hooks_map) = root
.as_object_mut()
.and_then(|o| o.get_mut("hooks"))
.and_then(|v| v.as_object_mut())
{
for (event, name) in &hook_keys {
if let Some(event_hooks) = hooks_map.get_mut(event)
&& let Some(arr) = event_hooks.as_array_mut()
{
arr.retain(|binding| {
let Some(inner_hooks) = binding.get("hooks").and_then(|h| h.as_array()) else {
return true;
};
!inner_hooks.iter().any(|h| {
h.get("command")
.and_then(|c| c.as_str())
.map(|cmd| {
is_managed_hook_command_for(cmd, name)
})
.unwrap_or(false)
})
});
}
}
}
let content = serde_json::to_string_pretty(&root).map_err(|e| {
MarsError::Config(crate::error::ConfigError::Invalid {
message: format!("failed to serialize {}: {e}", path.display()),
})
})?;
crate::fs::atomic_write(&path, content.as_bytes())?;
Ok(())
}
fn claude_hook_event(event: &str) -> Option<&'static str> {
match event {
"session.start" => Some("SessionStart"),
"session.end" => Some("SessionStop"),
"tool.pre" => Some("PreToolUse"),
"tool.post" => Some("PostToolUse"),
_ => None,
}
}
#[cfg(test)]
mod tests {
use super::*;
use indexmap::IndexMap;
use tempfile::TempDir;
fn make_mcp_entry(name: &str) -> ConfigEntry {
ConfigEntry::McpServer(McpServerEntry {
name: name.to_string(),
transport: McpTransport::Stdio,
command: Some("npx".to_string()),
args: vec!["-y".to_string(), "some-mcp@latest".to_string()],
env: IndexMap::new(),
url: None,
headers: IndexMap::new(),
})
}
fn make_mcp_entry_with_env(name: &str, env_key: &str, env_var: &str) -> ConfigEntry {
let mut env = IndexMap::new();
env.insert(env_key.to_string(), env_var.to_string());
ConfigEntry::McpServer(McpServerEntry {
name: name.to_string(),
transport: McpTransport::Stdio,
command: Some("npx".to_string()),
args: vec![],
env,
url: None,
headers: IndexMap::new(),
})
}
fn make_http_mcp_entry(name: &str) -> ConfigEntry {
let mut headers = IndexMap::new();
headers.insert(
"Authorization".to_string(),
HeaderValue::EnvRef(crate::compiler::mcp::EnvRef::Env {
var: "API_TOKEN".to_string(),
}),
);
headers.insert(
"X-Custom".to_string(),
HeaderValue::Plain("static-value".to_string()),
);
ConfigEntry::McpServer(McpServerEntry {
name: name.to_string(),
transport: McpTransport::Http,
command: None,
args: vec![],
env: IndexMap::new(),
url: Some("https://api.example.com/mcp".to_string()),
headers,
})
}
fn make_hook_entry(name: &str, event: &str, native: &str) -> ConfigEntry {
ConfigEntry::Hook(HookEntry {
name: name.to_string(),
event: event.to_string(),
native_event: native.to_string(),
script_path: format!("/hooks/{name}/run.sh"),
order: 0,
})
}
fn make_hook_entry_with_path(
name: &str,
event: &str,
native: &str,
script_path: &str,
) -> ConfigEntry {
ConfigEntry::Hook(HookEntry {
name: name.to_string(),
event: event.to_string(),
native_event: native.to_string(),
script_path: script_path.to_string(),
order: 0,
})
}
#[test]
fn write_mcp_creates_mcp_json() {
let tmp = TempDir::new().unwrap();
std::fs::create_dir_all(tmp.path()).unwrap();
let adapter = ClaudeAdapter;
let entries = vec![make_mcp_entry("context7")];
let written = adapter.write_config_entries(&entries, tmp.path()).unwrap();
assert_eq!(written.len(), 1);
assert!(tmp.path().join(".mcp.json").exists());
let raw = std::fs::read_to_string(tmp.path().join(".mcp.json")).unwrap();
let json: serde_json::Value = serde_json::from_str(&raw).unwrap();
assert!(json["mcpServers"]["context7"].is_object());
assert_eq!(json["mcpServers"]["context7"]["command"], "npx");
}
#[test]
fn write_mcp_merges_with_existing() {
let tmp = TempDir::new().unwrap();
let existing = serde_json::json!({
"mcpServers": { "existing-server": { "command": "old" } }
});
std::fs::write(
tmp.path().join(".mcp.json"),
serde_json::to_string_pretty(&existing).unwrap(),
)
.unwrap();
let adapter = ClaudeAdapter;
let entries = vec![make_mcp_entry("new-server")];
adapter.write_config_entries(&entries, tmp.path()).unwrap();
let raw = std::fs::read_to_string(tmp.path().join(".mcp.json")).unwrap();
let json: serde_json::Value = serde_json::from_str(&raw).unwrap();
assert!(json["mcpServers"]["existing-server"].is_object());
assert!(json["mcpServers"]["new-server"].is_object());
}
#[test]
fn write_mcp_env_renders_as_interpolation() {
let tmp = TempDir::new().unwrap();
let adapter = ClaudeAdapter;
let entries = vec![make_mcp_entry_with_env("server", "API_KEY", "MY_SECRET")];
adapter.write_config_entries(&entries, tmp.path()).unwrap();
let raw = std::fs::read_to_string(tmp.path().join(".mcp.json")).unwrap();
let json: serde_json::Value = serde_json::from_str(&raw).unwrap();
assert_eq!(
json["mcpServers"]["server"]["env"]["API_KEY"],
"${MY_SECRET}"
);
}
#[test]
fn write_mcp_http_renders_type_url_and_headers() {
let tmp = TempDir::new().unwrap();
let adapter = ClaudeAdapter;
adapter
.write_config_entries(&[make_http_mcp_entry("remote-server")], tmp.path())
.unwrap();
let raw = std::fs::read_to_string(tmp.path().join(".mcp.json")).unwrap();
let json: serde_json::Value = serde_json::from_str(&raw).unwrap();
let server = &json["mcpServers"]["remote-server"];
assert_eq!(server["type"], "http");
assert_eq!(server["url"], "https://api.example.com/mcp");
assert_eq!(server["headers"]["Authorization"], "${API_TOKEN}");
assert_eq!(server["headers"]["X-Custom"], "static-value");
assert!(server["command"].is_null());
}
#[test]
fn write_hooks_creates_settings_json() {
let tmp = TempDir::new().unwrap();
let adapter = ClaudeAdapter;
let entries = vec![make_hook_entry("audit", "tool.pre", "PreToolUse")];
let written = adapter.write_config_entries(&entries, tmp.path()).unwrap();
assert_eq!(written.len(), 1);
assert!(tmp.path().join("settings.json").exists());
let raw = std::fs::read_to_string(tmp.path().join("settings.json")).unwrap();
let json: serde_json::Value = serde_json::from_str(&raw).unwrap();
assert!(json["hooks"]["PreToolUse"].is_array());
assert!(!json["hooks"]["PreToolUse"].as_array().unwrap().is_empty());
}
#[test]
fn write_hooks_replaces_existing_managed_hook_with_same_event_and_name() {
let tmp = TempDir::new().unwrap();
let adapter = ClaudeAdapter;
adapter
.write_config_entries(
&[make_hook_entry_with_path(
"audit",
"tool.pre",
"PreToolUse",
"/old/hooks/audit/run.sh",
)],
tmp.path(),
)
.unwrap();
adapter
.write_config_entries(
&[make_hook_entry_with_path(
"audit",
"tool.pre",
"PreToolUse",
"/new/hooks/audit/run.sh",
)],
tmp.path(),
)
.unwrap();
let raw = std::fs::read_to_string(tmp.path().join("settings.json")).unwrap();
let json: serde_json::Value = serde_json::from_str(&raw).unwrap();
let hooks = json["hooks"]["PreToolUse"].as_array().unwrap();
assert_eq!(hooks.len(), 1);
let command = hooks[0]["hooks"][0]["command"].as_str().unwrap();
assert!(command.contains("/new/hooks/audit/"));
}
#[test]
fn remove_mcp_entries_removes_by_name() {
let tmp = TempDir::new().unwrap();
let adapter = ClaudeAdapter;
let entries = vec![make_mcp_entry("context7"), make_mcp_entry("other")];
adapter.write_config_entries(&entries, tmp.path()).unwrap();
adapter
.remove_config_entries(&["mcp:context7".to_string()], tmp.path())
.unwrap();
let raw = std::fs::read_to_string(tmp.path().join(".mcp.json")).unwrap();
let json: serde_json::Value = serde_json::from_str(&raw).unwrap();
assert!(json["mcpServers"]["context7"].is_null());
assert!(json["mcpServers"]["other"].is_object());
}
#[test]
fn write_mcp_and_hooks_both_written() {
let tmp = TempDir::new().unwrap();
let adapter = ClaudeAdapter;
let entries = vec![
make_mcp_entry("context7"),
make_hook_entry("audit", "tool.pre", "PreToolUse"),
];
let written = adapter.write_config_entries(&entries, tmp.path()).unwrap();
assert_eq!(written.len(), 2);
assert!(tmp.path().join(".mcp.json").exists());
assert!(tmp.path().join("settings.json").exists());
}
#[test]
fn remove_hook_entries_matches_backslash_commands() {
let tmp = TempDir::new().unwrap();
let existing = serde_json::json!({
"hooks": {
"PreToolUse": [
{
"matcher": "",
"hooks": [
{ "type": "command", "command": "bash \"C:\\\\pkg\\\\hooks\\\\audit\\\\run.sh\"" }
]
},
{
"matcher": "",
"hooks": [
{ "type": "command", "command": "bash \"C:\\\\pkg\\\\hooks\\\\audit-extended\\\\run.sh\"" }
]
}
]
}
});
std::fs::write(
tmp.path().join("settings.json"),
serde_json::to_string_pretty(&existing).unwrap(),
)
.unwrap();
remove_hook_entries_by_key(&["hook:tool.pre:audit".to_string()], tmp.path()).unwrap();
let raw = std::fs::read_to_string(tmp.path().join("settings.json")).unwrap();
let json: serde_json::Value = serde_json::from_str(&raw).unwrap();
let hooks = json["hooks"]["PreToolUse"].as_array().unwrap();
assert_eq!(hooks.len(), 1);
}
}