apm-cli 0.1.14

CLI project manager for running AI coding agents in parallel, isolated by design.
Documentation
use anyhow::Result;
use apm_core::{config::{Config, resolve_identity}, epic, ticket};
use std::path::Path;
use crate::ctx::CmdContext;

pub fn run(root: &Path, title: String, no_edit: bool, side_note: bool, context: Option<String>, context_section: Option<String>, no_aggressive: bool, sections: Vec<String>, sets: Vec<String>, epic: Option<String>, depends_on: Vec<String>) -> Result<()> {
    let config = CmdContext::load_config_only(root)?;

    if context_section.is_some() && context.is_none() {
        anyhow::bail!("--context-section requires --context");
    }

    if !sets.is_empty() && sections.is_empty() {
        anyhow::bail!("--set requires --section");
    }
    if sections.len() != sets.len() {
        anyhow::bail!(
            "--section and --set must be paired: {} --section flag(s) but {} --set flag(s)",
            sections.len(),
            sets.len()
        );
    }

    if !config.ticket.sections.is_empty() {
        for name in &sections {
            if !config.ticket.sections.iter().any(|s| s.name.eq_ignore_ascii_case(name)) {
                anyhow::bail!("unknown section {:?}; not defined in [ticket.sections]", name);
            }
        }
    }

    let aggressive = config.sync.aggressive && !no_aggressive;
    if side_note && !config.agents.side_tickets {
        anyhow::bail!("side tickets are disabled in apm.toml (agents.side_tickets = false)");
    }

    let author = resolve_identity(root);

    let (epic_id, target_branch, base_branch) = if let Some(ref id) = epic {
        match epic::find_epic_branch(root, id) {
            Some(branch) => (Some(id.clone()), Some(branch.clone()), Some(branch)),
            None => anyhow::bail!("No epic branch found for id '{id}'"),
        }
    } else {
        (None, None, None)
    };

    let depends_on_parsed: Option<Vec<String>> = if depends_on.is_empty() {
        None
    } else {
        Some(
            depends_on
                .iter()
                .flat_map(|s| s.split(','))
                .map(|s| s.trim().to_string())
                .filter(|s| !s.is_empty())
                .collect(),
        )
    };

    if let Some(ref dep_ids) = depends_on_parsed {
        if !dep_ids.is_empty() {
            let all_tickets = apm_core::ticket::load_all_from_git(root, &config.tickets.dir)?;
            let strategy = apm_core::validate::active_completion_strategy(&config);
            apm_core::validate::check_depends_on_rules(
                &strategy,
                epic_id.as_deref(),
                target_branch.as_deref(),
                dep_ids,
                &all_tickets,
                &config.project.default_branch,
            )?;
        }
    }

    let section_sets: Vec<(String, String)> = sections.into_iter().zip(sets).collect();
    let mut warnings = Vec::new();
    let t = ticket::create(root, &config, title, author, context, context_section, aggressive, section_sets, epic_id, target_branch, depends_on_parsed, base_branch, &mut warnings)?;
    for w in &warnings {
        eprintln!("{w}");
    }
    let id = &t.frontmatter.id;
    let branch = t.frontmatter.branch.as_deref().unwrap_or("");
    let filename = t.path.file_name().unwrap().to_string_lossy();
    let rel_path = format!("{}/{}", config.tickets.dir.to_string_lossy(), filename);

    println!("Created ticket {id}: {filename} (branch: {branch})");

    if !no_edit {
        open_editor(root, &config, branch, &rel_path)?;
    }

    Ok(())
}

fn open_editor(root: &Path, config: &Config, branch: &str, rel_path: &str) -> Result<()> {
    // Check out the ticket branch, open editor, commit result, return to previous branch.
    let prev_branch = std::process::Command::new("git")
        .args(["rev-parse", "--abbrev-ref", "HEAD"])
        .current_dir(root)
        .output()
        .ok()
        .and_then(|o| String::from_utf8(o.stdout).ok())
        .map(|s| s.trim().to_string())
        .unwrap_or_else(|| config.project.default_branch.clone());

    let _ = std::process::Command::new("git")
        .args(["checkout", branch])
        .current_dir(root)
        .status();

    let file_path = root.join(rel_path);
    // Commit whatever the user wrote, even if editor exited non-zero.
    let _ = crate::editor::open(&file_path);

    let _ = std::process::Command::new("git")
        .args(["-c", "commit.gpgsign=false", "add", rel_path])
        .current_dir(root)
        .status();
    let _ = std::process::Command::new("git")
        .args(["-c", "commit.gpgsign=false", "commit", "--allow-empty", "-m", "write spec"])
        .current_dir(root)
        .status();

    let _ = std::process::Command::new("git")
        .args(["checkout", &prev_branch])
        .current_dir(root)
        .status();

    Ok(())
}