use std::future::Future;
use std::pin::Pin;
use crate::context::CommandContext;
use crate::{CommandError, CommandHandler, CommandOutput, SlashCategory};
pub struct StatusCommand;
impl CommandHandler<CommandContext<'_>> for StatusCommand {
fn name(&self) -> &'static str {
"/status"
}
fn description(&self) -> &'static str {
"Show current session status (provider, model, tokens, uptime)"
}
fn category(&self) -> SlashCategory {
SlashCategory::Debugging
}
fn handle<'a>(
&'a self,
ctx: &'a mut CommandContext<'_>,
_args: &'a str,
) -> Pin<Box<dyn Future<Output = Result<CommandOutput, CommandError>> + Send + 'a>> {
use tracing::Instrument as _;
let span = tracing::info_span!("commands.status.handle");
Box::pin(
async move {
let result = ctx.agent.session_status().await?;
Ok(CommandOutput::Message(result))
}
.instrument(span),
)
}
}
pub struct GuardrailCommand;
impl CommandHandler<CommandContext<'_>> for GuardrailCommand {
fn name(&self) -> &'static str {
"/guardrail"
}
fn description(&self) -> &'static str {
"Show guardrail status (provider, model, action, timeout, stats)"
}
fn category(&self) -> SlashCategory {
SlashCategory::Debugging
}
fn feature_gate(&self) -> Option<&'static str> {
Some("guardrail")
}
fn handle<'a>(
&'a self,
ctx: &'a mut CommandContext<'_>,
_args: &'a str,
) -> Pin<Box<dyn Future<Output = Result<CommandOutput, CommandError>> + Send + 'a>> {
use tracing::Instrument as _;
let span = tracing::info_span!("commands.guardrail.handle");
Box::pin(
async move {
let result = ctx.agent.guardrail_status();
Ok(CommandOutput::Message(result))
}
.instrument(span),
)
}
}
pub struct FocusCommand;
impl CommandHandler<CommandContext<'_>> for FocusCommand {
fn name(&self) -> &'static str {
"/focus"
}
fn description(&self) -> &'static str {
"Show Focus Agent status (active session, knowledge block size)"
}
fn category(&self) -> SlashCategory {
SlashCategory::Advanced
}
fn feature_gate(&self) -> Option<&'static str> {
Some("context-compression")
}
fn handle<'a>(
&'a self,
ctx: &'a mut CommandContext<'_>,
_args: &'a str,
) -> Pin<Box<dyn Future<Output = Result<CommandOutput, CommandError>> + Send + 'a>> {
use tracing::Instrument as _;
let span = tracing::info_span!("commands.focus.handle");
Box::pin(
async move {
let result = ctx.agent.focus_status();
Ok(CommandOutput::Message(result))
}
.instrument(span),
)
}
}
pub struct SideQuestCommand;
impl CommandHandler<CommandContext<'_>> for SideQuestCommand {
fn name(&self) -> &'static str {
"/sidequest"
}
fn description(&self) -> &'static str {
"Show SideQuest eviction stats (passes run, tokens freed)"
}
fn category(&self) -> SlashCategory {
SlashCategory::Advanced
}
fn feature_gate(&self) -> Option<&'static str> {
Some("context-compression")
}
fn handle<'a>(
&'a self,
ctx: &'a mut CommandContext<'_>,
_args: &'a str,
) -> Pin<Box<dyn Future<Output = Result<CommandOutput, CommandError>> + Send + 'a>> {
use tracing::Instrument as _;
let span = tracing::info_span!("commands.sidequest.handle");
Box::pin(
async move {
let result = ctx.agent.sidequest_status();
Ok(CommandOutput::Message(result))
}
.instrument(span),
)
}
}
#[cfg(test)]
mod tests {
use super::*;
use crate::handlers::test_helpers::{MockDebug, MockMessages, MockSession, make_ctx};
use crate::sink::NullSink;
#[test]
fn status_name_and_description() {
assert_eq!(StatusCommand.name(), "/status");
assert!(!StatusCommand.description().is_empty());
}
#[test]
fn guardrail_name_and_description() {
assert_eq!(GuardrailCommand.name(), "/guardrail");
assert!(!GuardrailCommand.description().is_empty());
}
#[test]
fn focus_name_and_description() {
assert_eq!(FocusCommand.name(), "/focus");
assert!(!FocusCommand.description().is_empty());
}
#[test]
fn sidequest_name_and_description() {
assert_eq!(SideQuestCommand.name(), "/sidequest");
assert!(!SideQuestCommand.description().is_empty());
}
#[tokio::test]
async fn status_returns_message() {
let mut sink = NullSink;
let mut debug = MockDebug;
let mut messages = MockMessages;
let session = MockSession;
let mut agent = crate::NullAgent;
let mut ctx = make_ctx(&mut sink, &mut debug, &mut messages, &session, &mut agent);
let out = StatusCommand.handle(&mut ctx, "").await.unwrap();
assert!(matches!(out, CommandOutput::Message(_)));
}
#[tokio::test]
async fn guardrail_returns_message() {
let mut sink = NullSink;
let mut debug = MockDebug;
let mut messages = MockMessages;
let session = MockSession;
let mut agent = crate::NullAgent;
let mut ctx = make_ctx(&mut sink, &mut debug, &mut messages, &session, &mut agent);
let out = GuardrailCommand.handle(&mut ctx, "").await.unwrap();
assert!(matches!(out, CommandOutput::Message(_)));
}
#[tokio::test]
async fn focus_returns_message() {
let mut sink = NullSink;
let mut debug = MockDebug;
let mut messages = MockMessages;
let session = MockSession;
let mut agent = crate::NullAgent;
let mut ctx = make_ctx(&mut sink, &mut debug, &mut messages, &session, &mut agent);
let out = FocusCommand.handle(&mut ctx, "").await.unwrap();
assert!(matches!(out, CommandOutput::Message(_)));
}
#[tokio::test]
async fn sidequest_returns_message() {
let mut sink = NullSink;
let mut debug = MockDebug;
let mut messages = MockMessages;
let session = MockSession;
let mut agent = crate::NullAgent;
let mut ctx = make_ctx(&mut sink, &mut debug, &mut messages, &session, &mut agent);
let out = SideQuestCommand.handle(&mut ctx, "").await.unwrap();
assert!(matches!(out, CommandOutput::Message(_)));
}
}