use serde::{Deserialize, Serialize};
use std::collections::HashMap;
use std::path::PathBuf;
use std::sync::Arc;
use std::time::Duration;
use typed_builder::TypedBuilder;
use super::efficiency::EfficiencyConfig;
use super::hooks::{HookEvent, HookMatcher};
use super::mcp::McpServers;
use super::permissions::CanUseToolCallback;
use super::plugin::SdkPluginConfig;
#[derive(Clone, TypedBuilder)]
#[builder(doc)]
pub struct ClaudeAgentOptions {
#[builder(default, setter(strip_option))]
pub tools: Option<Tools>,
#[builder(default, setter(into))]
pub allowed_tools: Vec<String>,
#[builder(default, setter(into, strip_option))]
pub system_prompt: Option<SystemPrompt>,
#[builder(default)]
pub mcp_servers: McpServers,
#[builder(default, setter(strip_option))]
pub permission_mode: Option<PermissionMode>,
#[builder(default = false)]
pub continue_conversation: bool,
#[builder(default, setter(into, strip_option))]
pub resume: Option<String>,
#[builder(default, setter(strip_option))]
pub max_turns: Option<u32>,
#[builder(default, setter(into))]
pub disallowed_tools: Vec<String>,
#[builder(default, setter(into, strip_option))]
pub model: Option<String>,
#[builder(default, setter(into, strip_option))]
pub fallback_model: Option<String>,
#[builder(default, setter(into))]
pub betas: Vec<SdkBeta>,
#[builder(default, setter(strip_option))]
pub max_budget_usd: Option<f64>,
#[builder(default, setter(strip_option))]
pub max_thinking_tokens: Option<u32>,
#[builder(default, setter(into, strip_option))]
pub permission_prompt_tool_name: Option<String>,
#[builder(default, setter(into, strip_option))]
pub cwd: Option<PathBuf>,
#[builder(default, setter(into, strip_option))]
pub cli_path: Option<PathBuf>,
#[builder(default, setter(into, strip_option))]
pub settings: Option<String>,
#[builder(default, setter(into))]
pub add_dirs: Vec<PathBuf>,
#[builder(default)]
pub env: HashMap<String, String>,
#[builder(default)]
pub extra_args: HashMap<String, Option<String>>,
#[builder(default, setter(strip_option))]
pub max_buffer_size: Option<usize>,
#[builder(default, setter(strip_option))]
pub stderr_callback: Option<Arc<dyn Fn(String) + Send + Sync>>,
#[builder(default, setter(strip_option))]
pub can_use_tool: Option<CanUseToolCallback>,
#[builder(default, setter(strip_option))]
pub hooks: Option<HashMap<HookEvent, Vec<HookMatcher>>>,
#[builder(default, setter(into, strip_option))]
pub user: Option<String>,
#[builder(default = false)]
pub include_partial_messages: bool,
#[builder(default = false)]
pub fork_session: bool,
#[builder(default, setter(strip_option))]
pub agents: Option<HashMap<String, AgentDefinition>>,
#[builder(default, setter(strip_option))]
pub setting_sources: Option<Vec<SettingSource>>,
#[builder(default, setter(strip_option))]
pub sandbox: Option<SandboxSettings>,
#[builder(default, setter(into))]
pub plugins: Vec<SdkPluginConfig>,
#[builder(default, setter(strip_option))]
pub output_format: Option<serde_json::Value>,
#[builder(default = false)]
pub enable_file_checkpointing: bool,
#[builder(default = Some(Duration::from_secs(60)))]
pub control_request_timeout: Option<Duration>,
#[builder(default = false)]
pub skip_version_check: bool,
#[builder(default = true)]
pub verbose: bool,
#[builder(default = false)]
pub auto_download_cli: bool,
#[builder(default, setter(strip_option))]
pub efficiency: Option<EfficiencyConfig>,
}
impl Default for ClaudeAgentOptions {
fn default() -> Self {
Self::builder().build()
}
}
#[derive(Debug, Clone, Serialize, Deserialize)]
#[serde(untagged)]
pub enum SystemPrompt {
Text(String),
Preset(SystemPromptPreset),
}
impl From<String> for SystemPrompt {
fn from(text: String) -> Self {
SystemPrompt::Text(text)
}
}
impl From<&str> for SystemPrompt {
fn from(text: &str) -> Self {
SystemPrompt::Text(text.to_string())
}
}
#[derive(Debug, Clone, Serialize, Deserialize)]
pub struct SystemPromptPreset {
#[serde(rename = "type")]
pub type_: String,
pub preset: String,
#[serde(skip_serializing_if = "Option::is_none")]
pub append: Option<String>,
}
impl SystemPromptPreset {
pub fn new(preset: impl Into<String>) -> Self {
Self {
type_: "preset".to_string(),
preset: preset.into(),
append: None,
}
}
pub fn with_append(preset: impl Into<String>, append: impl Into<String>) -> Self {
Self {
type_: "preset".to_string(),
preset: preset.into(),
append: Some(append.into()),
}
}
}
#[derive(Debug, Clone, Copy, Serialize, Deserialize, PartialEq, Eq)]
#[serde(rename_all = "camelCase")]
pub enum PermissionMode {
#[serde(rename = "default")]
Default,
AcceptEdits,
#[serde(rename = "plan")]
Plan,
BypassPermissions,
}
#[derive(Debug, Clone, Copy, Serialize, Deserialize, PartialEq, Eq)]
#[serde(rename_all = "lowercase")]
pub enum SettingSource {
User,
Project,
Local,
}
#[derive(Debug, Clone, Serialize, Deserialize, TypedBuilder)]
#[builder(doc)]
pub struct AgentDefinition {
#[builder(setter(into))]
pub description: String,
#[builder(setter(into))]
pub prompt: String,
#[serde(skip_serializing_if = "Option::is_none")]
#[builder(default, setter(into, strip_option))]
pub tools: Option<Vec<String>>,
#[serde(skip_serializing_if = "Option::is_none")]
#[builder(default, setter(strip_option))]
pub model: Option<AgentModel>,
}
#[derive(Debug, Clone, Copy, Serialize, Deserialize, PartialEq, Eq)]
#[serde(rename_all = "lowercase")]
pub enum AgentModel {
Sonnet,
Opus,
Haiku,
Inherit,
}
#[derive(Debug, Clone, Serialize, Deserialize, PartialEq, Eq)]
pub enum SdkBeta {
#[serde(rename = "context-1m-2025-08-07")]
Context1M,
}
#[derive(Debug, Clone, Serialize, Deserialize)]
#[serde(untagged)]
pub enum Tools {
List(Vec<String>),
Preset(ToolsPreset),
}
#[derive(Debug, Clone, Serialize, Deserialize)]
pub struct ToolsPreset {
#[serde(rename = "type")]
pub type_: String,
pub preset: String,
}
impl ToolsPreset {
pub fn new(preset: impl Into<String>) -> Self {
Self {
type_: "preset".to_string(),
preset: preset.into(),
}
}
pub fn claude_code() -> Self {
Self::new("claude_code")
}
}
#[derive(Debug, Clone, Default, Serialize, Deserialize, TypedBuilder)]
#[builder(doc)]
pub struct SandboxNetworkConfig {
#[serde(skip_serializing_if = "Option::is_none", rename = "allowUnixSockets")]
#[builder(default, setter(into, strip_option))]
pub allow_unix_sockets: Option<Vec<String>>,
#[serde(
skip_serializing_if = "Option::is_none",
rename = "allowAllUnixSockets"
)]
#[builder(default, setter(strip_option))]
pub allow_all_unix_sockets: Option<bool>,
#[serde(skip_serializing_if = "Option::is_none", rename = "allowLocalBinding")]
#[builder(default, setter(strip_option))]
pub allow_local_binding: Option<bool>,
#[serde(skip_serializing_if = "Option::is_none", rename = "httpProxyPort")]
#[builder(default, setter(strip_option))]
pub http_proxy_port: Option<u16>,
#[serde(skip_serializing_if = "Option::is_none", rename = "socksProxyPort")]
#[builder(default, setter(strip_option))]
pub socks_proxy_port: Option<u16>,
}
#[derive(Debug, Clone, Default, Serialize, Deserialize, TypedBuilder)]
#[builder(doc)]
pub struct SandboxIgnoreViolations {
#[serde(skip_serializing_if = "Option::is_none")]
#[builder(default, setter(into, strip_option))]
pub file: Option<Vec<String>>,
#[serde(skip_serializing_if = "Option::is_none")]
#[builder(default, setter(into, strip_option))]
pub network: Option<Vec<String>>,
}
#[derive(Debug, Clone, Default, Serialize, Deserialize, TypedBuilder)]
#[builder(doc)]
pub struct SandboxSettings {
#[serde(skip_serializing_if = "Option::is_none")]
#[builder(default, setter(strip_option))]
pub enabled: Option<bool>,
#[serde(
skip_serializing_if = "Option::is_none",
rename = "autoAllowBashIfSandboxed"
)]
#[builder(default, setter(strip_option))]
pub auto_allow_bash_if_sandboxed: Option<bool>,
#[serde(skip_serializing_if = "Option::is_none", rename = "excludedCommands")]
#[builder(default, setter(into, strip_option))]
pub excluded_commands: Option<Vec<String>>,
#[serde(
skip_serializing_if = "Option::is_none",
rename = "allowUnsandboxedCommands"
)]
#[builder(default, setter(strip_option))]
pub allow_unsandboxed_commands: Option<bool>,
#[serde(skip_serializing_if = "Option::is_none")]
#[builder(default, setter(strip_option))]
pub network: Option<SandboxNetworkConfig>,
#[serde(skip_serializing_if = "Option::is_none", rename = "ignoreViolations")]
#[builder(default, setter(strip_option))]
pub ignore_violations: Option<SandboxIgnoreViolations>,
#[serde(
skip_serializing_if = "Option::is_none",
rename = "enableWeakerNestedSandbox"
)]
#[builder(default, setter(strip_option))]
pub enable_weaker_nested_sandbox: Option<bool>,
}
impl ClaudeAgentOptions {
pub fn use_acp_tools(&mut self, tools: &[&str]) {
use super::mcp::acp_tool_name;
for tool in tools {
let tool_str = (*tool).to_string();
if !self.disallowed_tools.contains(&tool_str) {
self.disallowed_tools.push(tool_str);
}
let mcp_name = acp_tool_name(tool);
if !self.allowed_tools.contains(&mcp_name) {
self.allowed_tools.push(mcp_name);
}
}
}
}
#[cfg(test)]
mod tests {
use super::*;
#[test]
fn test_control_request_timeout_default() {
let opts = ClaudeAgentOptions::default();
assert_eq!(opts.control_request_timeout, Some(Duration::from_secs(60)));
}
#[test]
fn test_control_request_timeout_custom() {
let opts = ClaudeAgentOptions::builder()
.control_request_timeout(Some(Duration::from_secs(30)))
.build();
assert_eq!(opts.control_request_timeout, Some(Duration::from_secs(30)));
}
#[test]
fn test_control_request_timeout_none() {
let opts = ClaudeAgentOptions::builder()
.control_request_timeout(None)
.build();
assert_eq!(opts.control_request_timeout, None);
}
}