import copy
import os
from datetime import datetime
from typing import Optional

import pandas as pd

from src.logger.cross_val_logger import ManyFoldLogger
from src.utils.path_io import get_path_up_to


class MultiInstanceLogger:
    """
    A class used to log training information for multiple instances of a model.
    """
    logger: Optional[ManyFoldLogger]

    def __init__(self, name: str = "") -> None:

        self.name = name
        self.logger = None
        self.start_time_str = datetime.now().strftime("%Y-%m-%d_%H-%M-%S")
        self.start_time = datetime.now()

        self.results = []

    def next_logger(self) -> ManyFoldLogger:
        """
        Initializes a new crossvalidation logger.

        Returns: CrossValLogger
        """

        self.logger = ManyFoldLogger(name=self.name)

        return self.logger

    def collect_final_results(self) -> None:
        """
        Collects the final results of the active loggers.

        This method collects the final results from the current logger, adds train and model params from passed
        arguments and stores them into the list of results. Depending on the class attribute report_max_scores, either
        the last scores or the scores of the model in the best epoch are collected.
        """
        setting_results = {}
        # Add the logger name and start time to the results
        setting_results.update({'name': f'{self.logger.start_time_str}_{self.name}'})

        # Insert dicts like config at the beginning of the results dict
        for _, top_d in self.logger.storage['dicts'].items():
            for top_key, d in top_d.items():
                if isinstance(d, dict):
                    for key, value in d.items():
                        setting_results[f"{key}"] = value
                else:
                    setting_results[top_key] = d

        # Insert the results
        setting_results.update( self.logger.get_max_scores())

        # Insert any texts that have been logged
        for key, value in self.logger.storage['texts'].items():
            setting_results[key] = value

        self.results.append(setting_results)

    def save_final_results(self) -> None:
        """
        Saves the final results of the active loggers to a pandas DataFrame.

        This method saves the final results from the current logger into a pandas DataFrame.
        """
        df_results = pd.DataFrame(self.results)
        dir_path = os.path.join(get_path_up_to(os.path.abspath(__file__), "src"), "data", "output", "multi_instance_results")

        if not os.path.exists(dir_path):
            os.makedirs(dir_path)

        file_path = os.path.join(dir_path, f'{self.start_time_str}_{self.name}_results.csv')

        if os.path.exists(file_path):
            df_loaded = pd.read_csv(file_path)
            df_results = pd.concat([df_loaded, df_results], ignore_index=True)

        df_results.to_csv(os.path.join(dir_path ,f'{self.start_time_str}_{self.name}_results.csv'), index=False)


    def close(self) -> None:
        """
        Closes multi instance logger and saves the results into a pandas DataFrame.
        """
        current_time = datetime.now()
        duration = current_time - self.start_time
        days, remainder = divmod(duration.total_seconds(), 86400)
        hours, remainder = divmod(duration.total_seconds(), 3600)
        minutes, seconds = divmod(remainder, 60)

        print(f"[LOGGER]: Finished multi instance logging after {int(days)} days, {int(hours)} hours, {int(minutes)} minutes, and {int(seconds)} seconds.")