File size: 5,187 Bytes
b7c2c9d
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
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
"""
QCrypt RNG Logging Configuration
Centralized logging setup with structured logging support
"""

import sys
import os
from typing import Optional
from pathlib import Path
from loguru import logger
from app.config import settings

# Remove default handler
logger.remove()

# Create logs directory if it doesn't exist
log_dir = Path("logs")
log_dir.mkdir(exist_ok=True)


def setup_logging():
    """Configure application logging"""
    
    # Console logging with color
    logger.add(
        sys.stdout,
        format="<green>{time:YYYY-MM-DD HH:mm:ss}</green> | <level>{level: <8}</level> | <cyan>{name}</cyan>:<cyan>{function}</cyan>:<cyan>{line}</cyan> - <level>{message}</level>",
        level=settings.log_level,
        colorize=True,
        backtrace=True,
        diagnose=settings.debug
    )
    
    # File logging for all logs
    logger.add(
        "logs/qcrypt_{time:YYYY-MM-DD}.log",
        format="{time:YYYY-MM-DD HH:mm:ss} | {level: <8} | {name}:{function}:{line} - {message}",
        level="DEBUG",
        rotation="00:00",  # Daily rotation
        retention="30 days",
        compression="zip",
        backtrace=True,
        diagnose=True
    )
    
    # Separate error log
    logger.add(
        "logs/errors_{time:YYYY-MM-DD}.log",
        format="{time:YYYY-MM-DD HH:mm:ss} | {level: <8} | {name}:{function}:{line} - {message}",
        level="ERROR",
        rotation="00:00",
        retention="90 days",
        compression="zip",
        backtrace=True,
        diagnose=True
    )
    
    # JSON logging for production (machine-readable)
    if not settings.debug:
        logger.add(
            "logs/qcrypt_json_{time:YYYY-MM-DD}.log",
            format="{message}",
            level="INFO",
            rotation="00:00",
            retention="30 days",
            compression="zip",
            serialize=True  # JSON format
        )
    
    # Performance logging
    logger.add(
        "logs/performance_{time:YYYY-MM-DD}.log",
        format="{time:YYYY-MM-DD HH:mm:ss} | PERF | {message}",
        level="INFO",
        filter=lambda record: "performance" in record["extra"],
        rotation="00:00",
        retention="7 days"
    )
    
    # Security audit log
    logger.add(
        "logs/security_{time:YYYY-MM-DD}.log",
        format="{time:YYYY-MM-DD HH:mm:ss} | SECURITY | {message}",
        level="INFO",
        filter=lambda record: "security" in record["extra"],
        rotation="00:00",
        retention="365 days",  # Keep for compliance
        compression="zip"
    )
    
    logger.info("Logging system initialized")
    logger.info(f"Log level: {settings.log_level}")
    logger.info(f"Debug mode: {settings.debug}")


# Specialized loggers
def get_performance_logger():
    """Get logger for performance metrics"""
    return logger.bind(performance=True)


def get_security_logger():
    """Get logger for security events"""
    return logger.bind(security=True)


def get_quantum_logger():
    """Get logger for quantum operations"""
    return logger.bind(quantum=True)


# Decorators for logging
def log_execution_time(func):
    """Decorator to log function execution time"""
    import functools
    import time
    
    @functools.wraps(func)
    async def async_wrapper(*args, **kwargs):
        start_time = time.time()
        try:
            result = await func(*args, **kwargs)
            execution_time = (time.time() - start_time) * 1000
            get_performance_logger().info(
                f"{func.__name__} executed in {execution_time:.2f}ms"
            )
            return result
        except Exception as e:
            execution_time = (time.time() - start_time) * 1000
            logger.error(
                f"{func.__name__} failed after {execution_time:.2f}ms: {str(e)}"
            )
            raise
    
    @functools.wraps(func)
    def sync_wrapper(*args, **kwargs):
        start_time = time.time()
        try:
            result = func(*args, **kwargs)
            execution_time = (time.time() - start_time) * 1000
            get_performance_logger().info(
                f"{func.__name__} executed in {execution_time:.2f}ms"
            )
            return result
        except Exception as e:
            execution_time = (time.time() - start_time) * 1000
            logger.error(
                f"{func.__name__} failed after {execution_time:.2f}ms: {str(e)}"
            )
            raise
    
    # Return appropriate wrapper based on function type
    import asyncio
    if asyncio.iscoroutinefunction(func):
        return async_wrapper
    else:
        return sync_wrapper


def log_api_request(endpoint: str, method: str, user_id: Optional[str] = None):
    """Log API request for audit trail"""
    get_security_logger().info(
        f"API Request | Method: {method} | Endpoint: {endpoint} | User: {user_id or 'anonymous'}"
    )


def log_quantum_generation(bytes_generated: int, qubits: int, backend: str, time_ms: float):
    """Log quantum generation event"""
    get_quantum_logger().info(
        f"Quantum Generation | Bytes: {bytes_generated} | Qubits: {qubits} | "
        f"Backend: {backend} | Time: {time_ms:.2f}ms"
    )