// JSON tool call filtering for display (moved from g3-core) pub mod filter_json; pub mod streaming_markdown; use anyhow::Result; use crossterm::style::{Color, ResetColor, SetForegroundColor}; use std::time::{Duration, Instant}; #[derive(Debug, Clone)] struct TurnMetrics { turn_number: usize, tokens_used: u32, wall_clock_time: Duration, } /// Generate a histogram showing tokens used and wall clock time per turn fn generate_turn_histogram(turn_metrics: &[TurnMetrics]) -> String { if turn_metrics.is_empty() { return " No turn data available".to_string(); } let mut histogram = String::new(); // Find max values for scaling let max_tokens = turn_metrics .iter() .map(|t| t.tokens_used) .max() .unwrap_or(1); let max_time_ms = turn_metrics .iter() .map(|t| t.wall_clock_time.as_millis().min(u32::MAX as u128) as u32) .max() .unwrap_or(1); // Constants for histogram display const MAX_BAR_WIDTH: usize = 40; const TOKEN_CHAR: char = 'โ–ˆ'; const TIME_CHAR: char = 'โ–“'; histogram.push_str("\n๐Ÿ“Š Per-Turn Performance Histogram:\n"); histogram.push_str(&format!( " {} = Tokens Used (max: {})\n", TOKEN_CHAR, max_tokens )); histogram.push_str(&format!( " {} = Wall Clock Time (max: {:.1}s)\n\n", TIME_CHAR, max_time_ms as f64 / 1000.0 )); for metrics in turn_metrics { let turn_time_ms = metrics.wall_clock_time.as_millis().min(u32::MAX as u128) as u32; // Calculate bar lengths (proportional to max values) let token_bar_len = if max_tokens > 0 { ((metrics.tokens_used as f64 / max_tokens as f64) * MAX_BAR_WIDTH as f64) as usize } else { 0 }; let time_bar_len = if max_time_ms > 0 { ((turn_time_ms as f64 / max_time_ms as f64) * MAX_BAR_WIDTH as f64) as usize } else { 0 }; // Format time duration let time_str = if turn_time_ms < 1000 { format!("{}ms", turn_time_ms) } else if turn_time_ms < 60_000 { format!("{:.1}s", turn_time_ms as f64 / 1000.0) } else { let minutes = turn_time_ms / 60_000; let seconds = (turn_time_ms % 60_000) as f64 / 1000.0; format!("{}m{:.1}s", minutes, seconds) }; // Create the bars let token_bar = TOKEN_CHAR.to_string().repeat(token_bar_len); let time_bar = TIME_CHAR.to_string().repeat(time_bar_len); // Add turn information histogram.push_str(&format!( " Turn {:2}: {:>6} tokens โ”‚{:<40}โ”‚\n", metrics.turn_number, metrics.tokens_used, token_bar )); histogram.push_str(&format!( " {:>6} โ”‚{:<40}โ”‚\n", time_str, time_bar )); // Add separator line between turns (except for last turn) if metrics.turn_number != turn_metrics.last().unwrap().turn_number { histogram .push_str(" โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”ผโ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”ค\n"); } } // Add summary statistics let total_tokens: u32 = turn_metrics.iter().map(|t| t.tokens_used).sum(); let total_time: Duration = turn_metrics.iter().map(|t| t.wall_clock_time).sum(); let avg_tokens = total_tokens as f64 / turn_metrics.len() as f64; let avg_time_ms = total_time.as_millis() as f64 / turn_metrics.len() as f64; histogram.push_str("\n๐Ÿ“ˆ Summary Statistics:\n"); histogram.push_str(&format!( " โ€ข Total Tokens: {} across {} turns\n", total_tokens, turn_metrics.len() )); histogram.push_str(&format!(" โ€ข Average Tokens/Turn: {:.1}\n", avg_tokens)); histogram.push_str(&format!( " โ€ข Total Time: {:.1}s\n", total_time.as_secs_f64() )); histogram.push_str(&format!( " โ€ข Average Time/Turn: {:.1}s\n", avg_time_ms / 1000.0 )); histogram } /// Format a Duration as human-readable elapsed time (e.g., "1h 23m 45s", "5m 30s", "45s") fn format_elapsed_time(duration: Duration) -> String { let total_secs = duration.as_secs(); let hours = total_secs / 3600; let minutes = (total_secs % 3600) / 60; let seconds = total_secs % 60; if hours > 0 { format!("{}h {}m {}s", hours, minutes, seconds) } else if minutes > 0 { format!("{}m {}s", minutes, seconds) } else if seconds > 0 { format!("{}s", seconds) } else { // For very short durations, show milliseconds format!("{}ms", duration.as_millis()) } } /// Extract coach feedback by reading from the coach agent's specific log file /// Uses the coach agent's session ID to find the exact log file fn extract_coach_feedback_from_logs( coach_result: &g3_core::TaskResult, coach_agent: &g3_core::Agent, output: &SimpleOutput, ) -> Result { // Get the coach agent's session ID let session_id = coach_agent .get_session_id() .ok_or_else(|| anyhow::anyhow!("Coach agent has no session ID"))?; // Try new .g3/sessions//session.json path first let log_file_path = g3_core::get_session_file(&session_id); // Fall back to old logs/ path if new path doesn't exist let log_file_path = if log_file_path.exists() { log_file_path } else { let logs_dir = std::path::Path::new("logs"); logs_dir.join(format!("g3_session_{}.json", session_id)) }; // Try to extract from session log if let Some(feedback) = try_extract_feedback_from_log(&log_file_path, output) { output.print(&format!( "โœ… Extracted coach feedback from session: {}", session_id )); return Ok(feedback); } // Fallback: use the TaskResult's extract_summary method let fallback = coach_result.extract_summary(); if !fallback.is_empty() { output.print(&format!( "โœ… Extracted coach feedback from response: {} chars", fallback.len() )); return Ok(fallback); } // Last resort: return an error instead of panicking Err(anyhow::anyhow!( "Could not extract coach feedback from session: {}\n\ Log file path: {:?}\n\ Log file exists: {}\n\ Coach result response length: {} chars", session_id, log_file_path, log_file_path.exists(), coach_result.response.len() )) } /// Helper function to extract feedback from a session log file /// Looks for the last assistant message with substantial text content fn try_extract_feedback_from_log( log_file_path: &std::path::Path, _output: &SimpleOutput, ) -> Option { if !log_file_path.exists() { return None; } let log_content = std::fs::read_to_string(log_file_path).ok()?; let log_json: serde_json::Value = serde_json::from_str(&log_content).ok()?; let messages = log_json .get("context_window")? .get("conversation_history")? .as_array()?; // Search backwards for the last assistant message with text content for msg in messages.iter().rev() { let role = msg.get("role").and_then(|v| v.as_str())?; if role.eq_ignore_ascii_case("assistant") { if let Some(content) = msg.get("content") { // Handle string content if let Some(content_str) = content.as_str() { let trimmed = content_str.trim(); // Skip empty or very short responses (likely just tool calls) if !trimmed.is_empty() && trimmed.len() > 10 { return Some(trimmed.to_string()); } } // Handle array content (native tool calling format) if let Some(content_array) = content.as_array() { for block in content_array { if block.get("type").and_then(|v| v.as_str()) == Some("text") { if let Some(text) = block.get("text").and_then(|v| v.as_str()) { let trimmed = text.trim(); if !trimmed.is_empty() && trimmed.len() > 10 { return Some(trimmed.to_string()); } } } } } } } } None } use clap::Parser; use g3_config::Config; use g3_core::{project::Project, ui_writer::UiWriter, Agent, DiscoveryOptions}; use rustyline::error::ReadlineError; use rustyline::DefaultEditor; use sha2::{Digest, Sha256}; use std::path::Path; use std::path::PathBuf; use std::process::exit; use tokio_util::sync::CancellationToken; use tracing::{debug, error}; use g3_core::error_handling::{classify_error, ErrorType, RecoverableError}; mod simple_output; mod ui_writer_impl; use simple_output::SimpleOutput; mod machine_ui_writer; use machine_ui_writer::MachineUiWriter; use ui_writer_impl::ConsoleUiWriter; #[derive(Parser, Clone)] #[command(name = "g3")] #[command(about = "A modular, composable AI coding agent")] #[command(version)] pub struct Cli { /// Enable verbose logging #[arg(short, long)] pub verbose: bool, /// Enable manual control of context compaction (disables auto-compact at 90%) #[arg(long = "manual-compact")] pub manual_compact: bool, /// Show the system prompt being sent to the LLM #[arg(long)] pub show_prompt: bool, /// Show the generated code before execution #[arg(long)] pub show_code: bool, /// Configuration file path #[arg(short, long)] pub config: Option, /// Workspace directory (defaults to current directory) #[arg(short, long)] pub workspace: Option, /// Task to execute (if provided, runs in single-shot mode instead of interactive) pub task: Option, /// Enable autonomous mode with coach-player feedback loop #[arg(long)] pub autonomous: bool, /// Maximum number of turns in autonomous mode (default: 5) #[arg(long, default_value = "5")] pub max_turns: usize, /// Override requirements text for autonomous mode (instead of reading from requirements.md) #[arg(long, value_name = "TEXT")] pub requirements: Option, /// Enable accumulative autonomous mode (default is chat mode) #[arg(long)] pub auto: bool, /// Enable interactive chat mode (no autonomous runs) #[arg(long)] pub chat: bool, /// Enable machine-friendly output mode with JSON markers and stats #[arg(long)] pub machine: bool, /// Override the configured provider (anthropic, databricks, embedded, openai) #[arg(long, value_name = "PROVIDER")] pub provider: Option, /// Override the model for the selected provider #[arg(long, value_name = "MODEL")] pub model: Option, /// Disable log file creation (no logs/ directory or session logs) #[arg(long)] pub quiet: bool, /// Enable WebDriver browser automation tools #[arg(long)] pub webdriver: bool, /// Use Chrome in headless mode for WebDriver (instead of Safari) #[arg(long)] pub chrome_headless: bool, /// Use Safari for WebDriver (this is the default) #[arg(long)] pub safari: bool, /// Enable flock mode - parallel multi-agent development #[arg(long, requires = "flock_workspace", requires = "segments")] pub project: Option, /// Flock workspace directory (where segment copies will be created) #[arg(long, requires = "project")] pub flock_workspace: Option, /// Number of segments to partition work into (for flock mode) #[arg(long, requires = "project")] pub segments: Option, /// Maximum turns per segment in flock mode (default: 5) #[arg(long, default_value = "5")] pub flock_max_turns: usize, /// Enable planning mode for requirements-driven development #[arg(long, conflicts_with_all = ["autonomous", "auto", "chat"])] pub planning: bool, /// Path to the codebase to work on (for planning mode) #[arg(long, value_name = "PATH")] pub codepath: Option, /// Disable git operations in planning mode #[arg(long)] pub no_git: bool, /// Enable fast codebase discovery before first LLM turn #[arg(long, value_name = "PATH")] pub codebase_fast_start: Option, /// Run as a specialized agent (loads prompt from agents/.md) #[arg(long, value_name = "NAME", conflicts_with_all = ["autonomous", "auto", "chat", "planning"])] pub agent: Option, /// Skip session resumption and force a new session (for agent mode) #[arg(long)] pub new_session: bool, } pub async fn run() -> Result<()> { let cli = Cli::parse(); // Check if flock mode is enabled if let (Some(project_dir), Some(flock_workspace), Some(num_segments)) = (&cli.project, &cli.flock_workspace, cli.segments) { // Run flock mode return run_flock_mode( project_dir.clone(), flock_workspace.clone(), num_segments, cli.flock_max_turns, ) .await; } if cli.codebase_fast_start.is_some() { print!("codebase_fast_start is temporarily disabled."); exit(1); } // Otherwise, continue with normal mode // Check if planning mode is enabled if cli.planning { // Expand ~ in codepath if provided // The expand_codepath function in g3_planner handles tilde expansion let codepath = cli.codepath.clone(); return g3_planner::run_planning_mode( codepath, cli.workspace.clone(), cli.no_git, cli.config.as_deref(), ) .await; } // Check if agent mode is enabled if let Some(agent_name) = &cli.agent { return run_agent_mode( agent_name, cli.workspace.clone(), cli.config.as_deref(), cli.quiet, cli.new_session, cli.task.clone(), cli.chrome_headless, cli.safari, ) .await; } // Check if agent mode is enabled if let Some(agent_name) = &cli.agent { return run_agent_mode( agent_name, cli.workspace.clone(), cli.config.as_deref(), cli.quiet, cli.new_session, cli.task.clone(), cli.chrome_headless, cli.safari, ) .await; } // Only initialize logging if not in retro mode if !cli.machine { // Initialize logging with filtering use tracing_subscriber::{layer::SubscriberExt, util::SubscriberInitExt, EnvFilter}; // Create a filter that suppresses llama_cpp logs unless in verbose mode let filter = if cli.verbose { EnvFilter::from_default_env() .add_directive(format!("{}=debug", env!("CARGO_PKG_NAME")).parse().unwrap()) .add_directive("g3_core=debug".parse().unwrap()) .add_directive("g3_cli=debug".parse().unwrap()) .add_directive("g3_execution=debug".parse().unwrap()) .add_directive("g3_providers=debug".parse().unwrap()) } else { EnvFilter::from_default_env() .add_directive(format!("{}=info", env!("CARGO_PKG_NAME")).parse().unwrap()) .add_directive("g3_core=info".parse().unwrap()) .add_directive("g3_cli=info".parse().unwrap()) .add_directive("g3_execution=info".parse().unwrap()) .add_directive("g3_providers=info".parse().unwrap()) .add_directive("llama_cpp=off".parse().unwrap()) // Suppress all llama_cpp logs .add_directive("llama=off".parse().unwrap()) // Suppress all llama.cpp logs }; tracing_subscriber::registry() .with(tracing_subscriber::fmt::layer()) .with(filter) .init(); } else { // In retro mode, we don't want any logging output to interfere with the TUI // We'll use a no-op subscriber use tracing_subscriber::{layer::SubscriberExt, util::SubscriberInitExt, EnvFilter}; // Create a filter that suppresses ALL logs in retro mode let filter = EnvFilter::from_default_env().add_directive("off".parse().unwrap()); // Turn off all logging tracing_subscriber::registry().with(filter).init(); } // Set up workspace directory let workspace_dir = if let Some(ws) = &cli.workspace { ws.clone() } else if cli.autonomous { // For autonomous mode, use G3_WORKSPACE env var or default setup_workspace_directory(cli.machine)? } else { // Default to current directory for interactive/single-shot mode std::env::current_dir()? }; // Check if we're in a project directory and read README and AGENTS.md if available // Load AGENTS.md first (if present) to provide agent-specific instructions let agents_content = read_agents_config(&workspace_dir); // Then load README for project context let readme_content = read_project_readme(&workspace_dir); // Create project model let project = if cli.autonomous { if let Some(requirements_text) = &cli.requirements { // Use requirements text override Project::new_autonomous_with_requirements( workspace_dir.clone(), requirements_text.clone(), )? } else { // Use traditional requirements.md file Project::new_autonomous(workspace_dir.clone())? } } else { Project::new(workspace_dir.clone()) }; // Ensure workspace exists and enter it project.ensure_workspace_exists()?; project.enter_workspace()?; // Load configuration with CLI overrides let mut config = Config::load_with_overrides( cli.config.as_deref(), cli.provider.clone(), cli.model.clone(), )?; // Apply webdriver flag override if cli.webdriver { config.webdriver.enabled = true; } // Apply chrome-headless flag override if cli.chrome_headless { config.webdriver.enabled = true; config.webdriver.browser = g3_config::WebDriverBrowser::ChromeHeadless; // Run Chrome diagnostics on first use let report = g3_computer_control::run_chrome_diagnostics( config.webdriver.chrome_binary.as_deref(), ); // Display the diagnostic report println!("{}", report.format_report()); // If there are errors, the user can ask g3 to help fix them // We continue anyway to let the user decide } // Apply safari flag override if cli.safari { config.webdriver.enabled = true; config.webdriver.browser = g3_config::WebDriverBrowser::Safari; } // Apply no-auto-compact flag override if cli.manual_compact { config.agent.auto_compact = false; } // Validate provider if specified if let Some(ref provider) = cli.provider { let valid_providers = ["anthropic", "databricks", "embedded", "openai"]; if !valid_providers.contains(&provider.as_str()) { return Err(anyhow::anyhow!( "Invalid provider '{}'. Valid options: {:?}", provider, valid_providers )); } } // Initialize agent // ui_writer will be created conditionally based on machine mode // Combine AGENTS.md and README content if both exist let combined_content = match (agents_content.clone(), readme_content.clone()) { (Some(agents), Some(readme)) => Some(format!("{}\n\n{}", agents, readme)), (Some(agents), None) => Some(agents), (None, Some(readme)) => Some(readme), (None, None) => None, }; // Execute task, autonomous mode, or start interactive mode based on machine mode if cli.machine { // Machine mode - use MachineUiWriter let ui_writer = MachineUiWriter::new(); let agent = if cli.autonomous { Agent::new_autonomous_with_readme_and_quiet( config.clone(), ui_writer, combined_content.clone(), cli.quiet, ) .await? } else { Agent::new_with_readme_and_quiet( config.clone(), ui_writer, combined_content.clone(), cli.quiet, ) .await? }; run_with_machine_mode(agent, cli, project).await?; } else { // Normal mode - use ConsoleUiWriter // DEFAULT: Chat mode for interactive sessions // It runs when: // 1. No task is provided (not single-shot) // 2. Not in autonomous mode // 3. Not explicitly enabled with --auto flag let use_accumulative = cli.task.is_none() && !cli.autonomous && cli.auto; if use_accumulative { // Run accumulative mode and return early run_accumulative_mode(workspace_dir.clone(), cli.clone(), combined_content.clone()) .await?; return Ok(()); } let ui_writer = ConsoleUiWriter::new(); let agent = if cli.autonomous { Agent::new_autonomous_with_readme_and_quiet( config.clone(), ui_writer, combined_content.clone(), cli.quiet, ) .await? } else { Agent::new_with_readme_and_quiet( config.clone(), ui_writer, combined_content.clone(), cli.quiet, ) .await? }; run_with_console_mode(agent, cli, project, combined_content).await?; } Ok(()) } /// Run agent mode - loads a specialized agent prompt and executes a single task async fn run_agent_mode( agent_name: &str, workspace: Option, config_path: Option<&str>, _quiet: bool, new_session: bool, task: Option, chrome_headless: bool, safari: bool, ) -> Result<()> { use g3_core::get_agent_system_prompt; use g3_core::find_incomplete_agent_session; // Initialize logging use tracing_subscriber::{layer::SubscriberExt, util::SubscriberInitExt, EnvFilter}; let filter = EnvFilter::from_default_env() .add_directive("g3_core=info".parse().unwrap()) .add_directive("g3_cli=info".parse().unwrap()) .add_directive("llama_cpp=off".parse().unwrap()) .add_directive("llama=off".parse().unwrap()); tracing_subscriber::registry() .with(tracing_subscriber::fmt::layer()) .with(filter) .init(); let output = SimpleOutput::new(); // Determine workspace directory (current dir if not specified) let workspace_dir = workspace.unwrap_or_else(|| std::env::current_dir().unwrap_or_default()); // Change to the workspace directory first so session scanning works correctly std::env::set_current_dir(&workspace_dir)?; // Check for incomplete agent sessions before starting a new one (unless --new-session is set) let resuming_session = if new_session { output.print("\n๐Ÿ†• Starting new session (--new-session flag set)"); output.print(""); None } else { find_incomplete_agent_session(agent_name).ok().flatten() }; if let Some(ref incomplete_session) = resuming_session { output.print(&format!( "\n๐Ÿ”„ Found incomplete session for agent '{}'", agent_name )); output.print(&format!( " Session: {}", incomplete_session.session_id )); output.print(&format!( " Created: {}", incomplete_session.created_at )); if let Some(ref todo) = incomplete_session.todo_snapshot { // Show first few lines of TODO let preview: String = todo.lines().take(5).collect::>().join("\n"); output.print(&format!(" TODO preview:\n{}", preview)); } output.print(""); output.print(" Resuming incomplete session..."); output.print(""); } // Load agent prompt from agents/.md let agent_prompt_path = workspace_dir.join("agents").join(format!("{}.md", agent_name)); // Also check in the g3 installation directory let agent_prompt = if agent_prompt_path.exists() { std::fs::read_to_string(&agent_prompt_path) .map_err(|e| anyhow::anyhow!("Failed to read agent prompt from {:?}: {}", agent_prompt_path, e))? } else { // Try to find agents/ relative to the executable or in common locations let exe_dir = std::env::current_exe() .ok() .and_then(|p| p.parent().map(|p| p.to_path_buf())); let possible_paths = [ exe_dir.as_ref().map(|d| d.join("agents").join(format!("{}.md", agent_name))), Some(PathBuf::from(format!("agents/{}.md", agent_name))), ]; let mut found_prompt = None; for path_opt in possible_paths.iter().flatten() { if path_opt.exists() { found_prompt = Some(std::fs::read_to_string(path_opt) .map_err(|e| anyhow::anyhow!("Failed to read agent prompt from {:?}: {}", path_opt, e))?); break; } } found_prompt.ok_or_else(|| anyhow::anyhow!( "Agent prompt not found: agents/{}.md\nSearched in: {:?} and current directory", agent_name, agent_prompt_path ))? }; output.print(&format!("๐Ÿค– Running as agent: {}", agent_name)); output.print(&format!("๐Ÿ“ Working directory: {:?}", workspace_dir)); // Load config let mut config = g3_config::Config::load(config_path)?; // Apply chrome-headless flag override if chrome_headless { config.webdriver.enabled = true; config.webdriver.browser = g3_config::WebDriverBrowser::ChromeHeadless; } // Apply safari flag override if safari { config.webdriver.enabled = true; config.webdriver.browser = g3_config::WebDriverBrowser::Safari; } // Generate the combined system prompt (agent prompt + tool instructions) // Note: allow_multiple_tool_calls parameter is deprecated but kept for API compatibility let system_prompt = get_agent_system_prompt(&agent_prompt, true); // Read README if present let readme_content = std::fs::read_to_string(workspace_dir.join("README.md")).ok(); let readme_for_prompt = readme_content.map(|content| { format!("๐Ÿ“š Project README (from README.md):\n\n{}", content) }); // Create agent with custom system prompt let ui_writer = ConsoleUiWriter::new(); // Set agent mode on UI writer for visual differentiation (royal blue tool names) ui_writer.set_agent_mode(true); let mut agent = Agent::new_with_custom_prompt( config, ui_writer, system_prompt, readme_for_prompt, ).await?; // Set agent mode for session tracking agent.set_agent_mode(agent_name); // If resuming a session, restore context and TODO let initial_task = if let Some(ref incomplete_session) = resuming_session { // Restore the session context match agent.restore_from_continuation(incomplete_session) { Ok(full_restore) => { if full_restore { output.print(" โœ… Full context restored from previous session"); } else { output.print(" โš ๏ธ Restored from summary (context was > 80%)"); } } Err(e) => { output.print(&format!(" โš ๏ธ Could not restore context: {}", e)); } } // Copy TODO from old session to new session directory let todo_content = if let Some(ref content) = incomplete_session.todo_snapshot { Some(content.clone()) } else { // Fallback: read from the actual todo.g3.md file in the old session directory let old_session_dir = std::path::Path::new(".g3/sessions").join(&incomplete_session.session_id); let old_todo_path = old_session_dir.join("todo.g3.md"); if old_todo_path.exists() { std::fs::read_to_string(&old_todo_path).ok() } else { None } }; if let Some(ref content) = todo_content { if let Some(session_id) = agent.get_session_id() { let new_todo_path = g3_core::paths::get_session_todo_path(session_id); let _ = g3_core::paths::ensure_session_dir(session_id); if let Err(e) = std::fs::write(&new_todo_path, content) { output.print(&format!(" โš ๏ธ Could not restore TODO: {}", e)); } else { output.print(" โœ… TODO list restored"); } } } output.print(""); // Resume message instead of fresh start "Continue working on the incomplete tasks. Use todo_read to see the current TODO list and resume from where you left off." } else { // Fresh start - the agent prompt should contain instructions to start working immediately "Begin your analysis and work on the current project. Follow your mission and workflow as specified in your instructions." }; // Use provided task if available, otherwise use the default initial_task let final_task = task.as_deref().unwrap_or(initial_task); let _result = agent.execute_task(final_task, None, true).await?; // Save session continuation for resume capability agent.save_session_continuation(None); // Don't print completion message for scout agent - it needs the last line // to be the report file path for the research tool to read if agent_name != "scout" { output.print("\nโœ… Agent mode completed"); } Ok(()) } /// Run flock mode - parallel multi-agent development async fn run_flock_mode( project_dir: PathBuf, flock_workspace: PathBuf, num_segments: usize, max_turns: usize, ) -> Result<()> { let output = SimpleOutput::new(); output.print(""); output.print("๐Ÿฆ… G3 FLOCK MODE - Parallel Multi-Agent Development"); output.print(""); output.print(&format!("๐Ÿ“ Project: {}", project_dir.display())); output.print(&format!("๐Ÿ—‚๏ธ Workspace: {}", flock_workspace.display())); output.print(&format!("๐Ÿ”ข Segments: {}", num_segments)); output.print(&format!("๐Ÿ”„ Max Turns per Segment: {}", max_turns)); output.print(""); // Create flock configuration let config = g3_ensembles::FlockConfig::new(project_dir, flock_workspace, num_segments)? .with_max_turns(max_turns); // Create and run flock mode let mut flock = g3_ensembles::FlockMode::new(config)?; match flock.run().await { Ok(_) => output.print("\nโœ… Flock mode completed successfully"), Err(e) => output.print(&format!("\nโŒ Flock mode failed: {}", e)), } Ok(()) } /// Accumulative autonomous mode: accumulates requirements from user input /// and runs autonomous mode after each input async fn run_accumulative_mode( workspace_dir: PathBuf, cli: Cli, combined_content: Option, ) -> Result<()> { let output = SimpleOutput::new(); output.print(""); output.print("g3 programming agent - autonomous mode"); output.print(" >> describe what you want, I'll build it iteratively"); output.print(""); print!( "{}workspace: {}{}\n", SetForegroundColor(Color::DarkGrey), workspace_dir.display(), ResetColor ); output.print(""); output.print("๐Ÿ’ก Each input you provide will be added to requirements"); output.print(" and I'll automatically work on implementing them. You can"); output.print(" interrupt at any time (Ctrl+C) to add clarifications or more requirements."); output.print(""); output.print(" Type '/help' for commands, 'exit' or 'quit' to stop, Ctrl+D to finish"); output.print(""); // Initialize rustyline editor with history let mut rl = DefaultEditor::new()?; let history_file = dirs::home_dir().map(|mut path| { path.push(".g3_accumulative_history"); path }); if let Some(ref history_path) = history_file { let _ = rl.load_history(history_path); } // Accumulated requirements stored in memory let mut accumulated_requirements = Vec::new(); let mut turn_number = 0; loop { output.print(&format!("\n{}", "=".repeat(60))); if accumulated_requirements.is_empty() { output.print("๐Ÿ“ What would you like me to build? (describe your requirements)"); } else { output.print(&format!( "๐Ÿ“ Turn {} - What's next? (add more requirements or refinements)", turn_number + 1 )); } output.print(&format!("{}", "=".repeat(60))); let readline = rl.readline("requirement> "); match readline { Ok(line) => { let input = line.trim().to_string(); if input.is_empty() { continue; } if input == "exit" || input == "quit" { output.print("\n๐Ÿ‘‹ Goodbye!"); break; } // Check for slash commands if input.starts_with('/') { match input.as_str() { "/help" => { output.print(""); output.print("๐Ÿ“– Available Commands:"); output.print(" /requirements - Show all accumulated requirements"); output.print(" /chat - Switch to interactive chat mode"); output.print(" /help - Show this help message"); output.print(" exit/quit - Exit the session"); output.print(""); continue; } "/requirements" => { output.print(""); if accumulated_requirements.is_empty() { output.print("๐Ÿ“‹ No requirements accumulated yet"); } else { output.print("๐Ÿ“‹ Accumulated Requirements:"); output.print(""); for req in &accumulated_requirements { output.print(&format!(" {}", req)); } } output.print(""); continue; } "/chat" => { output.print(""); output.print("๐Ÿ”„ Switching to interactive chat mode..."); output.print(""); // Build context message with accumulated requirements let requirements_context = if accumulated_requirements.is_empty() { None } else { Some(format!( "๐Ÿ“‹ Context from Accumulative Mode:\n\n\ We were working on these requirements. There may be unstaged or in-progress changes or recent changes to this branch. This is for your information.\n\n\ Requirements:\n{}\n", accumulated_requirements.join("\n") )) }; // Combine with existing content (README/AGENTS.md) let chat_combined_content = match (requirements_context, combined_content.clone()) { (Some(req_ctx), Some(existing)) => { Some(format!("{}\n\n{}", req_ctx, existing)) } (Some(req_ctx), None) => Some(req_ctx), (None, existing) => existing, }; // Load configuration let mut config = Config::load_with_overrides( cli.config.as_deref(), cli.provider.clone(), cli.model.clone(), )?; // Apply webdriver flag override if cli.webdriver { config.webdriver.enabled = true; } // Apply chrome-headless flag override if cli.chrome_headless { config.webdriver.enabled = true; config.webdriver.browser = g3_config::WebDriverBrowser::ChromeHeadless; } // Apply safari flag override if cli.safari { config.webdriver.enabled = true; config.webdriver.browser = g3_config::WebDriverBrowser::Safari; } // Apply no-auto-compact flag override if cli.manual_compact { config.agent.auto_compact = false; } // Create agent for interactive mode with requirements context let ui_writer = ConsoleUiWriter::new(); let agent = Agent::new_with_readme_and_quiet( config, ui_writer, chat_combined_content.clone(), cli.quiet, ) .await?; // Run interactive mode run_interactive( agent, cli.show_prompt, cli.show_code, chat_combined_content, &workspace_dir, ) .await?; // After returning from interactive mode, exit output.print("\n๐Ÿ‘‹ Goodbye!"); break; } _ => { output.print(&format!( "โŒ Unknown command: {}. Type /help for available commands.", input )); continue; } } } // Add to history rl.add_history_entry(&input)?; // Add this requirement to accumulated list turn_number += 1; accumulated_requirements.push(format!("{}. {}", turn_number, input)); // Build the complete requirements document let requirements_doc = format!( "# Project Requirements\n\n\ ## Current Instructions and Requirements:\n\n\ {}\n\n\ ## Latest Requirement (Turn {}):\n\n\ {}", accumulated_requirements.join("\n"), turn_number, input ); output.print(""); output.print(&format!( "๐Ÿ“‹ Current instructions and requirements (Turn {}):", turn_number )); output.print(&format!(" {}", input)); output.print(""); output.print("๐Ÿš€ Starting autonomous implementation..."); output.print(""); // Create a project with the accumulated requirements let project = Project::new_autonomous_with_requirements( workspace_dir.clone(), requirements_doc.clone(), )?; // Ensure workspace exists and enter it project.ensure_workspace_exists()?; project.enter_workspace()?; // Load configuration with CLI overrides let mut config = Config::load_with_overrides( cli.config.as_deref(), cli.provider.clone(), cli.model.clone(), )?; // Apply webdriver flag override if cli.webdriver { config.webdriver.enabled = true; } // Apply chrome-headless flag override if cli.chrome_headless { config.webdriver.enabled = true; config.webdriver.browser = g3_config::WebDriverBrowser::ChromeHeadless; } // Apply safari flag override if cli.safari { config.webdriver.enabled = true; config.webdriver.browser = g3_config::WebDriverBrowser::Safari; } // Apply no-auto-compact flag override if cli.manual_compact { config.agent.auto_compact = false; } // Create agent for this autonomous run let ui_writer = ConsoleUiWriter::new(); let agent = Agent::new_autonomous_with_readme_and_quiet( config.clone(), ui_writer, combined_content.clone(), cli.quiet, ) .await?; // Run autonomous mode with the accumulated requirements let autonomous_result = tokio::select! { result = run_autonomous( agent, project, cli.show_prompt, cli.show_code, cli.max_turns, cli.quiet, cli.codebase_fast_start.clone(), ) => result, _ = tokio::signal::ctrl_c() => { output.print("\nโš ๏ธ Autonomous run cancelled by user (Ctrl+C)"); Ok(()) } }; match autonomous_result { Ok(_) => { output.print(""); output.print("โœ… Autonomous run completed"); } Err(e) => { output.print(""); output.print(&format!("โŒ Autonomous run failed: {}", e)); output.print(" You can provide more requirements to continue."); } } } Err(ReadlineError::Interrupted) => { output.print("\n๐Ÿ‘‹ Interrupted. Goodbye!"); break; } Err(ReadlineError::Eof) => { output.print("\n๐Ÿ‘‹ Goodbye!"); break; } Err(err) => { error!("Error: {:?}", err); break; } } } // Save history before exiting if let Some(ref history_path) = history_file { let _ = rl.save_history(history_path); } Ok(()) } // Simplified machine mode version of autonomous mode async fn run_autonomous_machine( mut agent: Agent, project: Project, show_prompt: bool, show_code: bool, max_turns: usize, _quiet: bool, _codebase_fast_start: Option, ) -> Result<()> { println!("AUTONOMOUS_MODE_STARTED"); println!("WORKSPACE: {}", project.workspace().display()); println!("MAX_TURNS: {}", max_turns); // Check if requirements exist if !project.has_requirements() { println!("ERROR: requirements.md not found in workspace directory"); return Ok(()); } // Read requirements let requirements = match project.read_requirements()? { Some(content) => content, None => { println!("ERROR: Could not read requirements"); return Ok(()); } }; println!("REQUIREMENTS_LOADED"); // For now, just execute a simple autonomous loop // This is a simplified version - full implementation would need coach-player loop let task = format!( "You are G3 in implementation mode. Read and implement the following requirements:\n\n{}\n\nImplement this step by step, creating all necessary files and code.", requirements ); println!("TASK_START"); let result = agent .execute_task_with_timing(&task, None, false, show_prompt, show_code, true, None) .await?; println!("AGENT_RESPONSE:"); println!("{}", result.response); println!("END_AGENT_RESPONSE"); println!("TASK_END"); // Save session continuation for resume capability agent.save_session_continuation(Some(result.response.clone())); println!("AUTONOMOUS_MODE_ENDED"); Ok(()) } async fn run_with_console_mode( mut agent: Agent, cli: Cli, project: Project, combined_content: Option, ) -> Result<()> { // Execute task, autonomous mode, or start interactive mode if cli.autonomous { // Autonomous mode with coach-player feedback loop run_autonomous( agent, project, cli.show_prompt, cli.show_code, cli.max_turns, cli.quiet, cli.codebase_fast_start.clone(), ) .await?; } else if let Some(task) = cli.task { // Single-shot mode let output = SimpleOutput::new(); let result = agent .execute_task_with_timing( &task, None, false, cli.show_prompt, cli.show_code, true, None, ) .await?; output.print_smart(&result.response); // Save session continuation for resume capability agent.save_session_continuation(Some(result.response.clone())); } else { // Interactive mode (default) run_interactive( agent, cli.show_prompt, cli.show_code, combined_content, project.workspace(), ) .await?; } Ok(()) } async fn run_with_machine_mode( mut agent: Agent, cli: Cli, project: Project, ) -> Result<()> { if cli.autonomous { // Autonomous mode with coach-player feedback loop run_autonomous_machine( agent, project, cli.show_prompt, cli.show_code, cli.max_turns, cli.quiet, cli.codebase_fast_start.clone(), ) .await?; } else if let Some(task) = cli.task { // Single-shot mode let result = agent .execute_task_with_timing( &task, None, false, cli.show_prompt, cli.show_code, true, None, ) .await?; println!("AGENT_RESPONSE:"); println!("{}", result.response); println!("END_AGENT_RESPONSE"); // Save session continuation for resume capability agent.save_session_continuation(Some(result.response.clone())); } else { // Interactive mode run_interactive_machine(agent, cli.show_prompt, cli.show_code).await?; } Ok(()) } /// Check if we're in a project directory and read AGENTS.md if available fn read_agents_config(workspace_dir: &Path) -> Option { // Look for AGENTS.md in the current directory let agents_path = workspace_dir.join("AGENTS.md"); if agents_path.exists() { match std::fs::read_to_string(&agents_path) { Ok(content) => { // Return the content with a note about which file was read Some(format!( "๐Ÿค– Agent Configuration (from AGENTS.md):\n\n{}", content )) } Err(e) => { // Log the error but continue without the agents config error!("Failed to read AGENTS.md: {}", e); None } } } else { // Check for alternative names let alt_path = workspace_dir.join("agents.md"); if alt_path.exists() { match std::fs::read_to_string(&alt_path) { Ok(content) => Some(format!( "๐Ÿค– Agent Configuration (from agents.md):\n\n{}", content )), Err(e) => { error!("Failed to read agents.md: {}", e); None } } } else { None } } } /// Check if we're in a project directory and read README if available fn read_project_readme(workspace_dir: &Path) -> Option { // Check if we're in a project directory (contains .g3 or .git) let is_project_dir = workspace_dir.join(".g3").exists() || workspace_dir.join(".git").exists(); if !is_project_dir { return None; } // Look for README files in common formats let readme_names = [ "README.md", "README.MD", "readme.md", "Readme.md", "README", "README.txt", "README.rst", ]; for readme_name in &readme_names { let readme_path = workspace_dir.join(readme_name); if readme_path.exists() { match std::fs::read_to_string(&readme_path) { Ok(content) => { // Return the content with a note about which file was read return Some(format!( "๐Ÿ“š Project README (from {}):\n\n{}", readme_name, content )); } Err(e) => { // Log the error but continue looking for other README files error!("Failed to read {}: {}", readme_path.display(), e); } } } } None } /// Extract the main heading or title from README content fn extract_readme_heading(readme_content: &str) -> Option { // Find the README section in the combined content // The README section starts with "๐Ÿ“š Project README (from" let readme_start = readme_content.find("๐Ÿ“š Project README (from"); // If we can't find the README marker, the content might be just README // or might not contain README at all let content_to_search = match readme_start { Some(pos) => &readme_content[pos..], None => readme_content, }; // Process the content line by line, skipping the prefix line let mut content_lines = Vec::new(); for line in content_to_search.lines() { // Skip the "๐Ÿ“š Project README (from ...):" line if line.starts_with("๐Ÿ“š Project README") { continue; } content_lines.push(line); } let content = content_lines.join("\n"); // Look for the first markdown heading for line in content.lines() { let trimmed = line.trim(); // Check for H1 heading (# Title) if let Some(stripped) = trimmed.strip_prefix("# ") { let title = stripped.trim(); if !title.is_empty() { // Return the full title (including any description after dash) return Some(title.to_string()); } } // Skip other markdown headings for now (##, ###, etc.) // We're only looking for the main H1 heading } // If no H1 heading found, look for the first non-empty, non-metadata line as a fallback for line in content.lines().take(5) { let trimmed = line.trim(); // Skip empty lines, other heading markers, and metadata if !trimmed.is_empty() && !trimmed.starts_with("๐Ÿ“š") && !trimmed.starts_with('#') && !trimmed.starts_with("==") && !trimmed.starts_with("--") { // Limit length for display return Some(if trimmed.len() > 100 { format!("{}...", &trimmed[..97]) } else { trimmed.to_string() }); } } None } async fn run_interactive( mut agent: Agent, show_prompt: bool, show_code: bool, combined_content: Option, workspace_path: &Path, ) -> Result<()> { let output = SimpleOutput::new(); // Check for session continuation if let Ok(Some(continuation)) = g3_core::load_continuation() { output.print(""); output.print("๐Ÿ”„ Previous session detected!"); output.print(&format!( " Session: {}", &continuation.session_id[..continuation.session_id.len().min(20)] )); output.print(&format!( " Context: {:.1}% used", continuation.context_percentage )); if let Some(ref summary) = continuation.summary { let preview: String = summary.chars().take(80).collect(); output.print(&format!(" Last output: {}...", preview)); } output.print(""); output.print("Resume this session? [Y/n] "); // Read user input let mut input = String::new(); std::io::stdin().read_line(&mut input)?; let input = input.trim().to_lowercase(); if input.is_empty() || input == "y" || input == "yes" { // Resume the session match agent.restore_from_continuation(&continuation) { Ok(true) => { output.print("โœ… Full context restored from previous session"); } Ok(false) => { output.print("โœ… Session resumed with summary (context was > 80%)"); } Err(e) => { output.print(&format!("โš ๏ธ Could not restore session: {}", e)); output.print("Starting fresh session instead."); // Clear the invalid continuation let _ = g3_core::clear_continuation(); } } } else { // User declined, clear the continuation output.print("๐Ÿงน Starting fresh session..."); let _ = g3_core::clear_continuation(); } output.print(""); } output.print(""); output.print("g3 programming agent"); output.print(" >> what shall we build today?"); output.print(""); // Display provider and model information match agent.get_provider_info() { Ok((provider, model)) => { print!( "๐Ÿ”ง {}{}{} | {}{}{}\n", SetForegroundColor(Color::Cyan), provider, ResetColor, SetForegroundColor(Color::Yellow), model, ResetColor ); } Err(e) => { error!("Failed to get provider info: {}", e); } } // Display message if AGENTS.md or README was loaded if let Some(ref content) = combined_content { // Check what was loaded let has_agents = content.contains("Agent Configuration"); let has_readme = content.contains("Project README"); if has_agents { print!( "{}๐Ÿค– AGENTS.md configuration loaded{}\n", SetForegroundColor(Color::DarkGrey), ResetColor ); } if has_readme { // Extract the first heading or title from the README let readme_snippet = extract_readme_heading(content) .unwrap_or_else(|| "Project documentation loaded".to_string()); print!( "{}๐Ÿ“š detected: {}{}\n", SetForegroundColor(Color::DarkGrey), readme_snippet, ResetColor ); } } // Display workspace path print!( "{}workspace: {}{}\n", SetForegroundColor(Color::DarkGrey), workspace_path.display(), ResetColor ); output.print(""); // Initialize rustyline editor with history let mut rl = DefaultEditor::new()?; // Try to load history from a file in the user's home directory let history_file = dirs::home_dir().map(|mut path| { path.push(".g3_history"); path }); if let Some(ref history_path) = history_file { let _ = rl.load_history(history_path); } // Track multiline input let mut multiline_buffer = String::new(); let mut in_multiline = false; loop { // Display context window progress bar before each prompt display_context_progress(&agent, &output); // Adjust prompt based on whether we're in multi-line mode let prompt = if in_multiline { "... > " } else { "g3> " }; let readline = rl.readline(prompt); match readline { Ok(line) => { let trimmed = line.trim_end(); // Check if line ends with backslash for continuation if let Some(without_backslash) = trimmed.strip_suffix('\\') { // Remove the backslash and add to buffer multiline_buffer.push_str(without_backslash); multiline_buffer.push('\n'); in_multiline = true; continue; } // If we're in multiline mode and no backslash, this is the final line if in_multiline { multiline_buffer.push_str(&line); in_multiline = false; // Process the complete multiline input let input = multiline_buffer.trim().to_string(); multiline_buffer.clear(); if input.is_empty() { continue; } // Add complete multiline to history rl.add_history_entry(&input)?; if input == "exit" || input == "quit" { break; } // Process the multiline input execute_task(&mut agent, &input, show_prompt, show_code, &output).await; } else { // Single line input let input = line.trim().to_string(); if input.is_empty() { continue; } if input == "exit" || input == "quit" { break; } // Add to history rl.add_history_entry(&input)?; // Check for control commands if input.starts_with('/') { match input.as_str() { "/help" => { output.print(""); output.print("๐Ÿ“– Control Commands:"); output.print(" /compact - Trigger compaction (compacts conversation history)"); output.print(" /thinnify - Trigger context thinning (replaces large tool results with file references)"); output.print(" /skinnify - Trigger full context thinning (like /thinnify but for entire context, not just first third)"); output.print(" /clear - Clear session and start fresh (discards continuation artifacts)"); output.print(" /resume - List and switch to a previous session"); output.print( " /readme - Reload README.md and AGENTS.md from disk", ); output.print(" /stats - Show detailed context and performance statistics"); output.print(" /help - Show this help message"); output.print(" exit/quit - Exit the interactive session"); output.print(""); continue; } "/compact" => { output.print("๐Ÿ—œ๏ธ Triggering manual compaction..."); match agent.force_compact().await { Ok(true) => { output.print("โœ… Compaction completed successfully"); } Ok(false) => { output.print("โš ๏ธ Compaction failed"); } Err(e) => { output.print(&format!( "โŒ Error during compaction: {}", e )); } } continue; } "/thinnify" => { let summary = agent.force_thin(); println!("{}", summary); continue; } "/skinnify" => { let summary = agent.force_thin_all(); println!("{}", summary); continue; } "/clear" => { output.print("๐Ÿงน Clearing session..."); agent.clear_session(); output.print("โœ… Session cleared. Starting fresh."); continue; } "/readme" => { output.print("๐Ÿ“š Reloading README.md and AGENTS.md..."); match agent.reload_readme() { Ok(true) => { output.print("โœ… README content reloaded successfully") } Ok(false) => output .print("โš ๏ธ No README was loaded at startup, cannot reload"), Err(e) => { output.print(&format!("โŒ Error reloading README: {}", e)) } } continue; } "/stats" => { let stats = agent.get_stats(); output.print(&stats); continue; } "/resume" => { output.print("๐Ÿ“‹ Scanning for available sessions..."); match g3_core::list_sessions_for_directory() { Ok(sessions) => { if sessions.is_empty() { output.print("No sessions found for this directory."); continue; } // Get current session ID to mark it let current_session_id = agent.get_session_id().map(|s| s.to_string()); output.print(""); output.print("Available sessions:"); for (i, session) in sessions.iter().enumerate() { let time_str = g3_core::format_session_time(&session.created_at); let context_str = format!("{:.0}%", session.context_percentage); let current_marker = if current_session_id.as_deref() == Some(&session.session_id) { " (current)" } else { "" }; let todo_marker = if session.has_incomplete_todos() { " ๐Ÿ“" } else { "" }; // Truncate session ID for display let display_id = if session.session_id.len() > 40 { format!("{}...", &session.session_id[..40]) } else { session.session_id.clone() }; output.print(&format!( " {}. [{}] {} ({}){}{}", i + 1, time_str, display_id, context_str, todo_marker, current_marker )); } output.print(""); output.print("Enter session number to resume (or press Enter to cancel):"); // Read user selection if let Ok(selection) = rl.readline("> ") { let selection = selection.trim(); if selection.is_empty() { output.print("Resume cancelled."); } else if let Ok(num) = selection.parse::() { if num >= 1 && num <= sessions.len() { let selected = &sessions[num - 1]; output.print(&format!("๐Ÿ”„ Switching to session: {}", selected.session_id)); match agent.switch_to_session(selected) { Ok(true) => output.print("โœ… Full context restored from session."), Ok(false) => output.print("โœ… Session restored from summary."), Err(e) => output.print(&format!("โŒ Error restoring session: {}", e)), } } else { output.print("Invalid selection."); } } else { output.print("Invalid input. Please enter a number."); } } } Err(e) => output.print(&format!("โŒ Error listing sessions: {}", e)), } continue; } _ => { output.print(&format!( "โŒ Unknown command: {}. Type /help for available commands.", input )); continue; } } } // Process the single line input execute_task(&mut agent, &input, show_prompt, show_code, &output).await; } } Err(ReadlineError::Interrupted) => { // Ctrl-C pressed if in_multiline { // Cancel multiline input output.print("Multi-line input cancelled"); multiline_buffer.clear(); in_multiline = false; } else { output.print("CTRL-C"); } continue; } Err(ReadlineError::Eof) => { output.print("CTRL-D"); break; } Err(err) => { error!("Error: {:?}", err); break; } } } // Save history before exiting if let Some(ref history_path) = history_file { let _ = rl.save_history(history_path); } // Save session continuation for resume capability agent.save_session_continuation(None); output.print("๐Ÿ‘‹ Goodbye!"); Ok(()) } async fn execute_task( agent: &mut Agent, input: &str, show_prompt: bool, show_code: bool, output: &SimpleOutput, ) { const MAX_TIMEOUT_RETRIES: u32 = 3; let mut attempt = 0; // Show thinking indicator immediately output.print("๐Ÿค” Thinking..."); // Note: flush is handled internally by println // Create cancellation token for this request let cancellation_token = CancellationToken::new(); let cancel_token_clone = cancellation_token.clone(); loop { attempt += 1; // Execute task with cancellation support let execution_result = tokio::select! { result = agent.execute_task_with_timing_cancellable( input, None, false, show_prompt, show_code, true, cancellation_token.clone(), None ) => { result } _ = tokio::signal::ctrl_c() => { cancel_token_clone.cancel(); output.print("\nโš ๏ธ Operation cancelled by user (Ctrl+C)"); return; } }; match execution_result { Ok(result) => { if attempt > 1 { output.print(&format!("โœ… Request succeeded after {} attempts", attempt)); } output.print_smart(&result.response); return; } Err(e) => { if e.to_string().contains("cancelled") { output.print("โš ๏ธ Operation cancelled by user"); return; } // Check if this is a timeout error that we should retry let error_type = classify_error(&e); if matches!( error_type, ErrorType::Recoverable(RecoverableError::Timeout) ) && attempt < MAX_TIMEOUT_RETRIES { // Calculate retry delay with exponential backoff let delay_ms = 1000 * (2_u64.pow(attempt - 1)); let delay = std::time::Duration::from_millis(delay_ms); output.print(&format!( "โฑ๏ธ Timeout error detected (attempt {}/{}). Retrying in {:?}...", attempt, MAX_TIMEOUT_RETRIES, delay )); // Wait before retrying tokio::time::sleep(delay).await; continue; } // For non-timeout errors or after max retries, handle as before handle_execution_error(&e, input, output, attempt); return; } } } } async fn run_interactive_machine( mut agent: Agent, show_prompt: bool, show_code: bool, ) -> Result<()> { println!("INTERACTIVE_MODE_STARTED"); // Display provider and model information match agent.get_provider_info() { Ok((provider, model)) => { println!("PROVIDER: {}", provider); println!("MODEL: {}", model); } Err(e) => { println!("ERROR: Failed to get provider info: {}", e); } } // Initialize rustyline editor with history let mut rl = DefaultEditor::new()?; // Try to load history from a file in the user's home directory let history_file = dirs::home_dir().map(|mut path| { path.push(".g3_history"); path }); if let Some(ref history_path) = history_file { let _ = rl.load_history(history_path); } loop { let readline = rl.readline(""); match readline { Ok(line) => { let input = line.trim().to_string(); if input.is_empty() { continue; } if input == "exit" || input == "quit" { break; } // Add to history rl.add_history_entry(&input)?; // Check for control commands if input.starts_with('/') { match input.as_str() { "/compact" => { println!("COMMAND: compact"); match agent.force_compact().await { Ok(true) => println!("RESULT: Compaction completed"), Ok(false) => println!("RESULT: Compaction failed"), Err(e) => println!("ERROR: {}", e), } continue; } "/thinnify" => { println!("COMMAND: thinnify"); let summary = agent.force_thin(); println!("{}", summary); continue; } "/skinnify" => { println!("COMMAND: skinnify"); let summary = agent.force_thin_all(); println!("{}", summary); continue; } "/clear" => { println!("COMMAND: clear"); agent.clear_session(); println!("RESULT: Session cleared"); continue; } "/readme" => { println!("COMMAND: readme"); match agent.reload_readme() { Ok(true) => { println!("RESULT: README content reloaded successfully") } Ok(false) => println!( "RESULT: No README was loaded at startup, cannot reload" ), Err(e) => println!("ERROR: {}", e), } continue; } "/stats" => { println!("COMMAND: stats"); let stats = agent.get_stats(); // Emit stats as structured data (name: value pairs) println!("{}", stats); continue; } "/help" => { println!("COMMAND: help"); println!("AVAILABLE_COMMANDS: /compact /thinnify /skinnify /clear /resume /readme /stats /help"); continue; } "/resume" => { println!("COMMAND: resume"); match g3_core::list_sessions_for_directory() { Ok(sessions) => { if sessions.is_empty() { println!("RESULT: No sessions found"); continue; } println!("SESSIONS_START"); for (i, session) in sessions.iter().enumerate() { let time_str = g3_core::format_session_time(&session.created_at); let has_todos = if session.has_incomplete_todos() { "true" } else { "false" }; println!( "SESSION: {} | {} | {} | {:.0}% | {}", i + 1, session.session_id, time_str, session.context_percentage, has_todos ); } println!("SESSIONS_END"); println!("HINT: Use /resume to switch to a session"); } Err(e) => println!("ERROR: {}", e), } continue; } _ => { // Check for /resume pattern if input.starts_with("/resume ") { let num_str = input.strip_prefix("/resume ").unwrap().trim(); if let Ok(num) = num_str.parse::() { println!("COMMAND: resume {}", num); match g3_core::list_sessions_for_directory() { Ok(sessions) => { if num >= 1 && num <= sessions.len() { let selected = &sessions[num - 1]; match agent.switch_to_session(selected) { Ok(true) => println!("RESULT: Full context restored from session {}", selected.session_id), Ok(false) => println!("RESULT: Session {} restored from summary", selected.session_id), Err(e) => println!("ERROR: {}", e), } } else { println!("ERROR: Invalid session number"); } } Err(e) => println!("ERROR: {}", e), } continue; } } println!("ERROR: Unknown command: {}", input); continue; } } } // Execute task println!("TASK_START"); execute_task_machine(&mut agent, &input, show_prompt, show_code).await; println!("TASK_END"); } Err(ReadlineError::Interrupted) => continue, Err(ReadlineError::Eof) => break, Err(err) => { println!("ERROR: {:?}", err); break; } } } // Save history before exiting if let Some(ref history_path) = history_file { let _ = rl.save_history(history_path); } // Save session continuation for resume capability agent.save_session_continuation(None); println!("INTERACTIVE_MODE_ENDED"); Ok(()) } async fn execute_task_machine( agent: &mut Agent, input: &str, show_prompt: bool, show_code: bool, ) { const MAX_TIMEOUT_RETRIES: u32 = 3; let mut attempt = 0; // Create cancellation token for this request let cancellation_token = CancellationToken::new(); let cancel_token_clone = cancellation_token.clone(); loop { attempt += 1; // Execute task with cancellation support let execution_result = tokio::select! { result = agent.execute_task_with_timing_cancellable( input, None, false, show_prompt, show_code, true, cancellation_token.clone(), None ) => { result } _ = tokio::signal::ctrl_c() => { cancel_token_clone.cancel(); println!("CANCELLED"); return; } }; match execution_result { Ok(result) => { if attempt > 1 { println!("RETRY_SUCCESS: attempt {}", attempt); } println!("AGENT_RESPONSE:"); println!("{}", result.response); println!("END_AGENT_RESPONSE"); return; } Err(e) => { if e.to_string().contains("cancelled") { println!("CANCELLED"); return; } // Check if this is a timeout error that we should retry let error_type = classify_error(&e); if matches!( error_type, ErrorType::Recoverable(RecoverableError::Timeout) ) && attempt < MAX_TIMEOUT_RETRIES { // Calculate retry delay with exponential backoff let delay_ms = 1000 * (2_u64.pow(attempt - 1)); let delay = std::time::Duration::from_millis(delay_ms); println!( "TIMEOUT: attempt {} of {}, retrying in {:?}", attempt, MAX_TIMEOUT_RETRIES, delay ); // Wait before retrying tokio::time::sleep(delay).await; continue; } // For non-timeout errors or after max retries println!("ERROR: {}", e); if attempt > 1 { println!("FAILED_AFTER_RETRIES: {}", attempt); } return; } } } } fn handle_execution_error(e: &anyhow::Error, input: &str, output: &SimpleOutput, attempt: u32) { // Enhanced error logging with detailed information error!("=== TASK EXECUTION ERROR ==="); error!("Error: {}", e); if attempt > 1 { error!("Failed after {} attempts", attempt); } // Log error chain let mut source = e.source(); let mut depth = 1; while let Some(err) = source { error!(" Caused by [{}]: {}", depth, err); source = err.source(); depth += 1; } // Log additional context error!("Task input: {}", input); error!("Error type: {}", std::any::type_name_of_val(&e)); // Display user-friendly error message output.print(&format!("โŒ Error: {}", e)); // If it's a stream error, provide helpful guidance if e.to_string().contains("No response received") || e.to_string().contains("timed out") { output.print("๐Ÿ’ก This may be a temporary issue. Please try again or check the logs for more details."); output.print(" Log files are saved in the '.g3/sessions/' directory."); } } fn display_context_progress(agent: &Agent, _output: &SimpleOutput) { let context = agent.get_context_window(); let percentage = context.percentage_used(); // Create 10 dots representing context fullness let total_dots: usize = 10; let filled_dots = ((percentage / 100.0) * total_dots as f32).round() as usize; let empty_dots = total_dots.saturating_sub(filled_dots); let filled_str = "โ—".repeat(filled_dots); let empty_str = "โ—‹".repeat(empty_dots); // Determine color based on percentage let color = if percentage < 40.0 { Color::Green } else if percentage < 60.0 { Color::Yellow } else if percentage < 80.0 { Color::Rgb { r: 255, g: 165, b: 0, } // Orange } else { Color::Red }; // Format tokens as compact strings (e.g., "38.5k" instead of "38531") let format_tokens = |tokens: u32| -> String { if tokens >= 1_000_000 { format!("{:.1}m", tokens as f64 / 1_000_000.0) } else if tokens >= 1_000 { let k = tokens as f64 / 1000.0; if k >= 100.0 { format!("{:.0}k", k) } else { format!("{:.1}k", k) } } else { format!("{}", tokens) } }; // Print with colored dots (using print! directly to handle color codes) print!( "{}{}{}{} {}/{} โ—‰ | {:.0}%\n", SetForegroundColor(color), filled_str, empty_str, ResetColor, format_tokens(context.used_tokens), format_tokens(context.total_tokens), percentage ); } /// Set up the workspace directory for autonomous mode /// Uses G3_WORKSPACE environment variable or defaults to ~/tmp/workspace fn setup_workspace_directory(machine_mode: bool) -> Result { let workspace_dir = if let Ok(env_workspace) = std::env::var("G3_WORKSPACE") { PathBuf::from(env_workspace) } else { // Default to ~/tmp/workspace let home_dir = dirs::home_dir() .ok_or_else(|| anyhow::anyhow!("Could not determine home directory"))?; home_dir.join("tmp").join("workspace") }; // Create the directory if it doesn't exist if !workspace_dir.exists() { std::fs::create_dir_all(&workspace_dir)?; let output = SimpleOutput::new_with_mode(machine_mode); output.print(&format!( "๐Ÿ“ Created workspace directory: {}", workspace_dir.display() )); } Ok(workspace_dir) } // Simplified autonomous mode implementation async fn run_autonomous( mut agent: Agent, project: Project, show_prompt: bool, show_code: bool, max_turns: usize, quiet: bool, codebase_fast_start: Option, ) -> Result<()> { let start_time = std::time::Instant::now(); let output = SimpleOutput::new(); let mut turn_metrics: Vec = Vec::new(); output.print("g3 programming agent - autonomous mode"); output.print(&format!( "๐Ÿ“ Using workspace: {}", project.workspace().display() )); // Check if requirements exist if !project.has_requirements() { output.print("โŒ Error: requirements.md not found in workspace directory"); output.print(" Please either:"); output.print(" 1. Create a requirements.md file with your project requirements at:"); output.print(&format!( " {}/requirements.md", project.workspace().display() )); output.print(" 2. Or use the --requirements flag to provide requirements text directly:"); output.print(" g3 --autonomous --requirements \"Your requirements here\""); output.print(""); // Generate final report even for early exit let elapsed = start_time.elapsed(); let context_window = agent.get_context_window(); output.print(&format!("\n{}", "=".repeat(60))); output.print("๐Ÿ“Š AUTONOMOUS MODE SESSION REPORT"); output.print(&"=".repeat(60)); output.print(&format!( "โฑ๏ธ Total Duration: {:.2}s", elapsed.as_secs_f64() )); output.print(&format!("๐Ÿ”„ Turns Taken: 0/{}", max_turns)); output.print("๐Ÿ“ Final Status: โš ๏ธ NO REQUIREMENTS FILE"); output.print("\n๐Ÿ“ˆ Token Usage Statistics:"); output.print(&format!(" โ€ข Used Tokens: {}", context_window.used_tokens)); output.print(&format!( " โ€ข Total Available: {}", context_window.total_tokens )); output.print(&format!( " โ€ข Cumulative Tokens: {}", context_window.cumulative_tokens )); output.print(&format!( " โ€ข Usage Percentage: {:.1}%", context_window.percentage_used() )); // Add per-turn histogram output.print(&generate_turn_histogram(&turn_metrics)); output.print(&"=".repeat(60)); return Ok(()); } // Read requirements let requirements = match project.read_requirements()? { Some(content) => content, None => { output.print("โŒ Error: Could not read requirements (neither --requirements flag nor requirements.md file provided)"); // Generate final report even for early exit let elapsed = start_time.elapsed(); let context_window = agent.get_context_window(); output.print(&format!("\n{}", "=".repeat(60))); output.print("๐Ÿ“Š AUTONOMOUS MODE SESSION REPORT"); output.print(&"=".repeat(60)); output.print(&format!( "โฑ๏ธ Total Duration: {:.2}s", elapsed.as_secs_f64() )); output.print(&format!("๐Ÿ”„ Turns Taken: 0/{}", max_turns)); output.print("๐Ÿ“ Final Status: โš ๏ธ CANNOT READ REQUIREMENTS"); output.print("\n๐Ÿ“ˆ Token Usage Statistics:"); output.print(&format!(" โ€ข Used Tokens: {}", context_window.used_tokens)); output.print(&format!( " โ€ข Total Available: {}", context_window.total_tokens )); output.print(&format!( " โ€ข Cumulative Tokens: {}", context_window.cumulative_tokens )); output.print(&format!( " โ€ข Usage Percentage: {:.1}%", context_window.percentage_used() )); // Add per-turn histogram output.print(&generate_turn_histogram(&turn_metrics)); output.print(&"=".repeat(60)); return Ok(()); } }; // Display appropriate message based on requirements source if project.requirements_text.is_some() { output.print("๐Ÿ“‹ Requirements loaded from --requirements flag"); } else { output.print("๐Ÿ“‹ Requirements loaded from requirements.md"); } // Calculate SHA256 of requirements let mut hasher = Sha256::new(); hasher.update(requirements.as_bytes()); let requirements_sha = hex::encode(hasher.finalize()); output.print(&format!("๐Ÿ”’ Requirements SHA256: {}", requirements_sha)); // Pass SHA to agent for staleness checking agent.set_requirements_sha(requirements_sha.clone()); let loop_start = Instant::now(); output.print("๐Ÿ”„ Starting coach-player feedback loop..."); // Load fast-discovery messages before the loop starts (if enabled) let (discovery_messages, discovery_working_dir): (Vec, Option) = if let Some(ref codebase_path) = codebase_fast_start { // Canonicalize the path to ensure it's absolute let canonical_path = codebase_path .canonicalize() .unwrap_or_else(|_| codebase_path.clone()); let path_str = canonical_path.to_string_lossy(); output.print(&format!( "๐Ÿ” Fast-discovery mode: will explore codebase at {}", path_str )); // Get the provider from the agent and use async LLM-based discovery match agent.get_provider() { Ok(provider) => { // Create a status callback that prints to output let output_clone = output.clone(); let status_callback: g3_planner::StatusCallback = Box::new(move |msg: &str| { output_clone.print(msg); }); match g3_planner::get_initial_discovery_messages( &path_str, Some(&requirements), provider, Some(&status_callback), ) .await { Ok(messages) => (messages, Some(path_str.to_string())), Err(e) => { output.print(&format!( "โš ๏ธ LLM discovery failed: {}, skipping fast-start", e )); (Vec::new(), None) } } } Err(e) => { output.print(&format!( "โš ๏ธ Could not get provider: {}, skipping fast-start", e )); (Vec::new(), None) } } } else { (Vec::new(), None) }; let has_discovery = !discovery_messages.is_empty(); let mut turn = 1; let mut coach_feedback = String::new(); let mut implementation_approved = false; loop { let turn_start_time = Instant::now(); let turn_start_tokens = agent.get_context_window().used_tokens; output.print(&format!( "\n=== TURN {}/{} - PLAYER MODE ===", turn, max_turns )); // Surface provider info for player agent agent.print_provider_banner("Player"); // Player mode: implement requirements (with coach feedback if available) let player_prompt = if coach_feedback.is_empty() { format!( "You are G3 in implementation mode. Read and implement the following requirements:\n\n{}\n\nRequirements SHA256: {}\n\nImplement this step by step, creating all necessary files and code.", requirements, requirements_sha ) } else { format!( "You are G3 in implementation mode. Address the following specific feedback from the coach:\n\n{}\n\nContext: You are improving an implementation based on these requirements:\n{}\n\nFocus on fixing the issues mentioned in the coach feedback above.", coach_feedback, requirements ) }; output.print(&format!( "๐ŸŽฏ Starting player implementation... (elapsed: {})", format_elapsed_time(loop_start.elapsed()) )); // Display what feedback the player is receiving // If there's no coach feedback on subsequent turns, this is an error if coach_feedback.is_empty() { if turn > 1 { return Err(anyhow::anyhow!( "Player mode error: No coach feedback received on turn {}", turn )); } output.print("๐Ÿ“‹ Player starting initial implementation (no prior coach feedback)"); } else { output.print(&format!( "๐Ÿ“‹ Player received coach feedback ({} chars):", coach_feedback.len() )); output.print(&coach_feedback.to_string()); } output.print(""); // Empty line for readability // Execute player task with retry on error let mut _player_retry_count = 0; const MAX_PLAYER_RETRIES: u32 = 3; let mut player_failed = false; loop { match agent .execute_task_with_timing( &player_prompt, None, false, show_prompt, show_code, true, if has_discovery { Some(DiscoveryOptions { messages: &discovery_messages, fast_start_path: discovery_working_dir.as_deref(), }) } else { None }, ) .await { Ok(result) => { // Display player's implementation result output.print("๐Ÿ“ Player implementation completed:"); output.print_smart(&result.response); break; } Err(e) => { // Check if this is a context length exceeded error use g3_core::error_handling::{classify_error, ErrorType, RecoverableError}; let error_type = classify_error(&e); if matches!( error_type, ErrorType::Recoverable(RecoverableError::ContextLengthExceeded) ) { output.print(&format!("โš ๏ธ Context length exceeded in player turn: {}", e)); output.print("๐Ÿ“ Logging error to session and ending current turn..."); // Build forensic context let forensic_context = format!( "Turn: {}\n\ Role: Player\n\ Context tokens: {}\n\ Total available: {}\n\ Percentage used: {:.1}%\n\ Prompt length: {} chars\n\ Error occurred at: {}", turn, agent.get_context_window().used_tokens, agent.get_context_window().total_tokens, agent.get_context_window().percentage_used(), player_prompt.len(), chrono::Utc::now().to_rfc3339() ); // Log to session JSON agent.log_error_to_session(&e, "assistant", Some(forensic_context)); // Mark turn as failed and continue to next turn player_failed = true; break; } else if e.to_string().contains("panic") { output.print(&format!("๐Ÿ’ฅ Player panic detected: {}", e)); // Generate final report even for panic let elapsed = start_time.elapsed(); let context_window = agent.get_context_window(); output.print(&format!("\n{}", "=".repeat(60))); output.print("๐Ÿ“Š AUTONOMOUS MODE SESSION REPORT"); output.print(&"=".repeat(60)); output.print(&format!( "โฑ๏ธ Total Duration: {:.2}s", elapsed.as_secs_f64() )); output.print(&format!("๐Ÿ”„ Turns Taken: {}/{}", turn, max_turns)); output.print("๐Ÿ“ Final Status: ๐Ÿ’ฅ PLAYER PANIC"); output.print("\n๐Ÿ“ˆ Token Usage Statistics:"); output.print(&format!(" โ€ข Used Tokens: {}", context_window.used_tokens)); output.print(&format!( " โ€ข Total Available: {}", context_window.total_tokens )); output.print(&format!( " โ€ข Cumulative Tokens: {}", context_window.cumulative_tokens )); output.print(&format!( " โ€ข Usage Percentage: {:.1}%", context_window.percentage_used() )); // Add per-turn histogram output.print(&generate_turn_histogram(&turn_metrics)); output.print(&"=".repeat(60)); return Err(e); } _player_retry_count += 1; output.print(&format!( "โš ๏ธ Player error (attempt {}/{}): {}", _player_retry_count, MAX_PLAYER_RETRIES, e )); if _player_retry_count >= MAX_PLAYER_RETRIES { output .print("๐Ÿ”„ Max retries reached for player, marking turn as failed..."); player_failed = true; break; // Exit retry loop } output.print("๐Ÿ”„ Retrying player implementation..."); } } } // If player failed after max retries, increment turn and continue if player_failed { output.print(&format!( "โš ๏ธ Player turn {} failed after max retries. Moving to next turn.", turn )); // Record turn metrics before incrementing let turn_duration = turn_start_time.elapsed(); let turn_tokens = agent .get_context_window() .used_tokens .saturating_sub(turn_start_tokens); turn_metrics.push(TurnMetrics { turn_number: turn, tokens_used: turn_tokens, wall_clock_time: turn_duration, }); turn += 1; // Check if we've reached max turns if turn > max_turns { output.print("\n=== SESSION COMPLETED - MAX TURNS REACHED ==="); output.print(&format!("โฐ Maximum turns ({}) reached", max_turns)); break; } // Continue to next iteration with empty feedback (restart from scratch) coach_feedback = String::new(); continue; } // Give some time for file operations to complete tokio::time::sleep(tokio::time::Duration::from_millis(500)).await; // Create a new agent instance for coach mode to ensure fresh context // Use the same config with overrides that was passed to the player agent let base_config = agent.get_config().clone(); let coach_config = base_config.for_coach()?; // Reset filter suppression state before creating coach agent crate::filter_json::reset_json_tool_state(); let ui_writer = ConsoleUiWriter::new(); let mut coach_agent = Agent::new_autonomous_with_readme_and_quiet(coach_config, ui_writer, None, quiet) .await?; // Surface provider info for coach agent coach_agent.print_provider_banner("Coach"); // Ensure coach agent is also in the workspace directory project.enter_workspace()?; output.print(&format!( "\n=== TURN {}/{} - COACH MODE ===", turn, max_turns )); // Coach mode: critique the implementation let coach_prompt = format!( "You are G3 in coach mode. Your role is to critique and review implementations against requirements and provide concise, actionable feedback. REQUIREMENTS: {} IMPLEMENTATION REVIEW: Review the current state of the project and provide a concise critique focusing on: 1. Whether the requirements are correctly implemented 2. Whether the project compiles successfully 3. What requirements are missing or incorrect 4. Specific improvements needed to satisfy requirements 5. Use UI tools such as webdriver to test functionality thoroughly CRITICAL INSTRUCTIONS: 1. Provide your feedback as your final response message 2. Your feedback should be CONCISE and ACTIONABLE 3. Focus ONLY on what needs to be fixed or improved 4. Do NOT include your analysis process, file contents, or compilation output in your final feedback If the implementation thoroughly meets all requirements, compiles and is fully tested (especially UI flows) *WITHOUT* minor gaps or errors: - Respond with: 'IMPLEMENTATION_APPROVED' If improvements are needed: - Respond with a brief summary listing ONLY the specific issues to fix Remember: Be clear in your review and concise in your feedback. APPROVE iff the implementation works and thoroughly fits the requirements (implementation > 95% complete). Be rigorous, especially by testing that all UI features work.", requirements ); output.print(&format!( "๐ŸŽ“ Starting coach review... (elapsed: {})", format_elapsed_time(loop_start.elapsed()) )); // Execute coach task with retry on error let mut coach_retry_count = 0; const MAX_COACH_RETRIES: u32 = 3; let mut coach_failed = false; let coach_result_opt; loop { match coach_agent .execute_task_with_timing( &coach_prompt, None, false, show_prompt, show_code, true, if has_discovery { Some(DiscoveryOptions { messages: &discovery_messages, fast_start_path: discovery_working_dir.as_deref(), }) } else { None }, ) .await { Ok(result) => { coach_result_opt = Some(result); break; } Err(e) => { // Check if this is a context length exceeded error use g3_core::error_handling::{classify_error, ErrorType, RecoverableError}; let error_type = classify_error(&e); if matches!( error_type, ErrorType::Recoverable(RecoverableError::ContextLengthExceeded) ) { output.print(&format!("โš ๏ธ Context length exceeded in coach turn: {}", e)); output.print("๐Ÿ“ Logging error to session and ending current turn..."); // Build forensic context let forensic_context = format!( "Turn: {}\n\ Role: Coach\n\ Context tokens: {}\n\ Total available: {}\n\ Percentage used: {:.1}%\n\ Prompt length: {} chars\n\ Error occurred at: {}", turn, coach_agent.get_context_window().used_tokens, coach_agent.get_context_window().total_tokens, coach_agent.get_context_window().percentage_used(), coach_prompt.len(), chrono::Utc::now().to_rfc3339() ); // Log to coach's session JSON coach_agent.log_error_to_session(&e, "assistant", Some(forensic_context)); // Mark turn as failed and continue to next turn coach_result_opt = None; coach_failed = true; break; } else if e.to_string().contains("panic") { output.print(&format!("๐Ÿ’ฅ Coach panic detected: {}", e)); // Generate final report even for panic let elapsed = start_time.elapsed(); let context_window = agent.get_context_window(); output.print(&format!("\n{}", "=".repeat(60))); output.print("๐Ÿ“Š AUTONOMOUS MODE SESSION REPORT"); output.print(&"=".repeat(60)); output.print(&format!( "โฑ๏ธ Total Duration: {:.2}s", elapsed.as_secs_f64() )); output.print(&format!("๐Ÿ”„ Turns Taken: {}/{}", turn, max_turns)); output.print("๐Ÿ“ Final Status: ๐Ÿ’ฅ COACH PANIC"); output.print("\n๐Ÿ“ˆ Token Usage Statistics:"); output.print(&format!(" โ€ข Used Tokens: {}", context_window.used_tokens)); output.print(&format!( " โ€ข Total Available: {}", context_window.total_tokens )); output.print(&format!( " โ€ข Cumulative Tokens: {}", context_window.cumulative_tokens )); output.print(&format!( " โ€ข Usage Percentage: {:.1}%", context_window.percentage_used() )); // Add per-turn histogram output.print(&generate_turn_histogram(&turn_metrics)); output.print(&"=".repeat(60)); return Err(e); } coach_retry_count += 1; output.print(&format!( "โš ๏ธ Coach error (attempt {}/{}): {}", coach_retry_count, MAX_COACH_RETRIES, e )); if coach_retry_count >= MAX_COACH_RETRIES { output.print("๐Ÿ”„ Max retries reached for coach, using default feedback..."); // Provide default feedback and break out of retry loop coach_result_opt = None; coach_failed = true; break; // Exit retry loop with default feedback } output.print("๐Ÿ”„ Retrying coach review..."); } } } output.print("๐ŸŽ“ Coach review completed"); // If coach failed after max retries, increment turn and continue with default feedback if coach_failed { output.print(&format!( "โš ๏ธ Coach turn {} failed after max retries. Using default feedback.", turn )); coach_feedback = "The implementation needs review. Please ensure all requirements are met and the code compiles without errors.".to_string(); // Record turn metrics before incrementing let turn_duration = turn_start_time.elapsed(); let turn_tokens = agent .get_context_window() .used_tokens .saturating_sub(turn_start_tokens); turn_metrics.push(TurnMetrics { turn_number: turn, tokens_used: turn_tokens, wall_clock_time: turn_duration, }); turn += 1; if turn > max_turns { output.print("\n=== SESSION COMPLETED - MAX TURNS REACHED ==="); output.print(&format!("โฐ Maximum turns ({}) reached", max_turns)); break; } continue; // Continue to next iteration with default feedback } // We have a valid coach result, process it let coach_result = coach_result_opt.unwrap(); // Extract the complete coach feedback from the response let coach_feedback_text = extract_coach_feedback_from_logs(&coach_result, &coach_agent, &output)?; // Log the size of the feedback for debugging debug!( "Coach feedback extracted: {} characters (from {} total)", coach_feedback_text.len(), coach_result.response.len() ); // Check if we got empty feedback (this can happen if the coach doesn't provide substantive feedback) if coach_feedback_text.is_empty() { output.print("โš ๏ธ Coach did not provide feedback. This may be a model issue."); coach_feedback = "The implementation needs review. Please ensure all requirements are met and the code compiles without errors.".to_string(); // Record turn metrics before incrementing let turn_duration = turn_start_time.elapsed(); let turn_tokens = agent .get_context_window() .used_tokens .saturating_sub(turn_start_tokens); turn_metrics.push(TurnMetrics { turn_number: turn, tokens_used: turn_tokens, wall_clock_time: turn_duration, }); turn += 1; continue; } output.print_smart(&format!("Coach feedback:\n{}", coach_feedback_text)); // Check if coach approved the implementation if coach_result.is_approved() || coach_feedback_text.contains("IMPLEMENTATION_APPROVED") { output.print("\n=== SESSION COMPLETED - IMPLEMENTATION APPROVED ==="); output.print("โœ… Coach approved the implementation!"); implementation_approved = true; break; } // Check if we've reached max turns if turn >= max_turns { output.print("\n=== SESSION COMPLETED - MAX TURNS REACHED ==="); output.print(&format!("โฐ Maximum turns ({}) reached", max_turns)); break; } // Store coach feedback for next iteration coach_feedback = coach_feedback_text; // Record turn metrics before incrementing let turn_duration = turn_start_time.elapsed(); let turn_tokens = agent .get_context_window() .used_tokens .saturating_sub(turn_start_tokens); turn_metrics.push(TurnMetrics { turn_number: turn, tokens_used: turn_tokens, wall_clock_time: turn_duration, }); turn += 1; output.print("๐Ÿ”„ Coach provided feedback for next iteration"); } // Generate final report let elapsed = start_time.elapsed(); let context_window = agent.get_context_window(); output.print(&format!("\n{}", "=".repeat(60))); output.print("๐Ÿ“Š AUTONOMOUS MODE SESSION REPORT"); output.print(&"=".repeat(60)); output.print(&format!( "โฑ๏ธ Total Duration: {:.2}s", elapsed.as_secs_f64() )); output.print(&format!("๐Ÿ”„ Turns Taken: {}/{}", turn, max_turns)); output.print(&format!( "๐Ÿ“ Final Status: {}", if implementation_approved { "โœ… APPROVED" } else if turn >= max_turns { "โฐ MAX TURNS REACHED" } else { "โš ๏ธ INCOMPLETE" } )); output.print("\n๐Ÿ“ˆ Token Usage Statistics:"); output.print(&format!(" โ€ข Used Tokens: {}", context_window.used_tokens)); output.print(&format!( " โ€ข Total Available: {}", context_window.total_tokens )); output.print(&format!( " โ€ข Cumulative Tokens: {}", context_window.cumulative_tokens )); output.print(&format!( " โ€ข Usage Percentage: {:.1}%", context_window.percentage_used() )); // Add per-turn histogram output.print(&generate_turn_histogram(&turn_metrics)); output.print(&"=".repeat(60)); if implementation_approved { output.print(&format!( "\n๐ŸŽ‰ Autonomous mode completed successfully (total loop time: {})", format_elapsed_time(loop_start.elapsed()) )); } else { output.print(&format!( "\n๐Ÿ”„ Autonomous mode terminated (max iterations) (total loop time: {})", format_elapsed_time(loop_start.elapsed()) )); } // Save session continuation for resume capability agent.save_session_continuation(None); Ok(()) }