#![allow(clippy::missing_docs_in_private_items)]
use super::*;
use crate::routines::{new_store, slugify, Repository};
use std::collections::HashMap;
use std::sync::{Arc, Mutex};
struct TempHome(std::path::PathBuf);
impl TempHome {
fn set() -> TempHome {
let dir = std::env::temp_dir().join(format!("moadim-svctest-{}", uuid::Uuid::new_v4()));
std::fs::create_dir_all(&dir).expect("create temp home");
unsafe {
std::env::set_var("MOADIM_HOME_OVERRIDE", &dir);
}
TempHome(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 make_routine(id: &str, title: &str, created_at: u64, updated_at: u64) -> Routine {
Routine {
id: id.to_string(),
schedule: "@daily".to_string(),
title: title.to_string(),
agent: "claude".to_string(),
prompt: "do the thing".to_string(),
repositories: vec![],
machines: vec![crate::machine::current_machine()],
enabled: true,
source: "managed".to_string(),
created_at,
updated_at,
last_manual_trigger_at: None,
last_scheduled_trigger_at: None,
ttl_secs: None,
max_runtime_secs: None,
}
}
fn store_with(routines: Vec<Routine>) -> RoutineStore {
let mut map = HashMap::new();
for routine in routines {
map.insert(routine.id.clone(), routine);
}
Arc::new(Mutex::new(map))
}
#[test]
fn svc_list_sorts_by_updated_at() {
let _home = TempHome::set();
let store = store_with(vec![
make_routine("late", "Zeta", 100, 300),
make_routine("early", "Alpha", 100, 100),
make_routine("mid", "Mid", 100, 200),
]);
let query = RoutineListQuery {
sort: RoutineSort::Updated,
..Default::default()
};
let list = svc_list(&store, &query);
assert_eq!(list[0].routine.id, "early");
assert_eq!(list[1].routine.id, "mid");
assert_eq!(list[2].routine.id, "late");
}
#[test]
fn svc_list_sorts_by_title_case_insensitively() {
let _home = TempHome::set();
let store = store_with(vec![
make_routine("banana", "banana", 0, 0),
make_routine("apple", "Apple", 0, 0),
make_routine("cherry", "CHERRY", 0, 0),
]);
let query = RoutineListQuery {
sort: RoutineSort::Title,
..Default::default()
};
let list = svc_list(&store, &query);
assert_eq!(list[0].routine.id, "apple");
assert_eq!(list[1].routine.id, "banana");
assert_eq!(list[2].routine.id, "cherry");
}
fn valid_create_request() -> CreateRoutineRequest {
CreateRoutineRequest {
schedule: "@daily".into(),
title: "Valid Title".into(),
agent: "claude".into(),
prompt: "do the thing".into(),
repositories: vec![],
machines: vec![crate::machine::current_machine()],
enabled: true,
ttl_secs: None,
max_runtime_secs: None,
}
}
fn empty_update_request() -> UpdateRoutineRequest {
UpdateRoutineRequest {
schedule: None,
title: None,
agent: None,
prompt: None,
repositories: None,
machines: None,
enabled: None,
ttl_secs: None,
max_runtime_secs: None,
}
}
#[test]
fn svc_create_rejects_blank_title() {
let _home = TempHome::set();
let store = new_store();
let result = svc_create(
&store,
CreateRoutineRequest {
title: " ".into(),
..valid_create_request()
},
);
assert!(matches!(result, Err(AppError::BadRequest(_))));
}
#[test]
fn svc_create_rejects_blank_prompt() {
let _home = TempHome::set();
let store = new_store();
let result = svc_create(
&store,
CreateRoutineRequest {
prompt: String::new(),
..valid_create_request()
},
);
assert!(matches!(result, Err(AppError::BadRequest(_))));
}
#[test]
fn svc_create_rejects_zero_ttl_secs() {
let _home = TempHome::set();
let store = new_store();
let result = svc_create(
&store,
CreateRoutineRequest {
ttl_secs: Some(0),
..valid_create_request()
},
);
assert!(matches!(result, Err(AppError::BadRequest(_))));
}
#[test]
fn svc_create_rejects_zero_max_runtime_secs() {
let _home = TempHome::set();
let store = new_store();
let result = svc_create(
&store,
CreateRoutineRequest {
max_runtime_secs: Some(0),
..valid_create_request()
},
);
assert!(matches!(result, Err(AppError::BadRequest(_))));
}
#[test]
fn svc_create_persists_machines() {
let _home = TempHome::set();
let store = new_store();
let resp = svc_create(
&store,
CreateRoutineRequest {
machines: vec!["alpha".into(), "beta".into()],
..valid_create_request()
},
)
.expect("create");
assert_eq!(resp.routine.machines, vec!["alpha", "beta"]);
}
#[test]
fn svc_update_sets_machines() {
let _home = TempHome::set();
let store = store_with(vec![make_routine("upd-machines", "Keep", 1, 1)]);
let resp = svc_update(
&store,
"upd-machines",
UpdateRoutineRequest {
machines: Some(vec!["server".into()]),
..empty_update_request()
},
)
.expect("update");
assert_eq!(resp.routine.machines, vec!["server"]);
}
#[test]
fn svc_update_rejects_blank_title() {
let _home = TempHome::set();
let store = store_with(vec![make_routine("upd-blank-title", "Keep", 1, 1)]);
let result = svc_update(
&store,
"upd-blank-title",
UpdateRoutineRequest {
title: Some(" ".into()),
..empty_update_request()
},
);
assert!(matches!(result, Err(AppError::BadRequest(_))));
}
#[test]
fn svc_update_rejects_blank_prompt() {
let _home = TempHome::set();
let store = store_with(vec![make_routine("upd-blank-prompt", "Keep", 1, 1)]);
let result = svc_update(
&store,
"upd-blank-prompt",
UpdateRoutineRequest {
prompt: Some("\t\n".into()),
..empty_update_request()
},
);
assert!(matches!(result, Err(AppError::BadRequest(_))));
}
#[test]
fn svc_update_rejects_zero_durations() {
let _home = TempHome::set();
let store = store_with(vec![make_routine("upd-zero-secs", "Keep", 1, 1)]);
let ttl = svc_update(
&store,
"upd-zero-secs",
UpdateRoutineRequest {
ttl_secs: Some(0),
..empty_update_request()
},
);
assert!(matches!(ttl, Err(AppError::BadRequest(_))));
let max_runtime = svc_update(
&store,
"upd-zero-secs",
UpdateRoutineRequest {
max_runtime_secs: Some(0),
..empty_update_request()
},
);
assert!(matches!(max_runtime, Err(AppError::BadRequest(_))));
}
#[test]
fn svc_create_rejects_ttl_above_cron_ceiling() {
let _home = TempHome::set();
let store = new_store();
let result = svc_create(
&store,
CreateRoutineRequest {
schedule: "*/5 * * * *".into(),
ttl_secs: Some(1800),
..valid_create_request()
},
);
assert!(matches!(result, Err(AppError::BadRequest(_))));
}
#[test]
fn svc_create_rejects_max_runtime_above_cron_ceiling() {
let _home = TempHome::set();
let store = new_store();
let result = svc_create(
&store,
CreateRoutineRequest {
schedule: "*/5 * * * *".into(),
max_runtime_secs: Some(1800),
..valid_create_request()
},
);
assert!(matches!(result, Err(AppError::BadRequest(_))));
}
#[test]
fn svc_create_accepts_secs_at_cron_ceiling() {
let _home = TempHome::set();
let store = store_with(vec![make_routine(
"at-ceiling-dupe",
"At Ceiling ZZZ",
1,
1,
)]);
let result = svc_create(
&store,
CreateRoutineRequest {
schedule: "*/5 * * * *".into(),
title: " at ceiling ZZZ ".into(),
ttl_secs: Some(300),
max_runtime_secs: Some(300),
..valid_create_request()
},
);
assert!(matches!(result, Err(AppError::Conflict(_))));
}
#[test]
fn svc_update_rejects_ttl_above_current_schedule_ceiling() {
let _home = TempHome::set();
let store = store_with(vec![Routine {
schedule: "*/5 * * * *".to_string(),
..make_routine("upd-ttl-ceiling", "Keep Ceiling", 1, 1)
}]);
let result = svc_update(
&store,
"upd-ttl-ceiling",
UpdateRoutineRequest {
ttl_secs: Some(1800),
..empty_update_request()
},
);
assert!(matches!(result, Err(AppError::BadRequest(_))));
assert_eq!(
store
.lock()
.unwrap()
.get("upd-ttl-ceiling")
.unwrap()
.ttl_secs,
None
);
}
#[test]
fn svc_update_rejects_secs_above_new_schedule_ceiling() {
let _home = TempHome::set();
let store = store_with(vec![make_routine("upd-new-sched", "Keep New Sched", 1, 1)]);
let result = svc_update(
&store,
"upd-new-sched",
UpdateRoutineRequest {
schedule: Some("*/5 * * * *".into()),
max_runtime_secs: Some(1800),
..empty_update_request()
},
);
assert!(matches!(result, Err(AppError::BadRequest(_))));
}
#[test]
fn svc_create_rejects_duplicate_slug() {
let _home = TempHome::set();
let title = "Svc Create Dup ZZZ";
let store = new_store();
with_empty_path(|| {
let first = svc_create(
&store,
CreateRoutineRequest {
schedule: "@daily".into(),
title: title.into(),
agent: "claude".into(),
prompt: "p".into(),
repositories: vec![],
machines: vec![crate::machine::current_machine()],
enabled: true,
ttl_secs: None,
max_runtime_secs: None,
},
)
.unwrap();
let conflict = svc_create(
&store,
CreateRoutineRequest {
schedule: "@daily".into(),
title: " svc create DUP zzz ".into(),
agent: "claude".into(),
prompt: "p".into(),
repositories: vec![],
machines: vec![crate::machine::current_machine()],
enabled: true,
ttl_secs: None,
max_runtime_secs: None,
},
);
assert!(matches!(conflict, Err(AppError::Conflict(_))));
svc_delete(&store, &first.routine.id).unwrap();
});
}
#[test]
fn svc_create_rejects_malformed_agent_config() {
let _home = TempHome::set();
let agent_name = "svc-create-malformed-agent-zzz";
std::fs::create_dir_all(crate::paths::agents_dir()).unwrap();
let cfg = crate::paths::agent_toml_path(agent_name);
std::fs::write(&cfg, "command = [\n").unwrap();
let store = new_store();
let result = svc_create(
&store,
CreateRoutineRequest {
schedule: "@daily".into(),
title: "Svc Create Malformed ZZZ".into(),
agent: agent_name.into(),
prompt: "p".into(),
repositories: vec![],
machines: vec![crate::machine::current_machine()],
enabled: true,
ttl_secs: None,
max_runtime_secs: None,
},
);
match result {
Err(AppError::BadRequest(msg)) => assert!(msg.contains("malformed config")),
other => panic!("expected BadRequest, got {other:?}"),
}
}
#[test]
fn svc_update_rejects_malformed_agent_config() {
let _home = TempHome::set();
let agent_name = "svc-update-malformed-agent-zzz";
std::fs::create_dir_all(crate::paths::agents_dir()).unwrap();
let cfg = crate::paths::agent_toml_path(agent_name);
std::fs::write(&cfg, "command = [\n").unwrap();
let title = "Svc Update Malformed ZZZ";
let store = new_store();
let routine = make_routine("upd-mal-id", title, 1, 1);
crate::routine_storage::write_routine(&routine).unwrap();
store.lock().unwrap().insert("upd-mal-id".into(), routine);
let result = svc_update(
&store,
"upd-mal-id",
UpdateRoutineRequest {
schedule: None,
title: None,
agent: Some(agent_name.into()),
prompt: None,
repositories: None,
machines: None,
enabled: None,
ttl_secs: None,
max_runtime_secs: None,
},
);
match result {
Err(AppError::BadRequest(msg)) => assert!(msg.contains("malformed config")),
other => panic!("expected BadRequest, got {other:?}"),
}
}
#[test]
fn svc_update_rejects_renaming_into_existing_slug() {
let _home = TempHome::set();
let title_keep = "Svc Update Keep ZZZ";
let title_other = "Svc Update Other ZZZ";
let store = new_store();
let routine_keep = make_routine("keep-id", title_keep, 1, 1);
let routine_other = make_routine("other-id", title_other, 2, 2);
crate::routine_storage::write_routine(&routine_keep).unwrap();
crate::routine_storage::write_routine(&routine_other).unwrap();
store.lock().unwrap().insert("keep-id".into(), routine_keep);
store
.lock()
.unwrap()
.insert("other-id".into(), routine_other);
with_empty_path(|| {
let conflict = svc_update(
&store,
"other-id",
UpdateRoutineRequest {
schedule: None,
title: Some(title_keep.into()),
agent: None,
prompt: None,
repositories: None,
machines: None,
enabled: None,
ttl_secs: None,
max_runtime_secs: None,
},
);
assert!(matches!(conflict, Err(AppError::Conflict(_))));
});
}
#[test]
fn svc_update_sets_ttl_secs() {
let _home = TempHome::set();
let title = "Svc Update Ttl ZZZ";
let store = new_store();
let routine = make_routine("ttl-id", title, 1, 1);
crate::routine_storage::write_routine(&routine).unwrap();
store.lock().unwrap().insert("ttl-id".into(), routine);
with_empty_path(|| {
let updated = svc_update(
&store,
"ttl-id",
UpdateRoutineRequest {
schedule: None,
title: None,
agent: None,
prompt: None,
repositories: None,
machines: None,
enabled: None,
ttl_secs: Some(1800),
max_runtime_secs: None,
},
)
.unwrap();
assert_eq!(updated.routine.ttl_secs, Some(1800));
});
}
#[test]
fn svc_update_sets_max_runtime_secs() {
let _home = TempHome::set();
let title = "Svc Update Max Runtime ZZZ";
let store = new_store();
let routine = make_routine("max-runtime-id", title, 1, 1);
crate::routine_storage::write_routine(&routine).unwrap();
store
.lock()
.unwrap()
.insert("max-runtime-id".into(), routine);
with_empty_path(|| {
let updated = svc_update(
&store,
"max-runtime-id",
UpdateRoutineRequest {
schedule: None,
title: None,
agent: None,
prompt: None,
repositories: None,
machines: None,
enabled: None,
ttl_secs: None,
max_runtime_secs: Some(1234),
},
)
.unwrap();
assert_eq!(updated.routine.max_runtime_secs, Some(1234));
});
}
#[test]
fn svc_logs_returns_newest_workbench_log() {
let _home = TempHome::set();
let title = "Svc Logs Newest ZZZ";
let slug = slugify(title);
let store = new_store();
let mut routine = make_routine("logs-id", title, 1, 1);
routine.repositories = vec![Repository {
repository: "https://example.com/r.git".into(),
branch: None,
}];
store.lock().unwrap().insert("logs-id".into(), routine);
let workbenches = crate::paths::workbenches_dir();
std::fs::create_dir_all(&workbenches).unwrap();
let older = workbenches.join(format!("{slug}-1000"));
let newer = workbenches.join(format!("{slug}-2000"));
std::fs::create_dir_all(&older).unwrap();
std::fs::create_dir_all(&newer).unwrap();
std::fs::write(older.join("agent.log"), "old log contents").unwrap();
std::fs::write(newer.join("agent.log"), "new log contents").unwrap();
let logs = svc_logs(&store, "logs-id").unwrap();
assert_eq!(logs, "new log contents");
}
#[test]
fn svc_logs_skips_foreign_and_unparseable_workbenches() {
let _home = TempHome::set();
let title = "Svc Logs Mixed ZZQ";
let slug = slugify(title);
let store = new_store();
let routine = make_routine("logs-mixed-id", title, 1, 1);
store
.lock()
.unwrap()
.insert("logs-mixed-id".into(), routine);
let workbenches = crate::paths::workbenches_dir();
std::fs::create_dir_all(&workbenches).unwrap();
let unparseable = workbenches.join("not-a-workbench-name");
std::fs::create_dir_all(&unparseable).unwrap();
std::fs::write(unparseable.join("agent.log"), "ignored").unwrap();
let foreign = workbenches.join("some-other-routine-9999");
std::fs::create_dir_all(&foreign).unwrap();
std::fs::write(foreign.join("agent.log"), "foreign log").unwrap();
let mine = workbenches.join(format!("{slug}-4242"));
std::fs::create_dir_all(&mine).unwrap();
std::fs::write(mine.join("agent.log"), "mine log contents").unwrap();
let logs = svc_logs(&store, "logs-mixed-id").unwrap();
assert_eq!(logs, "mine log contents");
}
#[test]
fn svc_logs_empty_when_workbenches_dir_absent() {
let _home = TempHome::set();
let title = "Svc Logs No Workbenches ZZQ";
let store = new_store();
store.lock().unwrap().insert(
"logs-empty-id".into(),
make_routine("logs-empty-id", title, 1, 1),
);
assert!(!crate::paths::workbenches_dir().exists());
let logs = svc_logs(&store, "logs-empty-id").unwrap();
assert_eq!(logs, "");
}
#[test]
fn svc_logs_missing_routine_not_found() {
let _home = TempHome::set();
assert!(matches!(
svc_logs(&new_store(), "nope"),
Err(AppError::NotFound)
));
}
static PATH_GUARD: Mutex<()> = Mutex::new(());
fn with_empty_path(body: impl FnOnce()) {
let guard = PATH_GUARD
.lock()
.unwrap_or_else(|poison| poison.into_inner());
let saved = std::env::var_os("PATH");
std::env::set_var("PATH", "");
body();
match saved {
Some(value) => std::env::set_var("PATH", value),
None => std::env::remove_var("PATH"),
}
drop(guard);
}
#[test]
fn svc_create_warns_when_crontab_sync_fails() {
let _home = TempHome::set();
let title = "Svc Create Sync Fail ZZZ";
let store = new_store();
with_empty_path(|| {
let created = svc_create(
&store,
CreateRoutineRequest {
schedule: "@daily".into(),
title: title.into(),
agent: "claude".into(),
prompt: "p".into(),
repositories: vec![],
machines: vec![crate::machine::current_machine()],
enabled: true,
ttl_secs: None,
max_runtime_secs: None,
},
)
.unwrap();
assert_eq!(created.routine.title, title);
});
}
#[test]
fn svc_update_warns_when_crontab_sync_fails() {
let _home = TempHome::set();
let title = "Svc Update Sync Fail ZZZ";
let store = new_store();
let routine = make_routine("upd-sync-id", title, 1, 1);
crate::routine_storage::write_routine(&routine).unwrap();
store.lock().unwrap().insert("upd-sync-id".into(), routine);
with_empty_path(|| {
let updated = svc_update(
&store,
"upd-sync-id",
UpdateRoutineRequest {
schedule: None,
title: None,
agent: None,
prompt: Some("changed".into()),
repositories: None,
machines: None,
enabled: None,
ttl_secs: None,
max_runtime_secs: None,
},
)
.unwrap();
assert_eq!(updated.routine.prompt, "changed");
});
}
#[test]
fn svc_delete_warns_when_crontab_sync_fails() {
let _home = TempHome::set();
let title = "Svc Delete Sync Fail ZZZ";
let store = new_store();
let routine = make_routine("del-sync-id", title, 1, 1);
crate::routine_storage::write_routine(&routine).unwrap();
store.lock().unwrap().insert("del-sync-id".into(), routine);
with_empty_path(|| {
let deleted = svc_delete(&store, "del-sync-id").unwrap();
assert_eq!(deleted.routine.title, title);
});
}
fn with_working_crontab(body: impl FnOnce()) {
#[cfg(unix)]
use std::os::unix::fs::PermissionsExt as _;
let guard = PATH_GUARD
.lock()
.unwrap_or_else(|poison| poison.into_inner());
let base = std::env::temp_dir().join(format!("moadim-routcronok-{}", uuid::Uuid::new_v4()));
std::fs::create_dir_all(&base).unwrap();
let script = base.join("crontab-ok.sh");
std::fs::write(
&script,
"#!/bin/sh\nif [ \"$1\" = \"-\" ]; then cat > /dev/null; fi\nexit 0\n",
)
.unwrap();
#[cfg(unix)]
std::fs::set_permissions(&script, std::fs::Permissions::from_mode(0o755)).unwrap();
let saved = std::env::var_os("MOADIM_CRONTAB_BIN");
std::env::set_var("MOADIM_CRONTAB_BIN", &script);
body();
match saved {
Some(value) => std::env::set_var("MOADIM_CRONTAB_BIN", value),
None => std::env::remove_var("MOADIM_CRONTAB_BIN"),
}
let _ = std::fs::remove_dir_all(&base);
drop(guard);
}
#[test]
fn svc_create_syncs_crontab_on_success() {
let _home = TempHome::set();
let title = "Svc Create Sync OK ZZZ";
let store = new_store();
with_working_crontab(|| {
let created = svc_create(
&store,
CreateRoutineRequest {
schedule: "@daily".into(),
title: title.into(),
agent: "claude".into(),
prompt: "p".into(),
repositories: vec![],
machines: vec![crate::machine::current_machine()],
enabled: true,
ttl_secs: None,
max_runtime_secs: None,
},
)
.unwrap();
assert_eq!(created.routine.title, title);
});
}
#[test]
fn svc_update_syncs_crontab_on_success() {
let _home = TempHome::set();
let title = "Svc Update Sync OK ZZZ";
let store = new_store();
let routine = make_routine("upd-sync-ok-id", title, 1, 1);
crate::routine_storage::write_routine(&routine).unwrap();
store
.lock()
.unwrap()
.insert("upd-sync-ok-id".into(), routine);
with_working_crontab(|| {
let updated = svc_update(
&store,
"upd-sync-ok-id",
UpdateRoutineRequest {
schedule: None,
title: None,
agent: None,
prompt: Some("changed".into()),
repositories: None,
machines: None,
enabled: None,
ttl_secs: None,
max_runtime_secs: None,
},
)
.unwrap();
assert_eq!(updated.routine.prompt, "changed");
});
}
#[test]
fn svc_delete_syncs_crontab_on_success() {
let _home = TempHome::set();
let title = "Svc Delete Sync OK ZZZ";
let store = new_store();
let routine = make_routine("del-sync-ok-id", title, 1, 1);
crate::routine_storage::write_routine(&routine).unwrap();
store
.lock()
.unwrap()
.insert("del-sync-ok-id".into(), routine);
with_working_crontab(|| {
let deleted = svc_delete(&store, "del-sync-ok-id").unwrap();
assert_eq!(deleted.routine.title, title);
});
}
#[test]
fn svc_trigger_warns_when_spawn_fails() {
let _home = TempHome::set();
let agent_name = "svc-trigger-spawn-fail-agent-zzz";
std::fs::create_dir_all(crate::paths::agents_dir()).unwrap();
let cfg = crate::paths::agent_toml_path(agent_name);
std::fs::write(&cfg, "command = \"true\"\nargs = []\n").unwrap();
let title = "Svc Trigger Spawn Fail ZZZ";
let store = new_store();
let mut routine = make_routine("trig-spawn-id", title, 1, 1);
routine.agent = agent_name.into();
crate::routine_storage::write_routine(&routine).unwrap();
store
.lock()
.unwrap()
.insert("trig-spawn-id".into(), routine);
with_empty_path(|| {
let triggered = svc_trigger(&store, "trig-spawn-id").unwrap();
assert!(triggered.last_manual_trigger_at.is_some());
});
}
#[test]
fn svc_trigger_scheduled_missing_routine_not_found() {
let _home = TempHome::set();
assert!(matches!(
svc_trigger_scheduled(&new_store(), "nope"),
Err(AppError::NotFound)
));
}
#[test]
fn svc_trigger_scheduled_spawns_without_recording_manual_trigger() {
let _home = TempHome::set();
let agent_name = "svc-trigger-scheduled-agent-zzz";
std::fs::create_dir_all(crate::paths::agents_dir()).unwrap();
let cfg = crate::paths::agent_toml_path(agent_name);
std::fs::write(&cfg, "command = \"true\"\nargs = []\n").unwrap();
let title = "Svc Trigger Scheduled ZZZ";
let store = new_store();
let mut routine = make_routine("trig-sched-id", title, 1, 1);
routine.agent = agent_name.into();
crate::routine_storage::write_routine(&routine).unwrap();
store
.lock()
.unwrap()
.insert("trig-sched-id".into(), routine);
with_empty_path(|| {
let triggered = svc_trigger_scheduled(&store, "trig-sched-id").unwrap();
assert!(triggered.last_manual_trigger_at.is_none());
});
}
fn create_req_with_title(title: &str) -> CreateRoutineRequest {
CreateRoutineRequest {
schedule: "@daily".into(),
title: title.into(),
agent: "claude".into(),
prompt: "p".into(),
repositories: vec![],
machines: vec![crate::machine::current_machine()],
enabled: true,
ttl_secs: None,
max_runtime_secs: None,
}
}
#[test]
fn svc_create_rejects_blank_and_punctuation_titles() {
let _home = TempHome::set();
for title in ["", " \n\t", "!!!"] {
let store = new_store();
let result = svc_create(&store, create_req_with_title(title));
assert!(
matches!(result, Err(AppError::BadRequest(_))),
"title {title:?} should be rejected"
);
assert!(store.lock().unwrap().is_empty());
}
}
#[test]
fn svc_create_rejects_overlong_title() {
let _home = TempHome::set();
let store = new_store();
let title = "a".repeat(MAX_TITLE_LEN + 1);
let result = svc_create(&store, create_req_with_title(&title));
assert!(matches!(result, Err(AppError::BadRequest(_))));
assert!(store.lock().unwrap().is_empty());
}
#[test]
fn svc_create_rejects_unknown_agent() {
let _home = TempHome::set();
let store = new_store();
let result = svc_create(
&store,
CreateRoutineRequest {
schedule: "@daily".into(),
title: "Svc Create Unknown Agent ZZZ".into(),
agent: "no-such-agent-zzz".into(),
prompt: "p".into(),
repositories: vec![],
machines: vec![crate::machine::current_machine()],
enabled: true,
ttl_secs: None,
max_runtime_secs: None,
},
);
assert!(matches!(result, Err(AppError::BadRequest(_))));
assert!(store.lock().unwrap().is_empty());
}
#[test]
fn svc_update_rejects_blank_and_punctuation_titles() {
let _home = TempHome::set();
let original = "Svc Update Title Guard ZZZ";
for title in ["", " ", "!!!"] {
let store = new_store();
let routine = make_routine("title-guard-id", original, 1, 1);
crate::routine_storage::write_routine(&routine).unwrap();
store
.lock()
.unwrap()
.insert("title-guard-id".into(), routine);
let result = svc_update(
&store,
"title-guard-id",
UpdateRoutineRequest {
schedule: None,
title: Some(title.into()),
agent: None,
prompt: None,
repositories: None,
machines: None,
enabled: None,
ttl_secs: None,
max_runtime_secs: None,
},
);
assert!(
matches!(result, Err(AppError::BadRequest(_))),
"update to title {title:?} should be rejected"
);
assert_eq!(
store.lock().unwrap().get("title-guard-id").unwrap().title,
original
);
}
}
#[test]
fn svc_create_accepts_builtin_agent() {
let _home = TempHome::set();
crate::routines::ensure_default_agents();
let title = "Svc Create Valid Agent ZZZ";
let store = new_store();
let created = svc_create(
&store,
CreateRoutineRequest {
schedule: "@daily".into(),
title: title.into(),
agent: "claude".into(),
prompt: "p".into(),
repositories: vec![],
machines: vec![crate::machine::current_machine()],
enabled: true,
ttl_secs: None,
max_runtime_secs: None,
},
)
.unwrap();
assert_eq!(created.routine.agent, "claude");
svc_delete(&store, &created.routine.id).unwrap();
}
#[test]
fn svc_update_rejects_unknown_agent() {
let _home = TempHome::set();
let title = "Svc Update Unknown Agent ZZZ";
let store = new_store();
let routine = make_routine("upd-agent-id", title, 1, 1);
crate::routine_storage::write_routine(&routine).unwrap();
store.lock().unwrap().insert("upd-agent-id".into(), routine);
let result = svc_update(
&store,
"upd-agent-id",
UpdateRoutineRequest {
schedule: None,
title: None,
agent: Some("no-such-agent-zzz".into()),
prompt: None,
repositories: None,
machines: None,
enabled: None,
ttl_secs: None,
max_runtime_secs: None,
},
);
assert!(matches!(result, Err(AppError::BadRequest(_))));
assert_eq!(
store.lock().unwrap().get("upd-agent-id").unwrap().agent,
"claude"
);
}
#[test]
fn svc_create_rejects_blank_repository_url() {
let _home = TempHome::set();
let store = new_store();
for url in ["", " "] {
let result = svc_create(
&store,
CreateRoutineRequest {
schedule: "@daily".into(),
title: "Svc Create Blank Repo ZZZ".into(),
agent: "claude".into(),
prompt: "p".into(),
repositories: vec![Repository {
repository: url.into(),
branch: None,
}],
machines: vec![crate::machine::current_machine()],
enabled: true,
ttl_secs: None,
max_runtime_secs: None,
},
);
assert!(matches!(result, Err(AppError::BadRequest(_))));
}
assert!(store.lock().unwrap().is_empty());
}
#[test]
fn svc_create_rejects_blank_repository_branch() {
let _home = TempHome::set();
let store = new_store();
let result = svc_create(
&store,
CreateRoutineRequest {
schedule: "@daily".into(),
title: "Svc Create Blank Branch ZZZ".into(),
agent: "claude".into(),
prompt: "p".into(),
repositories: vec![Repository {
repository: "https://github.com/octocat/Hello-World".into(),
branch: Some(" ".into()),
}],
machines: vec![crate::machine::current_machine()],
enabled: true,
ttl_secs: None,
max_runtime_secs: None,
},
);
assert!(matches!(result, Err(AppError::BadRequest(_))));
assert!(store.lock().unwrap().is_empty());
}
#[test]
fn svc_create_trims_repository_entries() {
let _home = TempHome::set();
crate::routines::ensure_default_agents();
let title = "Svc Create Trim Repo ZZZ";
let store = new_store();
let created = svc_create(
&store,
CreateRoutineRequest {
schedule: "@daily".into(),
title: title.into(),
agent: "claude".into(),
prompt: "p".into(),
repositories: vec![Repository {
repository: " https://github.com/octocat/Hello-World ".into(),
branch: Some(" main ".into()),
}],
machines: vec![crate::machine::current_machine()],
enabled: true,
ttl_secs: None,
max_runtime_secs: None,
},
)
.unwrap();
let repo = &created.routine.repositories[0];
assert_eq!(repo.repository, "https://github.com/octocat/Hello-World");
assert_eq!(repo.branch.as_deref(), Some("main"));
svc_delete(&store, &created.routine.id).unwrap();
}
#[test]
fn svc_update_rejects_blank_repository_url() {
let _home = TempHome::set();
let title = "Svc Update Blank Repo ZZZ";
let store = new_store();
let routine = make_routine("upd-repo-id", title, 1, 1);
crate::routine_storage::write_routine(&routine).unwrap();
store.lock().unwrap().insert("upd-repo-id".into(), routine);
let result = svc_update(
&store,
"upd-repo-id",
UpdateRoutineRequest {
schedule: None,
title: None,
agent: None,
prompt: None,
repositories: Some(vec![Repository {
repository: " ".into(),
branch: None,
}]),
machines: None,
enabled: None,
ttl_secs: None,
max_runtime_secs: None,
},
);
assert!(matches!(result, Err(AppError::BadRequest(_))));
assert!(store
.lock()
.unwrap()
.get("upd-repo-id")
.unwrap()
.repositories
.is_empty());
}