use anyhow::{anyhow, Context as _, Result}; use assistant_tool::{ActionLog, Tool}; use gpui::{App, Entity, Task}; use language_model::LanguageModelRequestMessage; use project::Project; use schemars::JsonSchema; use serde::{Deserialize, Serialize}; use std::sync::Arc; use util::command::new_smol_command; #[derive(Debug, Serialize, Deserialize, JsonSchema)] pub struct BashToolInput { /// The bash command to execute as a one-liner. command: String, /// Working directory for the command. This must be one of the root directories of the project. cd: String, } pub struct BashTool; impl Tool for BashTool { fn name(&self) -> String { "bash".to_string() } fn description(&self) -> String { include_str!("./bash_tool/description.md").to_string() } fn input_schema(&self) -> serde_json::Value { let schema = schemars::schema_for!(BashToolInput); serde_json::to_value(&schema).unwrap() } fn ui_text(&self, input: &serde_json::Value) -> String { match serde_json::from_value::(input.clone()) { Ok(input) => format!("`$ {}`", input.command), Err(_) => "Run bash command".to_string(), } } fn run( self: Arc, input: serde_json::Value, _messages: &[LanguageModelRequestMessage], project: Entity, _action_log: Entity, cx: &mut App, ) -> Task> { let input: BashToolInput = match serde_json::from_value(input) { Ok(input) => input, Err(err) => return Task::ready(Err(anyhow!(err))), }; let Some(worktree) = project.read(cx).worktree_for_root_name(&input.cd, cx) else { return Task::ready(Err(anyhow!("Working directory not found in the project"))); }; let working_directory = worktree.read(cx).abs_path(); cx.spawn(async move |_| { // Add 2>&1 to merge stderr into stdout for proper interleaving. let command = format!("({}) 2>&1", input.command); let output = new_smol_command("bash") .arg("-c") .arg(&command) .current_dir(working_directory) .output() .await .context("Failed to execute bash command")?; let output_string = String::from_utf8_lossy(&output.stdout).to_string(); if output.status.success() { if output_string.is_empty() { Ok("Command executed successfully.".to_string()) } else { Ok(output_string) } } else { Ok(format!( "Command failed with exit code {}\n{}", output.status.code().unwrap_or(-1), &output_string )) } }) } }