Files
IdeA/crates/infrastructure/tests/project_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

140 lines
4.4 KiB
Rust

//! L2 integration tests for [`FsProjectStore`] against a real temp directory,
//! using a real [`LocalFileSystem`] so the full persistence path (JSON layout,
//! tolerant reads, upsert) is exercised end-to-end.
use std::path::PathBuf;
use std::sync::Arc;
use domain::ids::ProjectId;
use domain::layout::Workspace;
use domain::ports::{FileSystem, ProjectStore, RemotePath};
use domain::project::{Project, ProjectPath};
use domain::remote::RemoteRef;
use infrastructure::{FsProjectStore, 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-l2-store-{}", Uuid::new_v4()));
std::fs::create_dir_all(&p).unwrap();
Self(p)
}
/// The app-data dir as a plain string, as the composition root would pass it.
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) -> FsProjectStore {
let fs: Arc<dyn FileSystem> = Arc::new(LocalFileSystem::new());
FsProjectStore::new(fs, tmp.app_data_dir())
}
fn sample_project(id: ProjectId, name: &str, root: &str) -> Project {
Project::new(
id,
name,
ProjectPath::new(root).unwrap(),
RemoteRef::local(),
1_700_000_000_000,
)
.unwrap()
}
#[tokio::test]
async fn save_then_list_roundtrips() {
let tmp = TempDir::new();
let store = store(&tmp);
let p = sample_project(ProjectId::new_random(), "alpha", "/home/me/alpha");
store.save_project(&p).await.unwrap();
let listed = store.list_projects().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 id = ProjectId::new_random();
let first = sample_project(id, "before", "/home/me/proj");
store.save_project(&first).await.unwrap();
// Same id, changed fields: must update in place, not append.
let updated = sample_project(id, "after", "/home/me/proj-renamed");
store.save_project(&updated).await.unwrap();
let listed = store.list_projects().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 missing_registry_lists_empty() {
let tmp = TempDir::new();
let store = store(&tmp);
// No projects.json written yet: tolerant read returns an empty list.
let listed = store.list_projects().await.unwrap();
assert!(listed.is_empty());
}
#[tokio::test]
async fn workspace_save_then_load_roundtrips() {
let tmp = TempDir::new();
let store = store(&tmp);
// Missing workspace returns the default.
let loaded = store.load_workspace().await.unwrap();
assert_eq!(loaded, Workspace::default());
let ws = Workspace::default();
store.save_workspace(&ws).await.unwrap();
let back = store.load_workspace().await.unwrap();
assert_eq!(back, ws);
}
#[tokio::test]
async fn registry_file_is_camelcase_json() {
let tmp = TempDir::new();
let store = store(&tmp);
let p = sample_project(ProjectId::new_random(), "jsoncheck", "/srv/app");
store.save_project(&p).await.unwrap();
// Read the raw bytes the store wrote and assert the camelCase shape.
let fs = LocalFileSystem::new();
let bytes = fs.read(&tmp.child("projects.json")).await.unwrap();
let json: serde_json::Value = serde_json::from_slice(&bytes).unwrap();
assert!(json.get("version").is_some(), "top-level `version` present");
let projects = json
.get("projects")
.and_then(|v| v.as_array())
.expect("top-level `projects` array");
assert_eq!(projects.len(), 1);
let entry = &projects[0];
// camelCase serialization of `created_at`.
assert!(
entry.get("createdAt").is_some(),
"project uses camelCase `createdAt`, got {entry}"
);
assert!(entry.get("created_at").is_none(), "no snake_case leak");
assert_eq!(entry.get("name").and_then(|v| v.as_str()), Some("jsoncheck"));
assert_eq!(entry.get("root").and_then(|v| v.as_str()), Some("/srv/app"));
}