Files
IdeA/crates/infrastructure/tests/profile_store.rs
Blomios 307ae71857 feat: add main features
Agents for developpement added + frontend add + backend added. Git viewer created + agent and template creator + layout and project creator
2026-06-06 01:27:01 +02:00

170 lines
5.0 KiB
Rust

//! L5 integration tests for [`FsProfileStore`] against a real temp directory,
//! using a real [`LocalFileSystem`] so the full persistence path (camelCase
//! `profiles.json`, upsert, delete, first-run marker) is exercised end-to-end.
use std::path::PathBuf;
use std::sync::Arc;
use domain::ids::ProfileId;
use domain::ports::{FileSystem, ProfileStore, RemotePath, StoreError};
use domain::profile::{AgentProfile, ContextInjection};
use infrastructure::{FsProfileStore, LocalFileSystem};
use uuid::Uuid;
/// A unique scratch directory under the OS temp dir, cleaned up on drop.
struct TempDir(PathBuf);
impl TempDir {
fn new() -> Self {
let p = std::env::temp_dir().join(format!("idea-l5-profile-{}", Uuid::new_v4()));
std::fs::create_dir_all(&p).unwrap();
Self(p)
}
fn app_data_dir(&self) -> String {
self.0.to_string_lossy().into_owned()
}
fn child(&self, name: &str) -> RemotePath {
RemotePath::new(self.0.join(name).to_string_lossy().into_owned())
}
}
impl Drop for TempDir {
fn drop(&mut self) {
let _ = std::fs::remove_dir_all(&self.0);
}
}
fn store(tmp: &TempDir) -> FsProfileStore {
let fs: Arc<dyn FileSystem> = Arc::new(LocalFileSystem::new());
FsProfileStore::new(fs, tmp.app_data_dir())
}
fn sample(id: u128, name: &str, command: &str) -> AgentProfile {
AgentProfile::new(
ProfileId::from_uuid(Uuid::from_u128(id)),
name,
command,
Vec::new(),
ContextInjection::convention_file("CLAUDE.md").unwrap(),
Some(format!("{command} --version")),
"{projectRoot}",
)
.unwrap()
}
#[tokio::test]
async fn save_then_list_roundtrips() {
let tmp = TempDir::new();
let store = store(&tmp);
let p = sample(1, "Claude", "claude");
store.save(&p).await.unwrap();
let listed = store.list().await.unwrap();
assert_eq!(listed, vec![p]);
}
#[tokio::test]
async fn save_upserts_by_id_without_duplicating() {
let tmp = TempDir::new();
let store = store(&tmp);
let first = sample(1, "before", "claude");
store.save(&first).await.unwrap();
let updated = sample(1, "after", "claude-renamed");
store.save(&updated).await.unwrap();
let listed = store.list().await.unwrap();
assert_eq!(listed.len(), 1, "upsert must not duplicate by id");
assert_eq!(listed[0], updated);
assert_eq!(listed[0].name, "after");
}
#[tokio::test]
async fn delete_removes_profile() {
let tmp = TempDir::new();
let store = store(&tmp);
let a = sample(1, "A", "a");
let b = sample(2, "B", "b");
store.save(&a).await.unwrap();
store.save(&b).await.unwrap();
store.delete(a.id).await.unwrap();
let listed = store.list().await.unwrap();
assert_eq!(listed, vec![b]);
}
#[tokio::test]
async fn delete_unknown_is_not_found() {
let tmp = TempDir::new();
let store = store(&tmp);
store.save(&sample(1, "A", "a")).await.unwrap();
let err = store
.delete(ProfileId::from_uuid(Uuid::from_u128(999)))
.await
.expect_err("deleting unknown id fails");
assert!(matches!(err, StoreError::NotFound), "got {err:?}");
}
#[tokio::test]
async fn is_configured_false_before_any_write() {
let tmp = TempDir::new();
let store = store(&tmp);
// First run: no profiles.json yet.
assert!(!store.is_configured().await.unwrap());
assert!(store.list().await.unwrap().is_empty());
}
#[tokio::test]
async fn is_configured_true_after_save() {
let tmp = TempDir::new();
let store = store(&tmp);
store.save(&sample(1, "A", "a")).await.unwrap();
assert!(store.is_configured().await.unwrap());
}
#[tokio::test]
async fn mark_configured_creates_file_with_empty_profiles() {
let tmp = TempDir::new();
let store = store(&tmp);
assert!(!store.is_configured().await.unwrap());
store.mark_configured().await.unwrap();
assert!(store.is_configured().await.unwrap(), "marker materialised");
assert!(
store.list().await.unwrap().is_empty(),
"empty profile list recorded"
);
}
#[tokio::test]
async fn profiles_file_is_camelcase_versioned() {
let tmp = TempDir::new();
let store = store(&tmp);
let p = sample(1, "Claude", "claude");
store.save(&p).await.unwrap();
let fs = LocalFileSystem::new();
let bytes = fs.read(&tmp.child("profiles.json")).await.unwrap();
let json: serde_json::Value = serde_json::from_slice(&bytes).unwrap();
assert_eq!(json["version"], 1);
let profiles = json
.get("profiles")
.and_then(|v| v.as_array())
.expect("top-level `profiles` array");
assert_eq!(profiles.len(), 1);
let entry = &profiles[0];
assert_eq!(entry["name"], "Claude");
assert_eq!(entry["command"], "claude");
// camelCase fields, tagged contextInjection.
assert!(entry.get("cwdTemplate").is_some(), "camelCase cwdTemplate");
assert!(entry.get("cwd_template").is_none(), "no snake_case leak");
assert_eq!(entry["contextInjection"]["strategy"], "conventionFile");
assert_eq!(entry["contextInjection"]["target"], "CLAUDE.md");
}