pub mod claude;
pub(crate) mod claude_events;
pub mod codebuff;
pub mod codex;
pub mod copilot;
pub mod cursor;
pub mod droid;
pub mod gemini;
pub(crate) mod gemini_support;
pub mod kilo;
pub mod opencode;
pub mod oz;
pub mod qwen;
pub(crate) mod custom;
pub(crate) mod registry;
pub mod classifier;
pub(crate) mod selection;
pub(crate) mod truncate;
use anyhow::Result;
use std::collections::HashMap;
use std::process::Command;
use crate::prompt_scan::scan_for_injection;
use crate::store;
use crate::types::*;
pub(crate) mod env;
#[allow(unused_imports)]
pub use env::{
agent_has_fs_access, apply_run_env, is_rust_project, set_git_ceiling, shared_target_dir,
target_dir_for_worktree,
};
pub trait Agent: Send + Sync {
fn kind(&self) -> AgentKind;
fn streaming(&self) -> bool;
fn build_command(&self, prompt: &str, opts: &RunOpts) -> Result<Command>;
fn parse_event(&self, task_id: &TaskId, line: &str) -> Option<TaskEvent>;
fn parse_completion(&self, output: &str) -> CompletionInfo;
fn needs_pty(&self) -> bool {
false
}
}
#[derive(Debug, Clone)]
pub struct RunOpts {
pub dir: Option<String>,
pub output: Option<String>,
pub result_file: Option<String>,
pub model: Option<String>,
pub budget: bool,
pub read_only: bool,
pub context_files: Vec<String>,
pub session_id: Option<String>,
pub env: Option<HashMap<String, String>>,
pub env_forward: Option<Vec<String>>,
}
pub fn detect_agents() -> Vec<AgentKind> {
#[cfg(test)]
{
let maybe = DETECT_AGENTS_OVERRIDE.with(|cell| cell.borrow().clone());
if let Some(list) = maybe {
return list;
}
}
let mut found = Vec::new();
for (name, kind) in [
("gemini", AgentKind::Gemini),
("qwen", AgentKind::Qwen),
("codex", AgentKind::Codex),
("opencode", AgentKind::OpenCode),
("copilot", AgentKind::Copilot),
("agent", AgentKind::Cursor),
("cursor-agent", AgentKind::Cursor),
("droid", AgentKind::Droid),
("kilo", AgentKind::Kilo),
("aid-codebuff", AgentKind::Codebuff),
("oz", AgentKind::Oz),
("claude", AgentKind::Claude),
] {
if env::which_exists(name) && !found.contains(&kind) {
found.push(kind);
}
}
found
}
#[cfg(test)]
std::thread_local! {
static DETECT_AGENTS_OVERRIDE: std::cell::RefCell<Option<Vec<AgentKind>>> =
const { std::cell::RefCell::new(None) };
}
#[cfg(test)]
pub(crate) struct DetectAgentsGuard {
previous: Option<Vec<AgentKind>>,
}
#[cfg(test)]
impl DetectAgentsGuard {
pub fn set(agents: Vec<AgentKind>) -> Self {
let previous = DETECT_AGENTS_OVERRIDE.with(|cell| cell.borrow().clone());
DETECT_AGENTS_OVERRIDE.with(|cell| *cell.borrow_mut() = Some(agents));
Self { previous }
}
}
#[cfg(test)]
impl Drop for DetectAgentsGuard {
fn drop(&mut self) {
DETECT_AGENTS_OVERRIDE.with(|cell| *cell.borrow_mut() = self.previous.take());
}
}
pub(crate) fn ensure_agent_binary_available(agent_kind: AgentKind, agent_name: &str) -> Result<()> {
ensure_agent_binary_available_with(agent_kind, agent_name, env::which_exists)
}
pub(crate) fn ensure_agent_binary_available_with<F>(
agent_kind: AgentKind,
agent_name: &str,
which: F,
) -> Result<()>
where
F: Fn(&str) -> bool,
{
if built_in_agent_binary_exists(agent_kind, which) {
return Ok(());
}
anyhow::bail!("Agent '{}' not found: binary missing from PATH", agent_name);
}
pub(crate) fn built_in_agent_binary_exists<F>(agent_kind: AgentKind, which: F) -> bool
where
F: Fn(&str) -> bool,
{
match agent_kind {
AgentKind::Codex => which("codex"),
AgentKind::Copilot => which("copilot"),
AgentKind::Cursor => which("agent") || which("cursor-agent"),
AgentKind::Gemini => which("gemini"),
AgentKind::Qwen => which("qwen"),
AgentKind::OpenCode => which("opencode"),
AgentKind::Kilo => which("kilo"),
AgentKind::Codebuff => which("aid-codebuff"),
AgentKind::Droid => which("droid"),
AgentKind::Oz => which("oz"),
AgentKind::Claude => which("claude"),
AgentKind::Custom => true,
}
}
pub(crate) fn select_agent_with_reason(
prompt: &str, opts: &RunOpts, store: &store::Store,
team: Option<&crate::team::TeamConfig>,
) -> (String, String) {
selection::select_agent_with_reason(prompt, opts, store, team)
}
pub fn get_agent(kind: AgentKind) -> Box<dyn Agent> {
match kind {
AgentKind::Codex => Box::new(codex::CodexAgent),
AgentKind::Copilot => Box::new(copilot::CopilotAgent),
AgentKind::Cursor => Box::new(cursor::CursorAgent),
AgentKind::Gemini => Box::new(gemini::GeminiAgent),
AgentKind::Qwen => Box::new(qwen::QwenAgent),
AgentKind::OpenCode => Box::new(opencode::OpenCodeAgent),
AgentKind::Kilo => Box::new(kilo::KiloAgent),
AgentKind::Codebuff => Box::new(codebuff::CodebuffAgent),
AgentKind::Droid => Box::new(droid::DroidAgent),
AgentKind::Oz => Box::new(oz::OzAgent),
AgentKind::Claude => Box::new(claude::ClaudeAgent),
AgentKind::Custom => panic!("Custom agents must be resolved via resolve_agent()"),
}
}
pub fn embed_context_in_prompt(prompt: &str, context_files: &[String]) -> anyhow::Result<String> {
if context_files.is_empty() {
return Ok(prompt.to_string());
}
let mut combined = prompt.to_string();
for file in context_files {
let contents = std::fs::read_to_string(file)?;
let scan = scan_for_injection(&contents);
for warning in &scan.warnings {
aid_warn!(
"[aid] ⚠ Context file {file}: {} (line {})",
warning.pattern,
warning.line_num
);
}
if scan.has_critical {
aid_warn!("[aid] ⚠ Critical injection pattern detected in {file} — content may be adversarial");
}
combined.push_str("\n\n[Context File: ");
combined.push_str(file);
combined.push_str("]\n");
combined.push_str(&contents);
}
Ok(combined)
}
#[cfg(test)]
mod cursor_binary_tests;
#[cfg(test)]
mod binary_preflight_tests {
use super::{built_in_agent_binary_exists, ensure_agent_binary_available_with};
use crate::types::AgentKind;
#[test]
fn built_in_agent_binary_exists_rejects_missing_kilo_binary() {
assert!(!built_in_agent_binary_exists(AgentKind::Kilo, |_| false));
}
#[test]
fn built_in_agent_binary_exists_accepts_cursor_alias_binary() {
assert!(built_in_agent_binary_exists(AgentKind::Cursor, |name| {
name == "cursor-agent"
}));
}
#[test]
fn ensure_agent_binary_available_reports_missing_path_binary() {
let err = ensure_agent_binary_available_with(AgentKind::Kilo, "kilo", |_| false)
.unwrap_err();
assert_eq!(
err.to_string(),
"Agent 'kilo' not found: binary missing from PATH"
);
}
}
#[cfg(test)]
mod tests;