File size: 10,046 Bytes
77bcbf1
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
"""
CASCADE Log Manager
Orchestrates the tsunami of data into ordered causation troops.

Manages log levels, routing, and the beautiful display of system truth.
"""

import os
import sys
import time
from typing import Dict, List, Optional, Any
from dataclasses import dataclass
from enum import Enum

from .kleene_logger import KleeneLogger, LogLevel
from .interpretive_logger import InterpretiveLogger, ImpactLevel


class LogMode(Enum):
    """The two modes of logging excellence"""
    KLEENE = "kleene"      # Mathematical precision
    INTERPRETIVE = "interpretive"  # Human stories
    DUAL = "dual"          # Both simultaneously


@dataclass
class LogConfig:
    """Configuration for logging behavior"""
    mode: LogMode = LogMode.DUAL
    min_level_kleene: LogLevel = LogLevel.INFO
    min_level_interpretive: ImpactLevel = ImpactLevel.LOW
    show_metrics: bool = True
    show_timestamps: bool = True
    color_output: bool = True
    file_output: bool = False
    max_file_size_mb: int = 100


class CascadeLogManager:
    """The conductor of your causation orchestra"""
    
    def __init__(self, config: Optional[LogConfig] = None):
        self.config = config or LogConfig()
        self.kleene_loggers: Dict[str, KleeneLogger] = {}
        self.interpretive_loggers: Dict[str, InterpretiveLogger] = {}
        self.start_time = time.time()
        self.operation_count = 0
        
        # Initialize display
        self._setup_display()
        
    def _setup_display(self):
        """Setup beautiful terminal output"""
        if self.config.color_output:
            # Enable ANSI colors
            sys.stdout.reconfigure(encoding='utf-8')
        
        # Print header
        self._print_header()
    
    def _print_header(self):
        """Print beautiful cascade header with colors"""
        # ANSI color codes
        colors = {
            "WAVE": "\033[94m",      # Bright blue
            "BRIDGE": "\033[96m",    # Cyan
            "BOLD": "\033[1m",
            "DIM": "\033[2m",
            "RESET": "\033[0m",
            "GREEN": "\033[32m",
            "YELLOW": "\033[33m",
        }
        
        wave = colors["WAVE"]
        bridge = colors["BRIDGE"]
        bold = colors["BOLD"]
        dim = colors["DIM"]
        reset = colors["RESET"]
        green = colors["GREEN"]
        yellow = colors["YELLOW"]
        
        print(f"\n{bold}{'='*80}{reset}")
        print(f"{wave}🌊{reset} {bold}CASCADE // TRUTH INFRASTRUCTURE{reset} {bridge}🧠{reset}")
        print(f"{bold}{'='*80}{reset}")
        print(f"{bold}Mode:{reset} {green}{self.config.mode.value.upper()}{reset}")
        print(f"{bold}Started:{reset} {dim}{time.strftime('%Y-%m-%d %H:%M:%S')}{reset}")
        print(f"{bold}{'='*80}{reset}\n")
    
    def register_component(self, component: str, system: str = "CASCADE"):
        """Register a component for logging"""
        if self.config.mode in [LogMode.KLEENE, LogMode.DUAL]:
            kleene = KleeneLogger(component)
            self.kleene_loggers[component] = kleene
            
        if self.config.mode in [LogMode.INTERPRETIVE, LogMode.DUAL]:
            interpretive = InterpretiveLogger(system)
            self.interpretive_loggers[system] = interpretive
    
    def log_operation(self, component: str, operation: str, 
                     level: LogLevel = LogLevel.INFO,
                     impact: ImpactLevel = ImpactLevel.LOW,
                     details: Optional[Dict] = None):
        """Log an operation across all active loggers"""
        self.operation_count += 1
        
        if self.config.mode in [LogMode.KLEENE, LogMode.DUAL]:
            if component in self.kleene_loggers:
                self.kleene_loggers[component].log(
                    level, operation,
                    state_before=details.get("before") if details else None,
                    state_after=details.get("after") if details else None,
                    fixed_point=details.get("fixed_point", False) if details else False,
                    iterations=details.get("iterations", 0) if details else 0
                )
        
        if self.config.mode in [LogMode.INTERPRETIVE, LogMode.DUAL]:
            # Find interpretive logger for component
            system = details.get("system", "CASCADE") if details else "CASCADE"
            if system in self.interpretive_loggers:
                self.interpretive_loggers[system].log(
                    impact, component, operation,
                    context=details.get("context", "") if details else "",
                    consequence=details.get("consequence", "") if details else "",
                    metrics=details.get("metrics", {}) if details else {},
                    recommendation=details.get("recommendation") if details else None
                )
    
    def get_session_stats(self) -> Dict[str, Any]:
        """Get beautiful session statistics"""
        total_kleene = sum(len(logger.entries) for logger in self.kleene_loggers.values())
        total_interpretive = sum(len(logger.entries) for logger in self.interpretive_loggers.values())
        
        return {
            "uptime_seconds": time.time() - self.start_time,
            "operations": self.operation_count,
            "kleene_entries": total_kleene,
            "interpretive_entries": total_interpretive,
            "active_components": len(self.kleene_loggers),
            "active_systems": len(self.interpretive_loggers)
        }
    
    def print_summary(self):
        """Print beautiful session summary with colors"""
        stats = self.get_session_stats()
        
        # ANSI color codes
        colors = {
            "BOLD": "\033[1m",
            "DIM": "\033[2m",
            "RESET": "\033[0m",
            "CYAN": "\033[36m",
            "GREEN": "\033[32m",
            "YELLOW": "\033[33m",
            "BLUE": "\033[34m",
            "MAGENTA": "\033[35m",
        }
        
        bold = colors["BOLD"]
        dim = colors["DIM"]
        reset = colors["RESET"]
        cyan = colors["CYAN"]
        green = colors["GREEN"]
        yellow = colors["YELLOW"]
        blue = colors["BLUE"]
        magenta = colors["MAGENTA"]
        
        print(f"\n{bold}{'='*80}{reset}")
        print(f"{cyan}πŸ“Š CASCADE SESSION SUMMARY{reset}")
        print(f"{bold}{'='*80}{reset}")
        print(f"{bold}Uptime:{reset} {stats['uptime_seconds']:.1f} seconds")
        print(f"{bold}Operations:{reset} {green}{stats['operations']:,}{reset}")
        print(f"{bold}Kleene Entries:{reset} {yellow}{stats['kleene_entries']:,}{reset}")
        print(f"{bold}Interpretive Entries:{reset} {blue}{stats['interpretive_entries']:,}{reset}")
        print(f"{bold}Active Components:{reset} {magenta}{stats['active_components']}{reset}")
        print(f"{bold}Active Systems:{reset} {magenta}{stats['active_systems']}{reset}")
        
        if stats['kleene_entries'] > 0:
            # Get session hash from first logger
            first_logger = next(iter(self.kleene_loggers.values()))
            print(f"{bold}Session Hash:{reset} {dim}{first_logger.get_session_hash()}{reset}")
        
        print(f"{bold}{'='*80}{reset}")
    
    def set_mode(self, mode: LogMode):
        """Switch logging mode dynamically"""
        old_mode = self.config.mode
        self.config.mode = mode
        
        print(f"\nπŸ”„ Logging mode changed: {old_mode.value} β†’ {mode.value}")
    
    def enable_file_logging(self, filepath: str):
        """Enable logging to file"""
        self.config.file_output = True
        # TODO: Implement file logging
        print(f"πŸ“ File logging enabled: {filepath}")


