hippox 0.5.6

🦛A reliable, autonomous LLM runtime and skill orchestration engine, Capable of processing natural language and automatically executing OS-native atomic skills, fundamentally enabling the LLM to truly take over the computer.
//! Executor trait and state updater for tasks

use std::fmt::Debug;
use std::future::Future;
use std::pin::Pin;
use std::sync::Arc;
use tokio::sync::RwLock;

use super::types::{TASK_POOL, TaskPool, TaskStatus};
use crate::workflow::WorkflowCallback;

/// Task trait - each task must implement this to be executable
pub trait ExecutableTask: Send + Sync + Debug {
    /// Execute the task with the given state updater and external callback
    fn execute(
        &self,
        state_updater: TaskStateUpdater,
        callback: Option<Arc<dyn WorkflowCallback>>,
    ) -> Pin<Box<dyn Future<Output = ()> + Send + '_>>;

    /// Get the task type identifier
    fn task_type(&self) -> &str;

    /// Get the input for the task
    fn input(&self) -> &str;
}

/// Task state updater - updates internal task pool state
#[derive(Clone)]
pub struct TaskStateUpdater {
    task_id: String,
    task_pool: Arc<RwLock<TaskPool>>,
}

impl TaskStateUpdater {
    pub fn new(task_id: String, task_pool: Arc<RwLock<TaskPool>>) -> Self {
        Self { task_id, task_pool }
    }

    pub fn task_id(&self) -> &str {
        &self.task_id
    }

    /// Check if the task has been cancelled
    pub async fn is_cancelled(&self) -> bool {
        let pool = self.task_pool.read().await;
        if let Some(task) = pool.get_task(&self.task_id) {
            return task.status == TaskStatus::Cancelled;
        }
        false
    }

    /// Check if the task has been paused
    pub async fn is_paused(&self) -> bool {
        let pool = self.task_pool.read().await;
        if let Some(task) = pool.get_task(&self.task_id) {
            return task.status == TaskStatus::Paused;
        }
        false
    }

    /// Check if the task is still running (not cancelled or paused)
    pub async fn is_running(&self) -> bool {
        let pool = self.task_pool.read().await;
        if let Some(task) = pool.get_task(&self.task_id) {
            return task.status == TaskStatus::Running;
        }
        false
    }

    /// Get current task status
    pub async fn get_status(&self) -> Option<TaskStatus> {
        let pool = self.task_pool.read().await;
        pool.get_task(&self.task_id).map(|t| t.status)
    }

    /// Save checkpoint data for resume
    pub async fn save_checkpoint(&self, checkpoint_data: &str) -> bool {
        let mut pool = self.task_pool.write().await;
        if let Some(task) = pool.get_task_mut(&self.task_id) {
            task.resume_data = Some(checkpoint_data.to_string());
            return true;
        }
        false
    }

    /// Get checkpoint data for resume
    pub async fn get_checkpoint(&self) -> Option<String> {
        let pool = self.task_pool.read().await;
        pool.get_task(&self.task_id)
            .and_then(|t| t.resume_data.clone())
    }

    /// Update step start in internal state
    pub async fn update_step_start(&self, step_name: &str, step_index: usize) {
        let mut pool = self.task_pool.write().await;
        if let Some(task) = pool.get_task_mut(&self.task_id) {
            if step_index < task.steps.len() {
                task.steps[step_index].started();
            } else {
                let idx = task.add_step(step_name.to_string());
                if idx == step_index {
                    if let Some(step) = task.get_step_mut(step_index) {
                        step.started();
                    }
                }
            }
        }
    }

    /// Update step success in internal state
    pub async fn update_step_success(&self, step_name: &str, step_index: usize, output: &str) {
        let mut pool = self.task_pool.write().await;
        if let Some(task) = pool.get_task_mut(&self.task_id) {
            if step_index < task.steps.len() {
                task.steps[step_index].completed(output.to_string());
            } else {
                let idx = task.add_step(step_name.to_string());
                if idx == step_index {
                    if let Some(step) = task.get_step_mut(step_index) {
                        step.completed(output.to_string());
                    }
                }
            }
        }
    }

    /// Update step failure in internal state
    pub async fn update_step_failure(&self, step_name: &str, step_index: usize, error: &str) {
        let mut pool = self.task_pool.write().await;
        if let Some(task) = pool.get_task_mut(&self.task_id) {
            if step_index < task.steps.len() {
                task.steps[step_index].failed(error.to_string());
            } else {
                let idx = task.add_step(step_name.to_string());
                if idx == step_index {
                    if let Some(step) = task.get_step_mut(step_index) {
                        step.failed(error.to_string());
                    }
                }
            }
        }
    }

    /// Update workflow completion in internal state
    pub async fn update_workflow_complete(&self, final_output: &str) {
        let mut pool = self.task_pool.write().await;
        if let Some(task) = pool.get_task_mut(&self.task_id) {
            task.completed(final_output.to_string());
            pool.complete_task(&self.task_id);
        }
    }

    /// Update workflow failure in internal state
    pub async fn update_workflow_failed(&self, error: &str) {
        let mut pool = self.task_pool.write().await;
        if let Some(task) = pool.get_task_mut(&self.task_id) {
            task.failed(error.to_string());
            pool.complete_task(&self.task_id);
        }
    }
}