import os
import shutil
from typing import Dict, List

from src.domain_translator.translator import Concept, Translation, TranslationDictionary


def get_translations_that_need_more_images(
    translations_directory: str,
) -> Dict[int, List[Translation]]:
    return get_evaluated_translations(translations_directory, ["NEEDS_MORE_IMAGES"])


def get_successful_translations(
    translations_directory: str,
) -> Dict[int, List[Translation]]:
    return get_evaluated_translations(translations_directory, ["OK"])


def get_evaluated_translations(
    translations_directory: str,
    evaluations: List[str],
) -> Dict[int, List[Translation]]:
    translations_dict_path = os.path.join(translations_directory, "translations.json")
    translations_dict = TranslationDictionary.load(translations_dict_path)

    problem_to_successful_translations = {}

    for problem_id, translations in translations_dict.items():
        successful_translations = [
            translation
            for translation in translations
            if translation.evaluation in evaluations
        ]

        problem_to_successful_translations[problem_id] = successful_translations

    return problem_to_successful_translations


def get_translation_concept_files(
    data_path: str, problem_id: int, translation: Translation, concept: Concept
) -> List[str]:
    sanitized_left = translation.left.concept.replace(":", "")
    sanitized_right = translation.right.concept.replace(":", "")

    directory_path = os.path.join(
        data_path,
        str(problem_id),
        f"{sanitized_left} vs {sanitized_right}",
        concept.concept.replace(":", ""),
    )

    approved_images = [
        img_id
        for img_id, img_check in concept.img_checks.items()
        if img_check.decision == "OK"
    ]

    return [
        os.path.join(directory_path, file)
        for file in os.listdir(directory_path)
        if os.path.isfile(os.path.join(directory_path, file))
        and os.path.splitext(file)[0] in approved_images
    ]


def copy_picked_images(
    data_path: str,
    problem_id: int,
    translation: Translation,
    img_index: int,
    output_path: str,
    img_number: int,
):
    copy_side_image(
        data_path,
        problem_id,
        translation,
        img_index,
        output_path,
        "left",
        translation.left,
        img_number,
    )
    copy_side_image(
        data_path,
        problem_id,
        translation,
        img_index,
        output_path,
        "right",
        translation.right,
        img_number,
    )


def copy_side_image(
    data_path: str,
    problem_id: int,
    translation: Translation,
    img_index: int,
    output_path: str,
    side: str,
    concept: str,
    img_number: int,
):
    concept_files = get_translation_concept_files(
        data_path, problem_id, translation, concept
    )

    picked_file = concept_files[img_index]

    _, extension = os.path.splitext(picked_file)
    side_directory = os.path.join(output_path, side)
    os.makedirs(side_directory, exist_ok=True)
    shutil.copy(picked_file, os.path.join(side_directory, f"{img_number}{extension}"))


def generate_dataset_from_translations(
    data_path: str, output_path: str, n_mix_concepts: int = 1
):
    problem_to_successful_translations = get_evaluated_translations(
        data_path, ["OK", "NEEDS_MORE_IMAGES"]
    )

    for problem_id, translations in problem_to_successful_translations.items():
        if len(translations) < n_mix_concepts + 1:
            continue

        picked_translations = translations[:n_mix_concepts]

        problem_directory = os.path.join(output_path, str(problem_id))

        for i in range(6):
            picked_translation = picked_translations[i % n_mix_concepts]
            picked_img_index = i // n_mix_concepts

            copy_picked_images(
                data_path,
                problem_id,
                picked_translation,
                picked_img_index,
                problem_directory,
                i,
            )

        picked_translation = translations[n_mix_concepts]

        copy_picked_images(
            data_path,
            problem_id,
            picked_translation,
            0,
            problem_directory,
            6,
        )
