use crate::episode::Episode;
use super::constants::*;
pub struct EfficiencyCalculator {
pub duration_weight: f32,
pub step_count_weight: f32,
}
impl EfficiencyCalculator {
pub fn new(duration_weight: f32, step_count_weight: f32) -> Self {
Self {
duration_weight,
step_count_weight,
}
}
pub fn calculate(&self, episode: &Episode) -> f32 {
let duration_score = self.calculate_duration_efficiency(episode);
let step_count_score = self.calculate_step_count_efficiency(episode);
let combined =
(duration_score * self.duration_weight) + (step_count_score * self.step_count_weight);
combined.clamp(MIN_EFFICIENCY_MULTIPLIER, MAX_EFFICIENCY_MULTIPLIER)
}
fn calculate_duration_efficiency(&self, episode: &Episode) -> f32 {
if let Some(duration) = episode.duration() {
let duration_secs = duration.num_seconds() as f32;
if duration_secs <= 0.0 {
return MAX_EFFICIENCY_MULTIPLIER;
}
let ratio = duration_secs / EFFICIENT_DURATION_SECS;
let score = (-ratio / 2.0).exp();
MIN_EFFICIENCY_MULTIPLIER
+ (score * (MAX_EFFICIENCY_MULTIPLIER - MIN_EFFICIENCY_MULTIPLIER))
} else {
1.0 }
}
fn calculate_step_count_efficiency(&self, episode: &Episode) -> f32 {
let step_count = episode.steps.len();
if step_count == 0 {
return MIN_EFFICIENCY_MULTIPLIER;
}
let ratio = step_count as f32 / EFFICIENT_STEP_COUNT as f32;
let score = (-ratio / 2.0).exp();
MIN_EFFICIENCY_MULTIPLIER
+ (score * (MAX_EFFICIENCY_MULTIPLIER - MIN_EFFICIENCY_MULTIPLIER))
}
}
#[cfg(test)]
mod tests {
use super::*;
use crate::episode::ExecutionStep;
use crate::types::{ComplexityLevel, ExecutionResult, TaskContext, TaskOutcome, TaskType};
fn create_test_episode() -> Episode {
let context = TaskContext {
language: Some("rust".to_string()),
framework: None,
complexity: ComplexityLevel::Simple,
domain: "testing".to_string(),
tags: vec![],
};
Episode::new("Test task".to_string(), context, TaskType::Testing)
}
#[test]
fn test_efficiency_fast_execution() {
let calculator = EfficiencyCalculator::new(0.5, 0.5);
let mut episode = create_test_episode();
for i in 0..3 {
let mut step = ExecutionStep::new(i + 1, format!("tool_{}", i), "Action".to_string());
step.result = Some(ExecutionResult::Success {
output: "OK".to_string(),
});
episode.add_step(step);
}
episode.complete(TaskOutcome::Success {
verdict: "Quick".to_string(),
artifacts: vec![],
});
let efficiency = calculator.calculate(&episode);
assert!(efficiency > 1.0);
}
#[test]
fn test_efficiency_slow_execution() {
let calculator = EfficiencyCalculator::new(0.5, 0.5);
let mut episode = create_test_episode();
episode.start_time = chrono::Utc::now() - chrono::Duration::minutes(5);
for i in 0..50 {
let mut step = ExecutionStep::new(i + 1, format!("tool_{}", i), "Action".to_string());
step.result = Some(ExecutionResult::Success {
output: "OK".to_string(),
});
episode.add_step(step);
}
episode.complete(TaskOutcome::Success {
verdict: "Slow".to_string(),
artifacts: vec![],
});
let efficiency = calculator.calculate(&episode);
assert!(efficiency < 1.0);
}
}