#![allow(clippy::missing_docs_in_private_items)]
use super::*;
use crate::routines::model::Routine;
fn make_routine(title: &str) -> Routine {
Routine {
model: None,
id: "cmd-test-id".to_string(),
schedule: "@daily".to_string(),
title: title.to_string(),
agent: "claude".to_string(),
prompt: "do it".to_string(),
repositories: vec![],
machines: vec![crate::machine::current_machine()],
enabled: true,
source: "managed".to_string(),
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,
}
}
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"),
}
}
}
#[test]
fn build_routine_command_resolves_bin_dir_when_tool_on_path() {
let dir = std::env::temp_dir().join(format!("moadim-cmd-path-{}", uuid::Uuid::new_v4()));
std::fs::create_dir_all(&dir).unwrap();
let tmux = dir.join("tmux");
std::fs::write(&tmux, "#!/bin/sh\n").unwrap();
#[cfg(unix)]
{
use std::os::unix::fs::PermissionsExt as _;
std::fs::set_permissions(&tmux, std::fs::Permissions::from_mode(0o755)).unwrap();
}
let dir_str = dir.to_string_lossy().into_owned();
with_path(&dir, || {
let routine = make_routine("Cmd Path Routine");
let agent = AgentCommand {
command: "claude".to_string(),
args: vec![],
instructions_file: "CLAUDE.md".to_string(),
setup: None,
};
let cmd = build_routine_command(&routine, &agent);
assert!(
cmd.contains(&dir_str),
"expected resolved tmux dir {dir_str} in: {cmd}"
);
});
let _ = std::fs::remove_dir_all(&dir);
}
#[test]
fn build_routine_command_stamps_scheduled_trigger_sidecar() {
let routine = make_routine("Cmd Scheduled Stamp Routine");
let agent = AgentCommand {
command: "claude".to_string(),
args: vec![],
instructions_file: "CLAUDE.md".to_string(),
setup: None,
};
let cmd = build_routine_command(&routine, &agent);
let sidecar = crate::paths::routine_scheduled_state_path(&slugify(&routine.title))
.to_string_lossy()
.into_owned();
assert!(
cmd.contains(&format!(
r#"printf 'last_scheduled_trigger_at = %s\n' "$TS" > {} || true"#,
shell_quote(&sidecar)
)),
"expected scheduled-trigger sidecar stamp in: {cmd}"
);
let stamp = cmd.find("last_scheduled_trigger_at").unwrap();
let copy = cmd.find("/prompt.md\"").unwrap();
assert!(stamp < copy, "sidecar stamp must precede the prompt copy");
}
#[test]
fn build_routine_command_fail_fasts_when_disclosure_write_fails() {
let routine = make_routine("Cmd Disclosure Guard Routine");
let agent = AgentCommand {
command: "claude".to_string(),
args: vec![],
instructions_file: "CLAUDE.md".to_string(),
setup: None,
};
let cmd = build_routine_command(&routine, &agent);
let write = cmd.find(r#"> "$WB/CLAUDE.md" || {"#).unwrap();
assert!(
cmd.contains(
r#"> "$WB/CLAUDE.md" || { echo "moadim: failed to write agent instructions disclosure; aborting launch" | tee -a "$WB/agent.log" >&2; exit 1; }"#
),
"expected the CLAUDE.md disclosure write to fail-fast in: {cmd}"
);
let copy = cmd.find("/prompt.md\"").unwrap();
assert!(
write < copy,
"disclosure-write guard must precede the prompt copy"
);
assert!(
cmd.contains(r#">> "$WB/CLAUDE.md" || true"#),
"user-prompt append must remain best-effort in: {cmd}"
);
}
#[test]
fn build_routine_command_workbench_base_tracks_moadim_home_override() {
let dir = std::env::temp_dir().join(format!("moadim-cmd-home-{}", uuid::Uuid::new_v4()));
std::fs::create_dir_all(&dir).unwrap();
let previous = std::env::var_os("MOADIM_HOME_OVERRIDE");
unsafe {
std::env::set_var("MOADIM_HOME_OVERRIDE", &dir);
}
let expected_base = crate::paths::workbenches_dir()
.to_string_lossy()
.into_owned();
let routine = make_routine("Cmd Workbench Base Routine");
let agent = AgentCommand {
command: "claude".to_string(),
args: vec![],
instructions_file: "CLAUDE.md".to_string(),
setup: None,
};
let cmd = build_routine_command(&routine, &agent);
unsafe {
match previous {
Some(prev) => std::env::set_var("MOADIM_HOME_OVERRIDE", prev),
None => std::env::remove_var("MOADIM_HOME_OVERRIDE"),
}
}
let _ = std::fs::remove_dir_all(&dir);
assert!(
cmd.contains(&format!(
r#"WB={}/"$SLUG-$TS""#,
shell_quote(&expected_base)
)),
"expected WB base derived from paths::workbenches_dir() ({expected_base}) in: {cmd}"
);
assert!(
!cmd.contains(r#"WB="$HOME/.moadim/workbenches"#),
"expected the hardcoded $HOME/.moadim/workbenches literal to be gone, got: {cmd}"
);
}
#[test]
fn cron_path_falls_back_to_root_home_when_home_unset() {
let saved = std::env::var_os("HOME");
unsafe {
std::env::remove_var("HOME");
}
let path = cron_path("definitely-not-a-real-binary-xyz");
assert!(
path.contains("/root/.local/bin"),
"expected /root-anchored fallback dirs in: {path}"
);
unsafe {
match saved {
Some(prev) => std::env::set_var("HOME", prev),
None => std::env::remove_var("HOME"),
}
}
}
#[test]
fn tmux_available_in_true_when_fake_tmux_present() {
let dir = std::env::temp_dir().join(format!("moadim-tmux-present-{}", uuid::Uuid::new_v4()));
std::fs::create_dir_all(&dir).unwrap();
let tmux = dir.join("tmux");
std::fs::write(&tmux, "#!/bin/sh\n").unwrap();
#[cfg(unix)]
{
use std::os::unix::fs::PermissionsExt as _;
std::fs::set_permissions(&tmux, std::fs::Permissions::from_mode(0o755)).unwrap();
}
assert!(tmux_available_in(&dir.to_string_lossy()));
let _ = std::fs::remove_dir_all(&dir);
}
#[test]
fn tmux_available_in_false_when_dir_has_no_tmux() {
let dir = std::env::temp_dir().join(format!("moadim-tmux-missing-{}", uuid::Uuid::new_v4()));
std::fs::create_dir_all(&dir).unwrap();
assert!(!tmux_available_in(&dir.to_string_lossy()));
let _ = std::fs::remove_dir_all(&dir);
}
#[test]
fn tmux_available_reads_live_path_present() {
let dir = std::env::temp_dir().join(format!("moadim-tmux-live-{}", uuid::Uuid::new_v4()));
std::fs::create_dir_all(&dir).unwrap();
let tmux = dir.join("tmux");
std::fs::write(&tmux, "#!/bin/sh\n").unwrap();
#[cfg(unix)]
{
use std::os::unix::fs::PermissionsExt as _;
std::fs::set_permissions(&tmux, std::fs::Permissions::from_mode(0o755)).unwrap();
}
with_path(&dir, || assert!(tmux_available()));
let _ = std::fs::remove_dir_all(&dir);
}
#[test]
fn tmux_available_false_when_path_unset() {
let saved = std::env::var_os("PATH");
unsafe {
std::env::remove_var("PATH");
}
assert!(!tmux_available());
unsafe {
if let Some(prev) = saved {
std::env::set_var("PATH", prev);
}
}
}
#[test]
fn agent_command_available_in_true_when_fake_command_present() {
let dir =
std::env::temp_dir().join(format!("moadim-agentcmd-present-{}", 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();
}
assert!(agent_command_available_in(
&dir.to_string_lossy(),
"fake-agent-cmd"
));
let _ = std::fs::remove_dir_all(&dir);
}
#[test]
fn resolve_tmux_bin_from_prefers_path_over_fallbacks() {
let dir =
std::env::temp_dir().join(format!("moadim-resolve-tmux-path-{}", uuid::Uuid::new_v4()));
std::fs::create_dir_all(&dir).unwrap();
std::fs::write(dir.join("tmux"), "#!/bin/sh\n").unwrap();
let dir_str = dir.to_string_lossy().into_owned();
let resolved = resolve_tmux_bin_from(&dir_str, &["/definitely/not/here".to_string()]);
assert_eq!(resolved, format!("{dir_str}/tmux"));
let _ = std::fs::remove_dir_all(&dir);
}
#[test]
fn agent_command_available_in_false_when_dir_has_no_command() {
let dir =
std::env::temp_dir().join(format!("moadim-agentcmd-missing-{}", uuid::Uuid::new_v4()));
std::fs::create_dir_all(&dir).unwrap();
assert!(!agent_command_available_in(
&dir.to_string_lossy(),
"fake-agent-cmd"
));
let _ = std::fs::remove_dir_all(&dir);
}
#[test]
fn resolve_tmux_bin_from_falls_back_to_first_matching_fallback_dir() {
let base =
std::env::temp_dir().join(format!("moadim-resolve-tmux-fb-{}", uuid::Uuid::new_v4()));
let missing = base.join("missing");
let present = base.join("present");
std::fs::create_dir_all(&present).unwrap();
std::fs::write(present.join("tmux"), "#!/bin/sh\n").unwrap();
let resolved = resolve_tmux_bin_from(
"",
&[
missing.to_string_lossy().into_owned(),
present.to_string_lossy().into_owned(),
],
);
assert_eq!(resolved, format!("{}/tmux", present.to_string_lossy()));
let _ = std::fs::remove_dir_all(&base);
}
#[test]
fn resolve_tmux_bin_from_returns_bare_name_when_nowhere_found() {
let resolved = resolve_tmux_bin_from("", &["/definitely/not/here".to_string()]);
assert_eq!(resolved, "tmux");
}
#[test]
fn tmux_fallback_dirs_are_anchored_under_home() {
let dirs = tmux_fallback_dirs("/home/u");
assert!(dirs.contains(&"/opt/homebrew/bin".to_string()));
assert!(dirs.contains(&"/usr/local/bin".to_string()));
assert!(dirs.contains(&"/home/u/.local/bin".to_string()));
}
#[test]
fn resolve_tmux_bin_reads_live_path_and_home() {
let dir =
std::env::temp_dir().join(format!("moadim-resolve-tmux-live-{}", uuid::Uuid::new_v4()));
std::fs::create_dir_all(&dir).unwrap();
std::fs::write(dir.join("tmux"), "#!/bin/sh\n").unwrap();
let dir_str = dir.to_string_lossy().into_owned();
with_path(&dir, || {
assert_eq!(resolve_tmux_bin(), format!("{dir_str}/tmux"));
});
let _ = std::fs::remove_dir_all(&dir);
}
#[test]
fn agent_command_available_reads_live_path_present() {
let dir = std::env::temp_dir().join(format!("moadim-agentcmd-live-{}", 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();
}
with_path(&dir, || assert!(agent_command_available("fake-agent-cmd")));
let _ = std::fs::remove_dir_all(&dir);
}
#[test]
fn agent_command_available_false_when_path_unset() {
let saved = std::env::var_os("PATH");
unsafe {
std::env::remove_var("PATH");
}
assert!(!agent_command_available("definitely-not-a-real-binary-xyz"));
unsafe {
if let Some(prev) = saved {
std::env::set_var("PATH", prev);
}
}
}
#[test]
fn resolve_tmux_bin_falls_back_to_root_home_when_home_unset() {
let saved = std::env::var_os("HOME");
unsafe {
std::env::remove_var("HOME");
}
let _ = resolve_tmux_bin();
unsafe {
match saved {
Some(prev) => std::env::set_var("HOME", prev),
None => std::env::remove_var("HOME"),
}
}
}
#[test]
fn bin_dir_returns_none_when_path_unset() {
let saved = std::env::var_os("PATH");
unsafe {
std::env::remove_var("PATH");
}
assert!(bin_dir("definitely-not-a-real-binary-xyz").is_none());
unsafe {
if let Some(prev) = saved {
std::env::set_var("PATH", prev);
}
}
}
#[test]
fn build_routine_command_appends_model_override() {
let mut routine = make_routine("Cmd Model Routine");
routine.model = Some("claude-sonnet-4-6".to_string());
let agent = AgentCommand {
command: "claude".to_string(),
args: vec!["--permission-mode".to_string(), "auto".to_string()],
instructions_file: "CLAUDE.md".to_string(),
setup: None,
};
let cmd = build_routine_command(&routine, &agent);
let args_pos = cmd.find("--permission-mode auto").unwrap();
let model_pos = cmd.find("--model").unwrap();
assert!(
model_pos > args_pos,
"expected --model after the agent's own args in: {cmd}"
);
assert!(
cmd[model_pos..].contains("claude-sonnet-4-6"),
"expected model id after --model in: {cmd}"
);
}
#[test]
fn build_routine_command_omits_model_flag_when_unset() {
let routine = make_routine("Cmd No Model Routine");
let agent = AgentCommand {
command: "claude".to_string(),
args: vec![],
instructions_file: "CLAUDE.md".to_string(),
setup: None,
};
let cmd = build_routine_command(&routine, &agent);
assert!(
!cmd.contains("--model"),
"expected no --model flag in: {cmd}"
);
}