use crate::config::ContextMount;
use crate::config::Mount;
use crate::config::MountDirsV2;
use crate::config::ReferenceEntry;
use crate::config::ReferenceMount;
use crate::config::RepoConfigV2;
use crate::config::SyncStrategy;
use crate::config::ThoughtsMount;
use crate::mount::MountSpace;
use crate::utils::paths;
use anyhow::Context;
use anyhow::Result;
use atomicwrites::AtomicFile;
use atomicwrites::OverwriteBehavior;
use std::fs;
use std::io::Write;
use std::path::Path;
use std::path::PathBuf;
#[derive(Debug, Clone)]
pub struct DesiredState {
pub mount_dirs: MountDirsV2,
pub thoughts_mount: Option<ThoughtsMount>,
pub context_mounts: Vec<ContextMount>,
pub references: Vec<ReferenceMount>,
pub was_v1: bool, }
impl DesiredState {
pub fn find_mount(&self, space: &MountSpace) -> Option<Mount> {
match space {
MountSpace::Thoughts => self.thoughts_mount.as_ref().map(|tm| Mount::Git {
url: tm.remote.clone(),
sync: tm.sync,
subpath: tm.subpath.clone(),
}),
MountSpace::Context(mount_path) => self
.context_mounts
.iter()
.find(|cm| &cm.mount_path == mount_path)
.map(|cm| Mount::Git {
url: cm.remote.clone(),
sync: cm.sync,
subpath: cm.subpath.clone(),
}),
MountSpace::Reference {
org_path: _,
repo: _,
ref_key: _,
} => {
None
}
}
}
pub fn get_mount_target(&self, space: &MountSpace, repo_root: &Path) -> PathBuf {
repo_root
.join(".thoughts-data")
.join(space.relative_path(&self.mount_dirs))
}
}
pub struct RepoConfigManager {
repo_root: PathBuf,
}
impl RepoConfigManager {
#[expect(
clippy::expect_used,
reason = "current_dir failure indicates a fatal system state; panicking is appropriate"
)]
pub fn new(repo_root: PathBuf) -> Self {
let abs = if repo_root.is_absolute() {
repo_root
} else {
std::fs::canonicalize(&repo_root).unwrap_or_else(|_| {
std::env::current_dir()
.expect("Failed to determine current directory for path normalization")
.join(&repo_root)
})
};
Self { repo_root: abs }
}
pub fn load_desired_state(&self) -> Result<Option<DesiredState>> {
let config_path = paths::get_repo_config_path(&self.repo_root);
if !config_path.exists() {
return Ok(None);
}
let raw = std::fs::read_to_string(&config_path)?;
let v: serde_json::Value = serde_json::from_str(&raw)?;
let version = v.get("version").and_then(|x| x.as_str()).unwrap_or("1.0");
if version == "2.0" {
let v2: RepoConfigV2 = serde_json::from_str(&raw)?;
let refs = v2
.references
.into_iter()
.map(|e| match e {
ReferenceEntry::Simple(url) => ReferenceMount {
remote: url,
description: None,
ref_name: None,
},
ReferenceEntry::WithMetadata(rm) => rm,
})
.collect();
return Ok(Some(DesiredState {
mount_dirs: v2.mount_dirs,
thoughts_mount: v2.thoughts_mount,
context_mounts: v2.context_mounts,
references: refs,
was_v1: false,
}));
}
anyhow::bail!(
"Unsupported legacy config version (v1). V1 configurations are no longer supported. \
Please upgrade to a v2 configuration format."
);
}
fn validate_remote(remote: &str) -> Result<()> {
if remote.starts_with("./") {
return Ok(());
}
if !remote.starts_with("git@")
&& !remote.starts_with("https://")
&& !remote.starts_with("ssh://")
{
anyhow::bail!(
"Invalid remote URL: {remote}. Must be a git URL or relative path starting with ./"
);
}
Ok(())
}
pub fn load_v2_or_bail(&self) -> Result<RepoConfigV2> {
let config_path = paths::get_repo_config_path(&self.repo_root);
if !config_path.exists() {
anyhow::bail!("No repository configuration found. Run 'thoughts init' first.");
}
let raw = std::fs::read_to_string(&config_path)?;
let v: serde_json::Value = serde_json::from_str(&raw)?;
let version = v.get("version").and_then(|x| x.as_str()).unwrap_or("1.0");
if version == "2.0" {
let v2: RepoConfigV2 = serde_json::from_str(&raw)?;
Ok(v2)
} else {
anyhow::bail!(
"Repository is using v1 configuration. Please migrate to v2 configuration format."
);
}
}
pub fn save_v2(&self, config: &RepoConfigV2) -> Result<()> {
let config_path = paths::get_repo_config_path(&self.repo_root);
if let Some(parent) = config_path.parent() {
fs::create_dir_all(parent)
.with_context(|| format!("Failed to create directory {}", parent.display()))?;
}
let json =
serde_json::to_string_pretty(config).context("Failed to serialize configuration")?;
AtomicFile::new(&config_path, OverwriteBehavior::AllowOverwrite)
.write(|f| f.write_all(json.as_bytes()))
.with_context(|| format!("Failed to write config to {}", config_path.display()))?;
Ok(())
}
pub fn ensure_v2_default(&self) -> Result<RepoConfigV2> {
let config_path = paths::get_repo_config_path(&self.repo_root);
if config_path.exists() {
let raw = std::fs::read_to_string(&config_path)?;
let v: serde_json::Value = serde_json::from_str(&raw)?;
let version = v.get("version").and_then(|x| x.as_str()).unwrap_or("1.0");
if version == "2.0" {
return serde_json::from_str(&raw).context("Failed to parse v2 configuration");
}
anyhow::bail!(
"Unsupported legacy config version (v1). V1 configurations are no longer supported. \
Please manually migrate to v2 format or delete the config and reinitialize."
);
}
let default_config = RepoConfigV2 {
version: "2.0".to_string(),
mount_dirs: MountDirsV2::default(),
thoughts_mount: None,
context_mounts: vec![],
references: vec![],
};
self.save_v2(&default_config)?;
Ok(default_config)
}
pub fn validate_v2_soft(&self, cfg: &RepoConfigV2) -> Vec<String> {
let mut warnings = Vec::new();
for r in &cfg.references {
let url = match r {
ReferenceEntry::Simple(s) => s.as_str(),
ReferenceEntry::WithMetadata(rm) => rm.remote.as_str(),
};
if let Err(e) = crate::config::validation::validate_reference_url(url) {
warnings.push(format!("Invalid reference '{url}': {e}"));
}
}
warnings
}
pub fn peek_config_version(&self) -> Result<Option<String>> {
let config_path = paths::get_repo_config_path(&self.repo_root);
if !config_path.exists() {
return Ok(None);
}
let raw = std::fs::read_to_string(&config_path)?;
let v: serde_json::Value = serde_json::from_str(&raw)?;
Ok(v.get("version")
.and_then(|x| x.as_str())
.map(std::string::ToString::to_string))
}
pub fn validate_v2_hard(&self, cfg: &RepoConfigV2) -> Result<Vec<String>> {
use crate::config::validation::canonical_reference_instance_key;
use crate::config::validation::validate_pinned_ref_full_name;
use crate::config::validation::validate_reference_url;
if cfg.version != "2.0" {
anyhow::bail!("Unsupported configuration version: {}", cfg.version);
}
let m = &cfg.mount_dirs;
for (name, val) in [
("thoughts", &m.thoughts),
("context", &m.context),
("references", &m.references),
] {
if val.trim().is_empty() {
anyhow::bail!("Mount directory '{name}' cannot be empty");
}
if val == ".thoughts-data" {
anyhow::bail!("Mount directory '{name}' cannot be named '.thoughts-data'");
}
if val == "." || val == ".." {
anyhow::bail!("Mount directory '{name}' cannot be '.' or '..'");
}
if val.contains('/') || val.contains('\\') {
anyhow::bail!("Mount directory '{name}' must be a single path segment (got {val})");
}
}
if m.thoughts == m.context || m.thoughts == m.references || m.context == m.references {
anyhow::bail!("Mount directories must be distinct (thoughts/context/references)");
}
if let Some(tm) = &cfg.thoughts_mount {
Self::validate_remote(&tm.remote)?;
}
let mut warnings = Vec::new();
let mut seen_mount_paths = std::collections::HashSet::new();
for cm in &cfg.context_mounts {
if !seen_mount_paths.insert(&cm.mount_path) {
anyhow::bail!("Duplicate context mount path: {}", cm.mount_path);
}
let mp = cm.mount_path.trim();
if mp.is_empty() {
anyhow::bail!("Context mount path cannot be empty");
}
if mp == "." || mp == ".." {
anyhow::bail!("Context mount path cannot be '.' or '..'");
}
if mp.contains('/') || mp.contains('\\') {
anyhow::bail!(
"Context mount path must be a single path segment (got {})",
cm.mount_path
);
}
let m = &cfg.mount_dirs;
if mp == m.thoughts || mp == m.context || mp == m.references {
anyhow::bail!(
"Context mount path '{}' cannot conflict with configured mount_dirs names ('{}', '{}', '{}')",
cm.mount_path,
m.thoughts,
m.context,
m.references
);
}
Self::validate_remote(&cm.remote)?;
if matches!(cm.sync, SyncStrategy::None) {
warnings.push(format!(
"Context mount '{}' has sync:None; allowed but discouraged. Consider SyncStrategy::Auto.",
cm.mount_path
));
}
}
let mut seen_refs = std::collections::HashSet::new();
for r in &cfg.references {
let (url, ref_name) = match r {
ReferenceEntry::Simple(s) => (s.as_str(), None),
ReferenceEntry::WithMetadata(rm) => (rm.remote.as_str(), rm.ref_name.as_deref()),
};
validate_reference_url(url).with_context(|| format!("Invalid reference '{url}'"))?;
if let Some(ref_name) = ref_name {
if ref_name.starts_with("refs/remotes/") {
warnings.push(format!(
"Reference '{url}' uses legacy pinned ref '{ref_name}'. New references should use refs/heads/* or refs/tags/*; refs/remotes/* is a local remote-tracking namespace."
));
}
validate_pinned_ref_full_name(ref_name).with_context(|| {
format!("Invalid pinned ref '{ref_name}' for reference '{url}'")
})?;
}
let key = canonical_reference_instance_key(url, ref_name)?;
if !seen_refs.insert(key) {
anyhow::bail!("Duplicate reference detected: {url}");
}
}
Ok(warnings)
}
pub fn save_v2_validated(&self, config: &RepoConfigV2) -> Result<Vec<String>> {
let warnings = self.validate_v2_hard(config)?;
self.save_v2(config)?;
Ok(warnings)
}
}
#[cfg(test)]
mod tests {
use super::*;
use crate::utils::paths;
use tempfile::TempDir;
#[test]
fn test_load_desired_state_rejects_v1() {
let temp_dir = TempDir::new().unwrap();
let manager = RepoConfigManager::new(temp_dir.path().to_path_buf());
let v1_json = r#"{
"version": "1.0",
"mount_dirs": {"repository": "context", "personal": "personal"},
"requires": [],
"rules": []
}"#;
let config_path = paths::get_repo_config_path(temp_dir.path());
std::fs::create_dir_all(config_path.parent().unwrap()).unwrap();
std::fs::write(&config_path, v1_json).unwrap();
let result = manager.load_desired_state();
assert!(result.is_err());
assert!(result.unwrap_err().to_string().contains("v1"));
}
#[test]
fn test_v2_config_loading() {
let temp_dir = TempDir::new().unwrap();
let manager = RepoConfigManager::new(temp_dir.path().to_path_buf());
let v2_config = crate::config::RepoConfigV2 {
version: "2.0".to_string(),
mount_dirs: crate::config::MountDirsV2::default(),
thoughts_mount: Some(crate::config::ThoughtsMount {
remote: "git@github.com:user/thoughts.git".to_string(),
subpath: None,
sync: crate::config::SyncStrategy::Auto,
}),
context_mounts: vec![crate::config::ContextMount {
remote: "git@github.com:user/context.git".to_string(),
subpath: Some("docs".to_string()),
mount_path: "docs".to_string(),
sync: crate::config::SyncStrategy::Auto,
}],
references: vec![
ReferenceEntry::Simple("git@github.com:org/ref1.git".to_string()),
ReferenceEntry::Simple("https://github.com/org/ref2.git".to_string()),
],
};
let config_path = paths::get_repo_config_path(temp_dir.path());
std::fs::create_dir_all(config_path.parent().unwrap()).unwrap();
let json = serde_json::to_string_pretty(&v2_config).unwrap();
std::fs::write(&config_path, json).unwrap();
let desired_state = manager.load_desired_state().unwrap().unwrap();
assert!(!desired_state.was_v1);
assert!(desired_state.thoughts_mount.is_some());
assert_eq!(
desired_state.thoughts_mount.as_ref().unwrap().remote,
"git@github.com:user/thoughts.git"
);
assert_eq!(desired_state.context_mounts.len(), 1);
assert_eq!(desired_state.references.len(), 2);
}
#[test]
fn test_v2_references_normalize_to_reference_mount() {
let temp_dir = TempDir::new().unwrap();
let manager = RepoConfigManager::new(temp_dir.path().to_path_buf());
let json = r#"{
"version": "2.0",
"mount_dirs": {},
"context_mounts": [],
"references": [
"git@github.com:org/ref1.git",
{"remote": "https://github.com/org/ref2.git", "description": "Ref 2"}
]
}"#;
let config_path = paths::get_repo_config_path(temp_dir.path());
std::fs::create_dir_all(config_path.parent().unwrap()).unwrap();
std::fs::write(&config_path, json).unwrap();
let ds = manager.load_desired_state().unwrap().unwrap();
assert_eq!(ds.references.len(), 2);
assert_eq!(ds.references[0].remote, "git@github.com:org/ref1.git");
assert_eq!(ds.references[0].description, None);
assert_eq!(ds.references[1].remote, "https://github.com/org/ref2.git");
assert_eq!(ds.references[1].description.as_deref(), Some("Ref 2"));
}
#[test]
fn test_validate_v2_soft_handles_both_variants() {
let temp_dir = tempfile::TempDir::new().unwrap();
let mgr = RepoConfigManager::new(temp_dir.path().to_path_buf());
let cfg = RepoConfigV2 {
version: "2.0".into(),
mount_dirs: MountDirsV2::default(),
thoughts_mount: None,
context_mounts: vec![],
references: vec![
ReferenceEntry::Simple("https://github.com/org/repo".into()),
ReferenceEntry::WithMetadata(ReferenceMount {
remote: "git@github.com:org/repo.git:docs".into(), description: None,
ref_name: None,
}),
],
};
let warnings = mgr.validate_v2_soft(&cfg);
assert_eq!(warnings.len(), 1, "Expected one invalid reference warning");
assert!(warnings[0].contains("git@github.com:org/repo.git:docs"));
}
#[test]
fn test_peek_config_version_returns_none_when_no_config() {
let temp_dir = tempfile::TempDir::new().unwrap();
let mgr = RepoConfigManager::new(temp_dir.path().to_path_buf());
assert_eq!(mgr.peek_config_version().unwrap(), None);
}
#[test]
fn test_peek_config_version_returns_v1() {
let temp_dir = tempfile::TempDir::new().unwrap();
let mgr = RepoConfigManager::new(temp_dir.path().to_path_buf());
let v1_json = r#"{"version": "1.0", "mount_dirs": {}, "requires": [], "rules": []}"#;
let config_path = paths::get_repo_config_path(temp_dir.path());
std::fs::create_dir_all(config_path.parent().unwrap()).unwrap();
std::fs::write(&config_path, v1_json).unwrap();
assert_eq!(mgr.peek_config_version().unwrap(), Some("1.0".to_string()));
}
#[test]
fn test_peek_config_version_returns_v2() {
let temp_dir = tempfile::TempDir::new().unwrap();
let mgr = RepoConfigManager::new(temp_dir.path().to_path_buf());
let v2_config = RepoConfigV2 {
version: "2.0".to_string(),
mount_dirs: MountDirsV2::default(),
thoughts_mount: None,
context_mounts: vec![],
references: vec![],
};
mgr.save_v2(&v2_config).unwrap();
assert_eq!(mgr.peek_config_version().unwrap(), Some("2.0".to_string()));
}
#[test]
fn test_validate_v2_hard_rejects_invalid_version() {
let temp_dir = tempfile::TempDir::new().unwrap();
let mgr = RepoConfigManager::new(temp_dir.path().to_path_buf());
let cfg = RepoConfigV2 {
version: "3.0".to_string(),
mount_dirs: MountDirsV2::default(),
thoughts_mount: None,
context_mounts: vec![],
references: vec![],
};
let result = mgr.validate_v2_hard(&cfg);
assert!(result.is_err());
assert!(
result
.unwrap_err()
.to_string()
.contains("Unsupported configuration version: 3.0")
);
}
#[test]
fn test_validate_v2_hard_rejects_pinned_ref_with_whitespace_or_trailing_slash() {
let temp_dir = tempfile::TempDir::new().unwrap();
let mgr = RepoConfigManager::new(temp_dir.path().to_path_buf());
for bad_ref in [" refs/heads/main", "refs/heads/main ", "refs/heads/main/"] {
let cfg = RepoConfigV2 {
version: "2.0".to_string(),
mount_dirs: MountDirsV2::default(),
thoughts_mount: None,
context_mounts: vec![],
references: vec![ReferenceEntry::WithMetadata(ReferenceMount {
remote: "https://github.com/org/repo".to_string(),
description: None,
ref_name: Some(bad_ref.to_string()),
})],
};
assert!(
mgr.validate_v2_hard(&cfg).is_err(),
"expected {bad_ref:?} to fail hard validation"
);
}
}
#[test]
fn test_validate_v2_hard_rejects_empty_mount_dirs() {
let temp_dir = tempfile::TempDir::new().unwrap();
let mgr = RepoConfigManager::new(temp_dir.path().to_path_buf());
let cfg = RepoConfigV2 {
version: "2.0".to_string(),
mount_dirs: MountDirsV2 {
thoughts: String::new(),
context: "context".to_string(),
references: "references".to_string(),
},
thoughts_mount: None,
context_mounts: vec![],
references: vec![],
};
let result = mgr.validate_v2_hard(&cfg);
assert!(result.is_err());
assert!(result.unwrap_err().to_string().contains("cannot be empty"));
}
#[test]
fn test_validate_v2_hard_rejects_reserved_mount_dir_name() {
let temp_dir = tempfile::TempDir::new().unwrap();
let mgr = RepoConfigManager::new(temp_dir.path().to_path_buf());
let cfg = RepoConfigV2 {
version: "2.0".to_string(),
mount_dirs: MountDirsV2 {
thoughts: ".thoughts-data".to_string(),
context: "context".to_string(),
references: "references".to_string(),
},
thoughts_mount: None,
context_mounts: vec![],
references: vec![],
};
let result = mgr.validate_v2_hard(&cfg);
assert!(result.is_err());
assert!(result.unwrap_err().to_string().contains(".thoughts-data"));
}
#[test]
fn test_validate_v2_hard_rejects_dot_mount_dirs() {
let temp_dir = tempfile::TempDir::new().unwrap();
let mgr = RepoConfigManager::new(temp_dir.path().to_path_buf());
let cfg = RepoConfigV2 {
version: "2.0".to_string(),
mount_dirs: MountDirsV2 {
thoughts: ".".to_string(),
context: "context".to_string(),
references: "references".to_string(),
},
thoughts_mount: None,
context_mounts: vec![],
references: vec![],
};
let result = mgr.validate_v2_hard(&cfg);
assert!(result.is_err());
assert!(
result
.unwrap_err()
.to_string()
.contains("cannot be '.' or '..'")
);
}
#[test]
fn test_validate_v2_hard_rejects_multi_segment_mount_dirs() {
let temp_dir = tempfile::TempDir::new().unwrap();
let mgr = RepoConfigManager::new(temp_dir.path().to_path_buf());
let cfg = RepoConfigV2 {
version: "2.0".to_string(),
mount_dirs: MountDirsV2 {
thoughts: "sub/path".to_string(),
context: "context".to_string(),
references: "references".to_string(),
},
thoughts_mount: None,
context_mounts: vec![],
references: vec![],
};
let result = mgr.validate_v2_hard(&cfg);
assert!(result.is_err());
assert!(
result
.unwrap_err()
.to_string()
.contains("must be a single path segment")
);
}
#[test]
fn test_validate_v2_hard_rejects_duplicate_mount_dirs() {
let temp_dir = tempfile::TempDir::new().unwrap();
let mgr = RepoConfigManager::new(temp_dir.path().to_path_buf());
let cfg = RepoConfigV2 {
version: "2.0".to_string(),
mount_dirs: MountDirsV2 {
thoughts: "same".to_string(),
context: "same".to_string(),
references: "references".to_string(),
},
thoughts_mount: None,
context_mounts: vec![],
references: vec![],
};
let result = mgr.validate_v2_hard(&cfg);
assert!(result.is_err());
assert!(result.unwrap_err().to_string().contains("must be distinct"));
}
#[test]
fn test_validate_v2_hard_rejects_invalid_thoughts_mount_remote() {
let temp_dir = tempfile::TempDir::new().unwrap();
let mgr = RepoConfigManager::new(temp_dir.path().to_path_buf());
let cfg = RepoConfigV2 {
version: "2.0".to_string(),
mount_dirs: MountDirsV2::default(),
thoughts_mount: Some(ThoughtsMount {
remote: "invalid-url".to_string(),
subpath: None,
sync: SyncStrategy::Auto,
}),
context_mounts: vec![],
references: vec![],
};
let result = mgr.validate_v2_hard(&cfg);
assert!(result.is_err());
assert!(
result
.unwrap_err()
.to_string()
.contains("Invalid remote URL")
);
}
#[test]
fn test_validate_v2_hard_rejects_duplicate_context_mount_path() {
let temp_dir = tempfile::TempDir::new().unwrap();
let mgr = RepoConfigManager::new(temp_dir.path().to_path_buf());
let cfg = RepoConfigV2 {
version: "2.0".to_string(),
mount_dirs: MountDirsV2::default(),
thoughts_mount: None,
context_mounts: vec![
ContextMount {
remote: "git@github.com:org/repo1.git".to_string(),
subpath: None,
mount_path: "same".to_string(),
sync: SyncStrategy::Auto,
},
ContextMount {
remote: "git@github.com:org/repo2.git".to_string(),
subpath: None,
mount_path: "same".to_string(),
sync: SyncStrategy::Auto,
},
],
references: vec![],
};
let result = mgr.validate_v2_hard(&cfg);
assert!(result.is_err());
assert!(
result
.unwrap_err()
.to_string()
.contains("Duplicate context mount path")
);
}
#[test]
fn test_validate_v2_hard_rejects_invalid_context_remote() {
let temp_dir = tempfile::TempDir::new().unwrap();
let mgr = RepoConfigManager::new(temp_dir.path().to_path_buf());
let cfg = RepoConfigV2 {
version: "2.0".to_string(),
mount_dirs: MountDirsV2::default(),
thoughts_mount: None,
context_mounts: vec![ContextMount {
remote: "invalid-url".to_string(),
subpath: None,
mount_path: "mount1".to_string(),
sync: SyncStrategy::Auto,
}],
references: vec![],
};
let result = mgr.validate_v2_hard(&cfg);
assert!(result.is_err());
assert!(
result
.unwrap_err()
.to_string()
.contains("Invalid remote URL")
);
}
#[test]
fn test_validate_v2_hard_warns_on_sync_none() {
let temp_dir = tempfile::TempDir::new().unwrap();
let mgr = RepoConfigManager::new(temp_dir.path().to_path_buf());
let cfg = RepoConfigV2 {
version: "2.0".to_string(),
mount_dirs: MountDirsV2::default(),
thoughts_mount: None,
context_mounts: vec![ContextMount {
remote: "git@github.com:org/repo.git".to_string(),
subpath: None,
mount_path: "mount1".to_string(),
sync: SyncStrategy::None,
}],
references: vec![],
};
let result = mgr.validate_v2_hard(&cfg);
assert!(result.is_ok());
let warnings = result.unwrap();
assert_eq!(warnings.len(), 1);
assert!(warnings[0].contains("sync:None"));
assert!(warnings[0].contains("discouraged"));
}
#[test]
fn test_validate_v2_hard_rejects_invalid_reference_url() {
let temp_dir = tempfile::TempDir::new().unwrap();
let mgr = RepoConfigManager::new(temp_dir.path().to_path_buf());
let cfg = RepoConfigV2 {
version: "2.0".to_string(),
mount_dirs: MountDirsV2::default(),
thoughts_mount: None,
context_mounts: vec![],
references: vec![ReferenceEntry::Simple(
"git@github.com:org/repo.git:subpath".to_string(),
)],
};
let result = mgr.validate_v2_hard(&cfg);
assert!(result.is_err());
assert!(result.unwrap_err().to_string().contains("subpath"));
}
#[test]
fn test_validate_v2_hard_rejects_duplicate_references() {
let temp_dir = tempfile::TempDir::new().unwrap();
let mgr = RepoConfigManager::new(temp_dir.path().to_path_buf());
let cfg = RepoConfigV2 {
version: "2.0".to_string(),
mount_dirs: MountDirsV2::default(),
thoughts_mount: None,
context_mounts: vec![],
references: vec![
ReferenceEntry::Simple("git@github.com:Org/Repo.git".to_string()),
ReferenceEntry::Simple("https://github.com/org/repo".to_string()),
],
};
let result = mgr.validate_v2_hard(&cfg);
assert!(result.is_err());
assert!(
result
.unwrap_err()
.to_string()
.contains("Duplicate reference")
);
}
#[test]
fn test_validate_v2_hard_allows_same_repo_with_different_refs() {
let temp_dir = tempfile::TempDir::new().unwrap();
let mgr = RepoConfigManager::new(temp_dir.path().to_path_buf());
let cfg = RepoConfigV2 {
version: "2.0".to_string(),
mount_dirs: MountDirsV2::default(),
thoughts_mount: None,
context_mounts: vec![],
references: vec![
ReferenceEntry::WithMetadata(ReferenceMount {
remote: "https://github.com/org/repo".to_string(),
description: None,
ref_name: Some("refs/heads/main".to_string()),
}),
ReferenceEntry::WithMetadata(ReferenceMount {
remote: "git@github.com:Org/Repo.git".to_string(),
description: None,
ref_name: Some("refs/tags/v1.0.0".to_string()),
}),
],
};
let result = mgr.validate_v2_hard(&cfg);
assert!(result.is_ok());
}
#[test]
fn test_validate_v2_hard_rejects_duplicate_references_with_same_ref() {
let temp_dir = tempfile::TempDir::new().unwrap();
let mgr = RepoConfigManager::new(temp_dir.path().to_path_buf());
let cfg = RepoConfigV2 {
version: "2.0".to_string(),
mount_dirs: MountDirsV2::default(),
thoughts_mount: None,
context_mounts: vec![],
references: vec![
ReferenceEntry::WithMetadata(ReferenceMount {
remote: "https://github.com/org/repo".to_string(),
description: None,
ref_name: Some("refs/heads/main".to_string()),
}),
ReferenceEntry::WithMetadata(ReferenceMount {
remote: "git@github.com:Org/Repo.git".to_string(),
description: Some("duplicate".to_string()),
ref_name: Some("refs/heads/main".to_string()),
}),
],
};
let result = mgr.validate_v2_hard(&cfg);
assert!(result.is_err());
assert!(
result
.unwrap_err()
.to_string()
.contains("Duplicate reference")
);
}
#[test]
fn test_validate_v2_hard_rejects_duplicate_references_legacy_remotes_vs_heads() {
let temp_dir = tempfile::TempDir::new().unwrap();
let mgr = RepoConfigManager::new(temp_dir.path().to_path_buf());
let cfg = RepoConfigV2 {
version: "2.0".to_string(),
mount_dirs: MountDirsV2::default(),
thoughts_mount: None,
context_mounts: vec![],
references: vec![
ReferenceEntry::WithMetadata(ReferenceMount {
remote: "https://github.com/org/repo".to_string(),
description: None,
ref_name: Some("refs/remotes/origin/main".to_string()),
}),
ReferenceEntry::WithMetadata(ReferenceMount {
remote: "git@github.com:Org/Repo.git".to_string(),
description: None,
ref_name: Some("refs/heads/main".to_string()),
}),
],
};
let err = mgr.validate_v2_hard(&cfg).unwrap_err();
assert!(err.to_string().contains("Duplicate reference"));
}
#[test]
fn test_validate_v2_hard_rejects_shorthand_pinned_ref() {
let temp_dir = tempfile::TempDir::new().unwrap();
let mgr = RepoConfigManager::new(temp_dir.path().to_path_buf());
let cfg = RepoConfigV2 {
version: "2.0".to_string(),
mount_dirs: MountDirsV2::default(),
thoughts_mount: None,
context_mounts: vec![],
references: vec![ReferenceEntry::WithMetadata(ReferenceMount {
remote: "https://github.com/org/repo".to_string(),
description: None,
ref_name: Some("main".to_string()),
})],
};
let err = mgr.validate_v2_hard(&cfg).unwrap_err();
assert!(
format!("{err:#}").contains("Pinned refs must be full ref names"),
"unexpected error chain: {err:#}"
);
}
#[test]
fn test_validate_v2_hard_rejects_incomplete_pinned_ref_prefix() {
let temp_dir = tempfile::TempDir::new().unwrap();
let mgr = RepoConfigManager::new(temp_dir.path().to_path_buf());
let cfg = RepoConfigV2 {
version: "2.0".to_string(),
mount_dirs: MountDirsV2::default(),
thoughts_mount: None,
context_mounts: vec![],
references: vec![ReferenceEntry::WithMetadata(ReferenceMount {
remote: "https://github.com/org/repo".to_string(),
description: None,
ref_name: Some("refs/heads/".to_string()),
})],
};
assert!(mgr.validate_v2_hard(&cfg).is_err());
}
#[test]
fn test_validate_v2_hard_warns_on_legacy_refs_remotes() {
let temp_dir = tempfile::TempDir::new().unwrap();
let mgr = RepoConfigManager::new(temp_dir.path().to_path_buf());
let cfg = RepoConfigV2 {
version: "2.0".to_string(),
mount_dirs: MountDirsV2::default(),
thoughts_mount: None,
context_mounts: vec![],
references: vec![ReferenceEntry::WithMetadata(ReferenceMount {
remote: "https://github.com/org/repo".to_string(),
description: None,
ref_name: Some("refs/remotes/origin/main".to_string()),
})],
};
let warnings = mgr
.validate_v2_hard(&cfg)
.expect("legacy refs/remotes should warn");
assert_eq!(warnings.len(), 1);
assert!(warnings[0].contains("refs/remotes/origin/main"));
assert!(warnings[0].contains("legacy pinned ref"));
}
#[test]
fn test_validate_v2_hard_accepts_valid_config() {
let temp_dir = tempfile::TempDir::new().unwrap();
let mgr = RepoConfigManager::new(temp_dir.path().to_path_buf());
let cfg = RepoConfigV2 {
version: "2.0".to_string(),
mount_dirs: MountDirsV2::default(),
thoughts_mount: Some(ThoughtsMount {
remote: "git@github.com:user/thoughts.git".to_string(),
subpath: None,
sync: SyncStrategy::Auto,
}),
context_mounts: vec![ContextMount {
remote: "git@github.com:org/context.git".to_string(),
subpath: Some("docs".to_string()),
mount_path: "docs".to_string(),
sync: SyncStrategy::Auto,
}],
references: vec![
ReferenceEntry::Simple("git@github.com:org/repo1.git".to_string()),
ReferenceEntry::WithMetadata(ReferenceMount {
remote: "https://github.com/org/repo2".to_string(),
description: Some("Reference 2".to_string()),
ref_name: None,
}),
],
};
let result = mgr.validate_v2_hard(&cfg);
assert!(result.is_ok());
let warnings = result.unwrap();
assert_eq!(warnings.len(), 0);
}
#[test]
fn test_save_v2_validated_fails_before_write_on_invalid() {
let temp_dir = tempfile::TempDir::new().unwrap();
let mgr = RepoConfigManager::new(temp_dir.path().to_path_buf());
let cfg = RepoConfigV2 {
version: "2.0".to_string(),
mount_dirs: MountDirsV2 {
thoughts: "same".to_string(),
context: "same".to_string(),
references: "references".to_string(),
},
thoughts_mount: None,
context_mounts: vec![],
references: vec![],
};
let result = mgr.save_v2_validated(&cfg);
assert!(result.is_err());
let config_path = paths::get_repo_config_path(temp_dir.path());
assert!(!config_path.exists());
}
#[test]
fn test_save_v2_validated_returns_warnings_on_valid() {
let temp_dir = tempfile::TempDir::new().unwrap();
let mgr = RepoConfigManager::new(temp_dir.path().to_path_buf());
let cfg = RepoConfigV2 {
version: "2.0".to_string(),
mount_dirs: MountDirsV2::default(),
thoughts_mount: None,
context_mounts: vec![ContextMount {
remote: "git@github.com:org/repo.git".to_string(),
subpath: None,
mount_path: "mount1".to_string(),
sync: SyncStrategy::None,
}],
references: vec![],
};
let result = mgr.save_v2_validated(&cfg);
assert!(result.is_ok());
let warnings = result.unwrap();
assert_eq!(warnings.len(), 1);
assert!(warnings[0].contains("sync:None"));
let config_path = paths::get_repo_config_path(temp_dir.path());
assert!(config_path.exists());
}
#[test]
fn test_ensure_v2_default_rejects_v1() {
let temp_dir = tempfile::TempDir::new().unwrap();
let mgr = RepoConfigManager::new(temp_dir.path().to_path_buf());
let v1_json = r#"{"version": "1.0", "mount_dirs": {}, "requires": [], "rules": []}"#;
let config_path = paths::get_repo_config_path(temp_dir.path());
std::fs::create_dir_all(config_path.parent().unwrap()).unwrap();
std::fs::write(&config_path, v1_json).unwrap();
let result = mgr.ensure_v2_default();
assert!(result.is_err());
assert!(result.unwrap_err().to_string().contains("v1"));
}
#[test]
fn test_validate_v2_hard_rejects_empty_context_mount_path() {
let temp_dir = tempfile::TempDir::new().unwrap();
let mgr = RepoConfigManager::new(temp_dir.path().to_path_buf());
let cfg = RepoConfigV2 {
version: "2.0".to_string(),
mount_dirs: MountDirsV2::default(),
thoughts_mount: None,
context_mounts: vec![ContextMount {
remote: "git@github.com:org/repo.git".to_string(),
subpath: None,
mount_path: " ".to_string(), sync: SyncStrategy::Auto,
}],
references: vec![],
};
let result = mgr.validate_v2_hard(&cfg);
assert!(result.is_err());
assert!(result.unwrap_err().to_string().contains("cannot be empty"));
}
#[test]
fn test_validate_v2_hard_rejects_dot_context_mount_path() {
let temp_dir = tempfile::TempDir::new().unwrap();
let mgr = RepoConfigManager::new(temp_dir.path().to_path_buf());
let cfg = RepoConfigV2 {
version: "2.0".to_string(),
mount_dirs: MountDirsV2::default(),
thoughts_mount: None,
context_mounts: vec![ContextMount {
remote: "git@github.com:org/repo.git".to_string(),
subpath: None,
mount_path: ".".to_string(),
sync: SyncStrategy::Auto,
}],
references: vec![],
};
let result = mgr.validate_v2_hard(&cfg);
assert!(result.is_err());
assert!(
result
.unwrap_err()
.to_string()
.contains("cannot be '.' or '..'")
);
}
#[test]
fn test_validate_v2_hard_rejects_dotdot_context_mount_path() {
let temp_dir = tempfile::TempDir::new().unwrap();
let mgr = RepoConfigManager::new(temp_dir.path().to_path_buf());
let cfg = RepoConfigV2 {
version: "2.0".to_string(),
mount_dirs: MountDirsV2::default(),
thoughts_mount: None,
context_mounts: vec![ContextMount {
remote: "git@github.com:org/repo.git".to_string(),
subpath: None,
mount_path: "..".to_string(),
sync: SyncStrategy::Auto,
}],
references: vec![],
};
let result = mgr.validate_v2_hard(&cfg);
assert!(result.is_err());
assert!(
result
.unwrap_err()
.to_string()
.contains("cannot be '.' or '..'")
);
}
#[test]
fn test_validate_v2_hard_rejects_slash_in_context_mount_path() {
let temp_dir = tempfile::TempDir::new().unwrap();
let mgr = RepoConfigManager::new(temp_dir.path().to_path_buf());
let cfg = RepoConfigV2 {
version: "2.0".to_string(),
mount_dirs: MountDirsV2::default(),
thoughts_mount: None,
context_mounts: vec![ContextMount {
remote: "git@github.com:org/repo.git".to_string(),
subpath: None,
mount_path: "sub/path".to_string(),
sync: SyncStrategy::Auto,
}],
references: vec![],
};
let result = mgr.validate_v2_hard(&cfg);
assert!(result.is_err());
assert!(
result
.unwrap_err()
.to_string()
.contains("single path segment")
);
}
#[test]
fn test_validate_v2_hard_rejects_backslash_in_context_mount_path() {
let temp_dir = tempfile::TempDir::new().unwrap();
let mgr = RepoConfigManager::new(temp_dir.path().to_path_buf());
let cfg = RepoConfigV2 {
version: "2.0".to_string(),
mount_dirs: MountDirsV2::default(),
thoughts_mount: None,
context_mounts: vec![ContextMount {
remote: "git@github.com:org/repo.git".to_string(),
subpath: None,
mount_path: "sub\\path".to_string(),
sync: SyncStrategy::Auto,
}],
references: vec![],
};
let result = mgr.validate_v2_hard(&cfg);
assert!(result.is_err());
assert!(
result
.unwrap_err()
.to_string()
.contains("single path segment")
);
}
#[test]
fn test_validate_v2_hard_accepts_valid_context_mount_path() {
let temp_dir = tempfile::TempDir::new().unwrap();
let mgr = RepoConfigManager::new(temp_dir.path().to_path_buf());
let cfg = RepoConfigV2 {
version: "2.0".to_string(),
mount_dirs: MountDirsV2::default(),
thoughts_mount: None,
context_mounts: vec![ContextMount {
remote: "git@github.com:org/repo.git".to_string(),
subpath: None,
mount_path: "docs".to_string(),
sync: SyncStrategy::Auto,
}],
references: vec![],
};
let result = mgr.validate_v2_hard(&cfg);
assert!(result.is_ok());
let warnings = result.unwrap();
assert_eq!(warnings.len(), 0);
}
#[test]
fn test_new_makes_absolute_when_given_relative_repo_root() {
let temp_dir = TempDir::new().unwrap();
let cwd_before = std::env::current_dir().unwrap();
std::env::set_current_dir(temp_dir.path()).unwrap();
std::fs::create_dir_all("repo").unwrap();
let mgr = RepoConfigManager::new(PathBuf::from("repo"));
assert!(mgr.peek_config_version().is_ok());
std::env::set_current_dir(cwd_before).unwrap();
}
#[test]
fn test_validate_v2_hard_rejects_trailing_slash_in_mount_dirs() {
let temp_dir = tempfile::TempDir::new().unwrap();
let mgr = RepoConfigManager::new(temp_dir.path().to_path_buf());
let cfg = RepoConfigV2 {
version: "2.0".to_string(),
mount_dirs: MountDirsV2 {
thoughts: "thoughts/".to_string(),
context: "context".to_string(),
references: "references".to_string(),
},
thoughts_mount: None,
context_mounts: vec![],
references: vec![],
};
let result = mgr.validate_v2_hard(&cfg);
assert!(
result.is_err(),
"trailing slash on thoughts should be rejected"
);
assert!(
result
.unwrap_err()
.to_string()
.contains("single path segment"),
"error should mention single path segment requirement"
);
let cfg = RepoConfigV2 {
version: "2.0".to_string(),
mount_dirs: MountDirsV2 {
thoughts: "thoughts".to_string(),
context: "context/".to_string(),
references: "references".to_string(),
},
thoughts_mount: None,
context_mounts: vec![],
references: vec![],
};
let result = mgr.validate_v2_hard(&cfg);
assert!(
result.is_err(),
"trailing slash on context should be rejected"
);
let cfg = RepoConfigV2 {
version: "2.0".to_string(),
mount_dirs: MountDirsV2 {
thoughts: "thoughts".to_string(),
context: "context".to_string(),
references: "references/".to_string(),
},
thoughts_mount: None,
context_mounts: vec![],
references: vec![],
};
let result = mgr.validate_v2_hard(&cfg);
assert!(
result.is_err(),
"trailing slash on references should be rejected"
);
}
}