#!/usr/bin/env python3
"""
MULTI-TECHNIQUE Conflict Detection System
Multiple conflict detection techniques to choose from:

1. OLD_REJECTION_BASED - Default: Rejects new relationships if one exists (faster, more conservative)
2. OPTIMIZED_CONFIDENCE_BASED - Replaces relationships with higher confidence
3. CONSERVATIVE_CONFIDENCE - Only replaces if new confidence is significantly higher
4. AGGRESSIVE_CONFIDENCE - Always replaces with higher confidence, regardless of difference
5. NO_CONFLICT_RESOLUTION - No conflict resolution, just inference

Each technique can be selected via the conflict_resolution_strategy parameter.
"""

import time
import logging
from typing import Dict, List, Tuple, Any, Set, Optional
from enum import IntEnum, Enum
from collections import defaultdict, deque
import heapq

logging.basicConfig(level=logging.INFO, format='%(asctime)s - %(levelname)s - %(message)s')
logger = logging.getLogger(__name__)

class ConflictResolutionStrategy(Enum):
    """Different conflict resolution strategies"""
    OLD_REJECTION_BASED = "old_rejection_based"               # Default: Reject if exists (faster)
    OPTIMIZED_CONFIDENCE_BASED = "optimized_confidence_based"  # Replace with higher confidence
    CONSERVATIVE_CONFIDENCE = "conservative_confidence"        # Replace only if significantly higher
    AGGRESSIVE_CONFIDENCE = "aggressive_confidence"           # Always replace with higher
    NO_CONFLICT_RESOLUTION = "no_conflict_resolution"         # No conflict resolution

class RelationshipType(IntEnum):
    NULL = 0
    IMPLIES = 1       # →
    MUTUAL = 2        # ↔
    CONTRADICTS = 3   # ×
    EMPTY = -1        # ∅

class EvidenceType(IntEnum):
    DIRECT = 2
    INFERRED = 1
    PROVISIONAL = 0

class TraversalMode(IntEnum):
    SPARSE_FULL = 1
    DENSE_LIMITED = 2

class OptimizedTransitivityEngine:
    """Optimized transitivity engine with caching"""
    
    def __init__(self):
        # Cache for transitivity results
        self._transitivity_cache = {}
        
        self.transitivity_rules = {
            (RelationshipType.IMPLIES, RelationshipType.IMPLIES): RelationshipType.IMPLIES,
            (RelationshipType.IMPLIES, RelationshipType.MUTUAL): RelationshipType.IMPLIES,
            (RelationshipType.IMPLIES, RelationshipType.CONTRADICTS): RelationshipType.CONTRADICTS,
            (RelationshipType.MUTUAL, RelationshipType.IMPLIES): RelationshipType.IMPLIES,
            (RelationshipType.MUTUAL, RelationshipType.MUTUAL): RelationshipType.MUTUAL,
            (RelationshipType.MUTUAL, RelationshipType.CONTRADICTS): RelationshipType.CONTRADICTS,
            (RelationshipType.CONTRADICTS, RelationshipType.IMPLIES): RelationshipType.CONTRADICTS,
            (RelationshipType.CONTRADICTS, RelationshipType.MUTUAL): RelationshipType.CONTRADICTS,
            (RelationshipType.CONTRADICTS, RelationshipType.CONTRADICTS): RelationshipType.CONTRADICTS,
        }
    
    def apply_transitivity(self, rel_AB: RelationshipType, rel_BC: RelationshipType) -> Optional[RelationshipType]:
        """Cached transitivity application"""
        cache_key = (rel_AB, rel_BC)
        if cache_key in self._transitivity_cache:
            return self._transitivity_cache[cache_key]
            
        if rel_AB == RelationshipType.NULL or rel_BC == RelationshipType.NULL:
            result = None
        else:
            result = self.transitivity_rules.get(cache_key, None)
            
        self._transitivity_cache[cache_key] = result
        return result

