#![allow(clippy::missing_docs_in_private_items)]
use super::*;
struct TempHome(std::path::PathBuf);
impl TempHome {
fn set() -> Self {
let dir = std::env::temp_dir().join(format!("moadim-modeltest-{}", uuid::Uuid::new_v4()));
std::fs::create_dir_all(&dir).expect("create temp home");
unsafe {
std::env::set_var("MOADIM_HOME_OVERRIDE", &dir);
}
Self(dir)
}
}
impl Drop for TempHome {
fn drop(&mut self) {
unsafe {
std::env::remove_var("MOADIM_HOME_OVERRIDE");
}
let _ = std::fs::remove_dir_all(&self.0);
}
}
fn with_path(value: &std::path::Path, body: impl FnOnce()) {
let saved = std::env::var_os("PATH");
unsafe {
std::env::set_var("PATH", value);
}
body();
unsafe {
match saved {
Some(prev) => std::env::set_var("PATH", prev),
None => std::env::remove_var("PATH"),
}
}
}
fn make_routine(agent: &str) -> Routine {
Routine {
id: "model-test-id".into(),
schedule: "@daily".into(),
title: "Model Test Routine".into(),
agent: agent.into(),
model: None,
prompt: "p".into(),
repositories: vec![],
machines: vec![crate::machine::current_machine()],
enabled: true,
source: "managed".into(),
created_at: 0,
updated_at: 0,
last_manual_trigger_at: None,
last_scheduled_trigger_at: None,
tags: vec![],
ttl_secs: None,
max_runtime_secs: None,
}
}
#[test]
fn from_routine_agent_command_available_true_when_command_resolves() {
let _home = TempHome::set();
let dir = std::env::temp_dir().join(format!("moadim-model-bin-{}", uuid::Uuid::new_v4()));
std::fs::create_dir_all(&dir).unwrap();
let bin = dir.join("fake-agent-cmd");
std::fs::write(&bin, "#!/bin/sh\n").unwrap();
#[cfg(unix)]
{
use std::os::unix::fs::PermissionsExt as _;
std::fs::set_permissions(&bin, std::fs::Permissions::from_mode(0o755)).unwrap();
}
std::fs::create_dir_all(agent_toml_path("model-test-resolves").parent().unwrap()).unwrap();
std::fs::write(
agent_toml_path("model-test-resolves"),
r#"command = "fake-agent-cmd""#,
)
.unwrap();
with_path(&dir, || {
let resp = RoutineResponse::from_routine(make_routine("model-test-resolves"));
assert!(resp.agent_registered);
assert!(resp.agent_command_available);
});
let _ = std::fs::remove_dir_all(&dir);
}
#[test]
fn from_routine_agent_command_available_false_when_registered_but_not_on_path() {
let _home = TempHome::set();
std::fs::create_dir_all(agent_toml_path("model-test-unresolved").parent().unwrap()).unwrap();
std::fs::write(
agent_toml_path("model-test-unresolved"),
r#"command = "definitely-not-a-real-binary-xyz""#,
)
.unwrap();
let empty_dir =
std::env::temp_dir().join(format!("moadim-model-empty-bin-{}", uuid::Uuid::new_v4()));
std::fs::create_dir_all(&empty_dir).unwrap();
with_path(&empty_dir, || {
let resp = RoutineResponse::from_routine(make_routine("model-test-unresolved"));
assert!(resp.agent_registered);
assert!(!resp.agent_command_available);
});
let _ = std::fs::remove_dir_all(&empty_dir);
}
#[test]
fn from_routine_agent_command_available_false_when_agent_not_registered() {
let _home = TempHome::set();
let resp = RoutineResponse::from_routine(make_routine("model-test-unregistered-zzz"));
assert!(!resp.agent_registered);
assert!(!resp.agent_command_available);
}
#[test]
fn describe_schedule_appends_timezone_when_present() {
let desc = describe_schedule("@daily", Some("Asia/Jerusalem")).unwrap();
assert!(
desc.ends_with("(Asia/Jerusalem)"),
"expected timezone suffix in {desc}"
);
}
#[test]
fn describe_schedule_omits_timezone_when_none() {
let desc = describe_schedule("@daily", None).unwrap();
assert!(!desc.contains('('), "expected no timezone suffix in {desc}");
}
#[test]
fn describe_schedule_returns_none_for_unparseable() {
assert!(describe_schedule("@reboot", Some("UTC")).is_none());
assert!(describe_schedule("not a cron", None).is_none());
}
#[test]
fn from_routine_populates_derived_fields() {
let routine = Routine {
id: "rid".into(),
schedule: "@daily".into(),
title: "My Title".into(),
agent: "claude".into(),
model: None,
prompt: "p".into(),
repositories: vec![],
machines: vec![crate::machine::current_machine()],
enabled: true,
source: "managed".into(),
created_at: 0,
updated_at: 0,
last_manual_trigger_at: None,
last_scheduled_trigger_at: None,
tags: vec![],
ttl_secs: None,
max_runtime_secs: None,
};
let resp = RoutineResponse::from_routine(routine);
assert!(resp.schedule_description.is_some());
assert!(resp.file_path.contains("routine.toml"));
assert_eq!(resp.flag_count, 0);
}
#[test]
fn from_routine_counts_open_flags() {
let routine = Routine {
id: "rid2".into(),
schedule: "@daily".into(),
title: "Flag Count Model Test ZZZ".into(),
agent: "claude".into(),
model: None,
prompt: "p".into(),
repositories: vec![],
machines: vec![crate::machine::current_machine()],
enabled: true,
source: "managed".into(),
created_at: 0,
updated_at: 0,
last_manual_trigger_at: None,
last_scheduled_trigger_at: None,
tags: vec![],
ttl_secs: None,
max_runtime_secs: None,
};
let slug = slugify(&routine.title);
crate::routines::flags::create_flag(
&slug,
"bug",
"d1",
crate::routines::flags::FlagScope::General,
)
.unwrap();
crate::routines::flags::create_flag(
&slug,
"gap",
"d2",
crate::routines::flags::FlagScope::Local,
)
.unwrap();
let resp = RoutineResponse::from_routine(routine);
assert_eq!(resp.flag_count, 2);
crate::routine_storage::remove_routine_dir(&slug).unwrap();
}