rust_rule_engine/rete/
agenda.rs

1//! Advanced Agenda System (Drools-style)
2//!
3//! This module implements advanced agenda features similar to Drools:
4//! - Activation Groups: Only one rule in a group can fire
5//! - Agenda Groups: Sequential execution of rule groups
6//! - Ruleflow Groups: Workflow-based execution
7//! - Auto-focus: Automatic agenda group switching
8//! - Lock-on-active: Prevent re-activation during rule firing
9//! - Conflict Resolution Strategies: Multiple ordering strategies
10
11use std::collections::{HashMap, HashSet, BinaryHeap};
12use std::cmp::Ordering;
13
14/// Conflict Resolution Strategy
15///
16/// Determines how conflicting activations are ordered in the agenda.
17/// Similar to CLIPS and Drools conflict resolution strategies.
18#[derive(Debug, Clone, Copy, PartialEq, Eq)]
19pub enum ConflictResolutionStrategy {
20    /// Salience-based ordering (default) - Higher salience fires first
21    Salience,
22
23    /// LEX (Recency) - Most recently inserted facts fire first
24    /// Sorts by the timestamp of the most recent fact used in the activation
25    LEX,
26
27    /// MEA (Recency + Specificity) - LEX + more specific rules first
28    /// Combines recency with condition count (more conditions = more specific)
29    MEA,
30
31    /// Depth-first - Fire rule immediately after insertion
32    /// Re-evaluates agenda after each rule fires
33    Depth,
34
35    /// Breadth-first - Collect all activations before firing (default)
36    /// Fires all activations in current cycle before re-evaluating
37    Breadth,
38
39    /// Simplicity - Rules with fewer conditions fire first
40    /// Simpler rules are prioritized
41    Simplicity,
42
43    /// Complexity - Rules with more conditions fire first
44    /// More complex/specific rules are prioritized
45    Complexity,
46
47    /// Random - Random ordering
48    /// Useful for testing non-deterministic behavior
49    Random,
50}
51
52/// Activation represents a rule that is ready to fire
53#[derive(Debug, Clone)]
54pub struct Activation {
55    /// Rule name
56    pub rule_name: String,
57    /// Priority/salience (higher fires first)
58    pub salience: i32,
59    /// Activation group (only one rule in group can fire)
60    pub activation_group: Option<String>,
61    /// Agenda group (for sequential execution)
62    pub agenda_group: String,
63    /// Ruleflow group (for workflow execution)
64    pub ruleflow_group: Option<String>,
65    /// No-loop flag
66    pub no_loop: bool,
67    /// Lock-on-active flag
68    pub lock_on_active: bool,
69    /// Auto-focus flag
70    pub auto_focus: bool,
71    /// Creation timestamp (for conflict resolution)
72    pub created_at: std::time::Instant,
73    /// Number of conditions in the rule (for complexity/simplicity strategies)
74    pub condition_count: usize,
75    /// Matched fact handle (which fact triggered this activation)
76    pub matched_fact_handle: Option<super::FactHandle>,
77    /// Internal ID
78    id: usize,
79}
80
81impl Activation {
82    /// Create a new activation
83    pub fn new(rule_name: String, salience: i32) -> Self {
84        Self {
85            rule_name,
86            salience,
87            activation_group: None,
88            agenda_group: "MAIN".to_string(),
89            ruleflow_group: None,
90            no_loop: true,
91            lock_on_active: false,
92            auto_focus: false,
93            created_at: std::time::Instant::now(),
94            condition_count: 1, // Default to 1
95            matched_fact_handle: None,
96            id: 0,
97        }
98    }
99
100    /// Builder: Set matched fact handle
101    pub fn with_matched_fact(mut self, handle: super::FactHandle) -> Self {
102        self.matched_fact_handle = Some(handle);
103        self
104    }
105
106    /// Builder: Set condition count
107    pub fn with_condition_count(mut self, count: usize) -> Self {
108        self.condition_count = count;
109        self
110    }
111
112    /// Builder: Set activation group
113    pub fn with_activation_group(mut self, group: String) -> Self {
114        self.activation_group = Some(group);
115        self
116    }
117
118    /// Builder: Set agenda group
119    pub fn with_agenda_group(mut self, group: String) -> Self {
120        self.agenda_group = group;
121        self
122    }
123
124    /// Builder: Set ruleflow group
125    pub fn with_ruleflow_group(mut self, group: String) -> Self {
126        self.ruleflow_group = Some(group);
127        self
128    }
129
130    /// Builder: Set no-loop
131    pub fn with_no_loop(mut self, no_loop: bool) -> Self {
132        self.no_loop = no_loop;
133        self
134    }
135
136    /// Builder: Set lock-on-active
137    pub fn with_lock_on_active(mut self, lock: bool) -> Self {
138        self.lock_on_active = lock;
139        self
140    }
141
142    /// Builder: Set auto-focus
143    pub fn with_auto_focus(mut self, auto_focus: bool) -> Self {
144        self.auto_focus = auto_focus;
145        self
146    }
147}
148
149// Implement ordering for priority queue (higher salience first)
150impl PartialEq for Activation {
151    fn eq(&self, other: &Self) -> bool {
152        self.id == other.id
153    }
154}
155
156impl Eq for Activation {}
157
158impl PartialOrd for Activation {
159    fn partial_cmp(&self, other: &Self) -> Option<Ordering> {
160        Some(self.cmp(other))
161    }
162}
163
164impl Ord for Activation {
165    fn cmp(&self, other: &Self) -> Ordering {
166        // First compare by salience (higher is better)
167        match self.salience.cmp(&other.salience) {
168            Ordering::Equal => {
169                // Then by creation time (earlier is better = reverse)
170                other.created_at.cmp(&self.created_at)
171            }
172            other_order => other_order,
173        }
174    }
175}
176
177/// Advanced Agenda (Drools-style)
178pub struct AdvancedAgenda {
179    /// All activations by agenda group
180    activations: HashMap<String, BinaryHeap<Activation>>,
181    /// Current focus (agenda group)
182    focus: String,
183    /// Focus stack
184    focus_stack: Vec<String>,
185    /// Fired rules (for no-loop)
186    fired_rules: HashSet<String>,
187    /// Fired activation groups
188    fired_activation_groups: HashSet<String>,
189    /// Locked groups (lock-on-active)
190    locked_groups: HashSet<String>,
191    /// Active ruleflow groups
192    active_ruleflow_groups: HashSet<String>,
193    /// Next activation ID
194    next_id: usize,
195    /// Conflict resolution strategy
196    strategy: ConflictResolutionStrategy,
197}
198
199impl AdvancedAgenda {
200    /// Create a new agenda with "MAIN" as default focus
201    pub fn new() -> Self {
202        let mut agenda = Self {
203            activations: HashMap::new(),
204            focus: "MAIN".to_string(),
205            focus_stack: Vec::new(),
206            fired_rules: HashSet::new(),
207            fired_activation_groups: HashSet::new(),
208            locked_groups: HashSet::new(),
209            active_ruleflow_groups: HashSet::new(),
210            next_id: 0,
211            strategy: ConflictResolutionStrategy::Salience, // Default strategy
212        };
213        agenda.activations.insert("MAIN".to_string(), BinaryHeap::new());
214        agenda
215    }
216
217    /// Set the conflict resolution strategy
218    pub fn set_strategy(&mut self, strategy: ConflictResolutionStrategy) {
219        self.strategy = strategy;
220        // Re-sort all existing activations with new strategy
221        let current_strategy = self.strategy; // Copy strategy to avoid borrow issues
222        for heap in self.activations.values_mut() {
223            let mut activations: Vec<_> = heap.drain().collect();
224            Self::sort_with_strategy(current_strategy, &mut activations);
225            *heap = activations.into_iter().collect();
226        }
227    }
228
229    /// Get current strategy
230    pub fn strategy(&self) -> ConflictResolutionStrategy {
231        self.strategy
232    }
233
234    /// Sort activations according to given strategy (static method)
235    fn sort_with_strategy(strategy: ConflictResolutionStrategy, activations: &mut [Activation]) {
236        match strategy {
237            ConflictResolutionStrategy::Salience => {
238                // Default: sort by salience (higher first), then by recency
239                activations.sort_by(|a, b| {
240                    match b.salience.cmp(&a.salience) {
241                        Ordering::Equal => b.created_at.cmp(&a.created_at),
242                        other => other,
243                    }
244                });
245            }
246            ConflictResolutionStrategy::LEX => {
247                // Recency: most recent first
248                activations.sort_by(|a, b| b.created_at.cmp(&a.created_at));
249            }
250            ConflictResolutionStrategy::MEA => {
251                // Recency + Specificity: recent first, then more conditions
252                activations.sort_by(|a, b| {
253                    match b.created_at.cmp(&a.created_at) {
254                        Ordering::Equal => b.condition_count.cmp(&a.condition_count),
255                        other => other,
256                    }
257                });
258            }
259            ConflictResolutionStrategy::Depth => {
260                // Depth-first: same as salience (handled in fire loop)
261                activations.sort_by(|a, b| {
262                    match b.salience.cmp(&a.salience) {
263                        Ordering::Equal => b.created_at.cmp(&a.created_at),
264                        other => other,
265                    }
266                });
267            }
268            ConflictResolutionStrategy::Breadth => {
269                // Breadth-first: same as salience (default behavior)
270                activations.sort_by(|a, b| {
271                    match b.salience.cmp(&a.salience) {
272                        Ordering::Equal => b.created_at.cmp(&a.created_at),
273                        other => other,
274                    }
275                });
276            }
277            ConflictResolutionStrategy::Simplicity => {
278                // Simpler rules (fewer conditions) first
279                activations.sort_by(|a, b| {
280                    match a.condition_count.cmp(&b.condition_count) {
281                        Ordering::Equal => b.created_at.cmp(&a.created_at),
282                        other => other,
283                    }
284                });
285            }
286            ConflictResolutionStrategy::Complexity => {
287                // More complex rules (more conditions) first
288                activations.sort_by(|a, b| {
289                    match b.condition_count.cmp(&a.condition_count) {
290                        Ordering::Equal => b.created_at.cmp(&a.created_at),
291                        other => other,
292                    }
293                });
294            }
295            ConflictResolutionStrategy::Random => {
296                // Random ordering using fastrand
297                fastrand::shuffle(activations);
298            }
299        }
300    }
301
302    /// Add an activation to the agenda
303    pub fn add_activation(&mut self, mut activation: Activation) {
304        // Auto-focus: switch to this agenda group if requested
305        if activation.auto_focus && activation.agenda_group != self.focus {
306            self.set_focus(activation.agenda_group.clone());
307        }
308
309        // Check activation group: if group already fired, skip
310        if let Some(ref group) = activation.activation_group {
311            if self.fired_activation_groups.contains(group) {
312                return; // Skip this activation
313            }
314        }
315
316        // Check ruleflow group: if not active, skip
317        if let Some(ref group) = activation.ruleflow_group {
318            if !self.active_ruleflow_groups.contains(group) {
319                return; // Skip this activation
320            }
321        }
322
323        // Assign ID
324        activation.id = self.next_id;
325        self.next_id += 1;
326
327        // Add to appropriate agenda group
328        self.activations
329            .entry(activation.agenda_group.clone())
330            .or_insert_with(BinaryHeap::new)
331            .push(activation);
332    }
333
334    /// Get the next activation to fire (from current focus)
335    pub fn get_next_activation(&mut self) -> Option<Activation> {
336        loop {
337            // Try to get from current focus
338            if let Some(heap) = self.activations.get_mut(&self.focus) {
339                while let Some(activation) = heap.pop() {
340                    // Check no-loop
341                    if activation.no_loop && self.fired_rules.contains(&activation.rule_name) {
342                        continue;
343                    }
344
345                    // Check lock-on-active
346                    if activation.lock_on_active && self.locked_groups.contains(&activation.agenda_group) {
347                        continue;
348                    }
349
350                    // Check activation group
351                    if let Some(ref group) = activation.activation_group {
352                        if self.fired_activation_groups.contains(group) {
353                            continue;
354                        }
355                    }
356
357                    return Some(activation);
358                }
359            }
360
361            // No more activations in current focus, try to pop focus stack
362            if let Some(prev_focus) = self.focus_stack.pop() {
363                self.focus = prev_focus;
364            } else {
365                return None; // Agenda is empty
366            }
367        }
368    }
369
370    /// Mark a rule as fired
371    pub fn mark_rule_fired(&mut self, activation: &Activation) {
372        self.fired_rules.insert(activation.rule_name.clone());
373
374        // If has activation group, mark group as fired (no other rules in group can fire)
375        if let Some(ref group) = activation.activation_group {
376            self.fired_activation_groups.insert(group.clone());
377        }
378
379        // Lock the agenda group if lock-on-active
380        if activation.lock_on_active {
381            self.locked_groups.insert(activation.agenda_group.clone());
382        }
383    }
384
385    /// Check if a rule has already fired
386    pub fn has_fired(&self, rule_name: &str) -> bool {
387        self.fired_rules.contains(rule_name)
388    }
389
390    /// Set focus to a specific agenda group
391    pub fn set_focus(&mut self, group: String) {
392        if group != self.focus {
393            self.focus_stack.push(self.focus.clone());
394            self.focus = group;
395        }
396    }
397
398    /// Get current focus
399    pub fn get_focus(&self) -> &str {
400        &self.focus
401    }
402
403    /// Clear all agenda groups
404    pub fn clear(&mut self) {
405        self.activations.clear();
406        self.activations.insert("MAIN".to_string(), BinaryHeap::new());
407        self.focus = "MAIN".to_string();
408        self.focus_stack.clear();
409        self.fired_rules.clear();
410        self.fired_activation_groups.clear();
411        self.locked_groups.clear();
412    }
413
414    /// Reset fired flags (for re-evaluation)
415    pub fn reset_fired_flags(&mut self) {
416        self.fired_rules.clear();
417        self.fired_activation_groups.clear();
418        self.locked_groups.clear();
419    }
420
421    /// Activate a ruleflow group (make rules in this group eligible to fire)
422    pub fn activate_ruleflow_group(&mut self, group: String) {
423        self.active_ruleflow_groups.insert(group);
424    }
425
426    /// Deactivate a ruleflow group
427    pub fn deactivate_ruleflow_group(&mut self, group: &str) {
428        self.active_ruleflow_groups.remove(group);
429    }
430
431    /// Check if ruleflow group is active
432    pub fn is_ruleflow_group_active(&self, group: &str) -> bool {
433        self.active_ruleflow_groups.contains(group)
434    }
435
436    /// Get agenda statistics
437    pub fn stats(&self) -> AgendaStats {
438        let total_activations: usize = self.activations.values().map(|heap| heap.len()).sum();
439        let groups = self.activations.len();
440
441        AgendaStats {
442            total_activations,
443            groups,
444            focus: self.focus.clone(),
445            fired_rules: self.fired_rules.len(),
446            fired_activation_groups: self.fired_activation_groups.len(),
447            active_ruleflow_groups: self.active_ruleflow_groups.len(),
448        }
449    }
450}
451
452impl Default for AdvancedAgenda {
453    fn default() -> Self {
454        Self::new()
455    }
456}
457
458/// Agenda statistics
459#[derive(Debug, Clone)]
460pub struct AgendaStats {
461    pub total_activations: usize,
462    pub groups: usize,
463    pub focus: String,
464    pub fired_rules: usize,
465    pub fired_activation_groups: usize,
466    pub active_ruleflow_groups: usize,
467}
468
469impl std::fmt::Display for AgendaStats {
470    fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
471        write!(
472            f,
473            "Agenda Stats: {} activations, {} groups, focus='{}', {} fired rules",
474            self.total_activations, self.groups, self.focus, self.fired_rules
475        )
476    }
477}
478
479#[cfg(test)]
480mod tests {
481    use super::*;
482
483    #[test]
484    fn test_basic_activation() {
485        let mut agenda = AdvancedAgenda::new();
486
487        let act1 = Activation::new("Rule1".to_string(), 10);
488        let act2 = Activation::new("Rule2".to_string(), 20);
489
490        agenda.add_activation(act1);
491        agenda.add_activation(act2);
492
493        // Higher salience fires first
494        let next = agenda.get_next_activation().unwrap();
495        assert_eq!(next.rule_name, "Rule2");
496    }
497
498    #[test]
499    fn test_activation_groups() {
500        let mut agenda = AdvancedAgenda::new();
501
502        let act1 = Activation::new("Rule1".to_string(), 10)
503            .with_activation_group("group1".to_string());
504        let act2 = Activation::new("Rule2".to_string(), 20)
505            .with_activation_group("group1".to_string());
506
507        agenda.add_activation(act1);
508        agenda.add_activation(act2);
509
510        // First activation fires
511        let first = agenda.get_next_activation().unwrap();
512        agenda.mark_rule_fired(&first);
513
514        // Second activation should be skipped (same group)
515        let second = agenda.get_next_activation();
516        assert!(second.is_none());
517    }
518
519    #[test]
520    fn test_agenda_groups() {
521        let mut agenda = AdvancedAgenda::new();
522
523        let act1 = Activation::new("Rule1".to_string(), 10)
524            .with_agenda_group("group_a".to_string());
525        let act2 = Activation::new("Rule2".to_string(), 20)
526            .with_agenda_group("group_b".to_string());
527
528        agenda.add_activation(act1);
529        agenda.add_activation(act2);
530
531        // MAIN is empty, nothing fires
532        assert!(agenda.get_next_activation().is_none());
533
534        // Set focus to group_a
535        agenda.set_focus("group_a".to_string());
536        let next = agenda.get_next_activation().unwrap();
537        assert_eq!(next.rule_name, "Rule1");
538    }
539
540    #[test]
541    fn test_auto_focus() {
542        let mut agenda = AdvancedAgenda::new();
543
544        let act = Activation::new("Rule1".to_string(), 10)
545            .with_agenda_group("special".to_string())
546            .with_auto_focus(true);
547
548        agenda.add_activation(act);
549
550        // Auto-focus should switch to "special"
551        assert_eq!(agenda.get_focus(), "special");
552    }
553
554    #[test]
555    fn test_ruleflow_groups() {
556        let mut agenda = AdvancedAgenda::new();
557
558        let act = Activation::new("Rule1".to_string(), 10)
559            .with_ruleflow_group("flow1".to_string());
560
561        // Without activating ruleflow group, activation is not added
562        agenda.add_activation(act.clone());
563        assert_eq!(agenda.stats().total_activations, 0);
564
565        // Activate ruleflow group
566        agenda.activate_ruleflow_group("flow1".to_string());
567        agenda.add_activation(act);
568        assert_eq!(agenda.stats().total_activations, 1);
569    }
570}