"""Logging utilities for dual output (console + file).

This module provides classes for simultaneously writing output to
both console and log files, with proper resource management.
"""

import sys


class DualOutput:
    """Dual output stream that writes to both terminal and log file.
    
    Implements proper resource management to prevent file handle leaks.
    
    Usage:
        dual_out = DualOutput('/path/to/log.txt')
        sys.stdout = dual_out
        
        # Ensure cleanup on exit
        import atexit
        atexit.register(dual_out.close)
    """
    
    def __init__(self, file_path):
        self.terminal = sys.stdout
        self.log = open(file_path, 'a', buffering=1)  # Line buffering
        self._closed = False
    
    def write(self, message):
        if not self._closed:
            self.terminal.write(message)
            self.log.write(message)
            self.log.flush()
    
    def flush(self):
        if not self._closed:
            self.terminal.flush()
            self.log.flush()
    
    def close(self):
        """Close the log file handle."""
        if not self._closed and hasattr(self, 'log') and not self.log.closed:
            self.log.flush()
            self.log.close()
            self._closed = True
    
    def __del__(self):
        """Ensure file is closed when object is deleted."""
        self.close()


class DualError:
    """Dual error stream that writes to both terminal and log file.
    
    Implements proper resource management to prevent file handle leaks.
    
    Usage:
        dual_err = DualError('/path/to/log.txt')
        sys.stderr = dual_err
        
        # Ensure cleanup on exit
        import atexit
        atexit.register(dual_err.close)
    """
    
    def __init__(self, file_path):
        self.terminal = sys.stderr
        self.log = open(file_path, 'a', buffering=1)  # Line buffering
        self._closed = False
    
    def write(self, message):
        if not self._closed:
            self.terminal.write(message)
            self.log.write(message)
            self.log.flush()
    
    def flush(self):
        if not self._closed:
            self.terminal.flush()
            self.log.flush()
    
    def close(self):
        """Close the log file handle."""
        if not self._closed and hasattr(self, 'log') and not self.log.closed:
            self.log.flush()
            self.log.close()
            self._closed = True
    
    def __del__(self):
        """Ensure file is closed when object is deleted."""
        self.close()

