use serde::{Deserialize, Serialize};
use std::collections::HashMap;
use std::path::PathBuf;
use std::sync::Arc;
use typed_builder::TypedBuilder;
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(strip_option, into))]
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 buffer_config: Option<DynamicBufferConfig>,
#[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 = false)]
pub auto_discover_skills: bool,
#[builder(default, setter(into, strip_option))]
pub project_skills_dir: Option<PathBuf>,
#[builder(default, setter(into, strip_option))]
pub user_skills_dir: Option<PathBuf>,
#[builder(default = false)]
pub auto_install_cli: bool,
#[builder(default, setter(strip_option))]
pub cli_install_callback: Option<Arc<dyn Fn(crate::internal::cli_installer::InstallProgress) + Send + Sync>>,
#[builder(default, setter(strip_option))]
pub pool_config: Option<crate::internal::pool::PoolConfig>,
#[builder(default)]
pub parsing_mode: crate::internal::message_parser::ParsingMode,
}
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>,
}
#[derive(Debug, Clone, Serialize, Deserialize, TypedBuilder)]
#[builder(doc)]
pub struct DynamicBufferConfig {
#[builder(default = 64 * 1024)]
#[serde(default = "default_initial_size")]
pub initial_size: usize,
#[builder(default = 50 * 1024 * 1024)]
#[serde(default = "default_max_message_size")]
pub max_message_size: usize,
#[builder(default = 2.0)]
#[serde(default = "default_growth_factor")]
pub growth_factor: f64,
#[builder(default = true)]
#[serde(default = "default_enable_metrics")]
pub enable_metrics: bool,
}
fn default_initial_size() -> usize {
64 * 1024 }
fn default_max_message_size() -> usize {
50 * 1024 * 1024 }
fn default_growth_factor() -> f64 {
2.0
}
fn default_enable_metrics() -> bool {
true
}
impl Default for DynamicBufferConfig {
fn default() -> Self {
Self::builder().build()
}
}
#[derive(Debug, Clone, Default)]
pub struct BufferMetrics {
pub peak_size: usize,
pub message_count: usize,
pub total_bytes: usize,
pub resize_count: usize,
}
impl BufferMetrics {
pub fn average_message_size(&self) -> usize {
if self.message_count == 0 {
0
} else {
self.total_bytes / self.message_count
}
}
}