bamboo_agent_core/tools/mod.rs
1//! Tool execution system for Bamboo agents.
2//!
3//! This module provides a comprehensive framework for defining, registering, and executing
4//! tools that can be used by AI agents to interact with external systems.
5//!
6//! # Architecture
7//!
8//! The tools system is built around several key components:
9//!
10//! - **accumulator**: Accumulates partial tool calls from streaming responses
11//! - **agentic**: Agentic tool execution with multi-step capabilities
12//! - **executor**: Core tool execution logic
13//! - **registry**: Tool registration and lookup
14//! - **result_handler**: Processes tool results and handles agentic support
15//! - **smart_code_review**: Specialized tool for intelligent code review
16//! - **types**: Core type definitions for tools
17//!
18//! # Key Concepts
19//!
20//! ## Tool Registry
21//!
22//! Tools are registered in a central [`ToolRegistry`] that maps tool names to their
23//! implementations. The registry supports:
24//!
25//! - Dynamic tool registration
26//! - Tool name normalization
27//! - Global singleton access via [`global_registry`]
28//!
29//! ## Tool Execution
30//!
31//! Tools implement the [`ToolExecutor`] trait and can be executed via [`execute_tool_call`].
32//! The execution flow:
33//!
34//! 1. Parse tool arguments from JSON
35//! 2. Execute the tool logic
36//! 3. Return a [`ToolResult`] with success/failure status
37//!
38//! ## Agentic Tools
39//!
40//! Some tools support "agentic" behavior, allowing multi-step execution:
41//!
42//! - [`AgenticTool`]: Marker trait for agentic tools
43//! - [`AgenticContext`]: Context for agentic execution
44//! - [`AgenticToolResult`]: Extended result type with sub-actions
45//!
46//! # Example
47//!
48//! ```rust,ignore
49//! use async_trait::async_trait;
50//! use bamboo_agent::agent::core::tools::{
51//! execute_tool_call, FunctionCall, ToolCall, ToolError, ToolExecutor, ToolResult, ToolSchema,
52//! };
53//!
54//! struct NoopExecutor;
55//!
56//! #[async_trait]
57//! impl ToolExecutor for NoopExecutor {
58//! async fn execute(&self, call: &ToolCall) -> Result<ToolResult, ToolError> {
59//! Err(ToolError::NotFound(call.function.name.clone()))
60//! }
61//!
62//! fn list_tools(&self) -> Vec<ToolSchema> {
63//! Vec::new()
64//! }
65//! }
66//!
67//! #[tokio::main]
68//! async fn main() {
69//! // Execute a tool call (this example uses a no-op executor).
70//! let call = ToolCall {
71//! id: "call-1".to_string(),
72//! tool_type: "function".to_string(),
73//! function: FunctionCall {
74//! name: "read_file".to_string(),
75//! arguments: r#"{\"path\":\"/tmp/test.txt\"}"#.to_string(),
76//! },
77//! };
78//!
79//! let _ = execute_tool_call(&call, &NoopExecutor, None).await;
80//! }
81//! ```
82//!
83//! # Re-exports
84//!
85//! Key types and functions re-exported for convenience:
86//!
87//! - Accumulator: [`ToolCallAccumulator`], [`PartialToolCall`], [`finalize_tool_calls`]
88//! - Agentic: [`AgenticTool`], [`AgenticContext`], [`AgenticToolResult`], [`ToolGoal`]
89//! - Executor: [`ToolExecutor`], [`execute_tool_call`], [`ToolError`]
90//! - Registry: [`ToolRegistry`], [`Tool`], [`global_registry`]
91//! - Types: [`ToolCall`], [`ToolResult`], [`ToolSchema`]
92
93pub mod accumulator;
94pub mod agentic;
95pub mod context;
96pub mod executor;
97pub mod registry;
98pub mod result_handler;
99pub mod smart_code_review;
100pub mod types;
101
102pub use accumulator::{
103 finalize_tool_calls, update_partial_tool_call, PartialToolCall, ToolCallAccumulator,
104};
105pub use agentic::{
106 convert_from_standard_result, convert_to_standard_result, AgenticContext, AgenticTool,
107 AgenticToolExecutor, AgenticToolResult, Interaction, InteractionRole, ToolGoal,
108};
109pub use context::ToolExecutionContext;
110pub use executor::{execute_tool_call, execute_tool_call_with_context, ToolError, ToolExecutor};
111pub use registry::{
112 global_registry, normalize_tool_name, RegistryError, SharedTool, Tool, ToolRegistry,
113};
114pub use result_handler::{
115 execute_sub_actions, handle_tool_result_with_agentic_support, parse_tool_args,
116 parse_tool_args_best_effort, send_clarification_request, try_parse_agentic_result,
117 ToolHandlingOutcome, MAX_SUB_ACTIONS,
118};
119pub use smart_code_review::SmartCodeReviewTool;
120pub use types::{FunctionCall, FunctionSchema, ToolCall, ToolResult, ToolSchema};
121
122/// Classification of a tool call for approval purposes.
123#[derive(Debug, Clone, Copy, PartialEq, Eq)]
124pub enum ToolMutability {
125 ReadOnly,
126 Mutating,
127}
128
129/// Read-only tools that don't require user approval.
130const READ_ONLY_TOOLS: &[&str] = &[
131 "Read",
132 "GetFileInfo",
133 "Glob",
134 "Grep",
135 "WebFetch",
136 "WebSearch",
137 "Workspace",
138 "BashOutput",
139 "session_note",
140 "memory_note",
141 "recall",
142 "session_inspector",
143 "compact_context",
144 "Sleep",
145];
146
147/// Classify a tool call as read-only or mutating.
148pub fn classify_tool(tool_name: &str) -> ToolMutability {
149 if READ_ONLY_TOOLS
150 .iter()
151 .any(|&t| t.eq_ignore_ascii_case(tool_name))
152 {
153 ToolMutability::ReadOnly
154 } else {
155 ToolMutability::Mutating
156 }
157}
158
159#[cfg(test)]
160mod tests {
161 use super::*;
162
163 #[test]
164 fn classify_compact_context_as_read_only() {
165 assert_eq!(classify_tool("compact_context"), ToolMutability::ReadOnly);
166 }
167
168 #[test]
169 fn classify_compact_context_case_insensitive() {
170 assert_eq!(classify_tool("Compact_Context"), ToolMutability::ReadOnly);
171 assert_eq!(classify_tool("COMPACT_CONTEXT"), ToolMutability::ReadOnly);
172 }
173
174 #[test]
175 fn classify_write_as_mutating() {
176 assert_eq!(classify_tool("Write"), ToolMutability::Mutating);
177 }
178
179 #[test]
180 fn classify_unknown_as_mutating() {
181 assert_eq!(
182 classify_tool("totally_unknown_tool"),
183 ToolMutability::Mutating
184 );
185 }
186
187 #[test]
188 fn classify_all_read_only_tools() {
189 for name in READ_ONLY_TOOLS {
190 assert_eq!(
191 classify_tool(name),
192 ToolMutability::ReadOnly,
193 "{name} should be classified as read-only"
194 );
195 }
196 }
197}