use chrono::Utc;
use oxios_ouroboros::Seed;
use serde::{Deserialize, Serialize};
use uuid::Uuid;
#[derive(Debug, Clone, Copy, PartialEq, Eq, Serialize, Deserialize)]
pub enum OxiosAgentGroupStatus {
Pending,
Running,
Completed,
Failed,
}
#[derive(Debug, Clone, Serialize, Deserialize)]
pub struct OxiosGroupAgent {
pub id: Uuid,
pub seed: Seed,
pub status: OxiosAgentGroupStatus,
pub result: Option<String>,
}
#[derive(Debug, Clone, Serialize, Deserialize)]
pub struct OxiosAgentGroup {
pub id: Uuid,
pub parent_seed_id: Uuid,
pub agents: Vec<OxiosGroupAgent>,
}
impl OxiosAgentGroup {
pub fn new(parent_seed: &Seed, subtask_descriptions: Vec<String>) -> Self {
let agents = subtask_descriptions
.into_iter()
.map(|desc| {
let child_seed = Seed {
id: Uuid::new_v4(),
goal: desc,
constraints: parent_seed.constraints.clone(),
acceptance_criteria: vec!["Task completes successfully".into()],
ontology: parent_seed.ontology.clone(),
created_at: Utc::now(),
generation: parent_seed.generation + 1,
parent_seed_id: Some(parent_seed.id),
cspace_hint: parent_seed.cspace_hint.clone(),
original_request: parent_seed.original_request.clone(),
};
OxiosGroupAgent {
id: child_seed.id,
seed: child_seed,
status: OxiosAgentGroupStatus::Pending,
result: None,
}
})
.collect();
Self {
id: Uuid::new_v4(),
parent_seed_id: parent_seed.id,
agents,
}
}
pub fn pending_agents(&self) -> Vec<&OxiosGroupAgent> {
self.agents
.iter()
.filter(|a| a.status == OxiosAgentGroupStatus::Pending)
.collect()
}
pub fn completed_agents(&self) -> Vec<&OxiosGroupAgent> {
self.agents
.iter()
.filter(|a| a.status == OxiosAgentGroupStatus::Completed)
.collect()
}
pub fn failed_agents(&self) -> Vec<&OxiosGroupAgent> {
self.agents
.iter()
.filter(|a| a.status == OxiosAgentGroupStatus::Failed)
.collect()
}
pub fn all_completed(&self) -> bool {
self.agents
.iter()
.all(|a| a.status == OxiosAgentGroupStatus::Completed)
}
pub fn any_failed(&self) -> bool {
self.agents
.iter()
.any(|a| a.status == OxiosAgentGroupStatus::Failed)
}
pub fn completion_pct(&self) -> f64 {
if self.agents.is_empty() {
return 0.0;
}
let completed = self
.agents
.iter()
.filter(|a| a.status == OxiosAgentGroupStatus::Completed)
.count();
completed as f64 / self.agents.len() as f64
}
pub fn combined_results(&self) -> String {
self.completed_agents()
.iter()
.filter_map(|a| a.result.as_ref())
.map(|r| r.as_str())
.collect::<Vec<_>>()
.join("\n\n")
}
}
#[cfg(test)]
mod tests {
use super::*;
#[test]
fn test_group_new_splits_seed() {
let parent = Seed {
id: Uuid::new_v4(),
goal: "Test goal".into(),
constraints: vec!["constraint1".into()],
acceptance_criteria: vec!["Criterion".into()],
ontology: vec![],
created_at: Utc::now(),
generation: 0,
parent_seed_id: None,
cspace_hint: None,
original_request: String::new(),
};
let descriptions = vec!["subtask 1".into(), "subtask 2".into()];
let group = OxiosAgentGroup::new(&parent, descriptions);
assert_eq!(group.agents.len(), 2);
assert!(group.pending_agents().len() == 2);
assert!(!group.all_completed());
assert_eq!(group.parent_seed_id, parent.id);
}
#[test]
fn test_completion_pct_empty_group() {
let parent = Seed {
id: Uuid::new_v4(),
goal: "Test".into(),
constraints: vec![],
acceptance_criteria: vec![],
ontology: vec![],
created_at: Utc::now(),
generation: 0,
parent_seed_id: None,
cspace_hint: None,
original_request: String::new(),
};
let group = OxiosAgentGroup::new(&parent, vec![]);
assert!((group.completion_pct() - 0.0).abs() < f64::EPSILON);
}
}