Skip to main content

apm_core/
start.rs

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
25/// Delay inserted between `git fetch` and `git merge` in aggressive mode to let
26/// remote-propagation settle and reduce the fetch-race window.
27const 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    // Legacy command/args (kept for check_output_format_supported backward compat)
63    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    // Agent resolution: transition > profile > workers > default "claude"
71    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    // Emit deprecation warning when legacy fields present but agent absent
77    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    // Options merge: workers.options base, profile.options overrides on collision
88    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    // Model: options.model > legacy profile.model > legacy workers.model
96    let model = options.get("model").cloned()
97        .or_else(|| profile.and_then(|p| p.model.clone()))
98        .or_else(|| workers.model.clone());
99
100    // Env merge
101    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    // else: keep config-resolved agent unchanged
125}
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
149/// True when `agent` resolves to the built-in claude wrapper (no custom shadow).
150/// The compatibility probe is only meaningful in that case.
151pub(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    /// When set, denial scanning is run on drop (claude wrapper only).
188    /// Tuple: (log_path, worktree_path, ticket_id).
189    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
231/// Scan the worker transcript for permission denials, write the summary file,
232/// and emit a warning to the APM log when apm-command denials are found.
233fn 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(), &params.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, &params.agent) {
397        check_output_format_supported(&params.command)?;
398    }
399    let mut child = spawn_worker(&ctx, &params.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    // Filter out tickets whose epic already has the max number of active workers.
455    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(&current_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(), &params.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, &params.agent) {
611        check_output_format_supported(&params.command)?;
612    }
613    let mut child = spawn_worker(&ctx, &params.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(&current_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; // prompt used only for run_next, not spawn_next_worker
756
757    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(), &params.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, &params.agent) {
815        check_output_format_supported(&params.command)?;
816    }
817    let child = spawn_worker(&ctx, &params.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
840/// If the ticket has dependencies, prepend a dependency context bundle to the
841/// worker prompt content.  Tickets with no dependencies are unchanged.
842fn 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
853/// If the ticket belongs to an epic, prepend an epic context bundle to the
854/// worker prompt content.  Tickets without an epic are unchanged.
855fn 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    // Prepend agents.instructions prefix when configured and non-empty
893    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    // Level 0: .apm/agents/<agent>/apm.<role>.md (soft — no error if absent)
912    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    // Level 1: transition.instructions
919    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    // Level 2: profile.instructions
924    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    // Level 3: workers.instructions
933    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    // Level 4: built-in default
940    if let Some(s) = resolve_builtin_instructions(agent, role) {
941        return Ok(s.to_string());
942    }
943    // Level 5: hard error
944    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    // --- resolve_profile ---
1028
1029    #[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    // --- effective_spawn_params ---
1123
1124    #[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    // --- build_system_prompt ---
1205
1206    #[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    // --- agents_instructions prefix ---
1306
1307    #[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        // File ends with multiple newlines
1365        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        // Must have exactly one blank line between prefix and body
1373        let expected = format!("PREFIX\n\n{}", super::CLAUDE_WORKER_DEFAULT);
1374        assert_eq!(result, expected);
1375    }
1376
1377    // --- agent_role_prefix ---
1378
1379    #[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    // --- transition-level instruction overrides ---
1406
1407    #[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    // --- spawn worker cwd ---
1508
1509    #[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        // Write mock 'claude' script — reports pwd to a file
1518        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    // --- check_output_format_supported ---
1575
1576    #[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    // --- APM env vars on spawned process ---
1606
1607    #[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        // Mock 'claude' writes all env vars to a file then exits
1617        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        // APM_BIN must point to an existing file
1677        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    // --- temp file cleanup ---
1684
1685    #[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        // Mock 'claude' that just exits immediately
1694        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    // --- agent/options resolution ---
1748
1749    #[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        // Capture what would otherwise go to stderr — proves the message hits
1782        // the writer (i.e. stderr in production), not just an in-memory log.
1783        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    // --- APM_OPT_ env vars ---
1822
1823    #[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    // --- apply_frontmatter_agent ---
1884
1885    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    // --- mock wrapper integration tests ---
1932
1933    fn find_apm_bin() -> Option<String> {
1934        // 1. Explicit override wins
1935        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        // 2. Derive from the test binary path.
1941        //    current_exe() -> <workspace>/target/{profile}/deps/apm_core-<hash>
1942        //    two parents up -> <workspace>/target/{profile}/
1943        //    sibling "apm"  -> <workspace>/target/{profile}/apm
1944        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        // Create main branch with config files
2105        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        // Create the ticket branch and commit the ticket there
2116        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        // Switch back to main
2133        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        // Read ticket from the ticket branch (where apm commits changes)
2188        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        // Should have transitioned to some other state
2330        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            // Read state from ticket branch (where apm commits changes)
2374            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        // State should still be in_design (debug doesn't commit or transition)
2429        // Read from the ticket branch (HEAD of main won't have the ticket)
2430        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        // Log file should contain APM env vars and system prompt text
2442        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}