class OptimizedDirtyTracker:
    """Optimized dirty tracking with change detection and batching"""
    
    def __init__(self, batch_size: int = 50):
        self.dirty_nodes = set()
        self.priority_queue = []
        self.in_queue = set()
        self.visit_counts = defaultdict(int)
        self.batch_size = batch_size
        self.processed_in_batch = 0
        self.last_changes = defaultdict(int)  # Track changes per node
        self.convergence_threshold = 3  # Iterations without changes
        
    def mark_dirty(self, node_idx: int, change_count: int = 1):
        """Mark node as dirty only if it has new changes"""
        # Only mark dirty if there are actual changes
        if change_count > self.last_changes[node_idx]:
            if node_idx not in self.dirty_nodes:
                self.dirty_nodes.add(node_idx)
                
                if node_idx not in self.in_queue:
                    priority_score = self.visit_counts[node_idx]
                    heapq.heappush(self.priority_queue, (priority_score, node_idx))
                    self.in_queue.add(node_idx)
            
            self.last_changes[node_idx] = change_count
    
    def get_next_batch(self) -> List[int]:
        """Get next batch of nodes to process"""
        batch = []
        self.processed_in_batch = 0
        
        # Process all dirty nodes in this batch
        dirty_nodes_list = list(self.dirty_nodes)
        for node_idx in dirty_nodes_list[:self.batch_size]:
            batch.append(node_idx)
            self.dirty_nodes.discard(node_idx)
            self.visit_counts[node_idx] += 1
            self.processed_in_batch += 1
        
        return batch
    
    def has_dirty_nodes(self) -> bool:
        return len(self.dirty_nodes) > 0
    
    def reset(self):
        self.dirty_nodes.clear()
        self.priority_queue.clear()
        self.in_queue.clear()
        self.last_changes.clear()

