pub struct AgentLoop { /* private fields */ }Expand description
AgentLoop.
Implementations§
Source§impl AgentLoop
impl AgentLoop
Sourcepub fn new_with_resolver(
provider: Arc<dyn Provider>,
config: AgentLoopConfig,
tools: Arc<ToolRegistry>,
state: SharedState,
resolver: Arc<dyn ProviderResolver>,
) -> Self
pub fn new_with_resolver( provider: Arc<dyn Provider>, config: AgentLoopConfig, tools: Arc<ToolRegistry>, state: SharedState, resolver: Arc<dyn ProviderResolver>, ) -> Self
Creates a new AgentLoop with an explicit provider resolver.
Use this when the model ID needs to be resolved to a provider+model pair
using custom logic (e.g., per-session routing).
Sourcepub fn new(
provider: Arc<dyn Provider>,
config: AgentLoopConfig,
tools: Arc<ToolRegistry>,
state: SharedState,
) -> Self
pub fn new( provider: Arc<dyn Provider>, config: AgentLoopConfig, tools: Arc<ToolRegistry>, state: SharedState, ) -> Self
Create a new AgentLoop using the global resolver (backward compat).
Sourcepub fn with_before_tool_call(self, hook: BeforeToolCallHook) -> Self
pub fn with_before_tool_call(self, hook: BeforeToolCallHook) -> Self
Registers a hook called before every tool execution. The hook can inspect and modify tool arguments, or reject the call entirely.
Sourcepub fn with_after_tool_call(self, hook: AfterToolCallHook) -> Self
pub fn with_after_tool_call(self, hook: AfterToolCallHook) -> Self
Registers a hook called after every tool execution. The hook receives the tool name, arguments, and result.
Sourcepub fn steer(&self, message: Message)
pub fn steer(&self, message: Message)
Inject a steering message into the agent loop.
Steering messages are processed at the start of each turn, before the next LLM call. If the steering queue is at capacity (256 messages), the message is dropped and a warning is logged.
Sourcepub fn follow_up(&self, message: Message)
pub fn follow_up(&self, message: Message)
Enqueue a follow-up message to continue the conversation after all tool calls in the current batch are complete.
If the follow-up queue is at capacity (64 messages), the message is dropped and a warning is logged.
Sourcepub fn clear_steering_queue(&self)
pub fn clear_steering_queue(&self)
Removes all pending steering messages from the queue.
See steer() for an explanation of steering messages.
Sourcepub fn clear_follow_up_queue(&self)
pub fn clear_follow_up_queue(&self)
Removes all pending follow-up messages from the queue.
See follow_up() for an explanation of follow-up messages.
Sourcepub fn clear_all_queues(&self)
pub fn clear_all_queues(&self)
Removes all pending messages from both the steering and follow-up queues.
Sourcepub fn cancel_auto_retry(&self)
pub fn cancel_auto_retry(&self)
Cancels any in-progress auto-retry countdown. After calling this, the agent will not automatically retry on the next turn.
Sourcepub fn auto_retry_attempt(&self) -> usize
pub fn auto_retry_attempt(&self) -> usize
Returns the current auto-retry attempt number (0-based). Useful for displaying retry status in the UI.
Sourcepub fn state(&self) -> &SharedState
pub fn state(&self) -> &SharedState
Get a reference to the shared state. Used by Agent to sync state after loop execution.
Sourcepub fn external_stop(&self) -> &Arc<AtomicBool> ⓘ
pub fn external_stop(&self) -> &Arc<AtomicBool> ⓘ
Get the external stop flag.
Sourcepub fn set_cancel_signal(&mut self, flag: Arc<AtomicBool>)
pub fn set_cancel_signal(&mut self, flag: Arc<AtomicBool>)
Sets a shared cancel signal (typically Agent::cancel_flag).
The streaming loop checks this in its periodic wake-up timer,
ensuring cancellation is detected even when the provider stream
produces no events (e.g. waiting for first token).
Sourcepub fn cancel_signal(&self) -> Option<Arc<AtomicBool>>
pub fn cancel_signal(&self) -> Option<Arc<AtomicBool>>
Returns a clone of the loop’s cancel-signal flag, if one has been
installed via Self::set_cancel_signal. Used by tool execution
to bridge the loop’s AtomicBool cancellation into the per-tool
oneshot::Receiver cancellation channel (audit finding F-8).
Sourcepub fn is_cancelled(&self) -> bool
pub fn is_cancelled(&self) -> bool
Returns true if cancellation has been requested via either
external_stop or the direct cancel_signal.
Sourcepub fn cancel(&self)
pub fn cancel(&self)
Request cancellation from outside the loop (e.g. Ctrl+C).
Sets the external_stop flag which causes the streaming loop
to abort on its next periodic check (~500ms) and the agent loop
to exit after the current turn.
Sourcepub fn set_steering_hook(
&mut self,
hook: Arc<dyn Fn() -> Vec<String> + Send + Sync>,
)
pub fn set_steering_hook( &mut self, hook: Arc<dyn Fn() -> Vec<String> + Send + Sync>, )
Set the steering hook — called each turn to drain new messages from the session’s steering queue into the loop’s internal queue.
Sourcepub fn set_follow_up_hook(
&mut self,
hook: Arc<dyn Fn() -> Vec<String> + Send + Sync>,
)
pub fn set_follow_up_hook( &mut self, hook: Arc<dyn Fn() -> Vec<String> + Send + Sync>, )
Set the follow-up hook — called each turn to drain new messages from the session’s follow-up queue into the loop’s internal queue.
Sourcepub async fn run(
&self,
prompt: String,
emit: impl Fn(AgentEvent) + Send + Sync + 'static,
) -> Result<Vec<AgentEvent>>
pub async fn run( &self, prompt: String, emit: impl Fn(AgentEvent) + Send + Sync + 'static, ) -> Result<Vec<AgentEvent>>
Runs the agent loop with a single user prompt.
Convenience wrapper around run_messages().
Sourcepub async fn run_mut<S: Send + Debug + 'static>(
&self,
prompt: String,
state: S,
emit: impl FnMut(AgentEvent, &mut S) + Send + 'static,
) -> Result<(Vec<AgentEvent>, S)>
pub async fn run_mut<S: Send + Debug + 'static>( &self, prompt: String, state: S, emit: impl FnMut(AgentEvent, &mut S) + Send + 'static, ) -> Result<(Vec<AgentEvent>, S)>
Run with an FnMut callback and mutable state — no Arc<Mutex<>> needed.
Unlike run(), which takes Fn, this method accepts FnMut
and a user-provided state value S. The callback receives &mut S on each
event, so you can accumulate results without any locking overhead.
Returns the collected events and the final state value.
§Example
#[derive(Default)]
struct MyState { steps: usize, output: String }
let (events, state) = agent_loop.run_mut(
"do something".into(),
MyState::default(),
|event, s| {
match event {
AgentEvent::ToolExecutionEnd { is_error: false, .. } => s.steps += 1,
AgentEvent::AgentEnd { messages, .. } => {
if let Some(Message::Assistant(a)) = messages.last() {
s.output = a.text_content();
}
}
_ => {}
}
},
).await?;Sourcepub async fn run_messages(
&self,
prompts: Vec<Message>,
emit: Arc<dyn Fn(AgentEvent) + Send + Sync>,
) -> Result<Vec<AgentEvent>>
pub async fn run_messages( &self, prompts: Vec<Message>, emit: Arc<dyn Fn(AgentEvent) + Send + Sync>, ) -> Result<Vec<AgentEvent>>
Runs the agent loop with a list of pre-constructed messages. This is the primary entry point for executing agent turns.
Sourcepub async fn continue_loop(
&self,
emit: impl Fn(AgentEvent) + Send + Sync + 'static,
) -> Result<Vec<AgentEvent>>
pub async fn continue_loop( &self, emit: impl Fn(AgentEvent) + Send + Sync + 'static, ) -> Result<Vec<AgentEvent>>
Resumes the agent loop after a previous turn ended in a paused state (e.g., waiting for user confirmation). Emits events to the provided callback.
Auto Trait Implementations§
impl !Freeze for AgentLoop
impl !RefUnwindSafe for AgentLoop
impl !UnwindSafe for AgentLoop
impl Send for AgentLoop
impl Sync for AgentLoop
impl Unpin for AgentLoop
impl UnsafeUnpin for AgentLoop
Blanket Implementations§
Source§impl<T> BorrowMut<T> for Twhere
T: ?Sized,
impl<T> BorrowMut<T> for Twhere
T: ?Sized,
Source§fn borrow_mut(&mut self) -> &mut T
fn borrow_mut(&mut self) -> &mut T
impl<ST, DT> CastableFrom<ST, Initialized, Initialized> for DT
impl<ST, DT> CastableFrom<ST, Uninit, Uninit> for DT
impl<T> ErasedDestructor for Twhere
T: 'static,
Source§impl<T> Instrument for T
impl<T> Instrument for T
Source§fn instrument(self, span: Span) -> Instrumented<Self>
fn instrument(self, span: Span) -> Instrumented<Self>
Source§fn in_current_span(self) -> Instrumented<Self>
fn in_current_span(self) -> Instrumented<Self>
Source§impl<T> IntoEither for T
impl<T> IntoEither for T
Source§fn into_either(self, into_left: bool) -> Either<Self, Self>
fn into_either(self, into_left: bool) -> Either<Self, Self>
self into a Left variant of Either<Self, Self>
if into_left is true.
Converts self into a Right variant of Either<Self, Self>
otherwise. Read moreSource§fn into_either_with<F>(self, into_left: F) -> Either<Self, Self>
fn into_either_with<F>(self, into_left: F) -> Either<Self, Self>
self into a Left variant of Either<Self, Self>
if into_left(&self) returns true.
Converts self into a Right variant of Either<Self, Self>
otherwise. Read moreimpl<T> MaybeSendSync for T
Source§impl<T> Pipe for Twhere
T: ?Sized,
impl<T> Pipe for Twhere
T: ?Sized,
Source§fn pipe<R>(self, func: impl FnOnce(Self) -> R) -> Rwhere
Self: Sized,
fn pipe<R>(self, func: impl FnOnce(Self) -> R) -> Rwhere
Self: Sized,
Source§fn pipe_ref<'a, R>(&'a self, func: impl FnOnce(&'a Self) -> R) -> Rwhere
R: 'a,
fn pipe_ref<'a, R>(&'a self, func: impl FnOnce(&'a Self) -> R) -> Rwhere
R: 'a,
self and passes that borrow into the pipe function. Read moreSource§fn pipe_ref_mut<'a, R>(&'a mut self, func: impl FnOnce(&'a mut Self) -> R) -> Rwhere
R: 'a,
fn pipe_ref_mut<'a, R>(&'a mut self, func: impl FnOnce(&'a mut Self) -> R) -> Rwhere
R: 'a,
self and passes that borrow into the pipe function. Read moreSource§fn pipe_borrow<'a, B, R>(&'a self, func: impl FnOnce(&'a B) -> R) -> R
fn pipe_borrow<'a, B, R>(&'a self, func: impl FnOnce(&'a B) -> R) -> R
Source§fn pipe_borrow_mut<'a, B, R>(
&'a mut self,
func: impl FnOnce(&'a mut B) -> R,
) -> R
fn pipe_borrow_mut<'a, B, R>( &'a mut self, func: impl FnOnce(&'a mut B) -> R, ) -> R
Source§fn pipe_as_ref<'a, U, R>(&'a self, func: impl FnOnce(&'a U) -> R) -> R
fn pipe_as_ref<'a, U, R>(&'a self, func: impl FnOnce(&'a U) -> R) -> R
self, then passes self.as_ref() into the pipe function.Source§fn pipe_as_mut<'a, U, R>(&'a mut self, func: impl FnOnce(&'a mut U) -> R) -> R
fn pipe_as_mut<'a, U, R>(&'a mut self, func: impl FnOnce(&'a mut U) -> R) -> R
self, then passes self.as_mut() into the pipe
function.Source§fn pipe_deref<'a, T, R>(&'a self, func: impl FnOnce(&'a T) -> R) -> R
fn pipe_deref<'a, T, R>(&'a self, func: impl FnOnce(&'a T) -> R) -> R
self, then passes self.deref() into the pipe function.Source§impl<T> Pointable for T
impl<T> Pointable for T
Source§impl<T> PolicyExt for Twhere
T: ?Sized,
impl<T> PolicyExt for Twhere
T: ?Sized,
impl<T> Read<Exclusive, BecauseExclusive> for Twhere
T: ?Sized,
Source§impl<T> Tap for T
impl<T> Tap for T
Source§fn tap_borrow<B>(self, func: impl FnOnce(&B)) -> Self
fn tap_borrow<B>(self, func: impl FnOnce(&B)) -> Self
Borrow<B> of a value. Read moreSource§fn tap_borrow_mut<B>(self, func: impl FnOnce(&mut B)) -> Self
fn tap_borrow_mut<B>(self, func: impl FnOnce(&mut B)) -> Self
BorrowMut<B> of a value. Read moreSource§fn tap_ref<R>(self, func: impl FnOnce(&R)) -> Self
fn tap_ref<R>(self, func: impl FnOnce(&R)) -> Self
AsRef<R> view of a value. Read moreSource§fn tap_ref_mut<R>(self, func: impl FnOnce(&mut R)) -> Self
fn tap_ref_mut<R>(self, func: impl FnOnce(&mut R)) -> Self
AsMut<R> view of a value. Read moreSource§fn tap_deref<T>(self, func: impl FnOnce(&T)) -> Self
fn tap_deref<T>(self, func: impl FnOnce(&T)) -> Self
Deref::Target of a value. Read moreSource§fn tap_deref_mut<T>(self, func: impl FnOnce(&mut T)) -> Self
fn tap_deref_mut<T>(self, func: impl FnOnce(&mut T)) -> Self
Deref::Target of a value. Read moreSource§fn tap_dbg(self, func: impl FnOnce(&Self)) -> Self
fn tap_dbg(self, func: impl FnOnce(&Self)) -> Self
.tap() only in debug builds, and is erased in release builds.Source§fn tap_mut_dbg(self, func: impl FnOnce(&mut Self)) -> Self
fn tap_mut_dbg(self, func: impl FnOnce(&mut Self)) -> Self
.tap_mut() only in debug builds, and is erased in release
builds.Source§fn tap_borrow_dbg<B>(self, func: impl FnOnce(&B)) -> Self
fn tap_borrow_dbg<B>(self, func: impl FnOnce(&B)) -> Self
.tap_borrow() only in debug builds, and is erased in release
builds.Source§fn tap_borrow_mut_dbg<B>(self, func: impl FnOnce(&mut B)) -> Self
fn tap_borrow_mut_dbg<B>(self, func: impl FnOnce(&mut B)) -> Self
.tap_borrow_mut() only in debug builds, and is erased in release
builds.Source§fn tap_ref_dbg<R>(self, func: impl FnOnce(&R)) -> Self
fn tap_ref_dbg<R>(self, func: impl FnOnce(&R)) -> Self
.tap_ref() only in debug builds, and is erased in release
builds.Source§fn tap_ref_mut_dbg<R>(self, func: impl FnOnce(&mut R)) -> Self
fn tap_ref_mut_dbg<R>(self, func: impl FnOnce(&mut R)) -> Self
.tap_ref_mut() only in debug builds, and is erased in release
builds.Source§fn tap_deref_dbg<T>(self, func: impl FnOnce(&T)) -> Self
fn tap_deref_dbg<T>(self, func: impl FnOnce(&T)) -> Self
.tap_deref() only in debug builds, and is erased in release
builds.