use aim_core::domain::app::{AppRecord, InstallMetadata, InstallScope}; use aim_core::registry::model::Registry; use aim_core::registry::store::RegistryStore; use assert_cmd::Command; use predicates::prelude::PredicateBooleanExt; use predicates::str::contains; use tempfile::tempdir; const FIXTURE_MODE_ENV: &str = "AIM_GITHUB_FIXTURE_MODE"; #[test] fn list_command_runs_without_registry_entries() { let dir = tempdir().unwrap(); let registry_path = dir.path().join("registry.toml"); let mut cmd = Command::cargo_bin("aim").unwrap(); cmd.arg("list") .env("AIM_REGISTRY_PATH", ®istry_path) .assert() .success() .stdout(contains("No installed apps yet")); } #[test] fn list_command_reads_registered_apps_from_registry_file() { let dir = tempdir().unwrap(); let registry_path = dir.path().join("registry.toml"); std::fs::write( ®istry_path, "version = 1\n[[apps]]\nstable_id = \"bat\"\ndisplay_name = \"Bat\"\n", ) .unwrap(); let mut cmd = Command::cargo_bin("aim").unwrap(); cmd.arg("list") .env("AIM_REGISTRY_PATH", ®istry_path) .assert() .success() .stdout(contains("Name")) .stdout(contains("Version")) .stdout(contains("Source")) .stdout(contains("Bat")) .stdout(contains("Bat (bat)").not()); } #[test] fn remove_command_removes_registered_app_from_registry_file() { let dir = tempdir().unwrap(); let registry_path = dir.path().join("registry.toml"); std::fs::write( ®istry_path, "version = 1\n[[apps]]\nstable_id = \"bat\"\ndisplay_name = \"Bat\"\n", ) .unwrap(); let mut cmd = Command::cargo_bin("aim").unwrap(); cmd.args(["remove", "bat"]) .env("AIM_REGISTRY_PATH", ®istry_path) .assert() .success() .stdout(contains("Removed Bat")) .stdout(contains("Removal Summary").not()) .stdout(contains("Removed app:").not()); let contents = std::fs::read_to_string(®istry_path).unwrap(); assert!(!contents.contains("stable_id = \"bat\"")); } #[test] fn remove_command_uninstalls_managed_files() { let dir = tempdir().unwrap(); let registry_path = dir.path().join("registry.toml"); let install_home = dir.path().join("install-home"); let payload_path = install_home.join(".local/lib/aim/appimages/sharkdp-bat.AppImage"); let desktop_path = install_home.join(".local/share/applications/aim-sharkdp-bat.desktop"); let icon_path = install_home.join(".local/share/icons/hicolor/256x256/apps/sharkdp-bat.png"); let mut add_cmd = Command::cargo_bin("aim").unwrap(); add_cmd .arg("sharkdp/bat") .env("AIM_REGISTRY_PATH", ®istry_path) .env(FIXTURE_MODE_ENV, "1") .assert() .success(); assert!(payload_path.exists()); assert!(desktop_path.exists()); assert!(icon_path.exists()); let mut remove_cmd = Command::cargo_bin("aim").unwrap(); remove_cmd .args(["remove", "sharkdp-bat"]) .env("AIM_REGISTRY_PATH", ®istry_path) .assert() .success() .stdout(contains("\nRemoved bat")) .stdout(contains("Removed bat")) .stdout(contains("Removal Summary").not()) .stdout(contains("Removed app:").not()) .stdout(contains("Removed files")) .stdout(contains("sharkdp-bat.AppImage")) .stdout(contains("aim-sharkdp-bat.desktop")) .stdout(contains("sharkdp-bat.png")); assert!(!payload_path.exists()); assert!(!desktop_path.exists()); assert!(!icon_path.exists()); } #[test] fn query_command_registers_unambiguous_app_in_registry_file() { let dir = tempdir().unwrap(); let registry_path = dir.path().join("registry.toml"); let mut cmd = Command::cargo_bin("aim").unwrap(); cmd.arg("sharkdp/bat") .env("AIM_REGISTRY_PATH", ®istry_path) .env(FIXTURE_MODE_ENV, "1") .assert() .success() .stdout(contains("\nInstalled bat (user)")) .stdout(contains("Installed bat (user)")) .stdout(contains("Installation Summary").not()) .stdout(contains("Source: github sharkdp/bat")) .stdout(contains("Artifact:")) .stdout(contains("Selected artifact").not()) .stdout(contains("metadata-guided").not()) .stdout(contains("Installed files")) .stdout(contains("sharkdp-bat.AppImage")) .stdout(contains("Completed steps").not()); let contents = std::fs::read_to_string(®istry_path).unwrap(); assert!(contents.contains("stable_id = \"sharkdp-bat\"")); assert!(contents.contains("source_input = \"sharkdp/bat\"")); } #[test] fn old_release_query_renders_tracking_prompt_without_writing_registry() { let dir = tempdir().unwrap(); let registry_path = dir.path().join("registry.toml"); let mut cmd = Command::cargo_bin("aim").unwrap(); cmd.arg("https://github.com/pingdotgg/t3code/releases/download/v0.0.11/T3-Code-0.0.11-x86_64.AppImage") .env("AIM_REGISTRY_PATH", ®istry_path) .env(FIXTURE_MODE_ENV, "1") .assert() .success() .stdout(contains("Choose update tracking")) .stdout(contains("v0.0.11")) .stdout(contains("v0.0.12")); assert!(!registry_path.exists()); } #[test] fn old_release_query_can_track_latest_and_register_app() { let dir = tempdir().unwrap(); let registry_path = dir.path().join("registry.toml"); let mut cmd = Command::cargo_bin("aim").unwrap(); cmd.arg("https://github.com/pingdotgg/t3code/releases/download/v0.0.11/T3-Code-0.0.11-x86_64.AppImage") .env("AIM_REGISTRY_PATH", ®istry_path) .env(FIXTURE_MODE_ENV, "1") .env("AIM_TRACKING_PREFERENCE", "latest") .assert() .success() .stdout(contains("\nInstalled t3code (user)")) .stdout(contains("Installed t3code (user)")) .stdout(contains("Installation Summary").not()) .stdout(contains("Source: github pingdotgg/t3code")) .stdout(contains("Artifact: T3-Code-0.0.12-x86_64.AppImage")) .stdout(contains("Selected artifact").not()) .stdout(contains("metadata-guided").not()) .stdout(contains("Installed files")) .stdout(contains("pingdotgg-t3code.AppImage")) .stdout(contains("Completed steps").not()); let contents = std::fs::read_to_string(®istry_path).unwrap(); assert!(contents.contains("stable_id = \"pingdotgg-t3code\"")); assert!(contents.contains("locator = \"pingdotgg/t3code\"")); } #[test] fn cli_add_installs_and_renders_resolved_mode() { let dir = tempdir().unwrap(); let registry_path = dir.path().join("registry.toml"); let mut cmd = Command::cargo_bin("aim").unwrap(); cmd.arg("sharkdp/bat") .env("AIM_REGISTRY_PATH", ®istry_path) .env(FIXTURE_MODE_ENV, "1") .assert() .success() .stdout(contains("\nInstalled bat (user)")) .stdout(contains("Installed bat (user)")) .stdout(contains("Artifact:")) .stdout(contains("Installed files")) .stdout(contains("Completed steps").not()); } #[test] fn cli_add_emits_live_progress_to_stderr() { let dir = tempdir().unwrap(); let registry_path = dir.path().join("registry.toml"); let mut cmd = Command::cargo_bin("aim").unwrap(); cmd.arg("sharkdp/bat") .env("AIM_REGISTRY_PATH", ®istry_path) .env(FIXTURE_MODE_ENV, "1") .assert() .success() .stderr(contains("Installing sharkdp/bat")) .stderr(contains("Resolving source")) .stderr(contains("Discovering release")) .stderr(contains("Selecting artifact")) .stderr(contains("Downloading artifact")) .stderr(contains("Downloaded")) .stderr(contains("Payload Staged")) .stderr(contains("Desktop Entry Written")) .stderr(contains("Icon Extracted")) .stderr(contains("Desktop Integration Refreshed")) .stderr(contains("Registry Saved")); } #[test] fn bare_aim_review_renders_review_heading() { let dir = tempdir().unwrap(); let registry_path = dir.path().join("registry.toml"); let store = RegistryStore::new(registry_path.clone()); store .save(&Registry { version: 1, apps: vec![AppRecord { stable_id: "pingdotgg-t3code".to_owned(), display_name: "t3code".to_owned(), source_input: Some("pingdotgg/t3code".to_owned()), source: None, installed_version: Some("0.0.11".to_owned()), update_strategy: None, metadata: Vec::new(), install: Some(InstallMetadata { scope: InstallScope::User, payload_path: None, desktop_entry_path: None, icon_path: None, }), }], }) .unwrap(); let mut cmd = Command::cargo_bin("aim").unwrap(); cmd.env("AIM_REGISTRY_PATH", ®istry_path) .assert() .success() .stdout(contains("Update Review")) .stdout(contains("apps with updates")); } #[test] fn remove_command_emits_live_progress_to_stderr() { let dir = tempdir().unwrap(); let registry_path = dir.path().join("registry.toml"); std::fs::write( ®istry_path, "version = 1\n[[apps]]\nstable_id = \"bat\"\ndisplay_name = \"Bat\"\n", ) .unwrap(); let mut cmd = Command::cargo_bin("aim").unwrap(); cmd.args(["remove", "bat"]) .env("AIM_REGISTRY_PATH", ®istry_path) .assert() .success() .stderr(contains("Removing bat")) .stderr(contains("Resolving source: resolving bat")) .stderr(contains("Saving registry")); } #[test] fn system_request_on_immutable_host_falls_back_to_user_install() { let dir = tempdir().unwrap(); let registry_path = dir.path().join("registry.toml"); let os_release_path = dir.path().join("os-release"); std::fs::write(&os_release_path, "ID=fedora\nVARIANT_ID=silverblue\n").unwrap(); let mut cmd = Command::cargo_bin("aim").unwrap(); cmd.args(["--system", "sharkdp/bat"]) .env("AIM_REGISTRY_PATH", ®istry_path) .env("AIM_OS_RELEASE_PATH", &os_release_path) .env(FIXTURE_MODE_ENV, "1") .assert() .success() .stdout(contains("Installed bat (user)")) .stdout(contains("downgraded to user scope")); } #[test] fn update_command_applies_updates() { let dir = tempdir().unwrap(); let registry_path = dir.path().join("registry.toml"); let payload_path = dir .path() .join("install-home/.local/lib/aim/appimages/pingdotgg-t3code.AppImage"); let store = RegistryStore::new(registry_path.clone()); store .save(&Registry { version: 1, apps: vec![AppRecord { stable_id: "pingdotgg-t3code".to_owned(), display_name: "t3code".to_owned(), source_input: Some("pingdotgg/t3code".to_owned()), source: None, installed_version: Some("0.0.11".to_owned()), update_strategy: None, metadata: Vec::new(), install: Some(InstallMetadata { scope: InstallScope::User, payload_path: None, desktop_entry_path: None, icon_path: None, }), }], }) .unwrap(); let mut cmd = Command::cargo_bin("aim").unwrap(); cmd.arg("update") .env("AIM_REGISTRY_PATH", ®istry_path) .env(FIXTURE_MODE_ENV, "1") .assert() .success() .stdout(contains("updated apps: 1")) .stdout(contains("updates found:").not()); let updated = store.load().unwrap(); assert_eq!(updated.apps.len(), 1); assert_eq!(updated.apps[0].stable_id, "pingdotgg-t3code"); assert_eq!(updated.apps[0].installed_version.as_deref(), Some("0.0.12")); assert!(payload_path.exists()); } #[test] fn update_command_emits_live_progress_to_stderr() { let dir = tempdir().unwrap(); let registry_path = dir.path().join("registry.toml"); let store = RegistryStore::new(registry_path.clone()); store .save(&Registry { version: 1, apps: vec![AppRecord { stable_id: "pingdotgg-t3code".to_owned(), display_name: "t3code".to_owned(), source_input: Some("pingdotgg/t3code".to_owned()), source: None, installed_version: Some("0.0.11".to_owned()), update_strategy: None, metadata: Vec::new(), install: Some(InstallMetadata { scope: InstallScope::User, payload_path: None, desktop_entry_path: None, icon_path: None, }), }], }) .unwrap(); let mut cmd = Command::cargo_bin("aim").unwrap(); cmd.arg("update") .env("AIM_REGISTRY_PATH", ®istry_path) .env(FIXTURE_MODE_ENV, "1") .assert() .success() .stderr(contains("Updating 1 apps")) .stderr(contains("Resolving source: resolving pingdotgg-t3code")) .stderr(contains("Saving registry")); }