#![allow(clippy::missing_docs_in_private_items)]
use crate::cron_jobs::{new_registry, new_store};
use super::*;
fn make_handler() -> MoadimMcp {
MoadimMcp::new(
new_store(),
new_registry(),
crate::routines::new_store(),
0,
test_shutdown(),
)
}
fn test_shutdown() -> crate::cron_jobs::ShutdownSignal {
std::sync::Arc::new(tokio::sync::Notify::new())
}
struct TempHome;
impl TempHome {
fn set() -> TempHome {
let dir = std::env::temp_dir().join(format!("moadim-mcptest-{}", uuid::Uuid::new_v4()));
std::fs::create_dir_all(&dir).expect("create temp home");
unsafe {
std::env::set_var("MOADIM_HOME_OVERRIDE", &dir);
}
TempHome
}
}
impl Drop for TempHome {
fn drop(&mut self) {
unsafe {
std::env::remove_var("MOADIM_HOME_OVERRIDE");
}
}
}
#[test]
fn ok_helper_is_not_error() {
let result = ok(serde_json::json!({"status": "good"}));
assert!(!result.is_error.unwrap_or(false));
}
#[test]
fn err_helper_is_error() {
let result = err("something failed");
assert!(result.is_error.unwrap_or(false));
}
#[test]
fn health_returns_success() {
let handler = make_handler();
let result = handler.health().unwrap();
assert!(!result.is_error.unwrap_or(false));
}
#[test]
fn health_content_contains_status() {
let handler = make_handler();
let result = handler.health().unwrap();
let text = &result.content[0];
let json_str = match &text.raw {
rmcp::model::RawContent::Text(txt) => txt.text.clone(),
_ => panic!("expected text content"),
};
let val: serde_json::Value = serde_json::from_str(&json_str).unwrap();
assert_eq!(val["status"], "ok");
assert_eq!(val["running"], true);
assert_eq!(val["version"], crate::build_info::VERSION);
assert_eq!(val["git_sha"], crate::build_info::GIT_SHA);
assert_eq!(val["build_date"], crate::build_info::BUILD_DATE);
}
#[test]
fn echo_tool_returns_message() {
use rmcp::handler::server::wrapper::Parameters;
let handler = make_handler();
let result = handler
.echo(Parameters(EchoInput {
message: "test-msg".into(),
}))
.unwrap();
assert!(!result.is_error.unwrap_or(false));
let text = match &result.content[0].raw {
rmcp::model::RawContent::Text(txt) => txt.text.clone(),
_ => panic!("expected text content"),
};
let val: serde_json::Value = serde_json::from_str(&text).unwrap();
assert_eq!(val["message"], "test-msg");
}
#[test]
fn list_cron_jobs_empty() {
let handler = make_handler();
let result = handler.list_cron_jobs().unwrap();
assert!(!result.is_error.unwrap_or(false));
let text = match &result.content[0].raw {
rmcp::model::RawContent::Text(txt) => txt.text.clone(),
_ => panic!("expected text content"),
};
let val: serde_json::Value = serde_json::from_str(&text).unwrap();
assert!(val.as_array().unwrap().is_empty());
}
#[test]
fn get_cron_job_not_found_is_error() {
use rmcp::handler::server::wrapper::Parameters;
let handler = make_handler();
let result = handler
.get_cron_job(Parameters(IdInput {
id: "no-such".into(),
}))
.unwrap();
assert!(result.is_error.unwrap_or(false));
}
#[test]
fn delete_cron_job_not_found_is_error() {
use rmcp::handler::server::wrapper::Parameters;
let handler = make_handler();
let result = handler
.delete_cron_job(Parameters(IdInput {
id: "no-such".into(),
}))
.unwrap();
assert!(result.is_error.unwrap_or(false));
}
#[test]
fn trigger_cron_job_not_found_is_error() {
use rmcp::handler::server::wrapper::Parameters;
let handler = make_handler();
let result = handler
.trigger_cron_job(Parameters(IdInput {
id: "no-such".into(),
}))
.unwrap();
assert!(result.is_error.unwrap_or(false));
}
#[test]
fn create_cron_job_tool_invalid_cron_is_error() {
use rmcp::handler::server::wrapper::Parameters;
let store = crate::cron_jobs::new_store();
let handler = MoadimMcp::new(
store,
new_registry(),
crate::routines::new_store(),
0,
test_shutdown(),
);
let req = crate::cron_jobs::CreateRequest {
schedule: "not-a-cron".into(),
handler: "h".into(),
metadata: serde_json::Value::Null,
machines: vec![crate::machine::current_machine()],
enabled: true,
};
let result = handler.create_cron_job(Parameters(req)).unwrap();
assert!(result.is_error.unwrap_or(false));
}
#[test]
fn update_cron_job_tool_not_found_is_error() {
use rmcp::handler::server::wrapper::Parameters;
let handler = MoadimMcp::new(
crate::cron_jobs::new_store(),
new_registry(),
crate::routines::new_store(),
0,
test_shutdown(),
);
let result = handler
.update_cron_job(Parameters(UpdateInput {
id: "no-such".into(),
schedule: None,
handler: Some("h".into()),
metadata: None,
machines: None,
enabled: None,
}))
.unwrap();
assert!(result.is_error.unwrap_or(false));
}
#[test]
fn delete_cron_job_tool_success() {
use rmcp::handler::server::wrapper::Parameters;
let store = crate::cron_jobs::new_store();
let created = crate::cron_jobs::svc_create(
&store,
&new_registry(),
crate::cron_jobs::CreateRequest {
schedule: "@daily".into(),
handler: "h".into(),
metadata: serde_json::Value::Null,
machines: vec![crate::machine::current_machine()],
enabled: true,
},
)
.unwrap();
let id = created.job.id.clone();
let handler = MoadimMcp::new(
store,
new_registry(),
crate::routines::new_store(),
0,
test_shutdown(),
);
let result = handler.delete_cron_job(Parameters(IdInput { id })).unwrap();
assert!(!result.is_error.unwrap_or(false));
}
#[test]
fn trigger_cron_job_tool_success() {
use rmcp::handler::server::wrapper::Parameters;
let store = crate::cron_jobs::new_store();
let created = crate::cron_jobs::svc_create(
&store,
&new_registry(),
crate::cron_jobs::CreateRequest {
schedule: "@daily".into(),
handler: "h".into(),
metadata: serde_json::Value::Null,
machines: vec![crate::machine::current_machine()],
enabled: true,
},
)
.unwrap();
let id = created.job.id.clone();
let handler = MoadimMcp::new(
store,
new_registry(),
crate::routines::new_store(),
0,
test_shutdown(),
);
let result = handler
.trigger_cron_job(Parameters(IdInput { id: id.clone() }))
.unwrap();
assert!(!result.is_error.unwrap_or(false));
crate::storage::remove_job_dir(&id).unwrap();
}
#[test]
fn create_cron_job_tool_success() {
use rmcp::handler::server::wrapper::Parameters;
let store = crate::cron_jobs::new_store();
let handler = MoadimMcp::new(
store,
new_registry(),
crate::routines::new_store(),
0,
test_shutdown(),
);
let req = crate::cron_jobs::CreateRequest {
schedule: "@daily".into(),
handler: "mcp-handler".into(),
metadata: serde_json::Value::Null,
machines: vec![crate::machine::current_machine()],
enabled: true,
};
let result = handler.create_cron_job(Parameters(req)).unwrap();
assert!(!result.is_error.unwrap_or(false));
let text = match &result.content[0].raw {
rmcp::model::RawContent::Text(txt) => txt.text.clone(),
_ => panic!("expected text content"),
};
let val: serde_json::Value = serde_json::from_str(&text).unwrap();
let id = val["id"].as_str().unwrap().to_string();
crate::storage::remove_job_dir(&id).unwrap();
}
#[test]
fn get_cron_job_tool_success() {
use rmcp::handler::server::wrapper::Parameters;
let store = crate::cron_jobs::new_store();
let job = crate::cron_jobs::CronJob {
id: "get-test-id".into(),
schedule: "@daily".into(),
handler: "h".into(),
metadata: serde_json::Value::Null,
machines: vec![crate::machine::current_machine()],
enabled: true,
source: "managed".into(),
created_at: 0,
updated_at: 0,
last_manual_trigger_at: None,
};
store.lock().unwrap().insert("get-test-id".into(), job);
let handler = MoadimMcp::new(
store,
new_registry(),
crate::routines::new_store(),
0,
test_shutdown(),
);
let result = handler
.get_cron_job(Parameters(IdInput {
id: "get-test-id".into(),
}))
.unwrap();
assert!(!result.is_error.unwrap_or(false));
}
#[test]
fn update_cron_job_tool_success() {
use rmcp::handler::server::wrapper::Parameters;
let store = crate::cron_jobs::new_store();
let handler = MoadimMcp::new(
store.clone(),
new_registry(),
crate::routines::new_store(),
0,
test_shutdown(),
);
let created = crate::cron_jobs::svc_create(
&store,
&new_registry(),
crate::cron_jobs::CreateRequest {
schedule: "@daily".into(),
handler: "old".into(),
metadata: serde_json::Value::Null,
machines: vec![crate::machine::current_machine()],
enabled: true,
},
)
.unwrap();
let id = created.job.id.clone();
let result = handler
.update_cron_job(Parameters(UpdateInput {
id: id.clone(),
schedule: None,
handler: Some("new".into()),
metadata: None,
machines: None,
enabled: None,
}))
.unwrap();
assert!(!result.is_error.unwrap_or(false));
crate::storage::remove_job_dir(&id).unwrap();
}
fn make_create_routine_req() -> crate::routines::CreateRoutineRequest {
crate::routines::CreateRoutineRequest {
schedule: "@daily".into(),
title: "Mcp Routine".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 list_routines_empty() {
let handler = make_handler();
let result = handler.list_routines().unwrap();
assert!(!result.is_error.unwrap_or(false));
}
#[test]
fn get_routine_not_found_is_error() {
use rmcp::handler::server::wrapper::Parameters;
let handler = make_handler();
let result = handler
.get_routine(Parameters(IdInput {
id: "no-such".into(),
}))
.unwrap();
assert!(result.is_error.unwrap_or(false));
}
#[test]
fn create_routine_tool_invalid_cron_is_error() {
use rmcp::handler::server::wrapper::Parameters;
let handler = make_handler();
let mut req = make_create_routine_req();
req.schedule = "not-a-cron".into();
let result = handler.create_routine(Parameters(req)).unwrap();
assert!(result.is_error.unwrap_or(false));
}
#[test]
fn create_get_update_trigger_delete_routine_success() {
use rmcp::handler::server::wrapper::Parameters;
let _home = TempHome::set();
let routines = crate::routines::new_store();
let handler = MoadimMcp::new(
new_store(),
new_registry(),
routines.clone(),
0,
test_shutdown(),
);
let result = handler
.create_routine(Parameters(make_create_routine_req()))
.unwrap();
assert!(!result.is_error.unwrap_or(false));
let text = match &result.content[0].raw {
rmcp::model::RawContent::Text(txt) => txt.text.clone(),
_ => panic!("expected text content"),
};
let id = serde_json::from_str::<serde_json::Value>(&text).unwrap()["id"]
.as_str()
.unwrap()
.to_string();
let result = handler
.get_routine(Parameters(IdInput { id: id.clone() }))
.unwrap();
assert!(!result.is_error.unwrap_or(false));
let result = handler
.update_routine(Parameters(UpdateRoutineInput {
id: id.clone(),
schedule: None,
title: Some("Renamed".into()),
agent: None,
prompt: None,
repositories: None,
machines: None,
enabled: Some(false),
ttl_secs: None,
max_runtime_secs: None,
}))
.unwrap();
assert!(!result.is_error.unwrap_or(false));
let result = handler
.trigger_routine(Parameters(IdInput { id: id.clone() }))
.unwrap();
assert!(!result.is_error.unwrap_or(false));
let result = handler
.delete_routine(Parameters(IdInput { id: id.clone() }))
.unwrap();
assert!(!result.is_error.unwrap_or(false));
}
#[test]
fn cleanup_workbenches_tool_returns_removed_count() {
let handler = make_handler();
let result = handler.cleanup_workbenches().unwrap();
assert!(!result.is_error.unwrap_or(false));
let json_str = match &result.content[0].raw {
rmcp::model::RawContent::Text(txt) => txt.text.clone(),
_ => panic!("expected text content"),
};
let val: serde_json::Value = serde_json::from_str(&json_str).unwrap();
assert!(val["removed"].is_u64());
}
#[test]
fn update_routine_tool_not_found_is_error() {
use rmcp::handler::server::wrapper::Parameters;
let handler = make_handler();
let result = handler
.update_routine(Parameters(UpdateRoutineInput {
id: "no-such".into(),
schedule: None,
title: Some("x".into()),
agent: None,
prompt: None,
repositories: None,
machines: None,
enabled: None,
ttl_secs: None,
max_runtime_secs: None,
}))
.unwrap();
assert!(result.is_error.unwrap_or(false));
}
#[test]
fn delete_routine_tool_not_found_is_error() {
use rmcp::handler::server::wrapper::Parameters;
let handler = make_handler();
let result = handler
.delete_routine(Parameters(IdInput {
id: "no-such".into(),
}))
.unwrap();
assert!(result.is_error.unwrap_or(false));
}
#[test]
fn trigger_routine_tool_not_found_is_error() {
use rmcp::handler::server::wrapper::Parameters;
let handler = make_handler();
let result = handler
.trigger_routine(Parameters(IdInput {
id: "no-such".into(),
}))
.unwrap();
assert!(result.is_error.unwrap_or(false));
}
#[test]
fn list_agents_tool_returns_array() {
let _home = TempHome::set();
let handler = make_handler();
let result = handler.list_agents().unwrap();
assert!(!result.is_error.unwrap_or(false));
let text = match &result.content[0].raw {
rmcp::model::RawContent::Text(txt) => txt.text.clone(),
_ => panic!("expected text content"),
};
let val: serde_json::Value = serde_json::from_str(&text).unwrap();
assert!(val.is_array());
}
#[test]
fn cron_job_logs_tool_returns_logs_for_existing_job() {
use rmcp::handler::server::wrapper::Parameters;
let store = crate::cron_jobs::new_store();
let created = crate::cron_jobs::svc_create(
&store,
&new_registry(),
crate::cron_jobs::CreateRequest {
schedule: "@daily".into(),
handler: "h".into(),
metadata: serde_json::Value::Null,
machines: vec![crate::machine::current_machine()],
enabled: true,
},
)
.unwrap();
let id = created.job.id.clone();
let handler = MoadimMcp::new(
store,
new_registry(),
crate::routines::new_store(),
0,
test_shutdown(),
);
let result = handler
.cron_job_logs(Parameters(IdInput { id: id.clone() }))
.unwrap();
assert!(!result.is_error.unwrap_or(false));
let text = match &result.content[0].raw {
rmcp::model::RawContent::Text(txt) => txt.text.clone(),
_ => panic!("expected text content"),
};
let val: serde_json::Value = serde_json::from_str(&text).unwrap();
assert_eq!(val["logs"], "");
crate::storage::remove_job_dir(&id).unwrap();
}
#[test]
fn cron_job_logs_tool_not_found_is_error() {
use rmcp::handler::server::wrapper::Parameters;
let handler = make_handler();
let result = handler
.cron_job_logs(Parameters(IdInput {
id: "no-such".into(),
}))
.unwrap();
assert!(result.is_error.unwrap_or(false));
}
#[test]
fn routine_logs_tool_returns_logs_for_existing_routine() {
use rmcp::handler::server::wrapper::Parameters;
let _home = TempHome::set();
let routines = crate::routines::new_store();
let handler = MoadimMcp::new(
new_store(),
new_registry(),
routines.clone(),
0,
test_shutdown(),
);
let created = handler
.create_routine(Parameters(make_create_routine_req()))
.unwrap();
let text = match &created.content[0].raw {
rmcp::model::RawContent::Text(txt) => txt.text.clone(),
_ => panic!("expected text content"),
};
let id = serde_json::from_str::<serde_json::Value>(&text).unwrap()["id"]
.as_str()
.unwrap()
.to_string();
let result = handler
.routine_logs(Parameters(IdInput { id: id.clone() }))
.unwrap();
assert!(!result.is_error.unwrap_or(false));
let text = match &result.content[0].raw {
rmcp::model::RawContent::Text(txt) => txt.text.clone(),
_ => panic!("expected text content"),
};
let val: serde_json::Value = serde_json::from_str(&text).unwrap();
assert_eq!(val["logs"], "");
}
#[test]
fn routine_logs_tool_not_found_is_error() {
use rmcp::handler::server::wrapper::Parameters;
let handler = make_handler();
let result = handler
.routine_logs(Parameters(IdInput {
id: "no-such".into(),
}))
.unwrap();
assert!(result.is_error.unwrap_or(false));
}
#[test]
fn shutdown_tool_acknowledges() {
let handler = make_handler();
let result = handler.shutdown().unwrap();
assert!(!result.is_error.unwrap_or(false));
let text = match &result.content[0].raw {
rmcp::model::RawContent::Text(txt) => txt.text.clone(),
_ => panic!("expected text content"),
};
let val: serde_json::Value = serde_json::from_str(&text).unwrap();
assert_eq!(val["status"], "shutting down");
}
#[test]
fn restart_tool_spawns_helper_and_acknowledges() {
let _home = TempHome::set();
let handler = make_handler();
let result = handler.restart().unwrap();
assert!(!result.is_error.unwrap_or(false));
let text = match &result.content[0].raw {
rmcp::model::RawContent::Text(txt) => txt.text.clone(),
_ => panic!("expected text content"),
};
let val: serde_json::Value = serde_json::from_str(&text).unwrap();
assert_eq!(val["status"], "restarting");
assert!(val["helper_pid"].as_u64().unwrap() > 0);
}