pmat 3.15.0

PMAT - Zero-config AI context generation and code quality toolkit (CLI, MCP, HTTP)
#![cfg_attr(coverage_nightly, coverage(off))]
// Agent system with Actix actors
pub mod analyzer_actor;
pub mod messages;
pub mod messaging;
pub mod orchestrator_actor;
pub mod registry;
pub mod supervisor;
pub mod transformer_actor;
pub mod validator_actor;

#[cfg(test)]
mod registry_tests;

use actix::prelude::*;
use async_trait::async_trait;
use serde::{Deserialize, Serialize};
use uuid::Uuid;

pub type AgentId = Uuid;

#[derive(Debug, Clone, Serialize, Deserialize)]
/// Agent capabilities.
pub struct AgentCapabilities {
    pub name: String,
    pub version: String,
    pub capabilities: Vec<String>,
    pub max_concurrent_tasks: usize,
}

#[async_trait]
/// Trait defining Pmat agent behavior.
pub trait PmatAgent: Send + Sync + 'static {
    type Config: Send + Sync;
    type State: AgentState;
    type Message: AgentMessage;

    fn capabilities(&self) -> AgentCapabilities;
    async fn initialize(config: Self::Config) -> Result<Self, AgentError>
    where
        Self: Sized;
    async fn process(&mut self, msg: Self::Message) -> Result<AgentResponse, AgentError>;
    async fn checkpoint(&self) -> Result<Self::State, AgentError>;
}

/// Trait defining Agent state behavior.
pub trait AgentState: Send + Sync + Clone + Serialize + for<'de> Deserialize<'de> {
    fn last_event_id(&self) -> u64;
    fn events_since_snapshot(&self) -> usize;
    fn time_since_snapshot(&self) -> std::time::Duration;
}

/// Trait defining Agent message behavior.
pub trait AgentMessage: Send + Sync + Message<Result = Result<AgentResponse, AgentError>> {
    fn priority(&self) -> Priority;
}

#[derive(Debug, Clone, Serialize, Deserialize)]
/// Agent response.
pub enum AgentResponse {
    Success(serde_json::Value),
    Analyzed(crate::modules::analyzer::Metrics),
    Transformed(crate::modules::transformer::TransformResult),
    Validated(crate::modules::validator::ValidationResult),
    Error(String),
}

#[derive(Debug, thiserror::Error)]
/// Error variants for agent operations.
pub enum AgentError {
    #[error("Agent not found: {0}")]
    NotFound(AgentId),
    #[error("Agent initialization failed: {0}")]
    InitializationFailed(String),
    #[error("Agent processing failed: {0}")]
    ProcessingFailed(String),
    #[error("Agent communication failed: {0}")]
    CommunicationFailed(String),
    #[error("Agent timeout: {0}")]
    Timeout(String),
}

#[derive(Debug, Clone, Copy, PartialEq, Eq, PartialOrd, Ord, Serialize, Deserialize)]
/// Priority level for priority.
pub enum Priority {
    Critical = 0,
    High = 1,
    Normal = 2,
    Low = 3,
}

#[derive(Debug, Clone, PartialEq)]
/// Agent class.
pub enum AgentClass {
    Analyzer,
    Transformer,
    Validator,
    Orchestrator,
    Monitor,
}

#[derive(Debug, Clone, PartialEq)]
/// Agent spec.
pub struct AgentSpec {
    pub id: AgentId,
    pub class: AgentClass,
    pub config: serde_json::Value,
}

// Removed AgentHandle for now - will implement properly with specific actor types

// System initialization
// Note: actix::System::new() returns SystemRunner, not System
#[provable_contracts_macros::contract("pmat-core.yaml", equation = "check_compliance")]
/// Init agent system.
pub fn init_agent_system() {
    // actix::System::new() returns SystemRunner which auto-runs
}

#[cfg(test)]
mod tests {
    use super::*;

    #[test]
    fn test_agent_id_generation() {
        let id1 = Uuid::new_v4();
        let id2 = Uuid::new_v4();
        assert_ne!(id1, id2);
    }

    #[test]
    fn test_priority_ordering() {
        assert!(Priority::Critical < Priority::High);
        assert!(Priority::High < Priority::Normal);
        assert!(Priority::Normal < Priority::Low);
    }
}