use codex_config::ConfigLayerStack; use tokio::process::Command; use crate::engine::ClaudeHooksEngine; use crate::engine::CommandShell; use crate::events::post_tool_use::PostToolUseOutcome; use crate::events::post_tool_use::PostToolUseRequest; use crate::events::pre_tool_use::PreToolUseOutcome; use crate::events::pre_tool_use::PreToolUseRequest; use crate::events::session_start::SessionStartOutcome; use crate::events::session_start::SessionStartRequest; use crate::events::stop::StopOutcome; use crate::events::stop::StopRequest; use crate::events::user_prompt_submit::UserPromptSubmitOutcome; use crate::events::user_prompt_submit::UserPromptSubmitRequest; use crate::types::Hook; use crate::types::HookEvent; use crate::types::HookPayload; use crate::types::HookResponse; #[derive(Default, Clone)] pub struct HooksConfig { pub legacy_notify_argv: Option>, pub feature_enabled: bool, pub config_layer_stack: Option, pub shell_program: Option, pub shell_args: Vec, } #[derive(Clone)] pub struct Hooks { after_agent: Vec, after_tool_use: Vec, engine: ClaudeHooksEngine, } impl Default for Hooks { fn default() -> Self { Self::new(HooksConfig::default()) } } impl Hooks { pub fn new(config: HooksConfig) -> Self { let after_agent = config .legacy_notify_argv .filter(|argv| !argv.is_empty() && !argv[0].is_empty()) .map(crate::notify_hook) .into_iter() .collect(); let engine = ClaudeHooksEngine::new( config.feature_enabled, config.config_layer_stack.as_ref(), CommandShell { program: config.shell_program.unwrap_or_default(), args: config.shell_args, }, ); Self { after_agent, after_tool_use: Vec::new(), engine, } } pub fn startup_warnings(&self) -> &[String] { self.engine.warnings() } fn hooks_for_event(&self, hook_event: &HookEvent) -> &[Hook] { match hook_event { HookEvent::AfterAgent { .. } => &self.after_agent, HookEvent::AfterToolUse { .. } => &self.after_tool_use, } } pub async fn dispatch(&self, hook_payload: HookPayload) -> Vec { let hooks = self.hooks_for_event(&hook_payload.hook_event); let mut outcomes = Vec::with_capacity(hooks.len()); for hook in hooks { let outcome = hook.execute(&hook_payload).await; let should_abort_operation = outcome.result.should_abort_operation(); outcomes.push(outcome); if should_abort_operation { break; } } outcomes } pub fn preview_session_start( &self, request: &SessionStartRequest, ) -> Vec { self.engine.preview_session_start(request) } pub fn preview_pre_tool_use( &self, request: &PreToolUseRequest, ) -> Vec { self.engine.preview_pre_tool_use(request) } pub fn preview_post_tool_use( &self, request: &PostToolUseRequest, ) -> Vec { self.engine.preview_post_tool_use(request) } pub async fn run_session_start( &self, request: SessionStartRequest, turn_id: Option, ) -> SessionStartOutcome { self.engine.run_session_start(request, turn_id).await } pub async fn run_pre_tool_use(&self, request: PreToolUseRequest) -> PreToolUseOutcome { self.engine.run_pre_tool_use(request).await } pub async fn run_post_tool_use(&self, request: PostToolUseRequest) -> PostToolUseOutcome { self.engine.run_post_tool_use(request).await } pub fn preview_user_prompt_submit( &self, request: &UserPromptSubmitRequest, ) -> Vec { self.engine.preview_user_prompt_submit(request) } pub async fn run_user_prompt_submit( &self, request: UserPromptSubmitRequest, ) -> UserPromptSubmitOutcome { self.engine.run_user_prompt_submit(request).await } pub fn preview_stop( &self, request: &StopRequest, ) -> Vec { self.engine.preview_stop(request) } pub async fn run_stop(&self, request: StopRequest) -> StopOutcome { self.engine.run_stop(request).await } } pub fn command_from_argv(argv: &[String]) -> Option { let (program, args) = argv.split_first()?; if program.is_empty() { return None; } let mut command = Command::new(program); command.args(args); Some(command) }