class OptimizedSparseMatrix:
    """Performance-optimized sparse matrix system with multiple conflict resolution strategies"""
    
    def __init__(self, nodes: List[str], max_hops: int = 6, density_threshold: float = 0.15, 
                 conflict_resolution_strategy: ConflictResolutionStrategy = ConflictResolutionStrategy.OLD_REJECTION_BASED):
        self.nodes = nodes
        self.n = len(nodes)
        self.max_hops = max_hops  # Reduced from 8 to 6
        self.density_threshold = density_threshold
        self.conflict_resolution_strategy = conflict_resolution_strategy
        
        # Node mapping
        self.node_to_index = {node: i for i, node in enumerate(nodes)}
        self.index_to_node = {i: node for i, node in enumerate(nodes)}
        
        # Core data structures
        self.relationships = {}
        self.confidence_map = {}
        self.evidence_map = {}
        
        # Optimized adjacency lists
        self.implies_out = defaultdict(set)
        self.implies_in = defaultdict(set)
        self.mutual = defaultdict(set)
        self.contradicts_out = defaultdict(set)
        self.contradicts_in = defaultdict(set)
        self.null_pairs = set()
        
        # Symbol mappings
        self.enum_map = {
            RelationshipType.NULL: "null",
            RelationshipType.IMPLIES: "A_into_B",
            RelationshipType.MUTUAL: "mutual",
            RelationshipType.CONTRADICTS: "not_mergeable",
            RelationshipType.EMPTY: "∅"
        }
        
        self.symbol_map = {
            RelationshipType.IMPLIES: "→",
            RelationshipType.MUTUAL: "↔",
            RelationshipType.CONTRADICTS: "×",
            RelationshipType.NULL: "null",
            RelationshipType.EMPTY: "∅"
        }
        
        # Performance tracking
        self.relationship_count = 0
        self.inference_count = 0
        self.conflict_count = 0
        self.start_time = time.time()
        
        # Optimized components
        self.transitivity_engine = OptimizedTransitivityEngine()
        self.dirty_tracker = OptimizedDirtyTracker(batch_size=max(1, min(50, self.n // 10)))
        
        # Change tracking for convergence
        self.total_changes_by_iteration = []
        self.stagnant_iterations = 0
        self.max_stagnant_iterations = 5
        
        # Conflict tracking
        self.conflicts = []
        self.conflict_set = set()
        
        # Performance optimizations
        self.processed_pairs = set()  # Track processed pairs to avoid duplicates
        self.relationship_change_count = 0  # Global change counter
        
    def add_direct_relationship(self, node_a: str, node_b: str, relationship: str, confidence: float = 0.8):
        """Optimized relationship addition with minimal dirty marking"""
        if node_a not in self.node_to_index or node_b not in self.node_to_index:
            logger.warning(f"Invalid nodes: {node_a}, {node_b}")
            return
        
        idx_a = self.node_to_index[node_a]
        idx_b = self.node_to_index[node_b]
        
        if idx_a == idx_b:
            return
        
        rel_map = {
            "A_into_B": RelationshipType.IMPLIES, "→": RelationshipType.IMPLIES,
            "mutual": RelationshipType.MUTUAL, "↔": RelationshipType.MUTUAL,
            "not_mergeable": RelationshipType.CONTRADICTS, "×": RelationshipType.CONTRADICTS
        }
        
        if relationship not in rel_map:
            logger.warning(f"Invalid relationship: {relationship}")
            return
        
        rel_type = rel_map[relationship]
        existing_rel = self._get_relationship(idx_a, idx_b)
        
        # Quick exit if relationship already exists with same or better confidence
        if existing_rel == rel_type:
            existing_conf = self.confidence_map.get((idx_a, idx_b), 0.0)
            if confidence <= existing_conf:
                return  # No change needed
        
        # Apply relationship with optimized conflict handling
        if self._set_relationship_optimized(idx_a, idx_b, rel_type, confidence, EvidenceType.DIRECT):
            self.relationship_count += 1
            self.relationship_change_count += 1
            
            # Handle bidirectional relationships
            if rel_type == RelationshipType.MUTUAL:
                if self._set_relationship_optimized(idx_b, idx_a, RelationshipType.MUTUAL, confidence, EvidenceType.DIRECT):
                    self.relationship_count += 1
            elif rel_type == RelationshipType.CONTRADICTS:
                if self._set_relationship_optimized(idx_b, idx_a, RelationshipType.CONTRADICTS, confidence, EvidenceType.DIRECT):
                    self.relationship_count += 1
            elif rel_type == RelationshipType.IMPLIES:
                if self._get_relationship(idx_b, idx_a) == RelationshipType.EMPTY:
                    self._set_relationship_optimized(idx_b, idx_a, RelationshipType.NULL, 0.0, EvidenceType.DIRECT)
            
            # Optimized dirty marking - only mark if significant change
            self.dirty_tracker.mark_dirty(idx_a, self.relationship_change_count)
            self.dirty_tracker.mark_dirty(idx_b, self.relationship_change_count)
    
    def _set_relationship_optimized(self, i: int, j: int, rel_type: RelationshipType, 
                                   confidence: float, evidence: EvidenceType) -> bool:
        """Optimized relationship setting with configurable conflict resolution"""
        existing_rel = self._get_relationship(i, j)
        existing_conf = self.confidence_map.get((i, j), 0.0)
        
        # Apply conflict resolution strategy
        should_update = self._should_update_relationship(existing_rel, existing_conf, rel_type, confidence)
        
        if not should_update:
            return False  # No change needed
        
        self.relationships[(i, j)] = (rel_type, confidence, evidence)
        self.confidence_map[(i, j)] = confidence
        self.evidence_map[(i, j)] = evidence
        
        # Update adjacency lists efficiently
        self._update_adjacency_lists(i, j, rel_type, existing_rel)
        
        return True  # Change made
    
    def _should_update_relationship(self, existing_rel: RelationshipType, existing_conf: float, 
                                  new_rel: RelationshipType, new_conf: float) -> bool:
        """Determine if relationship should be updated based on conflict resolution strategy"""
        
        # If no existing relationship, always add
        if existing_rel == RelationshipType.EMPTY:
            return True
        
        # If same relationship type, check confidence
        if existing_rel == new_rel:
            if self.conflict_resolution_strategy == ConflictResolutionStrategy.OLD_REJECTION_BASED:
                return False  # Reject if relationship already exists
            elif self.conflict_resolution_strategy == ConflictResolutionStrategy.CONSERVATIVE_CONFIDENCE:
                return new_conf > existing_conf + 0.1  # Only if significantly higher
            elif self.conflict_resolution_strategy == ConflictResolutionStrategy.AGGRESSIVE_CONFIDENCE:
                return new_conf > existing_conf  # Always replace with higher
            else:  # OPTIMIZED_CONFIDENCE_BASED and NO_CONFLICT_RESOLUTION
                return new_conf > existing_conf
        
        # Different relationship types - conflict resolution
        if self.conflict_resolution_strategy == ConflictResolutionStrategy.OLD_REJECTION_BASED:
            return False  # Reject new relationship if any exists
        
        elif self.conflict_resolution_strategy == ConflictResolutionStrategy.NO_CONFLICT_RESOLUTION:
            return True  # Always add, no conflict resolution
        
        elif self.conflict_resolution_strategy == ConflictResolutionStrategy.CONSERVATIVE_CONFIDENCE:
            return new_conf > existing_conf + 0.2  # Only if significantly higher
        
        elif self.conflict_resolution_strategy == ConflictResolutionStrategy.AGGRESSIVE_CONFIDENCE:
            return new_conf > existing_conf  # Always replace with higher
        
        else:  # OPTIMIZED_CONFIDENCE_BASED (default)
            return new_conf > existing_conf
    
    def _update_adjacency_lists(self, i: int, j: int, new_rel: RelationshipType, old_rel: RelationshipType):
        """Efficiently update adjacency lists"""
        # Remove old relationships
        if old_rel == RelationshipType.IMPLIES:
            self.implies_out[i].discard(j)
            self.implies_in[j].discard(i)
        elif old_rel == RelationshipType.MUTUAL:
            self.mutual[i].discard(j)
            self.mutual[j].discard(i)
        elif old_rel == RelationshipType.CONTRADICTS:
            self.contradicts_out[i].discard(j)
            self.contradicts_in[j].discard(i)
        
        # Add new relationships
        if new_rel == RelationshipType.IMPLIES:
            self.implies_out[i].add(j)
            self.implies_in[j].add(i)
        elif new_rel == RelationshipType.MUTUAL:
            self.mutual[i].add(j)
            self.mutual[j].add(i)
        elif new_rel == RelationshipType.CONTRADICTS:
            self.contradicts_out[i].add(j)
            self.contradicts_in[j].add(i)
        elif new_rel == RelationshipType.NULL:
            self.null_pairs.add((i, j))
    
    def run_optimized_inference(self, timeout_seconds: int = 60):  # Reduced timeout
        """Optimized inference with proper convergence detection"""
        start_time = time.time()
        iteration = 0
        
        logger.info(f"🚀 Starting OPTIMIZED inference on {self.n} nodes")
        
        while True:
            iteration += 1
            current_time = time.time()
            
            if current_time - start_time > timeout_seconds:
                logger.warning(f"⏰ Timeout reached after {timeout_seconds}s")
                break
            
            # Process batches of dirty nodes
            total_changes = 0
            batch_count = 0
            
            while self.dirty_tracker.has_dirty_nodes() and batch_count < 10:  # Limit batches per iteration
                batch = self.dirty_tracker.get_next_batch()
                if not batch:
                    break
                
                batch_changes = self._process_node_batch(batch)
                total_changes += batch_changes
                batch_count += 1
            
            self.total_changes_by_iteration.append(total_changes)
            
            # Convergence detection
            if total_changes == 0:
                self.stagnant_iterations += 1
                if self.stagnant_iterations >= self.max_stagnant_iterations:
                    logger.info(f"✅ Convergence reached after {iteration} iterations")
                    break
            else:
                self.stagnant_iterations = 0
            
            # Check for oscillation
            if len(self.total_changes_by_iteration) >= 5:
                recent_changes = self.total_changes_by_iteration[-5:]
                if len(set(recent_changes)) <= 2 and all(c > 0 for c in recent_changes):
                    logger.warning(f"⚠️ Oscillation detected, stopping after {iteration} iterations")
                    break
            
            logger.info(f"📊 Iteration {iteration}: {total_changes} changes, {batch_count} batches")
            
            # Safety limits
            if iteration > 50:  # Reduced from 50
                logger.warning(f"⚠️ Max iterations reached ({iteration})")
                break
        
        end_time = time.time()
        self.inference_time = end_time - start_time
        
        logger.info(f"🎯 Optimized inference completed: {sum(self.total_changes_by_iteration)} total changes")
        logger.info(f"⏱️ Total time: {self.inference_time:.2f}s")
        logger.info(f"📈 Average changes per iteration: {sum(self.total_changes_by_iteration) / len(self.total_changes_by_iteration):.1f}")
    
    def run_sparse_inference(self, timeout_seconds: int = 300):
        """Alias for run_optimized_inference for compatibility with existing pipeline"""
        return self.run_optimized_inference(timeout_seconds)
    
    def detect_conflicts(self) -> List[Dict]:
        """Detect conflicts in the relationship matrix for compatibility"""
        conflicts = []
        
        # Check for bidirectional conflicts
        for (i, j), (rel_type, confidence, evidence) in self.relationships.items():
            if i != j:  # Skip self-relationships
                reverse_rel = self._get_relationship(j, i)
                
                # Check for conflicting relationships
                if rel_type == RelationshipType.IMPLIES and reverse_rel == RelationshipType.IMPLIES:
                    # Both directions are IMPLIES - this could be a conflict
                    conflicts.append({
                        'node_a': self.index_to_node[i],
                        'node_b': self.index_to_node[j],
                        'conflict_type': 'bidirectional_implies',
                        'explanation': f"Both {self.index_to_node[i]} → {self.index_to_node[j]} and {self.index_to_node[j]} → {self.index_to_node[i]}",
                        'confidence_a': confidence,
                        'confidence_b': self.confidence_map.get((j, i), 0.0)
                    })
                
                elif rel_type == RelationshipType.CONTRADICTS and reverse_rel == RelationshipType.CONTRADICTS:
                    # Both directions are CONTRADICTS - this is expected
                    pass
                
                elif rel_type == RelationshipType.MUTUAL and reverse_rel == RelationshipType.MUTUAL:
                    # Both directions are MUTUAL - this is expected
                    pass
                
                elif rel_type != reverse_rel and reverse_rel != RelationshipType.EMPTY:
                    # Different relationship types in opposite directions
                    conflicts.append({
                        'node_a': self.index_to_node[i],
                        'node_b': self.index_to_node[j],
                        'conflict_type': 'inconsistent_relationship',
                        'explanation': f"{self.index_to_node[i]} {self.symbol_map[rel_type]} {self.index_to_node[j]} vs {self.index_to_node[j]} {self.symbol_map[reverse_rel]} {self.index_to_node[i]}",
                        'confidence_a': confidence,
                        'confidence_b': self.confidence_map.get((j, i), 0.0)
                    })
        
        self.conflicts = conflicts
        self.conflict_count = len(conflicts)
        return conflicts
    
    def _process_node_batch(self, batch: List[int]) -> int:
        """Process a batch of nodes efficiently"""
        total_changes = 0
        
        for node in batch:
            # Bounded transitivity processing
            total_changes += self._bounded_transitivity(node, max_queue_size=1000)
            # Optimized mergeability
            total_changes += self._optimized_mergeability(node)
        
        return total_changes
    
    def _bounded_transitivity(self, start_node: int, max_queue_size: int = 1000) -> int:
        """Bounded BFS transitivity with queue size limits"""
        changes = 0
        queue = deque([(start_node, RelationshipType.EMPTY, 1.0, 0, "self", 0.0, 0)])
        visited = set()
        
        while queue and len(queue) < max_queue_size:
            current_node, path_rel, path_conf, depth, path_desc, conf_sum, hop_count = queue.popleft()
            
            if depth > self.max_hops or current_node in visited:
                continue
            
            visited.add(current_node)
            
            # Apply relationship for non-start nodes
            if current_node != start_node and path_rel != RelationshipType.EMPTY:
                # For OLD_REJECTION_BASED: no confidence threshold needed
                changes += self._apply_inferred_relationship_optimized(
                    start_node, current_node, path_rel, path_conf
                )
            
            # Get outgoing relationships (limited to prevent explosion)
            outgoing = self._get_outgoing_relationships_limited(current_node, max_edges=20)
            
            for target_node, rel_type in outgoing:
                if target_node == start_node or target_node in visited:
                    continue
                
                if current_node == start_node:
                    new_path_rel = rel_type
                    target_conf = self.confidence_map.get((current_node, target_node), 0)
                    new_path_conf = target_conf
                    new_conf_sum = target_conf
                    new_hop_count = 1
                else:
                    new_path_rel = self.transitivity_engine.apply_transitivity(path_rel, rel_type)
                    if new_path_rel is None:
                        continue
                    
                    target_conf = self.confidence_map.get((current_node, target_node), 0)
                    new_conf_sum = conf_sum + target_conf
                    new_hop_count = hop_count + 1
                    new_path_conf = new_conf_sum / new_hop_count
                
                if len(queue) < max_queue_size:  # Prevent queue explosion (no confidence threshold)
                    queue.append((target_node, new_path_rel, new_path_conf, depth + 1, 
                                f"{path_desc}→{self.index_to_node[target_node]}", 
                                new_conf_sum, new_hop_count))
        
        return changes
    
    def _get_outgoing_relationships_limited(self, node: int, max_edges: int = 20) -> List[Tuple[int, RelationshipType]]:
        """Get outgoing relationships with limits to prevent explosion"""
        relationships = []
        
        # Prioritize higher confidence relationships
        edges_with_conf = []
        
        for target in self.implies_out[node]:
            conf = self.confidence_map.get((node, target), 0)
            edges_with_conf.append((conf, target, RelationshipType.IMPLIES))
        
        for target in self.mutual[node]:
            conf = self.confidence_map.get((node, target), 0)
            edges_with_conf.append((conf, target, RelationshipType.MUTUAL))
        
        for target in self.contradicts_out[node]:
            conf = self.confidence_map.get((node, target), 0)
            edges_with_conf.append((conf, target, RelationshipType.CONTRADICTS))
        
        # Sort by confidence (highest first) and limit
        edges_with_conf.sort(reverse=True)
        
        for conf, target, rel_type in edges_with_conf[:max_edges]:
            relationships.append((target, rel_type))
        
        return relationships
    
    def _optimized_mergeability(self, start_node: int) -> int:
        """DISABLED: This was the buggy method causing mutual relationship explosion"""
        # BUG FIX: This method incorrectly created mutual relationships between nodes
        # that both point to the same target. This has been disabled to prevent
        # the mutual relationship explosion bug.
        return 0  # Return 0 changes - effectively disabled
    
    def _apply_inferred_relationship_optimized(self, node_a: int, node_b: int, 
                                             rel_type: RelationshipType, confidence: float) -> int:
        """Optimized relationship application with fewer checks"""
        pair_key = (min(node_a, node_b), max(node_a, node_b), rel_type)
        if pair_key in self.processed_pairs:
            return 0  # Already processed this inference
        
        existing_rel = self._get_relationship(node_a, node_b)
        existing_conf = self.confidence_map.get((node_a, node_b), 0.0)
        
        # For OLD_REJECTION_BASED: only check if relationship exists
        should_update = (existing_rel == RelationshipType.EMPTY)
        
        if should_update:
            if self._set_relationship_optimized(node_a, node_b, rel_type, confidence, EvidenceType.INFERRED):
                self.processed_pairs.add(pair_key)
                self.inference_count += 1
                return 1
        
        return 0
    
    def _get_relationship(self, i: int, j: int) -> RelationshipType:
        """Get relationship type"""
        if (i, j) in self.relationships:
            return self.relationships[(i, j)][0]
        return RelationshipType.EMPTY
    
    def get_performance_stats(self) -> Dict[str, Any]:
        """Get performance statistics"""
        end_time = time.time()
        total_time = end_time - self.start_time
        
        implies_count = sum(1 for rel_type, _, _ in self.relationships.values() if rel_type == RelationshipType.IMPLIES)
        mutual_count = sum(1 for rel_type, _, _ in self.relationships.values() if rel_type == RelationshipType.MUTUAL) // 2
        contradicts_count = sum(1 for rel_type, _, _ in self.relationships.values() if rel_type == RelationshipType.CONTRADICTS) // 2
        null_count = sum(1 for rel_type, _, _ in self.relationships.values() if rel_type == RelationshipType.NULL)
        
        return {
            'total_time': total_time,
            'inference_time': getattr(self, 'inference_time', 0.0),
            'node_count': self.n,
            'relationship_count': self.relationship_count,
            'inference_count': self.inference_count,
            'conflict_count': getattr(self, 'conflict_count', 0),
            'implies_count': implies_count,
            'mutual_count': mutual_count,
            'contradicts_count': contradicts_count,
            'null_count': null_count,
            'processed_pairs': len(self.processed_pairs),
            'memory_usage_mb': len(self.relationships) * 24 / (1024 * 1024),
            'avg_changes_per_iteration': sum(self.total_changes_by_iteration) / len(self.total_changes_by_iteration) if self.total_changes_by_iteration else 0,
            'total_iterations': len(self.total_changes_by_iteration)
        }
    
    def print_matrix(self):
        """Print optimized matrix results"""
        stats = self.get_performance_stats()
        
        print(f"\n🚀 OPTIMIZED SPARSE MATRIX ({self.n} nodes)")
        print("=" * 60)
        
        print(f"🎯 CONFLICT RESOLUTION STRATEGY: {self.conflict_resolution_strategy.value}")
        
        print("📊 PERFORMANCE IMPROVEMENTS:")
        print(f"   ✅ Bounded BFS (max queue: 1000)")
        print(f"   ✅ Batch processing ({self.dirty_tracker.batch_size} nodes/batch)")
        print(f"   ✅ Early convergence detection")
        print(f"   ✅ Duplicate inference prevention")
        print(f"   ✅ Optimized adjacency updates")
        
        print(f"\n📈 STATISTICS:")
        print(f"   Total Time: {stats['total_time']:.3f}s")
        print(f"   Relationships: {stats['relationship_count']}")
        print(f"   Inferences: {stats['inference_count']}")
        print(f"   Iterations: {stats['total_iterations']}")
        print(f"   Avg Changes/Iteration: {stats['avg_changes_per_iteration']:.1f}")
        print(f"   Memory Usage: {stats['memory_usage_mb']:.1f} MB")
        print(f"   Processed Pairs: {stats['processed_pairs']}")
        
        print(f"\n🔗 RELATIONSHIPS:")
        print(f"   → (IMPLIES): {stats['implies_count']}")
        print(f"   ↔ (MUTUAL): {stats['mutual_count']}")
        print(f"   × (CONTRADICTS): {stats['contradicts_count']}")

# Alias for compatibility
SparseScalableMatrix = OptimizedSparseMatrix
FastMatrix = OptimizedSparseMatrix

def create_matrix_with_strategy(nodes: List[str], strategy: ConflictResolutionStrategy) -> OptimizedSparseMatrix:
    """Create a matrix with a specific conflict resolution strategy"""
    return OptimizedSparseMatrix(nodes, conflict_resolution_strategy=strategy)

def test_conflict_resolution_strategies():
    """Test different conflict resolution strategies"""
    print("🧪 Testing CONFLICT RESOLUTION STRATEGIES")
    print("=" * 60)
    
    nodes = ['A', 'B', 'C', 'D', 'E']
    
    # Test relationships that will cause conflicts
    test_relationships = [
        ('A', 'B', '→', 0.8),  # A → B
        ('A', 'B', '×', 0.9),  # A × B (conflict with higher confidence)
        ('B', 'C', '→', 0.7),  # B → C
        ('B', 'C', '↔', 0.6),  # B ↔ C (conflict with lower confidence)
        ('C', 'D', '→', 0.8),  # C → D
        ('D', 'E', '×', 0.9),  # D × E
    ]
    
    strategies = [
        ConflictResolutionStrategy.OPTIMIZED_CONFIDENCE_BASED,
        ConflictResolutionStrategy.OLD_REJECTION_BASED,
        ConflictResolutionStrategy.CONSERVATIVE_CONFIDENCE,
        ConflictResolutionStrategy.AGGRESSIVE_CONFIDENCE,
        ConflictResolutionStrategy.NO_CONFLICT_RESOLUTION
    ]
    
    results = {}
    
    for strategy in strategies:
        print(f"\n🔍 Testing {strategy.value.upper()}")
        print("-" * 40)
        
        m = create_matrix_with_strategy(nodes, strategy)
        
        # Add relationships
        for node_a, node_b, rel, conf in test_relationships:
            m.add_direct_relationship(node_a, node_b, rel, conf)
        
        # Run inference
        m.run_optimized_inference(timeout_seconds=10)
        
        # Get results
        stats = m.get_performance_stats()
        results[strategy.value] = {
            'relationships': stats['relationship_count'],
            'implies': stats['implies_count'],
            'mutual': stats['mutual_count'],
            'contradicts': stats['contradicts_count']
        }
        
        print(f"   Final relationships: {stats['relationship_count']}")
        print(f"   → (IMPLIES): {stats['implies_count']}")
        print(f"   ↔ (MUTUAL): {stats['mutual_count']}")
        print(f"   × (CONTRADICTS): {stats['contradicts_count']}")
    
    # Print comparison
    print(f"\n📊 STRATEGY COMPARISON")
    print("=" * 60)
    print(f"{'Strategy':<30} {'Total':<8} {'→':<8} {'↔':<8} {'×':<8}")
    print("-" * 60)
    
    for strategy_name, result in results.items():
        print(f"{strategy_name:<30} {result['relationships']:<8} {result['implies']:<8} {result['mutual']:<8} {result['contradicts']:<8}")
    
    return results

def test_optimized_system():
    """Test the optimized system"""
    print("🧪 Testing OPTIMIZED System")
    print("=" * 50)
    
    nodes = ['A', 'B', 'C', 'D', 'E']
    m = OptimizedSparseMatrix(nodes)
    
    # Add test relationships
    m.add_direct_relationship('A', 'B', '→', 0.8)
    m.add_direct_relationship('B', 'C', '→', 0.8)
    m.add_direct_relationship('C', 'D', '→', 0.8)
    m.add_direct_relationship('A', 'E', '×', 0.8)
    m.add_direct_relationship('E', 'D', '→', 0.8)
    
    print("Running optimized inference...")
    m.run_optimized_inference()
    m.print_matrix()
    
    return m

def performance_comparison():
    """Compare performance with larger dataset"""
    print("🏃 PERFORMANCE COMPARISON")
    print("=" * 50)
    
    # Test with moderate size
    nodes = [f'Node_{i}' for i in range(100)]
    m = OptimizedSparseMatrix(nodes)
    
    # Add relationships
    import random
    random.seed(42)
    
    for _ in range(500):  # Moderate number of relationships
        i, j = random.sample(range(len(nodes)), 2)
        rel = random.choice(['→', '↔', '×'])
        conf = random.uniform(0.7, 1.0)
        m.add_direct_relationship(nodes[i], nodes[j], rel, conf)
    
    start_time = time.time()
    m.run_optimized_inference(timeout_seconds=30)
    end_time = time.time()
    
    stats = m.get_performance_stats()
    
    print(f"\n🎯 PERFORMANCE RESULTS:")
    print(f"   Nodes: {stats['node_count']}")
    print(f"   Direct Relationships: 500")
    print(f"   Total Relationships: {stats['relationship_count']}")
    print(f"   Inferences Made: {stats['inference_count']}")
    print(f"   Total Time: {end_time - start_time:.2f}s")
    print(f"   Inferences/Second: {stats['inference_count'] / (end_time - start_time):.0f}")
    print(f"   Memory Usage: {stats['memory_usage_mb']:.1f} MB")
    
    return m

if __name__ == "__main__":
    # Test different conflict resolution strategies
    test_conflict_resolution_strategies()
    print("\n" + "="*60 + "\n")
    
    # Test optimized system
    test_optimized_system()
    print("\n" + "="*50 + "\n")
    performance_comparison()