# Global log manager instance
_log_manager: Optional[CascadeLogManager] = None


def init_logging(config: Optional[LogConfig] = None) -> CascadeLogManager:
    """Initialize the global CASCADE logging system"""
    global _log_manager
    _log_manager = CascadeLogManager(config)
    return _log_manager


def get_log_manager() -> CascadeLogManager:
    """Get the global log manager"""
    global _log_manager
    if _log_manager is None:
        _log_manager = CascadeLogManager()
    return _log_manager


def log(component: str, operation: str, context: str = "", consequence: str = "", 
         metrics: Dict[str, Any] = None, impact: str = "LOW", **kwargs):
    """Quick log operation - convenience function"""
    manager = get_log_manager()
    manager.log_operation(component, operation, 
                         details={
                             "context": context,
                             "consequence": consequence,
                             "metrics": metrics or {},
                             "impact": impact,
                             **kwargs
                         })


def log_fixed_point(component: str, operation: str, iterations: int, **kwargs):
    """Log successful fixed point"""
    log(component, operation, 
        level=LogLevel.INFO,
        impact=ImpactLevel.LOW,
        details={
            "fixed_point": True,
            "iterations": iterations,
            **kwargs
        })


def log_error(component: str, operation: str, error: str, **kwargs):
    """Log error condition"""
    log(component, f"{operation}_error",
        level=LogLevel.ERROR,
        impact=ImpactLevel.HIGH,
        details={
            "context": f"Operation failed: {error}",
            "consequence": "System may be degraded",
            "metrics": {"error": error},
            **kwargs
        })


def log_performance(component: str, metric: str, value: float, threshold: float):
    """Log performance warning"""
    log(component, f"performance_{metric}",
        level=LogLevel.WARNING,
        impact=ImpactLevel.MEDIUM,
        details={
            "context": f"Performance metric {metric} exceeded threshold",
            "consequence": "May impact system performance",
            "metrics": {metric: value, "threshold": threshold},
            "recommendation": f"Optimize {metric} or scale resources"
        })