import dataclasses
import logging
import sys
from collections import defaultdict
from dataclasses import dataclass
from functools import partial
from typing import Tuple, Optional, Dict, Any, List, Iterable, Iterator

import numpy as np
import torch
from dpu_utils.utils import RichPath
from pyprojroot import here as project_root

sys.path.insert(0, str(project_root()))

from data import (
    DataFold,
    FSMolDataset,
    BindingAffinityTask,
    FSMolBatch,
    RandomTaskSampler,
    FSMolBatcher,
    MoleculeDatapoint,
    fsmol_batch_finalizer,
)
from utils.torch_utils import torchify


logger = logging.getLogger(__name__)


@dataclass(frozen=False)
class FSMolMultitaskBatch(FSMolBatch):
    sample_to_task_id: np.ndarray


def multitask_batcher_init_fn(batch_data: Dict[str, Any]):
    batch_data["sample_to_task_id"] = []


def multitask_batcher_add_sample_fn(
    batch_data: Dict[str, Any],
    sample_id: int,
    sample: MoleculeDatapoint,
    task_name_to_id: Dict[str, int],
):
    batch_data["sample_to_task_id"].append(task_name_to_id[sample.task_name])


def multitask_batcher_finalizer_fn(
    batch_data: Dict[str, Any]
) -> Tuple[FSMolMultitaskBatch, np.ndarray]:
    plain_batch = fsmol_batch_finalizer(batch_data)
    return (
        FSMolMultitaskBatch(
            sample_to_task_id=np.stack(batch_data["sample_to_task_id"], axis=0),
            **dataclasses.asdict(plain_batch),
        ),
        np.stack(batch_data["numeric_labels"], axis=0),
    )


def get_multitask_batcher(
    task_name_to_id: Dict[str, int],
    max_num_graphs: Optional[int] = None,
    max_num_nodes: Optional[int] = None,
    max_num_edges: Optional[int] = None,
    device: Optional[torch.device] = None,
) -> FSMolBatcher[FSMolMultitaskBatch, np.ndarray]:
    def finalizer(batch_data: Dict[str, Any]):
        finalized_batch = multitask_batcher_finalizer_fn(batch_data)
        if device is not None:
            finalized_batch = torchify(finalized_batch, device)

        return finalized_batch

    return FSMolBatcher(
        max_num_graphs=max_num_graphs,
        max_num_nodes=max_num_nodes,
        max_num_edges=max_num_edges,
        init_callback=multitask_batcher_init_fn,
        per_datapoint_callback=partial(
            multitask_batcher_add_sample_fn, task_name_to_id=task_name_to_id
        ),
        finalizer_callback=finalizer,
    )


def get_multitask_inference_batcher(
    max_num_graphs: int,
    device: torch.device,
) -> FSMolBatcher[FSMolMultitaskBatch, np.ndarray]:
    # In this setting, we only consider a single task at a time, so they just all get the same ID:
    task_name_to_const_id: Dict[str, int] = defaultdict(lambda: 0)
    return get_multitask_batcher(
        task_name_to_id=task_name_to_const_id,
        max_num_graphs=max_num_graphs,
        device=device,
    )


class MultitaskTaskSampleBatchIterable(Iterable[Tuple[FSMolMultitaskBatch, torch.Tensor]]):
    def __init__(
        self,
        dataset: FSMolDataset,
        data_fold: DataFold,
        task_name_to_id: Dict[str, int],
        device: torch.device,
        max_num_graphs: Optional[int] = None,
        max_num_nodes: Optional[int] = None,
        max_num_edges: Optional[int] = None,
        num_chunked_tasks: int = 8,
        repeat: bool = False,
    ):
        self._dataset = dataset
        self._data_fold = data_fold
        self._num_chunked_tasks = num_chunked_tasks
        self._repeat = repeat
        self._device = device

        self._task_sampler = RandomTaskSampler(
            train_size_or_ratio=1024, valid_size_or_ratio=0, test_size_or_ratio=0
        )
        self._batcher = get_multitask_batcher(
            task_name_to_id=task_name_to_id,
            max_num_graphs=max_num_graphs,
            max_num_nodes=max_num_nodes,
            max_num_edges=max_num_edges,
        )

    def __iter__(self) -> Iterator[Tuple[FSMolMultitaskBatch, torch.Tensor]]:
        def paths_to_mixed_samples(
            paths: List[RichPath], idx: int
        ) -> Iterable[Tuple[FSMolMultitaskBatch, np.ndarray]]:
            loaded_samples: List[MoleculeDatapoint] = []
            for i, path in enumerate(paths):
                task = BindingAffinityTask.load_from_file(path)
                task_sample = self._task_sampler.sample(task, seed=idx + i)
                loaded_samples.extend(task_sample.train_samples)
            if self._data_fold == DataFold.TRAIN:
                np.random.shuffle(loaded_samples)

            for features, labels in self._batcher.batch(loaded_samples):
                yield features, labels
        return map(
            partial(torchify, device=self._device),
            iter(
                self._dataset.get_task_reading_iterable(
                    data_fold=self._data_fold,
                    task_reader_fn=paths_to_mixed_samples,
                    repeat=self._repeat,
                    reader_chunk_size=self._num_chunked_tasks,
                )
            ),
        )
