import logging
import os
from logging.handlers import RotatingFileHandler
from typing import Optional

class Logger:
    def __init__(
            self,
            name: str,
            log_file: str = 'app.log',
            log_dir: str = './.logs',
            max_bytes: int = 1000000,
            backup_count: int = 5,
            log_level: int = logging.WARNING
    ):
        """
        Initialize the Logger object.
        """
        self.name = name
        self.log_file = log_file
        self.log_dir = log_dir
        self.max_bytes = max_bytes
        self.backup_count = backup_count
        self.log_level = log_level
        self.logger: Optional[logging.Logger] = None
        self.console_handler: Optional[logging.StreamHandler] = None
        self.file_handler: Optional[RotatingFileHandler] = None
        self.setup_logger()

    def setup_logger(self) -> None:
        """
        Set up the logger with rotation, formatting, and console output.
        """
        log_formatter = logging.Formatter('%(asctime)s - %(levelname)s - %(message)s')

        # Ensure the log directory exists
        os.makedirs(self.log_dir, exist_ok=True)

        log_path = os.path.join(self.log_dir, self.log_file)

        # Use a rotating file handler for file logging
        self.file_handler = RotatingFileHandler(log_path, maxBytes=self.max_bytes, backupCount=self.backup_count)
        self.file_handler.setFormatter(log_formatter)
        self.file_handler.setLevel(self.log_level)

        # Use a stream handler for console logging
        self.console_handler = logging.StreamHandler()
        self.console_handler.setFormatter(log_formatter)
        self.console_handler.setLevel(self.log_level)

        # Configure logger
        self.logger = logging.getLogger(self.name)
        self.logger.setLevel(self.log_level)

        # Remove any pre-existing handlers
        self.logger.handlers = []

        # Add both handlers
        self.logger.addHandler(self.file_handler)
        self.logger.addHandler(self.console_handler)

    def set_log_level(self, level: int) -> None:
        """
        Set the logging level for both console and file handlers.
        """
        self.logger.setLevel(level)
        if self.console_handler:
            self.console_handler.setLevel(level)
        if self.file_handler:
            self.file_handler.setLevel(level)

    def get_logger(self) -> logging.Logger:
        """
        Get the configured logger.
        """
        return self.logger


if __name__ == '__main__':
    # Example of setting up the logger and dynamically changing log level
    logger_obj = Logger('main_logger', log_dir='./logs')
    logger = logger_obj.get_logger()

    logger.debug('This debug message will not appear because the default level is INFO')

    # Change log level to DEBUG to see debug messages
    logger_obj.set_log_level(logging.DEBUG)
    logger.debug('Now this debug message will appear')
    logger.info('This is an info message')
    logger.warning('This is a warning message')