ironclaw 0.4.0

Secure personal AI assistant that protects your data and expands its capabilities on the fly
Documentation
//! Cost, time, and value estimation with continuous learning.
//!
//! Estimates are based on:
//! - Historical data from similar jobs
//! - Tool cost/time characteristics
//! - Statistical models that improve over time

mod cost;
mod learner;
mod time;
mod value;

pub use cost::CostEstimator;
pub use learner::{EstimationLearner, LearningModel};
pub use time::TimeEstimator;
pub use value::ValueEstimator;

use rust_decimal::Decimal;
use std::time::Duration;

/// Combined estimation for a job.
#[derive(Debug, Clone)]
pub struct JobEstimate {
    /// Estimated cost to complete the job.
    pub cost: Decimal,
    /// Estimated time to complete.
    pub duration: Duration,
    /// Estimated value/earnings.
    pub value: Decimal,
    /// Confidence in the estimate (0-1).
    pub confidence: f64,
    /// Breakdown by tool.
    pub tool_breakdown: Vec<ToolEstimate>,
}

/// Estimate for a single tool usage.
#[derive(Debug, Clone)]
pub struct ToolEstimate {
    pub tool_name: String,
    pub cost: Decimal,
    pub duration: Duration,
    pub confidence: f64,
}

/// Combined estimator.
pub struct Estimator {
    cost: CostEstimator,
    time: TimeEstimator,
    value: ValueEstimator,
    learner: EstimationLearner,
}

impl Estimator {
    /// Create a new estimator.
    pub fn new() -> Self {
        Self {
            cost: CostEstimator::new(),
            time: TimeEstimator::new(),
            value: ValueEstimator::new(),
            learner: EstimationLearner::new(),
        }
    }

    /// Estimate for a job.
    pub fn estimate_job(
        &self,
        description: &str,
        category: Option<&str>,
        tools: &[String],
    ) -> JobEstimate {
        let tool_estimates: Vec<ToolEstimate> = tools
            .iter()
            .map(|t| ToolEstimate {
                tool_name: t.clone(),
                cost: self.cost.estimate_tool(t),
                duration: self.time.estimate_tool(t),
                confidence: 0.7, // Default confidence
            })
            .collect();

        let total_cost: Decimal = tool_estimates.iter().map(|e| e.cost).sum();
        let total_duration: Duration = tool_estimates.iter().map(|e| e.duration).sum();

        // Apply learned adjustments
        let (adjusted_cost, adjusted_time) =
            self.learner
                .adjust(category.unwrap_or("general"), total_cost, total_duration);

        let value = self.value.estimate(description, adjusted_cost);
        let confidence = self.learner.confidence(category.unwrap_or("general"));

        JobEstimate {
            cost: adjusted_cost,
            duration: adjusted_time,
            value,
            confidence,
            tool_breakdown: tool_estimates,
        }
    }

    /// Record actual results for learning.
    pub fn record_actuals(
        &mut self,
        category: &str,
        estimated_cost: Decimal,
        actual_cost: Decimal,
        estimated_time: Duration,
        actual_time: Duration,
    ) {
        self.learner.record(
            category,
            estimated_cost,
            actual_cost,
            estimated_time,
            actual_time,
        );
    }

    /// Get the cost estimator.
    pub fn cost(&self) -> &CostEstimator {
        &self.cost
    }

    /// Get the time estimator.
    pub fn time(&self) -> &TimeEstimator {
        &self.time
    }

    /// Get the value estimator.
    pub fn value(&self) -> &ValueEstimator {
        &self.value
    }
}

impl Default for Estimator {
    fn default() -> Self {
        Self::new()
    }
}