lha 1.0.6

Long-Horizon Agent command-line package that installs the lha binary.
Documentation
use crate::product::agent::codex::Codex;
use crate::product::agent::error::Result as CodexResult;
use crate::product::agent::protocol::Event;
use crate::product::agent::protocol::Op;
use crate::product::agent::protocol::Submission;
use crate::product::protocol::config_types::IdentityKind;
use crate::product::protocol::config_types::Personality;
use crate::product::protocol::openai_models::ReasoningEffort;
use crate::product::protocol::protocol::AskForApproval;
use crate::product::protocol::protocol::SandboxPolicy;
use crate::product::protocol::protocol::SessionSource;
#[cfg(any(test, feature = "test-support"))]
use crate::product::protocol::workflow::WorkflowDefinition;
use lha_core::SessionSnapshot;
use lha_core::SessionStatus;
use lha_llm::RuntimeEndpoint;
use lha_llm::RuntimeMetadata;
use std::collections::hash_map::DefaultHasher;
use std::hash::Hash;
use std::hash::Hasher;
use std::path::PathBuf;

use crate::product::agent::state_db::StateDbHandle;

#[derive(Clone, Debug)]
pub struct ThreadConfigSnapshot {
    pub model: String,
    pub identity_kind: IdentityKind,
    pub model_provider_id: String,
    pub approval_policy: AskForApproval,
    pub sandbox_policy: SandboxPolicy,
    pub cwd: PathBuf,
    pub reasoning_effort: Option<ReasoningEffort>,
    pub personality: Option<Personality>,
    pub session_source: SessionSource,
}

pub struct CodexThread {
    codex: Codex,
    rollout_path: Option<PathBuf>,
}

/// Conduit for the bidirectional stream of messages that compose a thread
/// (formerly called a conversation) in LHA.
impl CodexThread {
    pub(crate) fn new(codex: Codex, rollout_path: Option<PathBuf>) -> Self {
        Self {
            codex,
            rollout_path,
        }
    }

    pub async fn submit(&self, op: Op) -> CodexResult<String> {
        self.codex.submit(op).await
    }

    /// Use sparingly: this is intended to be removed soon.
    pub async fn submit_with_id(&self, sub: Submission) -> CodexResult<()> {
        self.codex.submit_with_id(sub).await
    }

    pub async fn next_event(&self) -> CodexResult<Event> {
        self.codex.next_event().await
    }

    pub async fn session_status(&self) -> SessionStatus {
        self.codex.session_status().await
    }

    pub async fn wait_for_shutdown_complete(&self) -> CodexResult<()> {
        self.codex.wait_for_shutdown_complete().await
    }

    pub fn rollout_path(&self) -> Option<PathBuf> {
        self.rollout_path.clone()
    }

    pub fn state_db(&self) -> Option<StateDbHandle> {
        self.codex.state_db()
    }

    pub async fn config_snapshot(&self) -> ThreadConfigSnapshot {
        self.codex.thread_config_snapshot().await
    }

    pub async fn core_snapshot(&self) -> SessionSnapshot {
        let config = self.config_snapshot().await;
        let history = self.codex.session.clone_history().await;
        let mut hasher = DefaultHasher::new();
        self.codex.session.conversation_id.hash(&mut hasher);
        let status = self.session_status().await;

        SessionSnapshot {
            session_id: hasher.finish(),
            status,
            conversation: history.raw_items().to_vec(),
            steering_queue: Vec::new(),
            follow_up_queue: Vec::new(),
            runtime: RuntimeMetadata {
                endpoint_name: config.model_provider_id,
                model: config.model,
            },
            active_turn: None,
        }
    }

    pub async fn flush_rollout(&self) {
        self.codex.session.flush_rollout().await;
    }

    #[cfg(any(test, feature = "test-support"))]
    pub async fn set_workflow_for_testing(
        &self,
        definition: WorkflowDefinition,
    ) -> std::result::Result<(), Vec<crate::product::protocol::workflow::WorkflowValidationError>>
    {
        self.codex
            .session
            .set_workflow_for_testing(definition)
            .await
    }

    pub async fn update_model_provider(&self, provider: RuntimeEndpoint) {
        self.codex.update_model_provider(provider).await;
    }

    pub async fn update_tui_buddy(&self, buddy: crate::product::agent::config::types::TuiBuddy) {
        self.codex.update_tui_buddy(buddy).await;
    }

    pub async fn switch_provider_and_model(
        &self,
        model_provider_id: String,
        provider: RuntimeEndpoint,
        model: String,
    ) {
        self.codex
            .switch_provider_and_model(model_provider_id, provider, model)
            .await;
    }
}