1use anyhow::{bail, Context, Result};
2use crate::{config::{Config, WorkerProfileConfig, WorkersConfig}, git, ticket, ticket_fmt};
3use crate::wrapper::{WrapperContext, write_temp_file};
4use chrono::Utc;
5use std::path::{Path, PathBuf};
6
7const CLAUDE_WORKER_DEFAULT: &str = include_str!("default/agents/claude/apm.worker.md");
8const CLAUDE_SPEC_WRITER_DEFAULT: &str = include_str!("default/agents/claude/apm.spec-writer.md");
9const MOCK_HAPPY_WORKER_DEFAULT: &str = include_str!("default/agents/mock-happy/apm.worker.md");
10const MOCK_HAPPY_SPEC_WRITER_DEFAULT: &str = include_str!("default/agents/mock-happy/apm.spec-writer.md");
11const MOCK_SAD_WORKER_DEFAULT: &str = include_str!("default/agents/mock-sad/apm.worker.md");
12const MOCK_SAD_SPEC_WRITER_DEFAULT: &str = include_str!("default/agents/mock-sad/apm.spec-writer.md");
13const MOCK_RANDOM_WORKER_DEFAULT: &str = include_str!("default/agents/mock-random/apm.worker.md");
14const MOCK_RANDOM_SPEC_WRITER_DEFAULT: &str = include_str!("default/agents/mock-random/apm.spec-writer.md");
15const DEBUG_WORKER_DEFAULT: &str = include_str!("default/agents/debug/apm.worker.md");
16const DEBUG_SPEC_WRITER_DEFAULT: &str = include_str!("default/agents/debug/apm.spec-writer.md");
17
18static DEPRECATION_WARNED: std::sync::atomic::AtomicBool = std::sync::atomic::AtomicBool::new(false);
19
20#[cfg(test)]
21static DEPRECATION_TEST_LOCK: std::sync::Mutex<()> = std::sync::Mutex::new(());
22
23const DEPRECATION_MSG: &str = "apm: deprecated: `[workers] command`, `args`, and `model` fields are deprecated — migrate to `agent` and `[workers.options]`";
24
25const POST_FETCH_SETTLE_MS: u64 = 1_000;
28
29fn emit_deprecation_warning_to(out: &mut dyn std::io::Write) {
30 use std::sync::atomic::Ordering;
31 if DEPRECATION_WARNED.compare_exchange(false, true, Ordering::SeqCst, Ordering::SeqCst).is_ok() {
32 let _ = writeln!(out, "{DEPRECATION_MSG}");
33 }
34}
35
36fn emit_deprecation_warning() {
37 emit_deprecation_warning_to(&mut std::io::stderr().lock());
38}
39
40pub struct EffectiveWorkerParams {
41 pub command: String,
42 pub args: Vec<String>,
43 pub model: Option<String>,
44 pub env: std::collections::HashMap<String, String>,
45 pub container: Option<String>,
46 pub agent: String,
47 pub options: std::collections::HashMap<String, String>,
48}
49
50pub(crate) fn resolve_profile<'a>(transition: &crate::config::TransitionConfig, config: &'a Config, warnings: &mut Vec<String>) -> Option<&'a WorkerProfileConfig> {
51 let name = transition.profile.as_deref()?;
52 match config.worker_profiles.get(name) {
53 Some(p) => Some(p),
54 None => {
55 warnings.push(format!("warning: worker profile {name:?} not found — using global [workers] config"));
56 None
57 }
58 }
59}
60
61pub fn effective_spawn_params(transition_agent: Option<&str>, profile: Option<&WorkerProfileConfig>, workers: &WorkersConfig) -> EffectiveWorkerParams {
62 let command = profile.and_then(|p| p.command.clone())
64 .or_else(|| workers.command.clone())
65 .unwrap_or_else(|| "claude".to_string());
66 let args = profile.and_then(|p| p.args.clone())
67 .or_else(|| workers.args.clone())
68 .unwrap_or_else(|| vec!["--print".to_string()]);
69
70 let raw_agent = transition_agent
72 .map(str::to_owned)
73 .or_else(|| profile.and_then(|p| p.agent.clone()))
74 .or_else(|| workers.agent.clone());
75
76 let has_legacy = workers.command.is_some()
78 || workers.args.is_some()
79 || workers.model.is_some()
80 || profile.map(|p| p.command.is_some() || p.args.is_some() || p.model.is_some()).unwrap_or(false);
81 if raw_agent.is_none() && has_legacy {
82 emit_deprecation_warning();
83 }
84
85 let agent = raw_agent.unwrap_or_else(|| "claude".to_string());
86
87 let mut options = workers.options.clone();
89 if let Some(p) = profile {
90 for (k, v) in &p.options {
91 options.insert(k.clone(), v.clone());
92 }
93 }
94
95 let model = options.get("model").cloned()
97 .or_else(|| profile.and_then(|p| p.model.clone()))
98 .or_else(|| workers.model.clone());
99
100 let mut env = workers.env.clone();
102 if let Some(p) = profile {
103 for (k, v) in &p.env {
104 env.insert(k.clone(), v.clone());
105 }
106 }
107
108 let container = profile.and_then(|p| p.container.clone())
109 .or_else(|| workers.container.clone());
110
111 EffectiveWorkerParams { command, args, model, env, container, agent, options }
112}
113
114pub(crate) fn apply_frontmatter_agent(
115 agent: &mut String,
116 frontmatter: &ticket_fmt::Frontmatter,
117 profile_name: &str,
118) {
119 if let Some(ov) = frontmatter.agent_overrides.get(profile_name) {
120 *agent = ov.clone();
121 } else if let Some(a) = &frontmatter.agent {
122 *agent = a.clone();
123 }
124 }
126
127pub struct StartOutput {
128 pub id: String,
129 pub old_state: String,
130 pub new_state: String,
131 pub caller_name: String,
132 pub branch: String,
133 pub worktree_path: PathBuf,
134 pub merge_message: Option<String>,
135 pub worker_pid: Option<u32>,
136 pub log_path: Option<PathBuf>,
137 pub worker_name: Option<String>,
138 pub warnings: Vec<String>,
139}
140
141pub struct RunNextOutput {
142 pub ticket_id: Option<String>,
143 pub messages: Vec<String>,
144 pub warnings: Vec<String>,
145 pub worker_pid: Option<u32>,
146 pub log_path: Option<PathBuf>,
147}
148
149pub(crate) fn should_check_claude_compat(root: &Path, agent: &str) -> bool {
152 if agent != "claude" { return false; }
153 matches!(
154 crate::wrapper::resolve_wrapper(root, "claude"),
155 Ok(Some(crate::wrapper::WrapperKind::Builtin(_)))
156 )
157}
158
159pub(crate) fn check_output_format_supported(binary: &str) -> Result<()> {
160 let out = std::process::Command::new(binary)
161 .arg("--help")
162 .output()
163 .map_err(|e| anyhow::anyhow!(
164 "failed to run `{binary} --help` to check worker-driver compatibility: {e}"
165 ))?;
166 let combined = format!(
167 "{}{}",
168 String::from_utf8_lossy(&out.stdout),
169 String::from_utf8_lossy(&out.stderr)
170 );
171 if combined.contains("--output-format") {
172 Ok(())
173 } else {
174 bail!(
175 "worker binary `{binary}` does not advertise `--output-format` in its \
176 --help output; the flag `--output-format stream-json` is required for \
177 full transcript capture in .apm-worker.log.\n\
178 Upgrade the binary to a version that supports this flag, or configure \
179 an alternative worker command in your .apm/config.toml [workers] section."
180 )
181 }
182}
183
184pub struct ManagedChild {
185 pub inner: std::process::Child,
186 temp_files: Vec<PathBuf>,
187 denial_ctx: Option<(PathBuf, PathBuf, String)>,
190}
191
192impl std::ops::Deref for ManagedChild {
193 type Target = std::process::Child;
194 fn deref(&self) -> &std::process::Child { &self.inner }
195}
196
197impl std::ops::DerefMut for ManagedChild {
198 fn deref_mut(&mut self) -> &mut std::process::Child { &mut self.inner }
199}
200
201impl Drop for ManagedChild {
202 fn drop(&mut self) {
203 for f in &self.temp_files {
204 let _ = std::fs::remove_file(f);
205 }
206 if let Some((log_path, worktree_path, ticket_id)) = &self.denial_ctx {
207 run_denial_scan(log_path, worktree_path, ticket_id);
208 }
209 }
210}
211
212fn spawn_worker(ctx: &WrapperContext, agent: &str, project_root: &Path) -> Result<std::process::Child> {
213 use crate::wrapper::{resolve_wrapper, resolve_builtin, WrapperKind, Wrapper};
214 use crate::wrapper::custom::CustomWrapper;
215
216 match resolve_wrapper(project_root, agent)? {
217 Some(WrapperKind::Custom { script_path, manifest }) => {
218 CustomWrapper { script_path, manifest }.spawn(ctx)
219 }
220 Some(WrapperKind::Builtin(name)) => {
221 resolve_builtin(&name).expect("known built-in").spawn(ctx)
222 }
223 None => anyhow::bail!(
224 "agent {:?} not found: checked built-ins {{{}}} and '.apm/agents/{agent}/'",
225 agent,
226 crate::wrapper::list_builtin_names().join(", ")
227 ),
228 }
229}
230
231fn run_denial_scan(log_path: &Path, worktree: &Path, ticket_id: &str) {
234 let summary = crate::denial::scan_transcript(log_path, worktree, ticket_id);
235 let summary_path = crate::denial::summary_path_for(log_path);
236 crate::denial::write_summary(&summary_path, &summary);
237 let unique_cmds = crate::denial::collect_unique_apm_commands(&summary);
238 if !unique_cmds.is_empty() {
239 crate::logger::log(
240 "worker-diag",
241 &format!(
242 "apm_command_denial ticket {} denied apm commands: {}",
243 ticket_id,
244 unique_cmds.join(", ")
245 ),
246 );
247 }
248}
249
250pub fn run(root: &Path, id_arg: &str, no_aggressive: bool, spawn: bool, skip_permissions: bool, caller_name: &str) -> Result<StartOutput> {
251 let mut warnings: Vec<String> = Vec::new();
252 let config = Config::load(root)?;
253 let aggressive = config.sync.aggressive && !no_aggressive;
254 let skip_permissions = skip_permissions || config.agents.skip_permissions;
255
256 let startable: Vec<&str> = config.workflow.states.iter()
257 .filter(|s| s.transitions.iter().any(|tr| tr.trigger == "command:start"))
258 .map(|s| s.id.as_str())
259 .collect();
260
261 let mut tickets = ticket::load_all_from_git(root, &config.tickets.dir)?;
262 let id = ticket::resolve_id_in_slice(&tickets, id_arg)?;
263
264 let Some(t) = tickets.iter_mut().find(|t| t.frontmatter.id == id) else {
265 bail!("ticket {id:?} not found");
266 };
267
268 let ticket_epic_id = t.frontmatter.epic.clone();
269 let ticket_depends_on = t.frontmatter.depends_on.clone().unwrap_or_default();
270 let fm = &t.frontmatter;
271 if !startable.is_empty() && !startable.contains(&fm.state.as_str()) {
272 bail!(
273 "ticket {id:?} is in state {:?} — not startable\n\
274 Use `apm start` only from: {}",
275 fm.state,
276 startable.join(", ")
277 );
278 }
279
280 let now = Utc::now();
281 let old_state = t.frontmatter.state.clone();
282
283 let triggering_transition = config.workflow.states.iter()
284 .find(|s| s.id == old_state)
285 .and_then(|s| s.transitions.iter().find(|tr| tr.trigger == "command:start"));
286
287 let new_state = triggering_transition
288 .map(|tr| tr.to.clone())
289 .unwrap_or_else(|| "in_progress".into());
290
291 t.frontmatter.state = new_state.clone();
292 t.frontmatter.updated_at = Some(now);
293 let when = now.format("%Y-%m-%dT%H:%MZ").to_string();
294 crate::state::append_history(&mut t.body, &old_state, &new_state, &when, caller_name);
295
296 let content = t.serialize()?;
297 let rel_path = format!(
298 "{}/{}",
299 config.tickets.dir.to_string_lossy(),
300 t.path.file_name().unwrap().to_string_lossy()
301 );
302 let branch = t
303 .frontmatter
304 .branch
305 .clone()
306 .or_else(|| ticket_fmt::branch_name_from_path(&t.path))
307 .unwrap_or_else(|| format!("ticket/{id}"));
308
309 let default_branch = &config.project.default_branch;
310 let merge_base = t.frontmatter.target_branch.clone()
311 .unwrap_or_else(|| default_branch.to_string());
312
313 if aggressive {
314 if let Err(e) = git::fetch_branch(root, &branch) {
315 warnings.push(format!("warning: fetch failed: {e:#}"));
316 }
317 if let Err(e) = git::fetch_branch(root, default_branch) {
318 warnings.push(format!("warning: fetch {} failed: {e:#}", default_branch));
319 }
320 std::thread::sleep(std::time::Duration::from_millis(POST_FETCH_SETTLE_MS));
321 }
322
323 git::commit_to_branch(root, &branch, &rel_path, &content, &format!("ticket({id}): start — {old_state} → {new_state}"))?;
324
325 let wt_display = crate::worktree::provision_worktree(root, &config, &branch, &mut warnings)?;
326
327 let ref_to_merge = if crate::git_util::remote_branch_tip(&wt_display, &merge_base).is_some() {
328 format!("origin/{merge_base}")
329 } else {
330 merge_base.to_string()
331 };
332 let merge_message = crate::git_util::merge_ref(&wt_display, &ref_to_merge, &mut warnings);
333
334 if !spawn {
335 return Ok(StartOutput {
336 id,
337 old_state,
338 new_state,
339 caller_name: caller_name.to_string(),
340 branch,
341 worktree_path: wt_display,
342 merge_message,
343 worker_pid: None,
344 log_path: None,
345 worker_name: None,
346 warnings,
347 });
348 }
349
350 let profile_name = triggering_transition
351 .and_then(|tr| tr.profile.as_deref())
352 .unwrap_or("")
353 .to_string();
354 let profile = triggering_transition.and_then(|tr| resolve_profile(tr, &config, &mut warnings));
355 let role = profile.and_then(|p| p.role.as_deref()).unwrap_or("worker");
356 let mut params = effective_spawn_params(triggering_transition.and_then(|tr| tr.agent.as_deref()), profile, &config.workers);
357 apply_frontmatter_agent(&mut params.agent, &t.frontmatter, &profile_name);
358
359 let now_str = chrono::Utc::now().format("%m%d-%H%M").to_string();
360 let worker_name = format!("{}-{}-{:04x}", params.agent, now_str, rand_u16());
361 let tr_instructions = triggering_transition.and_then(|tr| tr.instructions.as_deref());
362 let tr_role_prefix = triggering_transition.and_then(|tr| tr.role_prefix.as_deref());
363 let worker_system = build_system_prompt(root, tr_instructions, profile, &config.workers, config.agents.instructions.as_deref(), ¶ms.agent, role)?;
364 let raw_prompt = format!("{}\n\n{content}", agent_role_prefix(tr_role_prefix, profile, &id));
365 let with_epic = with_epic_bundle(root, ticket_epic_id.as_deref(), &id, &config, raw_prompt);
366 let ticket_content = with_dependency_bundle(root, &ticket_depends_on, &config, with_epic);
367 let role_prefix = tr_role_prefix
368 .map(|p| p.replace("<id>", &id))
369 .or_else(|| profile.and_then(|p| p.role_prefix.clone()));
370
371 let log_path = wt_display.join(".apm-worker.log");
372
373 let sys_file = write_temp_file("sys", &worker_system)?;
374 let msg_file = write_temp_file("msg", &ticket_content)?;
375 let ctx = WrapperContext {
376 worker_name: worker_name.clone(),
377 agent_type: params.agent.clone(),
378 ticket_id: id.clone(),
379 ticket_branch: branch.clone(),
380 worktree_path: wt_display.clone(),
381 system_prompt_file: sys_file.clone(),
382 user_message_file: msg_file.clone(),
383 skip_permissions,
384 profile: profile_name,
385 role_prefix,
386 options: params.options.clone(),
387 model: params.model.clone(),
388 log_path: log_path.clone(),
389 container: params.container.clone(),
390 extra_env: params.env.clone(),
391 root: root.to_path_buf(),
392 keychain: config.workers.keychain.clone(),
393 current_state: new_state.clone(),
394 command: Some(params.command.clone()),
395 };
396 if should_check_claude_compat(root, ¶ms.agent) {
397 check_output_format_supported(¶ms.command)?;
398 }
399 let mut child = spawn_worker(&ctx, ¶ms.agent, root)?;
400 let pid = child.id();
401
402 let pid_path = wt_display.join(".apm-worker.pid");
403 write_pid_file(&pid_path, pid, &id)?;
404
405 let enforce_isolation = skip_permissions || config.isolation.enforce_worktree_isolation;
406 let wt_for_cleanup = wt_display.clone();
407 let denial_log_path = log_path.clone();
408 let denial_worktree = wt_display.clone();
409 let denial_ticket_id = id.clone();
410 let agent_for_diag = params.agent.clone();
411 std::thread::spawn(move || {
412 let _ = child.wait();
413 let _ = std::fs::remove_file(&sys_file);
414 let _ = std::fs::remove_file(&msg_file);
415 if agent_for_diag == "claude" {
416 run_denial_scan(&denial_log_path, &denial_worktree, &denial_ticket_id);
417 }
418 if enforce_isolation {
419 let _ = crate::wrapper::hook_config::remove_hook_config(&wt_for_cleanup);
420 }
421 });
422
423 Ok(StartOutput {
424 id,
425 old_state,
426 new_state,
427 caller_name: caller_name.to_string(),
428 branch,
429 worktree_path: wt_display,
430 merge_message,
431 worker_pid: Some(pid),
432 log_path: Some(log_path),
433 worker_name: Some(worker_name),
434 warnings,
435 })
436}
437
438pub fn run_next(root: &Path, no_aggressive: bool, spawn: bool, skip_permissions: bool) -> Result<RunNextOutput> {
439 let mut messages: Vec<String> = Vec::new();
440 let mut warnings: Vec<String> = Vec::new();
441 let config = Config::load(root)?;
442 let skip_permissions = skip_permissions || config.agents.skip_permissions;
443 let p = &config.workflow.prioritization;
444 let startable: Vec<&str> = config.workflow.states.iter()
445 .filter(|s| s.transitions.iter().any(|tr| tr.trigger == "command:start"))
446 .map(|s| s.id.as_str())
447 .collect();
448 let actionable_owned = config.actionable_states_for("agent");
449 let actionable: Vec<&str> = actionable_owned.iter().map(|s| s.as_str()).collect();
450 let all_tickets = ticket::load_all_from_git(root, &config.tickets.dir)?;
451 let caller_name = crate::config::resolve_caller_name();
452 let current_user = crate::config::resolve_identity(root);
453
454 let active_epic_ids: Vec<Option<String>> = all_tickets.iter()
456 .filter(|t| {
457 let s = t.frontmatter.state.as_str();
458 actionable.contains(&s) && !startable.contains(&s)
459 })
460 .map(|t| t.frontmatter.epic.clone())
461 .collect();
462 let blocked = config.blocked_epics(&active_epic_ids);
463 let default_blocked = config.is_default_branch_blocked(&active_epic_ids);
464 let tickets: Vec<_> = all_tickets.into_iter()
465 .filter(|t| match t.frontmatter.epic.as_deref() {
466 Some(eid) => !blocked.iter().any(|b| b == eid),
467 None => !default_blocked,
468 })
469 .collect();
470
471 let Some(candidate) = ticket::pick_next(&tickets, &actionable, &startable, p.priority_weight, p.effort_weight, p.risk_weight, &config, Some(&caller_name), Some(¤t_user)) else {
472 messages.push("No actionable tickets.".to_string());
473 return Ok(RunNextOutput { ticket_id: None, messages, warnings, worker_pid: None, log_path: None });
474 };
475
476 let id = candidate.frontmatter.id.clone();
477 let old_state = candidate.frontmatter.state.clone();
478
479 let triggering_transition_owned = config.workflow.states.iter()
480 .find(|s| s.id == old_state)
481 .and_then(|s| s.transitions.iter().find(|tr| tr.trigger == "command:start"))
482 .cloned();
483 let profile = triggering_transition_owned.as_ref().and_then(|tr| resolve_profile(tr, &config, &mut warnings));
484 let state_instructions = config.workflow.states.iter()
485 .find(|s| s.id == old_state)
486 .and_then(|sc| sc.instructions.as_deref())
487 .map(|s| s.to_string());
488 let instructions_text = profile
489 .and_then(|p| p.instructions.as_deref())
490 .map(|path| {
491 match std::fs::read_to_string(root.join(path)) {
492 Ok(s) => s,
493 Err(_) => { warnings.push("warning: instructions file not found".to_string()); String::new() }
494 }
495 })
496 .filter(|s| !s.is_empty())
497 .or_else(|| state_instructions.as_deref()
498 .and_then(|path| {
499 std::fs::read_to_string(root.join(path)).ok()
500 .or_else(|| { warnings.push("warning: instructions file not found".to_string()); None })
501 }));
502 let start_out = run(root, &id, no_aggressive, false, false, &caller_name)?;
503 warnings.extend(start_out.warnings);
504
505 if let Some(ref msg) = start_out.merge_message {
506 messages.push(msg.clone());
507 }
508 messages.push(format!("{}: {} → {} (caller: {}, branch: {})", start_out.id, start_out.old_state, start_out.new_state, start_out.caller_name, start_out.branch));
509 messages.push(format!("Worktree: {}", start_out.worktree_path.display()));
510
511 let tickets2 = ticket::load_all_from_git(root, &config.tickets.dir)?;
512 let Some(t) = tickets2.iter().find(|t| t.frontmatter.id == id) else {
513 return Ok(RunNextOutput { ticket_id: Some(id), messages, warnings, worker_pid: None, log_path: None });
514 };
515
516 let focus_hint = if let Some(ref section) = t.frontmatter.focus_section {
517 let hint = format!("Pay special attention to section: {section}");
518 let rel_path = format!(
519 "{}/{}",
520 config.tickets.dir.to_string_lossy(),
521 t.path.file_name().unwrap().to_string_lossy()
522 );
523 let branch = t.frontmatter.branch.clone()
524 .or_else(|| ticket_fmt::branch_name_from_path(&t.path))
525 .unwrap_or_else(|| format!("ticket/{id}"));
526 let mut t_mut = t.clone();
527 t_mut.frontmatter.focus_section = None;
528 let cleared = t_mut.serialize()?;
529 git::commit_to_branch(root, &branch, &rel_path, &cleared, &format!("ticket({id}): clear focus_section"))?;
530 Some(hint)
531 } else {
532 None
533 };
534
535 let mut prompt = String::new();
536 if let Some(ref instr) = instructions_text {
537 prompt.push_str(instr.trim());
538 prompt.push('\n');
539 }
540 if let Some(ref hint) = focus_hint {
541 if !prompt.is_empty() { prompt.push('\n'); }
542 prompt.push_str(hint);
543 prompt.push('\n');
544 }
545
546 if !spawn {
547 if !prompt.is_empty() {
548 messages.push(format!("Prompt:\n{prompt}"));
549 }
550 return Ok(RunNextOutput { ticket_id: Some(id), messages, warnings, worker_pid: None, log_path: None });
551 }
552
553 let profile_name2 = triggering_transition_owned.as_ref()
554 .and_then(|tr| tr.profile.as_deref())
555 .unwrap_or("")
556 .to_string();
557 let profile2 = triggering_transition_owned.as_ref().and_then(|tr| resolve_profile(tr, &config, &mut warnings));
558 let role2 = profile2.and_then(|p| p.role.as_deref()).unwrap_or("worker");
559 let mut params = effective_spawn_params(triggering_transition_owned.as_ref().and_then(|tr| tr.agent.as_deref()), profile2, &config.workers);
560 apply_frontmatter_agent(&mut params.agent, &t.frontmatter, &profile_name2);
561
562 let now_str = chrono::Utc::now().format("%m%d-%H%M").to_string();
563 let worker_name = format!("{}-{}-{:04x}", params.agent, now_str, rand_u16());
564 let tr_instructions2 = triggering_transition_owned.as_ref().and_then(|tr| tr.instructions.as_deref());
565 let tr_role_prefix2 = triggering_transition_owned.as_ref().and_then(|tr| tr.role_prefix.as_deref());
566 let worker_system = build_system_prompt(root, tr_instructions2, profile2, &config.workers, config.agents.instructions.as_deref(), ¶ms.agent, role2)?;
567
568 let raw = t.serialize()?;
569 let dep_ids_next = t.frontmatter.depends_on.clone().unwrap_or_default();
570 let raw_prompt_next = format!("{}\n\n{raw}", agent_role_prefix(tr_role_prefix2, profile2, &id));
571 let with_epic_next = with_epic_bundle(root, t.frontmatter.epic.as_deref(), &id, &config, raw_prompt_next);
572 let ticket_content = with_dependency_bundle(root, &dep_ids_next, &config, with_epic_next);
573 let role_prefix2 = tr_role_prefix2
574 .map(|p| p.replace("<id>", &id))
575 .or_else(|| profile2.and_then(|p| p.role_prefix.clone()));
576
577 let branch = t.frontmatter.branch.clone()
578 .or_else(|| ticket_fmt::branch_name_from_path(&t.path))
579 .unwrap_or_else(|| format!("ticket/{id}"));
580 let wt_name = branch.replace('/', "-");
581 let main_root = crate::git_util::main_worktree_root(root).unwrap_or_else(|| root.to_path_buf());
582 let wt_path = main_root.join(&config.worktrees.dir).join(&wt_name);
583 let wt_display = crate::worktree::find_worktree_for_branch(root, &branch).unwrap_or(wt_path);
584
585 let log_path = wt_display.join(".apm-worker.log");
586
587 let sys_file = write_temp_file("sys", &worker_system)?;
588 let msg_file = write_temp_file("msg", &ticket_content)?;
589 let ctx = WrapperContext {
590 worker_name: worker_name.clone(),
591 agent_type: params.agent.clone(),
592 ticket_id: id.clone(),
593 ticket_branch: branch.clone(),
594 worktree_path: wt_display.clone(),
595 system_prompt_file: sys_file.clone(),
596 user_message_file: msg_file.clone(),
597 skip_permissions,
598 profile: profile_name2,
599 role_prefix: role_prefix2,
600 options: params.options.clone(),
601 model: params.model.clone(),
602 log_path: log_path.clone(),
603 container: params.container.clone(),
604 extra_env: params.env.clone(),
605 root: root.to_path_buf(),
606 keychain: config.workers.keychain.clone(),
607 current_state: t.frontmatter.state.clone(),
608 command: Some(params.command.clone()),
609 };
610 if should_check_claude_compat(root, ¶ms.agent) {
611 check_output_format_supported(¶ms.command)?;
612 }
613 let mut child = spawn_worker(&ctx, ¶ms.agent, root)?;
614 let pid = child.id();
615
616 let pid_path = wt_display.join(".apm-worker.pid");
617 write_pid_file(&pid_path, pid, &id)?;
618 let enforce_isolation_next = skip_permissions || config.isolation.enforce_worktree_isolation;
619 let wt_for_cleanup_next = wt_display.clone();
620 let denial_log_path2 = log_path.clone();
621 let denial_worktree2 = wt_display.clone();
622 let denial_ticket_id2 = id.clone();
623 let agent_for_diag2 = params.agent.clone();
624 std::thread::spawn(move || {
625 let _ = child.wait();
626 let _ = std::fs::remove_file(&sys_file);
627 let _ = std::fs::remove_file(&msg_file);
628 if agent_for_diag2 == "claude" {
629 run_denial_scan(&denial_log_path2, &denial_worktree2, &denial_ticket_id2);
630 }
631 if enforce_isolation_next {
632 let _ = crate::wrapper::hook_config::remove_hook_config(&wt_for_cleanup_next);
633 }
634 });
635
636 messages.push(format!("Worker spawned: PID={pid}, log={}", log_path.display()));
637 messages.push(format!("Agent name: {worker_name}"));
638
639 Ok(RunNextOutput { ticket_id: Some(id), messages, warnings, worker_pid: Some(pid), log_path: Some(log_path) })
640}
641
642#[allow(clippy::type_complexity)]
643pub fn spawn_next_worker(
644 root: &Path,
645 no_aggressive: bool,
646 skip_permissions: bool,
647 epic_filter: Option<&str>,
648 blocked_epics: &[String],
649 default_blocked: bool,
650 messages: &mut Vec<String>,
651 warnings: &mut Vec<String>,
652) -> Result<Option<(String, Option<String>, ManagedChild, PathBuf)>> {
653 let config = Config::load(root)?;
654 let skip_permissions = skip_permissions || config.agents.skip_permissions;
655 let p = &config.workflow.prioritization;
656 let startable: Vec<&str> = config.workflow.states.iter()
657 .filter(|s| s.transitions.iter().any(|tr| tr.trigger == "command:start"))
658 .map(|s| s.id.as_str())
659 .collect();
660 let actionable_owned = config.actionable_states_for("agent");
661 let actionable: Vec<&str> = actionable_owned.iter().map(|s| s.as_str()).collect();
662 let all_tickets = ticket::load_all_from_git(root, &config.tickets.dir)?;
663 let tickets: Vec<ticket::Ticket> = {
664 let epic_filtered: Vec<ticket::Ticket> = match epic_filter {
665 Some(epic_id) => all_tickets.into_iter()
666 .filter(|t| t.frontmatter.epic.as_deref() == Some(epic_id))
667 .collect(),
668 None => all_tickets,
669 };
670 epic_filtered.into_iter()
671 .filter(|t| match t.frontmatter.epic.as_deref() {
672 Some(eid) => !blocked_epics.iter().any(|b| b == eid),
673 None => !default_blocked,
674 })
675 .collect()
676 };
677 let caller_name = crate::config::resolve_caller_name();
678 let current_user = crate::config::resolve_identity(root);
679
680 let Some(candidate) = ticket::pick_next(&tickets, &actionable, &startable, p.priority_weight, p.effort_weight, p.risk_weight, &config, Some(&caller_name), Some(¤t_user)) else {
681 return Ok(None);
682 };
683
684 let id = candidate.frontmatter.id.clone();
685 let epic_id = candidate.frontmatter.epic.clone();
686 let old_state = candidate.frontmatter.state.clone();
687
688 let triggering_transition_owned = config.workflow.states.iter()
689 .find(|s| s.id == old_state)
690 .and_then(|s| s.transitions.iter().find(|tr| tr.trigger == "command:start"))
691 .cloned();
692 let profile = triggering_transition_owned.as_ref().and_then(|tr| resolve_profile(tr, &config, warnings));
693 let state_instructions = config.workflow.states.iter()
694 .find(|s| s.id == old_state)
695 .and_then(|sc| sc.instructions.as_deref())
696 .map(|s| s.to_string());
697 let instructions_text = profile
698 .and_then(|p| p.instructions.as_deref())
699 .map(|path| {
700 match std::fs::read_to_string(root.join(path)) {
701 Ok(s) => s,
702 Err(_) => { warnings.push("warning: instructions file not found".to_string()); String::new() }
703 }
704 })
705 .filter(|s| !s.is_empty())
706 .or_else(|| state_instructions.as_deref()
707 .and_then(|path| {
708 std::fs::read_to_string(root.join(path)).ok()
709 .or_else(|| { warnings.push("warning: instructions file not found".to_string()); None })
710 }));
711 let start_out = run(root, &id, no_aggressive, false, false, &caller_name)?;
712 warnings.extend(start_out.warnings);
713
714 if let Some(ref msg) = start_out.merge_message {
715 messages.push(msg.clone());
716 }
717 messages.push(format!("{}: {} → {} (caller: {}, branch: {})", start_out.id, start_out.old_state, start_out.new_state, start_out.caller_name, start_out.branch));
718 messages.push(format!("Worktree: {}", start_out.worktree_path.display()));
719
720 let tickets2 = ticket::load_all_from_git(root, &config.tickets.dir)?;
721 let Some(t) = tickets2.iter().find(|t| t.frontmatter.id == id) else {
722 return Ok(None);
723 };
724
725 let focus_hint = if let Some(ref section) = t.frontmatter.focus_section {
726 let hint = format!("Pay special attention to section: {section}");
727 let rel_path = format!(
728 "{}/{}",
729 config.tickets.dir.to_string_lossy(),
730 t.path.file_name().unwrap().to_string_lossy()
731 );
732 let branch = t.frontmatter.branch.clone()
733 .or_else(|| ticket_fmt::branch_name_from_path(&t.path))
734 .unwrap_or_else(|| format!("ticket/{id}"));
735 let mut t_mut = t.clone();
736 t_mut.frontmatter.focus_section = None;
737 let cleared = t_mut.serialize()?;
738 git::commit_to_branch(root, &branch, &rel_path, &cleared,
739 &format!("ticket({id}): clear focus_section"))?;
740 Some(hint)
741 } else {
742 None
743 };
744
745 let mut prompt = String::new();
746 if let Some(ref instr) = instructions_text {
747 prompt.push_str(instr.trim());
748 prompt.push('\n');
749 }
750 if let Some(ref hint) = focus_hint {
751 if !prompt.is_empty() { prompt.push('\n'); }
752 prompt.push_str(hint);
753 prompt.push('\n');
754 }
755 let _ = prompt; let profile_name2 = triggering_transition_owned.as_ref()
758 .and_then(|tr| tr.profile.as_deref())
759 .unwrap_or("")
760 .to_string();
761 let profile2 = triggering_transition_owned.as_ref().and_then(|tr| resolve_profile(tr, &config, warnings));
762 let role2 = profile2.and_then(|p| p.role.as_deref()).unwrap_or("worker");
763 let mut params = effective_spawn_params(triggering_transition_owned.as_ref().and_then(|tr| tr.agent.as_deref()), profile2, &config.workers);
764 apply_frontmatter_agent(&mut params.agent, &t.frontmatter, &profile_name2);
765
766 let now_str = chrono::Utc::now().format("%m%d-%H%M").to_string();
767 let worker_name = format!("{}-{}-{:04x}", params.agent, now_str, rand_u16());
768 let tr_instructions_snw = triggering_transition_owned.as_ref().and_then(|tr| tr.instructions.as_deref());
769 let tr_role_prefix_snw = triggering_transition_owned.as_ref().and_then(|tr| tr.role_prefix.as_deref());
770 let worker_system = build_system_prompt(root, tr_instructions_snw, profile2, &config.workers, config.agents.instructions.as_deref(), ¶ms.agent, role2)?;
771
772 let raw = t.serialize()?;
773 let dep_ids_snw = t.frontmatter.depends_on.clone().unwrap_or_default();
774 let raw_prompt_snw = format!("{}\n\n{raw}", agent_role_prefix(tr_role_prefix_snw, profile2, &id));
775 let with_epic_snw = with_epic_bundle(root, t.frontmatter.epic.as_deref(), &id, &config, raw_prompt_snw);
776 let ticket_content = with_dependency_bundle(root, &dep_ids_snw, &config, with_epic_snw);
777 let role_prefix2 = tr_role_prefix_snw
778 .map(|p| p.replace("<id>", &id))
779 .or_else(|| profile2.and_then(|p| p.role_prefix.clone()));
780
781 let branch = t.frontmatter.branch.clone()
782 .or_else(|| ticket_fmt::branch_name_from_path(&t.path))
783 .unwrap_or_else(|| format!("ticket/{id}"));
784 let wt_name = branch.replace('/', "-");
785 let main_root = crate::git_util::main_worktree_root(root).unwrap_or_else(|| root.to_path_buf());
786 let wt_path = main_root.join(&config.worktrees.dir).join(&wt_name);
787 let wt_display = crate::worktree::find_worktree_for_branch(root, &branch).unwrap_or(wt_path);
788
789 let log_path = wt_display.join(".apm-worker.log");
790
791 let sys_file = write_temp_file("sys", &worker_system)?;
792 let msg_file = write_temp_file("msg", &ticket_content)?;
793 let ctx = WrapperContext {
794 worker_name: worker_name.clone(),
795 agent_type: params.agent.clone(),
796 ticket_id: id.clone(),
797 ticket_branch: branch.clone(),
798 worktree_path: wt_display.clone(),
799 system_prompt_file: sys_file.clone(),
800 user_message_file: msg_file.clone(),
801 skip_permissions,
802 profile: profile_name2,
803 role_prefix: role_prefix2,
804 options: params.options.clone(),
805 model: params.model.clone(),
806 log_path: log_path.clone(),
807 container: params.container.clone(),
808 extra_env: params.env.clone(),
809 root: root.to_path_buf(),
810 keychain: config.workers.keychain.clone(),
811 current_state: t.frontmatter.state.clone(),
812 command: Some(params.command.clone()),
813 };
814 if should_check_claude_compat(root, ¶ms.agent) {
815 check_output_format_supported(¶ms.command)?;
816 }
817 let child = spawn_worker(&ctx, ¶ms.agent, root)?;
818 let pid = child.id();
819
820 let denial_ctx = if params.agent == "claude" {
821 Some((log_path.clone(), wt_display.clone(), id.clone()))
822 } else {
823 None
824 };
825 let managed = ManagedChild {
826 inner: child,
827 temp_files: vec![sys_file, msg_file],
828 denial_ctx,
829 };
830
831 let pid_path = wt_display.join(".apm-worker.pid");
832 write_pid_file(&pid_path, pid, &id)?;
833
834 messages.push(format!("Worker spawned: PID={pid}, log={}", log_path.display()));
835 messages.push(format!("Agent name: {worker_name}"));
836
837 Ok(Some((id, epic_id, managed, pid_path)))
838}
839
840fn with_dependency_bundle(root: &Path, depends_on: &[String], config: &Config, content: String) -> String {
843 if depends_on.is_empty() {
844 return content;
845 }
846 let bundle = crate::context::build_dependency_bundle(root, depends_on, config);
847 if bundle.is_empty() {
848 return content;
849 }
850 format!("{bundle}\n{content}")
851}
852
853fn with_epic_bundle(root: &Path, epic_id: Option<&str>, ticket_id: &str, config: &Config, content: String) -> String {
856 match epic_id {
857 Some(eid) => {
858 let bundle = crate::context::build_epic_bundle(root, eid, ticket_id, config);
859 format!("{bundle}\n{content}")
860 }
861 None => content,
862 }
863}
864
865pub(crate) fn resolve_builtin_instructions(agent: &str, role: &str) -> Option<&'static str> {
866 match (agent, role) {
867 ("claude", "worker") => Some(CLAUDE_WORKER_DEFAULT),
868 ("claude", "spec-writer") => Some(CLAUDE_SPEC_WRITER_DEFAULT),
869 ("mock-happy", "worker") => Some(MOCK_HAPPY_WORKER_DEFAULT),
870 ("mock-happy", "spec-writer") => Some(MOCK_HAPPY_SPEC_WRITER_DEFAULT),
871 ("mock-sad", "worker") => Some(MOCK_SAD_WORKER_DEFAULT),
872 ("mock-sad", "spec-writer") => Some(MOCK_SAD_SPEC_WRITER_DEFAULT),
873 ("mock-random", "worker") => Some(MOCK_RANDOM_WORKER_DEFAULT),
874 ("mock-random", "spec-writer") => Some(MOCK_RANDOM_SPEC_WRITER_DEFAULT),
875 ("debug", "worker") => Some(DEBUG_WORKER_DEFAULT),
876 ("debug", "spec-writer") => Some(DEBUG_SPEC_WRITER_DEFAULT),
877 _ => None,
878 }
879}
880
881pub(crate) fn build_system_prompt(
882 root: &Path,
883 transition_instructions: Option<&str>,
884 profile: Option<&WorkerProfileConfig>,
885 workers: &WorkersConfig,
886 agents_instructions: Option<&Path>,
887 agent: &str,
888 role: &str,
889) -> Result<String> {
890 let base = build_system_prompt_body(root, transition_instructions, profile, workers, agent, role)?;
891
892 let Some(path) = agents_instructions else { return Ok(base); };
894 if path.as_os_str().is_empty() {
895 return Ok(base);
896 }
897 let prefix = std::fs::read_to_string(root.join(path))
898 .map_err(|_| anyhow::anyhow!("agents.instructions: file not found: {}", path.display()))?;
899 let prefix = prefix.trim_end();
900 Ok(format!("{prefix}\n\n{base}"))
901}
902
903fn build_system_prompt_body(
904 root: &Path,
905 transition_instructions: Option<&str>,
906 profile: Option<&WorkerProfileConfig>,
907 workers: &WorkersConfig,
908 agent: &str,
909 role: &str,
910) -> Result<String> {
911 let per_agent = root.join(format!(".apm/agents/{agent}/apm.{role}.md"));
913 if per_agent.exists() {
914 if let Ok(content) = std::fs::read_to_string(&per_agent) {
915 return Ok(content);
916 }
917 }
918 if let Some(path) = transition_instructions {
920 return std::fs::read_to_string(root.join(path))
921 .with_context(|| format!("transition.instructions: file not found: {path}"));
922 }
923 if let Some(p) = profile {
925 if let Some(ref instr_path) = p.instructions {
926 match std::fs::read_to_string(root.join(instr_path)) {
927 Ok(content) => return Ok(content),
928 Err(_) => bail!("[worker_profiles.*].instructions: file not found: {instr_path}"),
929 }
930 }
931 }
932 if let Some(ref instr_path) = workers.instructions {
934 match std::fs::read_to_string(root.join(instr_path)) {
935 Ok(content) => return Ok(content),
936 Err(_) => bail!("[workers].instructions: file not found: {instr_path}"),
937 }
938 }
939 if let Some(s) = resolve_builtin_instructions(agent, role) {
941 return Ok(s.to_string());
942 }
943 bail!(
945 "no instructions found for agent '{agent}' role '{role}': \
946 set [workers].instructions in .apm/config.toml or add \
947 .apm/agents/{agent}/apm.{role}.md"
948 )
949}
950
951fn agent_role_prefix(transition_role_prefix: Option<&str>, profile: Option<&WorkerProfileConfig>, id: &str) -> String {
952 if let Some(prefix) = transition_role_prefix {
953 return prefix.replace("<id>", id);
954 }
955 if let Some(p) = profile {
956 if let Some(ref prefix) = p.role_prefix {
957 return prefix.replace("<id>", id);
958 }
959 }
960 format!("You are a Worker agent assigned to ticket #{id}.")
961}
962
963fn write_pid_file(path: &Path, pid: u32, ticket_id: &str) -> Result<()> {
964 let started_at = chrono::Utc::now().format("%Y-%m-%dT%H:%MZ").to_string();
965 let content = serde_json::json!({
966 "pid": pid,
967 "ticket_id": ticket_id,
968 "started_at": started_at,
969 })
970 .to_string();
971 std::fs::write(path, content)?;
972 Ok(())
973}
974
975fn rand_u16() -> u16 {
976 use std::time::{SystemTime, UNIX_EPOCH};
977 SystemTime::now().duration_since(UNIX_EPOCH).unwrap_or_default().subsec_nanos() as u16
978}
979
980#[cfg(test)]
981mod tests {
982 use super::{build_system_prompt, agent_role_prefix, resolve_profile, effective_spawn_params, check_output_format_supported, apply_frontmatter_agent, ManagedChild, DEPRECATION_WARNED, DEPRECATION_MSG, DEPRECATION_TEST_LOCK, emit_deprecation_warning_to};
983 use crate::config::{WorkerProfileConfig, WorkersConfig, TransitionConfig, CompletionStrategy};
984 use std::collections::HashMap;
985
986 fn make_transition(profile: Option<&str>) -> TransitionConfig {
987 TransitionConfig {
988 to: "in_progress".into(),
989 trigger: "command:start".into(),
990 label: String::new(),
991 hint: String::new(),
992 completion: CompletionStrategy::None,
993 focus_section: None,
994 context_section: None,
995 warning: None,
996 profile: profile.map(|s| s.to_string()),
997 instructions: None,
998 role_prefix: None,
999 agent: None,
1000 on_failure: None,
1001 outcome: None,
1002 }
1003 }
1004
1005 fn make_profile(instructions: Option<&str>, role_prefix: Option<&str>) -> WorkerProfileConfig {
1006 WorkerProfileConfig {
1007 instructions: instructions.map(|s| s.to_string()),
1008 role_prefix: role_prefix.map(|s| s.to_string()),
1009 ..Default::default()
1010 }
1011 }
1012
1013 fn make_workers(command: &str, model: Option<&str>) -> WorkersConfig {
1014 WorkersConfig {
1015 command: Some(command.to_string()),
1016 args: None,
1017 model: model.map(|s| s.to_string()),
1018 env: HashMap::new(),
1019 container: None,
1020 keychain: HashMap::new(),
1021 agent: None,
1022 options: HashMap::new(),
1023 instructions: None,
1024 }
1025 }
1026
1027 #[test]
1030 fn resolve_profile_returns_profile_when_found() {
1031 let mut config = crate::config::Config {
1032 project: crate::config::ProjectConfig {
1033 name: "test".into(),
1034 description: String::new(),
1035 default_branch: "main".into(),
1036 collaborators: vec![],
1037 },
1038 ticket: Default::default(),
1039 tickets: Default::default(),
1040 workflow: Default::default(),
1041 agents: Default::default(),
1042 worktrees: Default::default(),
1043 sync: Default::default(),
1044 logging: Default::default(),
1045 workers: make_workers("claude", None),
1046 work: Default::default(),
1047 server: Default::default(),
1048 git_host: Default::default(),
1049 worker_profiles: HashMap::new(),
1050 context: Default::default(),
1051 isolation: Default::default(),
1052 load_warnings: vec![],
1053 };
1054 let profile = make_profile(Some(".apm/spec.md"), Some("Spec-Writer for #<id>"));
1055 config.worker_profiles.insert("spec_agent".into(), profile);
1056
1057 let tr = make_transition(Some("spec_agent"));
1058 let mut w = Vec::new();
1059 assert!(resolve_profile(&tr, &config, &mut w).is_some());
1060 }
1061
1062 #[test]
1063 fn resolve_profile_returns_none_for_missing_profile() {
1064 let config = crate::config::Config {
1065 project: crate::config::ProjectConfig {
1066 name: "test".into(),
1067 description: String::new(),
1068 default_branch: "main".into(),
1069 collaborators: vec![],
1070 },
1071 ticket: Default::default(),
1072 tickets: Default::default(),
1073 workflow: Default::default(),
1074 agents: Default::default(),
1075 worktrees: Default::default(),
1076 sync: Default::default(),
1077 logging: Default::default(),
1078 workers: make_workers("claude", None),
1079 work: Default::default(),
1080 server: Default::default(),
1081 git_host: Default::default(),
1082 worker_profiles: HashMap::new(),
1083 context: Default::default(),
1084 isolation: Default::default(),
1085 load_warnings: vec![],
1086 };
1087 let tr = make_transition(Some("nonexistent_profile"));
1088 let mut w = Vec::new();
1089 assert!(resolve_profile(&tr, &config, &mut w).is_none());
1090 }
1091
1092 #[test]
1093 fn resolve_profile_returns_none_when_no_profile_on_transition() {
1094 let config = crate::config::Config {
1095 project: crate::config::ProjectConfig {
1096 name: "test".into(),
1097 description: String::new(),
1098 default_branch: "main".into(),
1099 collaborators: vec![],
1100 },
1101 ticket: Default::default(),
1102 tickets: Default::default(),
1103 workflow: Default::default(),
1104 agents: Default::default(),
1105 worktrees: Default::default(),
1106 sync: Default::default(),
1107 logging: Default::default(),
1108 workers: make_workers("claude", None),
1109 work: Default::default(),
1110 server: Default::default(),
1111 git_host: Default::default(),
1112 worker_profiles: HashMap::new(),
1113 context: Default::default(),
1114 isolation: Default::default(),
1115 load_warnings: vec![],
1116 };
1117 let tr = make_transition(None);
1118 let mut w = Vec::new();
1119 assert!(resolve_profile(&tr, &config, &mut w).is_none());
1120 }
1121
1122 #[test]
1125 fn effective_spawn_params_profile_command_overrides_global() {
1126 let workers = make_workers("claude", Some("sonnet"));
1127 let profile = WorkerProfileConfig {
1128 command: Some("my-claude".into()),
1129 ..Default::default()
1130 };
1131 let params = effective_spawn_params(None, Some(&profile), &workers);
1132 assert_eq!(params.command, "my-claude");
1133 }
1134
1135 #[test]
1136 fn effective_spawn_params_falls_back_to_global_command() {
1137 let workers = make_workers("claude", None);
1138 let params = effective_spawn_params(None, None, &workers);
1139 assert_eq!(params.command, "claude");
1140 }
1141
1142 #[test]
1143 fn effective_spawn_params_profile_model_overrides_global() {
1144 let workers = make_workers("claude", Some("sonnet"));
1145 let profile = WorkerProfileConfig {
1146 model: Some("opus".into()),
1147 ..Default::default()
1148 };
1149 let params = effective_spawn_params(None, Some(&profile), &workers);
1150 assert_eq!(params.model.as_deref(), Some("opus"));
1151 }
1152
1153 #[test]
1154 fn effective_spawn_params_falls_back_to_global_model() {
1155 let workers = make_workers("claude", Some("sonnet"));
1156 let params = effective_spawn_params(None, None, &workers);
1157 assert_eq!(params.model.as_deref(), Some("sonnet"));
1158 }
1159
1160 #[test]
1161 fn effective_spawn_params_profile_env_merged_over_global() {
1162 let mut workers = make_workers("claude", None);
1163 workers.env.insert("FOO".into(), "global".into());
1164 workers.env.insert("BAR".into(), "bar".into());
1165
1166 let mut profile_env = HashMap::new();
1167 profile_env.insert("FOO".into(), "profile".into());
1168 let profile = WorkerProfileConfig {
1169 env: profile_env,
1170 ..Default::default()
1171 };
1172 let params = effective_spawn_params(None, Some(&profile), &workers);
1173 assert_eq!(params.env.get("FOO").map(|s| s.as_str()), Some("profile"));
1174 assert_eq!(params.env.get("BAR").map(|s| s.as_str()), Some("bar"));
1175 }
1176
1177 #[test]
1178 fn effective_spawn_params_profile_container_overrides_global() {
1179 let mut workers = make_workers("claude", None);
1180 workers.container = Some("global-image".into());
1181 let profile = WorkerProfileConfig {
1182 container: Some("profile-image".into()),
1183 ..Default::default()
1184 };
1185 let params = effective_spawn_params(None, Some(&profile), &workers);
1186 assert_eq!(params.container.as_deref(), Some("profile-image"));
1187 }
1188
1189 #[test]
1190 fn transition_agent_takes_precedence_over_profile() {
1191 let workers = WorkersConfig::default();
1192 let profile = WorkerProfileConfig { agent: Some("other".into()), ..Default::default() };
1193 let params = effective_spawn_params(Some("custom"), Some(&profile), &workers);
1194 assert_eq!(params.agent, "custom");
1195 }
1196
1197 #[test]
1198 fn effective_agent_defaults_to_claude() {
1199 let workers = WorkersConfig::default();
1200 let params = effective_spawn_params(None, None, &workers);
1201 assert_eq!(params.agent, "claude");
1202 }
1203
1204 #[test]
1207 fn build_system_prompt_uses_profile_instructions() {
1208 let dir = tempfile::tempdir().unwrap();
1209 let p = dir.path();
1210 std::fs::create_dir_all(p.join(".apm")).unwrap();
1211 std::fs::write(p.join(".apm/spec.md"), "SPEC WRITER").unwrap();
1212 let profile = make_profile(Some(".apm/spec.md"), None);
1213 let workers = WorkersConfig::default();
1214 assert_eq!(
1215 build_system_prompt(p, None, Some(&profile), &workers, None, "claude", "worker").unwrap(),
1216 "SPEC WRITER"
1217 );
1218 }
1219
1220 #[test]
1221 fn build_system_prompt_uses_workers_instructions_when_no_profile() {
1222 let dir = tempfile::tempdir().unwrap();
1223 let p = dir.path();
1224 std::fs::create_dir_all(p.join(".apm")).unwrap();
1225 std::fs::write(p.join(".apm/global.md"), "GLOBAL INSTRUCTIONS").unwrap();
1226 let workers = WorkersConfig {
1227 instructions: Some(".apm/global.md".to_string()),
1228 ..WorkersConfig::default()
1229 };
1230 assert_eq!(
1231 build_system_prompt(p, None, None, &workers, None, "claude", "worker").unwrap(),
1232 "GLOBAL INSTRUCTIONS"
1233 );
1234 }
1235
1236 #[test]
1237 fn build_system_prompt_uses_per_agent_file() {
1238 let dir = tempfile::tempdir().unwrap();
1239 let p = dir.path();
1240 std::fs::create_dir_all(p.join(".apm/agents/claude")).unwrap();
1241 std::fs::write(p.join(".apm/agents/claude/apm.worker.md"), "PER AGENT WORKER").unwrap();
1242 let workers = WorkersConfig::default();
1243 assert_eq!(
1244 build_system_prompt(p, None, None, &workers, None, "claude", "worker").unwrap(),
1245 "PER AGENT WORKER"
1246 );
1247 }
1248
1249 #[test]
1250 fn build_system_prompt_falls_back_to_builtin_default() {
1251 let dir = tempfile::tempdir().unwrap();
1252 let p = dir.path();
1253 let workers = WorkersConfig::default();
1254 let result = build_system_prompt(p, None, None, &workers, None, "claude", "worker").unwrap();
1255 assert_eq!(result, super::CLAUDE_WORKER_DEFAULT);
1256 }
1257
1258 #[test]
1259 fn build_system_prompt_falls_back_to_builtin_spec_writer() {
1260 let dir = tempfile::tempdir().unwrap();
1261 let p = dir.path();
1262 let workers = WorkersConfig::default();
1263 let result = build_system_prompt(p, None, None, &workers, None, "claude", "spec-writer").unwrap();
1264 assert_eq!(result, super::CLAUDE_SPEC_WRITER_DEFAULT);
1265 }
1266
1267 #[test]
1268 fn build_system_prompt_errors_for_unknown_agent() {
1269 let dir = tempfile::tempdir().unwrap();
1270 let p = dir.path();
1271 let workers = WorkersConfig::default();
1272 let result = build_system_prompt(p, None, None, &workers, None, "custom-bot", "worker");
1273 assert!(result.is_err());
1274 let msg = result.unwrap_err().to_string();
1275 assert!(msg.contains("custom-bot"), "error should name the agent: {msg}");
1276 assert!(msg.contains("worker"), "error should name the role: {msg}");
1277 }
1278
1279 #[test]
1280 fn build_system_prompt_profile_instructions_missing_file_is_error() {
1281 let dir = tempfile::tempdir().unwrap();
1282 let p = dir.path();
1283 let profile = make_profile(Some(".apm/nonexistent.md"), None);
1284 let workers = WorkersConfig::default();
1285 let result = build_system_prompt(p, None, Some(&profile), &workers, None, "claude", "worker");
1286 assert!(result.is_err());
1287 let msg = result.unwrap_err().to_string();
1288 assert!(msg.contains("nonexistent.md"), "error should name the file: {msg}");
1289 }
1290
1291 #[test]
1292 fn build_system_prompt_backward_compat() {
1293 let dir = tempfile::tempdir().unwrap();
1294 let p = dir.path();
1295 std::fs::create_dir_all(p.join(".apm")).unwrap();
1296 std::fs::write(p.join(".apm/apm.worker.md"), "LEGACY WORKER CONTENT").unwrap();
1297 let profile = make_profile(Some(".apm/apm.worker.md"), None);
1298 let workers = WorkersConfig::default();
1299 assert_eq!(
1300 build_system_prompt(p, None, Some(&profile), &workers, None, "claude", "worker").unwrap(),
1301 "LEGACY WORKER CONTENT"
1302 );
1303 }
1304
1305 #[test]
1308 fn agents_instructions_prepended_with_blank_line() {
1309 let dir = tempfile::tempdir().unwrap();
1310 let p = dir.path();
1311 std::fs::write(p.join("prefix.md"), "PREFIX CONTENT\n").unwrap();
1312 let workers = WorkersConfig::default();
1313 let result = build_system_prompt(
1314 p, None, None, &workers,
1315 Some(std::path::Path::new("prefix.md")),
1316 "claude", "worker",
1317 ).unwrap();
1318 let expected = format!("PREFIX CONTENT\n\n{}", super::CLAUDE_WORKER_DEFAULT);
1319 assert_eq!(result, expected);
1320 }
1321
1322 #[test]
1323 fn agents_instructions_none_is_no_op() {
1324 let dir = tempfile::tempdir().unwrap();
1325 let p = dir.path();
1326 let workers = WorkersConfig::default();
1327 let without_prefix = build_system_prompt(p, None, None, &workers, None, "claude", "worker").unwrap();
1328 assert_eq!(without_prefix, super::CLAUDE_WORKER_DEFAULT);
1329 }
1330
1331 #[test]
1332 fn agents_instructions_empty_path_is_no_op() {
1333 let dir = tempfile::tempdir().unwrap();
1334 let p = dir.path();
1335 let workers = WorkersConfig::default();
1336 let result = build_system_prompt(
1337 p, None, None, &workers,
1338 Some(std::path::Path::new("")),
1339 "claude", "worker",
1340 ).unwrap();
1341 assert_eq!(result, super::CLAUDE_WORKER_DEFAULT);
1342 }
1343
1344 #[test]
1345 fn agents_instructions_missing_file_is_hard_error() {
1346 let dir = tempfile::tempdir().unwrap();
1347 let p = dir.path();
1348 let workers = WorkersConfig::default();
1349 let result = build_system_prompt(
1350 p, None, None, &workers,
1351 Some(std::path::Path::new("no-such-file.md")),
1352 "claude", "worker",
1353 );
1354 assert!(result.is_err());
1355 let msg = result.unwrap_err().to_string();
1356 assert!(msg.contains("agents.instructions"), "error should mention agents.instructions: {msg}");
1357 assert!(msg.contains("no-such-file.md"), "error should name the file: {msg}");
1358 }
1359
1360 #[test]
1361 fn agents_instructions_trailing_whitespace_trimmed() {
1362 let dir = tempfile::tempdir().unwrap();
1363 let p = dir.path();
1364 std::fs::write(p.join("prefix.md"), "PREFIX\n\n\n").unwrap();
1366 let workers = WorkersConfig::default();
1367 let result = build_system_prompt(
1368 p, None, None, &workers,
1369 Some(std::path::Path::new("prefix.md")),
1370 "claude", "worker",
1371 ).unwrap();
1372 let expected = format!("PREFIX\n\n{}", super::CLAUDE_WORKER_DEFAULT);
1374 assert_eq!(result, expected);
1375 }
1376
1377 #[test]
1380 fn agent_role_prefix_uses_profile_role_prefix() {
1381 let profile = make_profile(None, Some("You are a Spec-Writer agent assigned to ticket #<id>."));
1382 assert_eq!(
1383 agent_role_prefix(None, Some(&profile), "abc123"),
1384 "You are a Spec-Writer agent assigned to ticket #abc123."
1385 );
1386 }
1387
1388 #[test]
1389 fn agent_role_prefix_falls_back_to_worker_default() {
1390 assert_eq!(
1391 agent_role_prefix(None, None, "abc123"),
1392 "You are a Worker agent assigned to ticket #abc123."
1393 );
1394 }
1395
1396 #[test]
1397 fn agent_role_prefix_transition_takes_precedence_over_profile() {
1398 let profile = make_profile(None, Some("You are a Spec-Writer agent assigned to ticket #<id>."));
1399 assert_eq!(
1400 agent_role_prefix(Some("You are a Custom agent for ticket #<id>."), Some(&profile), "abc123"),
1401 "You are a Custom agent for ticket #abc123."
1402 );
1403 }
1404
1405 #[test]
1408 fn transition_instructions_takes_precedence_over_profile() {
1409 let dir = tempfile::tempdir().unwrap();
1410 let p = dir.path();
1411 std::fs::create_dir_all(p.join(".apm")).unwrap();
1412 std::fs::write(p.join(".apm/transition.md"), "TRANSITION CONTENT").unwrap();
1413 std::fs::write(p.join(".apm/profile.md"), "PROFILE CONTENT").unwrap();
1414 let profile = make_profile(Some(".apm/profile.md"), None);
1415 let workers = WorkersConfig::default();
1416 assert_eq!(
1417 build_system_prompt(p, Some(".apm/transition.md"), Some(&profile), &workers, None, "claude", "worker").unwrap(),
1418 "TRANSITION CONTENT"
1419 );
1420 }
1421
1422 #[test]
1423 fn transition_instructions_no_profile_required() {
1424 let dir = tempfile::tempdir().unwrap();
1425 let p = dir.path();
1426 std::fs::create_dir_all(p.join(".apm")).unwrap();
1427 std::fs::write(p.join(".apm/transition.md"), "TRANSITION ONLY").unwrap();
1428 let workers = WorkersConfig::default();
1429 assert_eq!(
1430 build_system_prompt(p, Some(".apm/transition.md"), None, &workers, None, "claude", "worker").unwrap(),
1431 "TRANSITION ONLY"
1432 );
1433 }
1434
1435 #[test]
1436 fn per_agent_file_beats_transition_instructions() {
1437 let dir = tempfile::tempdir().unwrap();
1438 let p = dir.path();
1439 std::fs::create_dir_all(p.join(".apm/agents/claude")).unwrap();
1440 std::fs::write(p.join(".apm/agents/claude/apm.worker.md"), "PER AGENT WINS").unwrap();
1441 std::fs::write(p.join(".apm/transition.md"), "TRANSITION CONTENT").unwrap();
1442 let workers = WorkersConfig::default();
1443 assert_eq!(
1444 build_system_prompt(p, Some(".apm/transition.md"), None, &workers, None, "claude", "worker").unwrap(),
1445 "PER AGENT WINS"
1446 );
1447 }
1448
1449 #[test]
1450 fn epic_filter_keeps_only_matching_tickets() {
1451 use crate::ticket::Ticket;
1452 use std::path::Path;
1453
1454 let make_ticket = |id: &str, epic: Option<&str>| {
1455 let epic_line = epic.map(|e| format!("epic = \"{e}\"\n")).unwrap_or_default();
1456 let raw = format!(
1457 "+++\nid = \"{id}\"\ntitle = \"T\"\nstate = \"ready\"\n{epic_line}+++\n"
1458 );
1459 Ticket::parse(Path::new("tickets/dummy.md"), &raw).unwrap()
1460 };
1461
1462 let all_tickets = vec![
1463 make_ticket("aaa", Some("epic1")),
1464 make_ticket("bbb", Some("epic2")),
1465 make_ticket("ccc", None),
1466 ];
1467
1468 let epic_id = "epic1";
1469 let filtered: Vec<Ticket> = all_tickets.into_iter()
1470 .filter(|t| t.frontmatter.epic.as_deref() == Some(epic_id))
1471 .collect();
1472
1473 assert_eq!(filtered.len(), 1);
1474 assert_eq!(filtered[0].frontmatter.id, "aaa");
1475 }
1476
1477 #[test]
1478 fn no_epic_filter_keeps_all_tickets() {
1479 use crate::ticket::Ticket;
1480 use std::path::Path;
1481
1482 let make_ticket = |id: &str, epic: Option<&str>| {
1483 let epic_line = epic.map(|e| format!("epic = \"{e}\"\n")).unwrap_or_default();
1484 let raw = format!(
1485 "+++\nid = \"{id}\"\ntitle = \"T\"\nstate = \"ready\"\n{epic_line}+++\n"
1486 );
1487 Ticket::parse(Path::new("tickets/dummy.md"), &raw).unwrap()
1488 };
1489
1490 let all_tickets: Vec<Ticket> = vec![
1491 make_ticket("aaa", Some("epic1")),
1492 make_ticket("bbb", Some("epic2")),
1493 make_ticket("ccc", None),
1494 ];
1495
1496 let count = all_tickets.len();
1497 let epic_filter: Option<&str> = None;
1498 let filtered: Vec<Ticket> = match epic_filter {
1499 Some(eid) => all_tickets.into_iter()
1500 .filter(|t| t.frontmatter.epic.as_deref() == Some(eid))
1501 .collect(),
1502 None => all_tickets,
1503 };
1504 assert_eq!(filtered.len(), count);
1505 }
1506
1507 #[test]
1510 fn spawn_worker_cwd_is_ticket_worktree() {
1511 use std::os::unix::fs::PermissionsExt;
1512
1513 let wt = tempfile::tempdir().unwrap();
1514 let log_dir = tempfile::tempdir().unwrap();
1515 let mock_dir = tempfile::tempdir().unwrap();
1516
1517 let mock_claude = mock_dir.path().join("claude");
1519 let cwd_file = wt.path().join("cwd-output.txt");
1520 let script = format!(concat!(
1521 "#!/bin/sh\n",
1522 "pwd > \"{}\"\n",
1523 ), cwd_file.display());
1524 std::fs::write(&mock_claude, &script).unwrap();
1525 std::fs::set_permissions(&mock_claude, std::fs::Permissions::from_mode(0o755)).unwrap();
1526
1527 let sys_file = crate::wrapper::write_temp_file("sys", "system").unwrap();
1528 let msg_file = crate::wrapper::write_temp_file("msg", "ticket content").unwrap();
1529
1530 let mut extra_env = HashMap::new();
1531 extra_env.insert(
1532 "PATH".to_string(),
1533 format!("{}:{}", mock_dir.path().display(), std::env::var("PATH").unwrap_or_default()),
1534 );
1535
1536 let ctx = crate::wrapper::WrapperContext {
1537 worker_name: "test-worker".to_string(),
1538 agent_type: "test".to_string(),
1539 ticket_id: "test-id".to_string(),
1540 ticket_branch: "ticket/test-id".to_string(),
1541 worktree_path: wt.path().to_path_buf(),
1542 system_prompt_file: sys_file.clone(),
1543 user_message_file: msg_file.clone(),
1544 skip_permissions: false,
1545 profile: "default".to_string(),
1546 role_prefix: None,
1547 options: HashMap::new(),
1548 model: None,
1549 log_path: log_dir.path().join("worker.log"),
1550 container: None,
1551 extra_env,
1552 root: wt.path().to_path_buf(),
1553 keychain: HashMap::new(),
1554 current_state: "in_progress".to_string(),
1555 command: None,
1556 };
1557
1558 let wrapper = crate::wrapper::resolve_builtin("claude").unwrap();
1559 let mut child = wrapper.spawn(&ctx).unwrap();
1560 child.wait().unwrap();
1561 let _ = std::fs::remove_file(&sys_file);
1562 let _ = std::fs::remove_file(&msg_file);
1563
1564 let cwd_out = std::fs::read_to_string(&cwd_file)
1565 .expect("cwd-output.txt not written — mock claude did not run in expected cwd");
1566 let expected = wt.path().canonicalize().unwrap();
1567 assert_eq!(
1568 cwd_out.trim(),
1569 expected.to_str().unwrap(),
1570 "spawned worker CWD must equal the ticket worktree path"
1571 );
1572 }
1573
1574 #[test]
1577 fn check_output_format_supported_passes_when_flag_present() {
1578 use std::os::unix::fs::PermissionsExt;
1579 let dir = tempfile::tempdir().unwrap();
1580 let bin = dir.path().join("fake-claude");
1581 std::fs::write(&bin, "#!/bin/sh\necho '--output-format stream-json'\n").unwrap();
1582 std::fs::set_permissions(&bin, std::fs::Permissions::from_mode(0o755)).unwrap();
1583 assert!(check_output_format_supported(bin.to_str().unwrap()).is_ok());
1584 }
1585
1586 #[test]
1587 fn check_output_format_supported_errors_when_flag_absent() {
1588 use std::os::unix::fs::PermissionsExt;
1589 let dir = tempfile::tempdir().unwrap();
1590 let bin = dir.path().join("old-claude");
1591 std::fs::write(&bin, "#!/bin/sh\necho 'Usage: old-claude [options]'\n").unwrap();
1592 std::fs::set_permissions(&bin, std::fs::Permissions::from_mode(0o755)).unwrap();
1593 let err = check_output_format_supported(bin.to_str().unwrap()).unwrap_err();
1594 let msg = err.to_string();
1595 assert!(
1596 msg.contains("--output-format"),
1597 "error message must name the missing flag: {msg}"
1598 );
1599 assert!(
1600 msg.contains(bin.to_str().unwrap()),
1601 "error message must include binary path: {msg}"
1602 );
1603 }
1604
1605 #[test]
1608 fn claude_wrapper_sets_apm_env_vars() {
1609 use std::os::unix::fs::PermissionsExt;
1610
1611 let wt = tempfile::tempdir().unwrap();
1612 let log_dir = tempfile::tempdir().unwrap();
1613 let mock_dir = tempfile::tempdir().unwrap();
1614 let env_output = wt.path().join("env-output.txt");
1615
1616 let mock_claude = mock_dir.path().join("claude");
1618 let script = format!(
1619 "#!/bin/sh\nprintenv > \"{}\"\n",
1620 env_output.display()
1621 );
1622 std::fs::write(&mock_claude, &script).unwrap();
1623 std::fs::set_permissions(&mock_claude, std::fs::Permissions::from_mode(0o755)).unwrap();
1624
1625 let sys_file = crate::wrapper::write_temp_file("sys", "system prompt").unwrap();
1626 let msg_file = crate::wrapper::write_temp_file("msg", "ticket content").unwrap();
1627
1628 let mut extra_env = HashMap::new();
1629 extra_env.insert(
1630 "PATH".to_string(),
1631 format!("{}:{}", mock_dir.path().display(), std::env::var("PATH").unwrap_or_default()),
1632 );
1633
1634 let ctx = crate::wrapper::WrapperContext {
1635 worker_name: "test-worker".to_string(),
1636 agent_type: "test".to_string(),
1637 ticket_id: "abc123".to_string(),
1638 ticket_branch: "ticket/abc123-some-feature".to_string(),
1639 worktree_path: wt.path().to_path_buf(),
1640 system_prompt_file: sys_file.clone(),
1641 user_message_file: msg_file.clone(),
1642 skip_permissions: false,
1643 profile: "my-profile".to_string(),
1644 role_prefix: None,
1645 options: HashMap::new(),
1646 model: None,
1647 log_path: log_dir.path().join("worker.log"),
1648 container: None,
1649 extra_env,
1650 root: wt.path().to_path_buf(),
1651 keychain: HashMap::new(),
1652 current_state: "in_progress".to_string(),
1653 command: None,
1654 };
1655
1656 let wrapper = crate::wrapper::resolve_builtin("claude").unwrap();
1657 let mut child = wrapper.spawn(&ctx).unwrap();
1658 child.wait().unwrap();
1659 let _ = std::fs::remove_file(&sys_file);
1660 let _ = std::fs::remove_file(&msg_file);
1661
1662 let env_content = std::fs::read_to_string(&env_output)
1663 .expect("env-output.txt not written — mock claude did not run");
1664
1665 assert!(env_content.contains("APM_AGENT_NAME=test-worker"), "missing APM_AGENT_NAME\n{env_content}");
1666 assert!(env_content.contains("APM_TICKET_ID=abc123"), "missing APM_TICKET_ID\n{env_content}");
1667 assert!(env_content.contains("APM_TICKET_BRANCH=ticket/abc123-some-feature"), "missing APM_TICKET_BRANCH\n{env_content}");
1668 assert!(env_content.contains("APM_TICKET_WORKTREE="), "missing APM_TICKET_WORKTREE\n{env_content}");
1669 assert!(env_content.contains("APM_SYSTEM_PROMPT_FILE="), "missing APM_SYSTEM_PROMPT_FILE\n{env_content}");
1670 assert!(env_content.contains("APM_USER_MESSAGE_FILE="), "missing APM_USER_MESSAGE_FILE\n{env_content}");
1671 assert!(env_content.contains("APM_SKIP_PERMISSIONS=0"), "missing APM_SKIP_PERMISSIONS\n{env_content}");
1672 assert!(env_content.contains("APM_PROFILE=my-profile"), "missing APM_PROFILE\n{env_content}");
1673 assert!(env_content.contains("APM_WRAPPER_VERSION=1"), "missing APM_WRAPPER_VERSION\n{env_content}");
1674 assert!(env_content.contains("APM_BIN="), "missing APM_BIN\n{env_content}");
1675
1676 if let Some(line) = env_content.lines().find(|l| l.starts_with("APM_BIN=")) {
1678 let path = line.trim_start_matches("APM_BIN=");
1679 assert!(std::path::Path::new(path).exists(), "APM_BIN path does not exist: {path}");
1680 }
1681 }
1682
1683 #[test]
1686 fn temp_files_removed_after_child_exits() {
1687 use std::os::unix::fs::PermissionsExt;
1688
1689 let wt = tempfile::tempdir().unwrap();
1690 let log_dir = tempfile::tempdir().unwrap();
1691 let mock_dir = tempfile::tempdir().unwrap();
1692
1693 let mock_claude = mock_dir.path().join("claude");
1695 std::fs::write(&mock_claude, "#!/bin/sh\nexit 0\n").unwrap();
1696 std::fs::set_permissions(&mock_claude, std::fs::Permissions::from_mode(0o755)).unwrap();
1697
1698 let sys_file = crate::wrapper::write_temp_file("sys", "system").unwrap();
1699 let msg_file = crate::wrapper::write_temp_file("msg", "message").unwrap();
1700
1701 assert!(sys_file.exists(), "sys_file should exist before spawn");
1702 assert!(msg_file.exists(), "msg_file should exist before spawn");
1703
1704 let mut extra_env = HashMap::new();
1705 extra_env.insert(
1706 "PATH".to_string(),
1707 format!("{}:{}", mock_dir.path().display(), std::env::var("PATH").unwrap_or_default()),
1708 );
1709
1710 let ctx = crate::wrapper::WrapperContext {
1711 worker_name: "test".to_string(),
1712 agent_type: "test".to_string(),
1713 ticket_id: "test123".to_string(),
1714 ticket_branch: "ticket/test123".to_string(),
1715 worktree_path: wt.path().to_path_buf(),
1716 system_prompt_file: sys_file.clone(),
1717 user_message_file: msg_file.clone(),
1718 skip_permissions: false,
1719 profile: "default".to_string(),
1720 role_prefix: None,
1721 options: HashMap::new(),
1722 model: None,
1723 log_path: log_dir.path().join("worker.log"),
1724 container: None,
1725 extra_env,
1726 root: wt.path().to_path_buf(),
1727 keychain: HashMap::new(),
1728 current_state: "in_progress".to_string(),
1729 command: None,
1730 };
1731
1732 let wrapper = crate::wrapper::resolve_builtin("claude").unwrap();
1733 let child = wrapper.spawn(&ctx).unwrap();
1734
1735 let mut managed = ManagedChild {
1736 inner: child,
1737 temp_files: vec![sys_file.clone(), msg_file.clone()],
1738 denial_ctx: None,
1739 };
1740 managed.inner.wait().unwrap();
1741 drop(managed);
1742
1743 assert!(!sys_file.exists(), "sys_file should be removed after ManagedChild is dropped");
1744 assert!(!msg_file.exists(), "msg_file should be removed after ManagedChild is dropped");
1745 }
1746
1747 #[test]
1750 fn resolution_agent_profile_overrides_global() {
1751 let workers = WorkersConfig { agent: Some("codex".into()), ..Default::default() };
1752 let profile = WorkerProfileConfig { agent: Some("mock-happy".into()), ..Default::default() };
1753 let params = effective_spawn_params(None, Some(&profile), &workers);
1754 assert_eq!(params.agent, "mock-happy");
1755 }
1756
1757 #[test]
1758 fn resolution_agent_falls_back_to_claude() {
1759 let params = effective_spawn_params(None, None, &WorkersConfig::default());
1760 assert_eq!(params.agent, "claude");
1761 }
1762
1763 #[test]
1764 fn resolution_options_merge() {
1765 let mut workers = WorkersConfig { agent: Some("claude".into()), ..Default::default() };
1766 workers.options.insert("model".into(), "opus".into());
1767 workers.options.insert("timeout".into(), "30".into());
1768 let mut profile_opts = HashMap::new();
1769 profile_opts.insert("model".into(), "sonnet".into());
1770 let profile = WorkerProfileConfig { options: profile_opts, ..Default::default() };
1771 let params = effective_spawn_params(None, Some(&profile), &workers);
1772 assert_eq!(params.options.get("model").map(|s| s.as_str()), Some("sonnet"), "profile model should override workers model");
1773 assert_eq!(params.options.get("timeout").map(|s| s.as_str()), Some("30"), "non-overlapping key should survive");
1774 }
1775
1776 #[test]
1777 fn deprecation_warning_writes_to_stream_once() {
1778 let _guard = DEPRECATION_TEST_LOCK.lock().unwrap();
1779 DEPRECATION_WARNED.store(false, std::sync::atomic::Ordering::SeqCst);
1780
1781 let mut buf: Vec<u8> = Vec::new();
1784 emit_deprecation_warning_to(&mut buf);
1785 emit_deprecation_warning_to(&mut buf);
1786
1787 let captured = String::from_utf8(buf).unwrap();
1788 let count = captured.matches(DEPRECATION_MSG).count();
1789 assert_eq!(count, 1, "deprecated message should appear exactly once on the writer, found {count}\n{captured}");
1790 }
1791
1792 #[test]
1793 fn deprecation_warning_triggered_by_legacy_workers_config() {
1794 let _guard = DEPRECATION_TEST_LOCK.lock().unwrap();
1795 DEPRECATION_WARNED.store(false, std::sync::atomic::Ordering::SeqCst);
1796
1797 let workers = WorkersConfig { command: Some("claude".into()), ..Default::default() };
1798 effective_spawn_params(None, None, &workers);
1799
1800 assert!(
1801 DEPRECATION_WARNED.load(std::sync::atomic::Ordering::SeqCst),
1802 "legacy [workers].command must trigger the deprecation warning"
1803 );
1804 }
1805
1806 #[test]
1807 fn legacy_model_forwarded_to_ctx() {
1808 let workers = WorkersConfig { model: Some("opus".into()), ..Default::default() };
1809 let params = effective_spawn_params(None, None, &workers);
1810 assert_eq!(params.model.as_deref(), Some("opus"));
1811 }
1812
1813 #[test]
1814 fn options_model_takes_precedence_over_legacy() {
1815 let mut workers = WorkersConfig { model: Some("opus".into()), agent: Some("claude".into()), ..Default::default() };
1816 workers.options.insert("model".into(), "sonnet".into());
1817 let params = effective_spawn_params(None, None, &workers);
1818 assert_eq!(params.model.as_deref(), Some("sonnet"));
1819 }
1820
1821 #[test]
1824 fn apm_opt_env_vars_set() {
1825 use std::os::unix::fs::PermissionsExt;
1826
1827 let wt = tempfile::tempdir().unwrap();
1828 let log_dir = tempfile::tempdir().unwrap();
1829 let mock_dir = tempfile::tempdir().unwrap();
1830 let env_output = wt.path().join("env-output.txt");
1831
1832 let mock_claude = mock_dir.path().join("claude");
1833 let script = format!("#!/bin/sh\nprintenv > \"{}\"\n", env_output.display());
1834 std::fs::write(&mock_claude, &script).unwrap();
1835 std::fs::set_permissions(&mock_claude, std::fs::Permissions::from_mode(0o755)).unwrap();
1836
1837 let sys_file = crate::wrapper::write_temp_file("sys", "system prompt").unwrap();
1838 let msg_file = crate::wrapper::write_temp_file("msg", "ticket content").unwrap();
1839
1840 let mut extra_env = HashMap::new();
1841 extra_env.insert(
1842 "PATH".to_string(),
1843 format!("{}:{}", mock_dir.path().display(), std::env::var("PATH").unwrap_or_default()),
1844 );
1845
1846 let mut options = HashMap::new();
1847 options.insert("model".to_string(), "sonnet".to_string());
1848
1849 let ctx = crate::wrapper::WrapperContext {
1850 worker_name: "test-worker".to_string(),
1851 agent_type: "test".to_string(),
1852 ticket_id: "abc123".to_string(),
1853 ticket_branch: "ticket/abc123".to_string(),
1854 worktree_path: wt.path().to_path_buf(),
1855 system_prompt_file: sys_file.clone(),
1856 user_message_file: msg_file.clone(),
1857 skip_permissions: false,
1858 profile: "default".to_string(),
1859 role_prefix: None,
1860 options,
1861 model: None,
1862 log_path: log_dir.path().join("worker.log"),
1863 container: None,
1864 extra_env,
1865 root: wt.path().to_path_buf(),
1866 keychain: HashMap::new(),
1867 current_state: "in_progress".to_string(),
1868 command: None,
1869 };
1870
1871 let wrapper = crate::wrapper::resolve_builtin("claude").unwrap();
1872 let mut child = wrapper.spawn(&ctx).unwrap();
1873 child.wait().unwrap();
1874 let _ = std::fs::remove_file(&sys_file);
1875 let _ = std::fs::remove_file(&msg_file);
1876
1877 let env_content = std::fs::read_to_string(&env_output)
1878 .expect("env-output.txt not written");
1879
1880 assert!(env_content.contains("APM_OPT_MODEL=sonnet"), "APM_OPT_MODEL=sonnet must be set\n{env_content}");
1881 }
1882
1883 fn make_frontmatter_with_agent(agent: Option<&str>, overrides: &[(&str, &str)]) -> crate::ticket_fmt::Frontmatter {
1886 let agent_line = agent.map(|a| format!("agent = \"{a}\"\n")).unwrap_or_default();
1887 let overrides_section = if overrides.is_empty() {
1888 String::new()
1889 } else {
1890 let pairs: Vec<String> = overrides.iter()
1891 .map(|(k, v)| format!("{k} = \"{v}\""))
1892 .collect();
1893 format!("[agent_overrides]\n{}\n", pairs.join("\n"))
1894 };
1895 let toml_str = format!("id = \"t\"\ntitle = \"T\"\nstate = \"new\"\n{agent_line}{overrides_section}");
1896 toml::from_str(&toml_str).unwrap()
1897 }
1898
1899 #[test]
1900 fn apply_fm_profile_override_wins() {
1901 let fm = make_frontmatter_with_agent(Some("mock-sad"), &[("impl_agent", "mock-happy")]);
1902 let mut agent = "claude".to_string();
1903 apply_frontmatter_agent(&mut agent, &fm, "impl_agent");
1904 assert_eq!(agent, "mock-happy");
1905 }
1906
1907 #[test]
1908 fn apply_fm_agent_field_wins_when_no_profile_match() {
1909 let fm = make_frontmatter_with_agent(Some("mock-sad"), &[]);
1910 let mut agent = "claude".to_string();
1911 apply_frontmatter_agent(&mut agent, &fm, "impl_agent");
1912 assert_eq!(agent, "mock-sad");
1913 }
1914
1915 #[test]
1916 fn apply_fm_profile_override_beats_agent_field() {
1917 let fm = make_frontmatter_with_agent(Some("mock-random"), &[("impl_agent", "claude")]);
1918 let mut agent = "other".to_string();
1919 apply_frontmatter_agent(&mut agent, &fm, "impl_agent");
1920 assert_eq!(agent, "claude");
1921 }
1922
1923 #[test]
1924 fn apply_fm_no_fields_unchanged() {
1925 let fm = make_frontmatter_with_agent(None, &[]);
1926 let mut agent = "claude".to_string();
1927 apply_frontmatter_agent(&mut agent, &fm, "impl_agent");
1928 assert_eq!(agent, "claude");
1929 }
1930
1931 fn find_apm_bin() -> Option<String> {
1934 if let Ok(v) = std::env::var("APM_BIN") {
1936 if !v.is_empty() && std::path::Path::new(&v).exists() {
1937 return Some(v);
1938 }
1939 }
1940 if let Ok(exe) = std::env::current_exe() {
1945 if let Some(target_dir) = exe.parent().and_then(|p| p.parent()) {
1946 let candidate = target_dir.join("apm");
1947 if candidate.is_file() {
1948 return Some(candidate.to_string_lossy().into_owned());
1949 }
1950 }
1951 }
1952 None
1953 }
1954
1955 fn make_mock_project(root: &std::path::Path, ticket_state: &str, ticket_id: &str) {
1956 use std::fs;
1957
1958 fs::create_dir_all(root.join(".apm/agents/claude")).unwrap();
1959 fs::create_dir_all(root.join("tickets")).unwrap();
1960
1961 fs::write(root.join(".apm/config.toml"), r#"
1962[project]
1963name = "test-project"
1964default_branch = "main"
1965
1966[workers]
1967agent = "mock-happy"
1968
1969[tickets]
1970dir = "tickets"
1971"#).unwrap();
1972
1973 fs::write(root.join(".apm/workflow.toml"), r#"
1974[[workflow.states]]
1975id = "in_design"
1976label = "In Design"
1977actionable = ["agent"]
1978instructions = ".apm/apm.spec-writer.md"
1979
1980 [[workflow.states.transitions]]
1981 to = "specd"
1982 trigger = "manual"
1983 outcome = "success"
1984
1985 [[workflow.states.transitions]]
1986 to = "closed"
1987 trigger = "manual"
1988 outcome = "cancelled"
1989
1990[[workflow.states]]
1991id = "specd"
1992label = "Specd"
1993actionable = ["supervisor"]
1994satisfies_deps = true
1995worker_end = true
1996
1997 [[workflow.states.transitions]]
1998 to = "in_progress"
1999 trigger = "manual"
2000 outcome = "success"
2001
2002 [[workflow.states.transitions]]
2003 to = "closed"
2004 trigger = "manual"
2005 outcome = "cancelled"
2006
2007[[workflow.states]]
2008id = "in_progress"
2009label = "In Progress"
2010instructions = ".apm/apm.worker.md"
2011
2012 [[workflow.states.transitions]]
2013 to = "implemented"
2014 trigger = "manual"
2015 outcome = "success"
2016
2017 [[workflow.states.transitions]]
2018 to = "closed"
2019 trigger = "manual"
2020 outcome = "cancelled"
2021
2022[[workflow.states]]
2023id = "implemented"
2024label = "Implemented"
2025actionable = ["supervisor"]
2026satisfies_deps = true
2027worker_end = true
2028terminal = false
2029
2030 [[workflow.states.transitions]]
2031 to = "closed"
2032 trigger = "manual"
2033 outcome = "cancelled"
2034
2035[[workflow.states]]
2036id = "closed"
2037label = "Closed"
2038terminal = true
2039"#).unwrap();
2040
2041 fs::write(root.join(".apm/apm.worker.md"), "Worker instructions.").unwrap();
2042 fs::write(root.join(".apm/apm.spec-writer.md"), "Spec writer instructions.").unwrap();
2043
2044 let ticket_content = format!(r#"+++
2045id = "{ticket_id}"
2046title = "Test Ticket"
2047state = "{ticket_state}"
2048priority = 0
2049effort = 5
2050risk = 3
2051author = "test"
2052owner = "test"
2053branch = "ticket/{ticket_id}-test"
2054created_at = "2026-01-01T00:00:00Z"
2055updated_at = "2026-01-01T00:00:00Z"
2056+++
2057
2058## Spec
2059
2060### Problem
2061
2062Original problem.
2063
2064### Acceptance criteria
2065
2066- [ ] Some criterion
2067
2068### Out of scope
2069
2070Nothing.
2071
2072### Approach
2073
2074Some approach.
2075
2076### Open questions
2077
2078### Amendment requests
2079
2080### Code review
2081
2082## History
2083
2084| When | From | To | By |
2085|------|------|----|----|
2086"#);
2087 fs::write(root.join(format!("tickets/{ticket_id}-test.md")), ticket_content).unwrap();
2088
2089 std::process::Command::new("git")
2090 .arg("init")
2091 .current_dir(root)
2092 .output()
2093 .unwrap();
2094 std::process::Command::new("git")
2095 .args(["config", "user.email", "test@test.com"])
2096 .current_dir(root)
2097 .output()
2098 .unwrap();
2099 std::process::Command::new("git")
2100 .args(["config", "user.name", "Test"])
2101 .current_dir(root)
2102 .output()
2103 .unwrap();
2104 std::process::Command::new("git")
2106 .args(["add", ".apm"])
2107 .current_dir(root)
2108 .output()
2109 .unwrap();
2110 std::process::Command::new("git")
2111 .args(["commit", "-m", "initial commit", "--allow-empty"])
2112 .current_dir(root)
2113 .output()
2114 .unwrap();
2115 let branch_name = format!("ticket/{ticket_id}-test");
2117 std::process::Command::new("git")
2118 .args(["checkout", "-b", &branch_name])
2119 .current_dir(root)
2120 .output()
2121 .unwrap();
2122 std::process::Command::new("git")
2123 .args(["add", &format!("tickets/{ticket_id}-test.md")])
2124 .current_dir(root)
2125 .output()
2126 .unwrap();
2127 std::process::Command::new("git")
2128 .args(["commit", "-m", &format!("ticket({ticket_id}): created")])
2129 .current_dir(root)
2130 .output()
2131 .unwrap();
2132 std::process::Command::new("git")
2134 .args(["checkout", "main"])
2135 .current_dir(root)
2136 .output()
2137 .unwrap();
2138 }
2139
2140 fn make_wrapper_ctx_for_mock(
2141 project_root: &std::path::Path,
2142 ticket_id: &str,
2143 ticket_state: &str,
2144 apm_bin: &str,
2145 log_path: std::path::PathBuf,
2146 ) -> crate::wrapper::WrapperContext {
2147 let sys_file = crate::wrapper::write_temp_file("sys", "system prompt").unwrap();
2148 let msg_file = crate::wrapper::write_temp_file("msg", "ticket content").unwrap();
2149 let mut options = HashMap::new();
2150 options.insert("apm_bin".to_string(), apm_bin.to_string());
2151 crate::wrapper::WrapperContext {
2152 worker_name: "test-worker".to_string(),
2153 agent_type: "test".to_string(),
2154 ticket_id: ticket_id.to_string(),
2155 ticket_branch: format!("ticket/{ticket_id}-test"),
2156 worktree_path: project_root.to_path_buf(),
2157 system_prompt_file: sys_file,
2158 user_message_file: msg_file,
2159 skip_permissions: false,
2160 profile: "default".to_string(),
2161 role_prefix: None,
2162 options,
2163 model: None,
2164 log_path,
2165 container: None,
2166 extra_env: HashMap::new(),
2167 root: project_root.to_path_buf(),
2168 keychain: HashMap::new(),
2169 current_state: ticket_state.to_string(),
2170 command: None,
2171 }
2172 }
2173
2174 #[test]
2175 fn mock_happy_spec_mode_transitions_to_specd() {
2176 let apm_bin = match find_apm_bin() { Some(b) => b, None => return };
2177 let dir = tempfile::tempdir().unwrap();
2178 let root = dir.path();
2179 make_mock_project(root, "in_design", "aaaa0001");
2180 let log_path = root.join("test-worker.log");
2181 let ctx = make_wrapper_ctx_for_mock(root, "aaaa0001", "in_design", &apm_bin, log_path.clone());
2182 let wrapper = crate::wrapper::resolve_builtin("mock-happy").unwrap();
2183 let mut child = wrapper.spawn(&ctx).unwrap();
2184 child.wait().unwrap();
2185
2186 let log_content = std::fs::read_to_string(&log_path).unwrap_or_default();
2187 let ticket_from_branch = {
2189 let out = std::process::Command::new("git")
2190 .args(["show", "ticket/aaaa0001-test:tickets/aaaa0001-test.md"])
2191 .current_dir(root)
2192 .output()
2193 .unwrap();
2194 String::from_utf8_lossy(&out.stdout).to_string()
2195 };
2196 assert!(ticket_from_branch.contains("state = \"specd\""),
2197 "ticket should be in specd state\nticket_from_branch: {ticket_from_branch}\nlog: {log_content}");
2198 assert!(ticket_from_branch.contains("### Problem"),
2199 "ticket should have Problem section\n{ticket_from_branch}");
2200 assert!(ticket_from_branch.contains("effort = 1"),
2201 "effort should be 1\n{ticket_from_branch}");
2202 assert!(ticket_from_branch.contains("risk = 1"),
2203 "risk should be 1\n{ticket_from_branch}");
2204 }
2205
2206 #[test]
2207 fn mock_happy_zero_success_transitions_returns_err() {
2208 use std::fs;
2209 let dir = tempfile::tempdir().unwrap();
2210 let root = dir.path();
2211
2212 fs::create_dir_all(root.join(".apm/agents/claude")).unwrap();
2213 fs::create_dir_all(root.join("tickets")).unwrap();
2214 fs::write(root.join(".apm/config.toml"), r#"
2215[project]
2216name = "test"
2217default_branch = "main"
2218[workers]
2219agent = "mock-happy"
2220[tickets]
2221dir = "tickets"
2222"#).unwrap();
2223 fs::write(root.join(".apm/workflow.toml"), r#"
2224[[workflow.states]]
2225id = "in_design"
2226label = "In Design"
2227actionable = ["agent"]
2228
2229 [[workflow.states.transitions]]
2230 to = "closed"
2231 trigger = "manual"
2232 outcome = "needs_input"
2233
2234[[workflow.states]]
2235id = "closed"
2236label = "Closed"
2237terminal = true
2238"#).unwrap();
2239 fs::write(root.join(".apm/apm.worker.md"), "instructions").unwrap();
2240 fs::write(root.join(".apm/apm.spec-writer.md"), "instructions").unwrap();
2241 let ticket_content = r#"+++
2242id = "aaaa0002"
2243title = "Test"
2244state = "in_design"
2245priority = 0
2246effort = 5
2247risk = 3
2248author = "test"
2249owner = "test"
2250branch = "ticket/aaaa0002-test"
2251created_at = "2026-01-01T00:00:00Z"
2252updated_at = "2026-01-01T00:00:00Z"
2253+++
2254
2255## Spec
2256
2257### Problem
2258
2259### Acceptance criteria
2260
2261### Out of scope
2262
2263### Approach
2264
2265## History
2266
2267| When | From | To | By |
2268|------|------|----|----|
2269"#;
2270 fs::write(root.join("tickets/aaaa0002-test.md"), ticket_content).unwrap();
2271 std::process::Command::new("git").args(["init"]).current_dir(root).output().unwrap();
2272 std::process::Command::new("git").args(["config", "user.email", "t@t.com"]).current_dir(root).output().unwrap();
2273 std::process::Command::new("git").args(["config", "user.name", "T"]).current_dir(root).output().unwrap();
2274 std::process::Command::new("git").args(["add", "."]).current_dir(root).output().unwrap();
2275 std::process::Command::new("git").args(["commit", "-m", "init"]).current_dir(root).output().unwrap();
2276
2277 let log_path = root.join("test.log");
2278 let sys_file = crate::wrapper::write_temp_file("sys", "sys").unwrap();
2279 let msg_file = crate::wrapper::write_temp_file("msg", "msg").unwrap();
2280 let ctx = crate::wrapper::WrapperContext {
2281 worker_name: "test".to_string(),
2282 agent_type: "test".to_string(),
2283 ticket_id: "aaaa0002".to_string(),
2284 ticket_branch: "ticket/aaaa0002-test".to_string(),
2285 worktree_path: root.to_path_buf(),
2286 system_prompt_file: sys_file,
2287 user_message_file: msg_file,
2288 skip_permissions: false,
2289 profile: "default".to_string(),
2290 role_prefix: None,
2291 options: HashMap::new(),
2292 model: None,
2293 log_path,
2294 container: None,
2295 extra_env: HashMap::new(),
2296 root: root.to_path_buf(),
2297 keychain: HashMap::new(),
2298 current_state: "in_design".to_string(),
2299 command: None,
2300 };
2301 let wrapper = crate::wrapper::resolve_builtin("mock-happy").unwrap();
2302 let result = wrapper.spawn(&ctx);
2303 assert!(result.is_err(), "mock-happy should return Err when no success transitions");
2304 let msg = result.unwrap_err().to_string();
2305 assert!(msg.contains("no success-outcome transition"), "error should mention no success transition: {msg}");
2306 }
2307
2308 #[test]
2309 fn mock_sad_transitions_to_non_success_state() {
2310 let apm_bin = match find_apm_bin() { Some(b) => b, None => return };
2311 let dir = tempfile::tempdir().unwrap();
2312 let root = dir.path();
2313 make_mock_project(root, "in_design", "aaaa0003");
2314 let log_path = root.join("test.log");
2315 let ctx = make_wrapper_ctx_for_mock(root, "aaaa0003", "in_design", &apm_bin, log_path.clone());
2316 let wrapper = crate::wrapper::resolve_builtin("mock-sad").unwrap();
2317 let mut child = wrapper.spawn(&ctx).unwrap();
2318 child.wait().unwrap();
2319
2320 let log_content = std::fs::read_to_string(&log_path).unwrap_or_default();
2321 let out = std::process::Command::new("git")
2322 .args(["show", "ticket/aaaa0003-test:tickets/aaaa0003-test.md"])
2323 .current_dir(root)
2324 .output()
2325 .unwrap();
2326 let ticket_from_branch = String::from_utf8_lossy(&out.stdout).to_string();
2327 assert!(!ticket_from_branch.contains("state = \"specd\""),
2328 "mock-sad should NOT transition to specd\n{ticket_from_branch}\nlog: {log_content}");
2329 assert!(ticket_from_branch.contains("state = \"closed\"") || ticket_from_branch.contains("state = \"in_design\""),
2331 "mock-sad should transition to a non-success state\n{ticket_from_branch}\nlog: {log_content}");
2332 }
2333
2334 #[test]
2335 fn mock_sad_seed_reproducibility() {
2336 let apm_bin = match find_apm_bin() { Some(b) => b, None => return };
2337
2338 let run_mock_sad = |ticket_id: &str, seed: &str| -> String {
2339 let dir = tempfile::tempdir().unwrap();
2340 let root = dir.path();
2341 make_mock_project(root, "in_design", ticket_id);
2342 let log_path = root.join("test.log");
2343 let mut options = HashMap::new();
2344 options.insert("apm_bin".to_string(), apm_bin.clone());
2345 options.insert("seed".to_string(), seed.to_string());
2346 let sys_file = crate::wrapper::write_temp_file("sys", "sys").unwrap();
2347 let msg_file = crate::wrapper::write_temp_file("msg", "msg").unwrap();
2348 let ctx = crate::wrapper::WrapperContext {
2349 worker_name: "test".to_string(),
2350 agent_type: "test".to_string(),
2351 ticket_id: ticket_id.to_string(),
2352 ticket_branch: format!("ticket/{ticket_id}-test"),
2353 worktree_path: root.to_path_buf(),
2354 system_prompt_file: sys_file,
2355 user_message_file: msg_file,
2356 skip_permissions: false,
2357 profile: "default".to_string(),
2358 role_prefix: None,
2359 options,
2360 model: None,
2361 log_path,
2362 container: None,
2363 extra_env: HashMap::new(),
2364 root: root.to_path_buf(),
2365 keychain: HashMap::new(),
2366 current_state: "in_design".to_string(),
2367 command: None,
2368 };
2369 let wrapper = crate::wrapper::resolve_builtin("mock-sad").unwrap();
2370 let mut child = wrapper.spawn(&ctx).unwrap();
2371 child.wait().unwrap();
2372
2373 let git_content = {
2375 let o = std::process::Command::new("git")
2376 .args(["show", &format!("ticket/{ticket_id}-test:tickets/{ticket_id}-test.md")])
2377 .current_dir(root)
2378 .output()
2379 .unwrap();
2380 String::from_utf8_lossy(&o.stdout).to_string()
2381 };
2382 for line in git_content.lines() {
2383 if line.starts_with("state = ") {
2384 return line.to_string();
2385 }
2386 }
2387 "unknown".to_string()
2388 };
2389
2390 let state1 = run_mock_sad("aaaa000a", "42");
2391 let state2 = run_mock_sad("aaaa000b", "42");
2392 assert_eq!(state1, state2, "mock-sad with same seed should pick same target state");
2393 }
2394
2395 #[test]
2396 fn debug_does_not_change_state() {
2397 let dir = tempfile::tempdir().unwrap();
2398 let root = dir.path();
2399 make_mock_project(root, "in_design", "aaaa0005");
2400 let log_path = root.join("test.log");
2401 let sys_file = crate::wrapper::write_temp_file("sys", "debug-system-prompt-unique-text").unwrap();
2402 let msg_file = crate::wrapper::write_temp_file("msg", "debug-message").unwrap();
2403 let ctx = crate::wrapper::WrapperContext {
2404 worker_name: "test-worker".to_string(),
2405 agent_type: "test".to_string(),
2406 ticket_id: "aaaa0005".to_string(),
2407 ticket_branch: "ticket/aaaa0005-test".to_string(),
2408 worktree_path: root.to_path_buf(),
2409 system_prompt_file: sys_file,
2410 user_message_file: msg_file,
2411 skip_permissions: false,
2412 profile: "default".to_string(),
2413 role_prefix: None,
2414 options: HashMap::new(),
2415 model: None,
2416 log_path: log_path.clone(),
2417 container: None,
2418 extra_env: HashMap::new(),
2419 root: root.to_path_buf(),
2420 keychain: HashMap::new(),
2421 current_state: "in_design".to_string(),
2422 command: None,
2423 };
2424 let wrapper = crate::wrapper::resolve_builtin("debug").unwrap();
2425 let mut child = wrapper.spawn(&ctx).unwrap();
2426 child.wait().unwrap();
2427
2428 let git_content = {
2431 let o = std::process::Command::new("git")
2432 .args(["show", "ticket/aaaa0005-test:tickets/aaaa0005-test.md"])
2433 .current_dir(root)
2434 .output()
2435 .unwrap();
2436 String::from_utf8_lossy(&o.stdout).to_string()
2437 };
2438 assert!(git_content.contains("state = \"in_design\""),
2439 "debug should not change ticket state\n{git_content}");
2440
2441 let log_content = std::fs::read_to_string(&log_path).unwrap_or_default();
2443 assert!(log_content.contains("APM_TICKET_ID"),
2444 "log should contain APM_TICKET_ID\n{log_content}");
2445 assert!(log_content.contains("debug-system-prompt-unique-text"),
2446 "log should contain system prompt text\n{log_content}");
2447 assert!(log_content.contains("\"type\":\"tool_use\""),
2448 "log should contain tool_use JSONL\n{log_content}");
2449 }
2450}