butterfly-bot 0.2.2

Butterfly Bot is an opinionated personal-ops AI assistant built for people who want results, not setup overhead.
Documentation
use async_trait::async_trait;
use tokio::sync::Mutex;

use crate::error::Result;
use crate::interfaces::brain::{BrainContext, BrainEvent, BrainPlugin};

#[derive(Debug, Clone)]
pub struct SelfCritiqueResult {
    pub passed: bool,
    pub severity: String,
    pub concerns: Vec<String>,
}

pub struct MandatorySelfCritiqueBrain {
    last_result: Mutex<Option<SelfCritiqueResult>>,
}

impl MandatorySelfCritiqueBrain {
    pub fn new() -> Self {
        Self {
            last_result: Mutex::new(None),
        }
    }

    pub async fn last_result(&self) -> Option<SelfCritiqueResult> {
        let guard = self.last_result.lock().await;
        guard.clone()
    }

    fn critique(text: &str) -> SelfCritiqueResult {
        let lower = text.to_lowercase();
        let mut concerns = Vec::new();
        if ["harm", "kill", "suicide", "violence"]
            .iter()
            .any(|kw| lower.contains(kw))
        {
            concerns.push("potential harm".to_string());
        }
        if ["vote", "election", "party"]
            .iter()
            .any(|kw| lower.contains(kw))
        {
            concerns.push("political bias".to_string());
        }
        let passed = concerns.is_empty();
        let severity = if passed { "none" } else { "moderate" };
        SelfCritiqueResult {
            passed,
            severity: severity.to_string(),
            concerns,
        }
    }
}

#[async_trait]
impl BrainPlugin for MandatorySelfCritiqueBrain {
    fn name(&self) -> &str {
        "mandatory_self_critique"
    }

    fn description(&self) -> &str {
        "Runs a lightweight self-critique on assistant responses"
    }

    async fn on_event(&self, event: BrainEvent, _ctx: &BrainContext) -> Result<()> {
        if let BrainEvent::AssistantResponse { text, .. } = event {
            let result = Self::critique(&text);
            let mut guard = self.last_result.lock().await;
            *guard = Some(result);
        }
        Ok(())
    }
}