/// inference/graph.rs, BAYES STAR, (c) coppola.ai 2024 use std::{ collections::{HashMap, HashSet}, error::Error, rc::Rc, }; use env_logger::init; use serde::{Deserialize, Serialize}; use crate::{ common::{graph::InferenceGraph, redis::RedisManager}, model::{ choose::{compute_search_predicates, extract_backimplications_from_proposition}, objects::{GroupRoleMap, PredicateFactor, Proposition, PropositionGroup}, }, print_green, }; use super::table::{GenericNodeType, PropositionNode}; #[derive(Serialize, Deserialize, Debug, Clone)] pub struct PropositionFactor { pub premise: PropositionGroup, pub conclusion: Proposition, pub inference: PredicateFactor, } impl PropositionFactor { pub fn debug_string(&self) -> String { format!( "{} -> {}", self.premise.hash_string(), self.conclusion.hash_string() ) } } pub struct PropositionGraph { pub predicate_graph: Rc, pub single_forward: HashMap>, pub single_backward: HashMap>, pub group_forward: HashMap>, pub inference_used: HashMap<(PropositionGroup, Proposition), PredicateFactor>, pub roots: HashSet, pub all_nodes: HashSet, } fn initialize_visit_single( graph: &mut PropositionGraph, single: &Proposition, ) -> Result<(), Box> { // Green for starting a new operation info!( "\x1b[32mInitializing visit for proposition: {:?}\x1b[0m", single.hash_string() ); graph .all_nodes .insert(PropositionNode::from_single(single)); let inference_factors = extract_backimplications_from_proposition(&graph.predicate_graph, single)?; // Yellow for showing counts or lengths info!( "\x1b[33mInference factors count: {}\x1b[0m", inference_factors.len() ); if inference_factors.is_empty() { // Blue for specific condition-related messages info!("\x1b[34mNo inference factors. Adding to roots.\x1b[0m"); graph.roots.insert(single.clone()); } else { for inference_factor in &inference_factors { // Cyan for loop iteration info!( "\x1b[36mProcessing inference factor: {:?}\x1b[0m", inference_factor.debug_string() ); let inference_used_key = (inference_factor.premise.clone(), inference_factor.conclusion.clone()); graph.inference_used.insert(inference_used_key, inference_factor.inference.clone()); info!( "\x1b[36mUpdating single_backward for conclusion: {:?}\x1b[0m", inference_factor.conclusion.hash_string() ); graph .single_backward .entry(inference_factor.conclusion.clone()) .or_insert_with(HashSet::new) .insert(inference_factor.premise.clone()); info!( "\x1b[36mUpdating group_forward for premise: {:?}\x1b[0m", inference_factor.premise.hash_string() ); graph .group_forward .entry(inference_factor.premise.clone()) .or_insert_with(HashSet::new) .insert(inference_factor.conclusion.clone()); graph .all_nodes .insert(PropositionNode::from_group(&inference_factor.premise)); for term in &inference_factor.premise.terms { info!("\x1b[35mProcessing term: {:?}\x1b[0m", term.hash_string()); graph .single_forward .entry(term.clone()) .or_insert_with(HashSet::new) .insert(inference_factor.premise.clone()); info!( "\x1b[35mRecursively initializing visit for term: {:?}\x1b[0m", term.hash_string() ); initialize_visit_single(graph, term)?; } } } // Green for completion messages info!( "\x1b[32mFinished initializing visit for proposition: {:?}\x1b[0m", single.hash_string() ); Ok(()) } impl PropositionGraph { pub fn new_shared( predicate_graph: Rc, target: &Proposition, ) -> Result, Box> { let mut graph = PropositionGraph { predicate_graph, single_forward: HashMap::new(), single_backward: HashMap::new(), group_forward: HashMap::new(), inference_used: HashMap::new(), roots: HashSet::new(), all_nodes: HashSet::new(), }; initialize_visit_single(&mut graph, target)?; Ok(Rc::new(graph)) } pub fn get_inference_used(&self, premise:&PropositionGroup, conclusion: &Proposition) -> PredicateFactor { let key = (premise.clone(), conclusion.clone()); self.inference_used .get(&key).unwrap().clone() } pub fn get_single_forward(&self, key: &Proposition) -> HashSet { self.single_forward .get(key) .cloned() .unwrap_or_else(HashSet::new) } pub fn get_single_backward(&self, key: &Proposition) -> HashSet { self.single_backward .get(key) .cloned() .unwrap_or_else(HashSet::new) } pub fn get_group_forward(&self, key: &PropositionGroup) -> HashSet { self.group_forward.get(key).unwrap().clone() } pub fn get_group_backward(&self, key: &PropositionGroup) -> Vec { key.terms.clone() } pub fn get_all_backward(&self, node: &PropositionNode) -> Vec { print_green!("get_all_backward called for node: {:?}", node.debug_string()); let mut r = vec![]; match &node.node { GenericNodeType::Single(proposition) => { print_green!("Processing as Single: {:?}", proposition.debug_string()); let initial = self.get_single_backward(proposition); print_green!("Initial singles: {}", initial.len()); for group in &initial { print_green!("Adding group from initial singles: {:?}", group.debug_string()); r.push(PropositionNode::from_group(group)); } } GenericNodeType::Group(group) => { print_green!("Processing as Group: {:?}", group.debug_string()); let initial = self.get_group_backward(group); print_green!("Initial groups: {}", initial.len()); for single in &initial { print_green!("Adding single from initial groups: {:?}", single.debug_string()); r.push(PropositionNode::from_single(single)); } } } info!("Resulting vector: {:?}", r); r } pub fn get_all_forward(&self, node: &PropositionNode) -> Vec { print_green!("get_all_backward called for node: {:?}", node.debug_string()); let mut r = vec![]; match &node.node { GenericNodeType::Single(proposition) => { print_green!("Processing as Single: {:?}", proposition.debug_string()); let initial = self.get_single_forward(proposition); print_green!("Initial singles: {}", initial.len()); for group in &initial { print_green!("Adding group from initial singles: {:?}", group.debug_string()); r.push(PropositionNode::from_group(group)); } } GenericNodeType::Group(group) => { print_green!("Processing as Group: {:?}", group.debug_string()); let initial = self.get_group_forward(group); print_green!("Initial groups: {}", initial.len()); for single in &initial { print_green!("Adding single from initial groups: {:?}", single.debug_string()); r.push(PropositionNode::from_single(single)); } } } info!("Resulting vector: {:?}", r); r } pub fn get_roots(&self) -> HashSet { self.roots.clone() } }