#!/usr/bin/env python3
"""
Map embeddings to predictions for every downstream task and store
test predictions to disk.

Model selection over the validation score.

TODO:
    * Profiling should occur here (both embedding time AFTER loading
    to GPU, and complete wall time include disk writes).
    * If disk speed is the limiting factor maybe we should train
    many models simultaneously with one disk read?
"""

import copy
import json
import logging
import multiprocessing
import pickle
import random
import sys
import time
from collections import defaultdict
from pathlib import Path
from typing import Any, DefaultDict, Dict, List, Optional, Tuple, Union

import more_itertools
import numpy as np
import pandas as pd
import pytorch_lightning as pl
import torch
import torchinfo

# import wandb
from pytorch_lightning import seed_everything
from pytorch_lightning.callbacks.early_stopping import EarlyStopping
from pytorch_lightning.callbacks.model_checkpoint import ModelCheckpoint
from pytorch_lightning.loggers import CSVLogger
from scipy.ndimage import median_filter
from sklearn.model_selection import ParameterGrid
from torch.utils.data import ConcatDataset, DataLoader, Dataset
from tqdm.auto import tqdm

from heareval.score import (
    ScoreFunction,
    available_scores,
    label_to_binary_vector,
    label_vocab_as_dict,
    validate_score_return_type,
)

TASK_SPECIFIC_PARAM_GRID = {
    "dcase2016_task2": {
        # sed_eval is very slow
        "check_val_every_n_epoch": [10],
    }
}

PARAM_GRID = {
    "hidden_layers": [1, 2],
    # "hidden_layers": [0, 1, 2],
    # "hidden_layers": [1, 2, 3],
    "hidden_dim": [1024],
    # "hidden_dim": [256, 512, 1024],
    # "hidden_dim": [1024, 512],
    # Encourage 0.5
    "dropout": [0.1],
    # "dropout": [0.1, 0.5],
    # "dropout": [0.1, 0.3],
    # "dropout": [0.1, 0.3, 0.5],
    # "dropout": [0.0, 0.1, 0.2, 0.3, 0.4, 0.5],
    # "dropout": [0.0, 0.1, 0.2, 0.3, 0.4, 0.5, 0.6, 0.7, 0.8],
    "lr": [3.2e-3, 1e-3, 3.2e-4, 1e-4],
    # "lr": [3.2e-3, 1e-3, 3.2e-4, 1e-4, 3.2e-5, 1e-5],
    # "lr": [1e-2, 3.2e-3, 1e-3, 3.2e-4, 1e-4],
    # "lr": [1e-1, 1e-2, 1e-3, 1e-4, 1e-5],
    "patience": [20],
    "max_epochs": [500],
    # "max_epochs": [500, 1000],
    "check_val_every_n_epoch": [3],
    # "check_val_every_n_epoch": [1, 3, 10],
    "batch_size": [1024],
    # "batch_size": [1024, 2048],
    # "batch_size": [256, 512, 1024],
    # "batch_size": [256, 512, 1024, 2048, 4096, 8192],
    "hidden_norm": [torch.nn.BatchNorm1d],
    # "hidden_norm": [torch.nn.Identity, torch.nn.BatchNorm1d, torch.nn.LayerNorm],
    "norm_after_activation": [False],
    # "norm_after_activation": [False, True],
    "embedding_norm": [torch.nn.Identity],
    # "embedding_norm": [torch.nn.Identity, torch.nn.BatchNorm1d],
    # "embedding_norm": [torch.nn.Identity, torch.nn.BatchNorm1d, torch.nn.LayerNorm],
    "initialization": [torch.nn.init.xavier_uniform_, torch.nn.init.xavier_normal_],
    "optim": [torch.optim.Adam],
    # "optim": [torch.optim.Adam, torch.optim.SGD],
}

FAST_PARAM_GRID = copy.deepcopy(PARAM_GRID)
FAST_PARAM_GRID.update(
    {
        "max_epochs": [50],
        "check_val_every_n_epoch": [3, 10],
    }
)

FASTER_PARAM_GRID = copy.deepcopy(PARAM_GRID)
FASTER_PARAM_GRID.update(
    {
        "hidden_layers": [0, 1],
        "hidden_dim": [64, 128],
        "patience": [1, 3],
        "max_epochs": [10],
        "check_val_every_n_epoch": [1],
    }
)

# These are good for dcase, change for other event-based secret tasks
EVENT_POSTPROCESSING_GRID = {
    "median_filter_ms": [250],
    "min_duration": [125, 250],
    #    "median_filter_ms": [0, 62, 125, 250, 500, 1000],
    #    "min_duration": [0, 62, 125, 250, 500, 1000],
}

NUM_WORKERS = int(multiprocessing.cpu_count() / (max(1, torch.cuda.device_count())))


class OneHotToCrossEntropyLoss(pl.LightningModule):
    def __init__(self):
        super().__init__()
        self.loss = torch.nn.CrossEntropyLoss()

    def forward(self, y_hat: torch.Tensor, y: torch.Tensor) -> torch.Tensor:
        # One and only one label per class
        assert torch.all(
            torch.sum(y, dim=1) == torch.ones(y.shape[0], device=self.device)
        )
        y = y.argmax(dim=1)
        return self.loss(y_hat, y)


class FullyConnectedPrediction(torch.nn.Module):
    def __init__(self, nfeatures: int, nlabels: int, prediction_type: str, conf: Dict):
        super().__init__()

        hidden_modules: List[torch.nn.Module] = []
        curdim = nfeatures
        # Honestly, we don't really know what activation preceded
        # us for the final embedding.
        last_activation = "linear"
        if conf["hidden_layers"]:
            for i in range(conf["hidden_layers"]):
                linear = torch.nn.Linear(curdim, conf["hidden_dim"])
                conf["initialization"](
                    linear.weight,
                    gain=torch.nn.init.calculate_gain(last_activation),
                )
                hidden_modules.append(linear)
                if not conf["norm_after_activation"]:
                    hidden_modules.append(conf["hidden_norm"](conf["hidden_dim"]))
                hidden_modules.append(torch.nn.Dropout(conf["dropout"]))
                hidden_modules.append(torch.nn.ReLU())
                last_activation = "relu"

                if conf["norm_after_activation"]:
                    hidden_modules.append(conf["hidden_norm"](conf["hidden_dim"]))
                curdim = conf["hidden_dim"]

            self.hidden = torch.nn.Sequential(*hidden_modules)
        else:
            self.hidden = torch.nn.Identity()  # type: ignore
        self.projection = torch.nn.Linear(curdim, nlabels)

        gain = torch.nn.init.calculate_gain(last_activation)
        conf["initialization"](self.projection.weight, gain=gain)
        self.logit_loss: torch.nn.Module
        if prediction_type == "multilabel":
            self.activation: torch.nn.Module = torch.nn.Sigmoid()
            self.logit_loss = torch.nn.BCEWithLogitsLoss()
        elif prediction_type == "multiclass" or (
            prediction_type == "polar-classification"
        ):
            self.activation = torch.nn.Softmax(dim=1)
            self.logit_loss = OneHotToCrossEntropyLoss()
        elif prediction_type == "cartesian-regression":
            self.activation = torch.nn.Tanh()
            self.logit_loss = torch.nn.MSELoss()
        elif prediction_type == "polar-regression":
            self.activation = torch.nn.Identity()
            self.logit_loss = torch.nn.MSELoss()
        else:
            raise ValueError(f"Unknown prediction_type {prediction_type}")

    def forward_logit(self, x: torch.Tensor) -> torch.Tensor:
        x = self.hidden(x)
        x = self.projection(x)
        return x

    def forward(self, x: torch.Tensor) -> torch.Tensor:
        x = self.forward_logit(x)
        x = self.activation(x)
        return x


class AbstractPredictionModel(pl.LightningModule):
    def __init__(
        self,
        nfeatures: int,
        label_to_idx: Dict[str, int],
        nlabels: int,
        prediction_type: str,
        scores: List[ScoreFunction],
        conf: Dict,
        use_scoring_for_early_stopping: bool = True,
    ):
        super().__init__()

        self.save_hyperparameters(conf)
        self.use_scoring_for_early_stopping = use_scoring_for_early_stopping

        # Since we don't know how these embeddings are scaled
        self.layernorm = conf["embedding_norm"](nfeatures)
        self.predictor = FullyConnectedPrediction(
            nfeatures, nlabels, prediction_type, conf
        )
        torchinfo.summary(self.predictor, input_size=(64, nfeatures))
        self.label_to_idx = label_to_idx
        self.idx_to_label: Dict[int, str] = {
            idx: label for (label, idx) in self.label_to_idx.items()
        }
        self.scores = scores

    def forward(self, x):
        # x = self.layernorm(x)
        x = self.predictor(x)
        return x

    def training_step(self, batch, batch_idx):
        # training_step defined the train loop.
        # It is independent of forward
        x, y, _ = batch
        y = y.float()
        y_hat = self.predictor.forward_logit(x)
        loss = self.predictor.logit_loss(y_hat, y)
        # Logging to TensorBoard by default
        self.log("train_loss", loss)
        return loss

    def _step(self, batch, batch_idx):
        # -> Dict[str, Union[torch.Tensor, List(str)]]:
        x, y, metadata = batch
        y = y.float()
        y_hat = self.predictor.forward_logit(x)
        y_pr = self.predictor(x)
        z = {
            "prediction": y_pr,
            "prediction_logit": y_hat,
            "target": y,
        }
        # https://stackoverflow.com/questions/38987/how-do-i-merge-two-dictionaries-in-a-single-expression-taking-union-of-dictiona
        return {**z, **metadata}

    def validation_step(self, batch, batch_idx):
        return self._step(batch, batch_idx)

    def test_step(self, batch, batch_idx):
        return self._step(batch, batch_idx)

    def log_scores(self, name: str, score_args):
        """Logs the metric score value for each score defined for the model"""
        assert hasattr(self, "scores"), "Scores for the model should be defined"
        end_scores = {}
        # The first score in the first `self.scores` is the optimization criterion
        for score in self.scores:
            score_ret = score(*score_args)
            validate_score_return_type(score_ret)
            # If the returned score is a tuple, store each subscore as separate entry
            if isinstance(score_ret, tuple):
                end_scores[f"{name}_{score}"] = score_ret[0][1]
                # All other scores will also be logged
                for subscore, value in score_ret:
                    end_scores[f"{name}_{score}_{subscore}"] = value
            elif isinstance(score_ret, float):
                end_scores[f"{name}_{score}"] = score_ret
            else:
                raise ValueError(
                    f"Return type {type(score_ret)} is unexpected. Return type of "
                    "the score function should either be a "
                    "tuple(tuple) or float."
                )

        self.log(
            f"{name}_score", end_scores[f"{name}_{str(self.scores[0])}"], logger=True
        )
        for score_name in end_scores:
            self.log(score_name, end_scores[score_name], prog_bar=True, logger=True)

    # Implement this for each inheriting class
    # TODO: Can we combine the boilerplate for both of these?
    def _score_epoch_end(self, name: str, outputs: List[Dict[str, List[Any]]]):
        """
        Return at the end of every validation and test epoch.
        :param name: "val" or "test"
        :param outputs: Unflattened minibatches from {name}_step,
            each with "target", "prediction", and additional metadata,
            with a list of values for each instance in the batch.
        :return:
        """
        raise NotImplementedError("Implement this in children")

    def validation_epoch_end(self, outputs: List[Dict[str, List[Any]]]):
        self._score_epoch_end("val", outputs)

    def test_epoch_end(self, outputs: List[Dict[str, List[Any]]]):
        self._score_epoch_end("test", outputs)

    def _flatten_batched_outputs(
        self,
        outputs,  #: Union[torch.Tensor, List[str]],
        keys: List[str],
        dont_stack: List[str] = [],
    ) -> Dict:
        # ) -> Dict[str, Union[torch.Tensor, List[str]]]:
        flat_outputs_default: DefaultDict = defaultdict(list)
        for output in outputs:
            assert set(output.keys()) == set(keys), f"{output.keys()} != {keys}"
            for key in keys:
                flat_outputs_default[key] += output[key]
        flat_outputs = dict(flat_outputs_default)
        for key in keys:
            if key in dont_stack:
                continue
            else:
                flat_outputs[key] = torch.stack(flat_outputs[key])
        return flat_outputs

    def configure_optimizers(self):
        optimizer = self.hparams.optim(self.parameters(), lr=self.hparams.lr)
        return optimizer


class ScenePredictionModel(AbstractPredictionModel):
    """
    Prediction model with simple scoring over entire audio scenes.
    """

    def __init__(
        self,
        nfeatures: int,
        label_to_idx: Dict[str, int],
        nlabels: int,
        prediction_type: str,
        scores: List[ScoreFunction],
        conf: Dict,
        use_scoring_for_early_stopping: bool = True,
    ):
        super().__init__(
            nfeatures=nfeatures,
            label_to_idx=label_to_idx,
            nlabels=nlabels,
            prediction_type=prediction_type,
            scores=scores,
            conf=conf,
            use_scoring_for_early_stopping=use_scoring_for_early_stopping,
        )

    def _score_epoch_end(self, name: str, outputs: List[Dict[str, List[Any]]]):
        flat_outputs = self._flatten_batched_outputs(
            outputs, keys=["target", "prediction", "prediction_logit"]
        )
        target, prediction, prediction_logit = (
            flat_outputs[key] for key in ["target", "prediction", "prediction_logit"]
        )

        self.log(
            f"{name}_loss",
            self.predictor.logit_loss(prediction_logit, target),
            prog_bar=True,
            logger=True,
        )

        if name == "test":
            # Cache all predictions for later serialization
            self.test_predictions = {
                "target": target.detach().cpu(),
                "prediction": prediction.detach().cpu(),
                "prediction_logit": prediction_logit.detach().cpu(),
            }

        if name == "test" or self.use_scoring_for_early_stopping:
            self.log_scores(
                name,
                score_args=(
                    prediction.detach().cpu().numpy(),
                    target.detach().cpu().numpy(),
                ),
            )


class EventPredictionModel(AbstractPredictionModel):
    """
    Event prediction model. For validation (and test),
    we combine timestamp events that are adjacent,
    but discard ones that are too short.
    """

    def __init__(
        self,
        nfeatures: int,
        label_to_idx: Dict[str, int],
        nlabels: int,
        prediction_type: str,
        scores: List[ScoreFunction],
        validation_target_events: Dict[str, List[Dict[str, Any]]],
        test_target_events: Dict[str, List[Dict[str, Any]]],
        postprocessing_grid: Dict[str, List[float]],
        conf: Dict,
        use_scoring_for_early_stopping: bool = True,
    ):
        super().__init__(
            nfeatures=nfeatures,
            label_to_idx=label_to_idx,
            nlabels=nlabels,
            prediction_type=prediction_type,
            scores=scores,
            conf=conf,
            use_scoring_for_early_stopping=use_scoring_for_early_stopping,
        )
        self.target_events = {
            "val": validation_target_events,
            "test": test_target_events,
        }
        # For each epoch, what postprocessing parameters were best
        self.epoch_best_postprocessing: Dict[int, Tuple[Tuple[str, Any], ...]] = {}
        self.postprocessing_grid = postprocessing_grid

    def epoch_best_postprocessing_or_default(
        self, epoch: int
    ) -> Tuple[Tuple[str, Any], ...]:
        if self.use_scoring_for_early_stopping:
            try:
                return self.epoch_best_postprocessing[epoch]
            except KeyError:
                print("Got key erorr with epoch number : {k} using the last epoch")
                return self.epoch_best_postprocessing[epoch - 1]
        else:
            postprocessing_confs = list(ParameterGrid(self.postprocessing_grid))
            # There can only be one kind of postprocessing
            assert len(postprocessing_confs) == 1
            return tuple(postprocessing_confs[0].items())

    def _score_epoch_end(self, name: str, outputs: List[Dict[str, List[Any]]]):
        flat_outputs = self._flatten_batched_outputs(
            outputs,
            keys=["target", "prediction", "prediction_logit", "filename", "timestamp"],
            # This is a list of string, not tensor, so we don't need to stack it
            dont_stack=["filename"],
        )
        target, prediction, prediction_logit, filename, timestamp = (
            flat_outputs[key]
            for key in [
                "target",
                "prediction",
                "prediction_logit",
                "filename",
                "timestamp",
            ]
        )

        self.log(
            f"{name}_loss",
            self.predictor.logit_loss(prediction_logit, target),
            prog_bar=True,
            logger=True,
        )

        epoch = self.current_epoch
        if name == "val":
            postprocessing_cached = None
        elif name == "test":
            postprocessing_cached = self.epoch_best_postprocessing_or_default(epoch)
        else:
            raise ValueError
        # print("\n\n\n", epoch)

        if name == "test" or self.use_scoring_for_early_stopping:
            predicted_events_by_postprocessing = get_events_for_all_files(
                prediction,
                filename,
                timestamp,
                self.idx_to_label,
                self.postprocessing_grid,
                postprocessing_cached,
            )

            score_and_postprocessing = []
            for postprocessing in tqdm(predicted_events_by_postprocessing):
                predicted_events = predicted_events_by_postprocessing[postprocessing]
                primary_score_fn = self.scores[0]
                primary_score_ret = primary_score_fn(
                    # predicted_events, self.target_events[name]
                    predicted_events,
                    self.target_events[name],
                )
                # If the score returns a tuple of scores, the first score
                # is used
                if isinstance(primary_score_ret, tuple):
                    primary_score = primary_score_ret[0][1]
                elif isinstance(primary_score_ret, float):
                    primary_score = primary_score_ret
                else:
                    raise ValueError(
                        f"Return type {type(primary_score_ret)} is unexpected. "
                        "Return type of the score function should either be a "
                        "tuple(tuple) or float. "
                    )
                if np.isnan(primary_score):
                    primary_score = 0.0
                score_and_postprocessing.append((primary_score, postprocessing))
            score_and_postprocessing.sort(reverse=True)

            # for vs in score_and_postprocessing:
            #    print(vs)

            best_postprocessing = score_and_postprocessing[0][1]
            if name == "val":
                print("BEST POSTPROCESSING", best_postprocessing)
                for k, v in best_postprocessing:
                    self.log(f"postprocessing/{k}", v, logger=True)
                print(epoch, flush=True)
                self.epoch_best_postprocessing[epoch] = best_postprocessing
            predicted_events = predicted_events_by_postprocessing[best_postprocessing]

            if name == "test":
                # Cache all predictions for later serialization
                self.test_predictions = {
                    "target": target.detach().cpu(),
                    "prediction": prediction.detach().cpu(),
                    "prediction_logit": prediction_logit.detach().cpu(),
                    "target_events": self.target_events[name],
                    "predicted_events": predicted_events,
                    "timestamp": timestamp,
                }

            self.log_scores(
                name, score_args=(predicted_events, self.target_events[name])
            )


class SplitMemmapDataset(Dataset):
    """
    Embeddings are memmap'ed, unless in-memory = True.

    WARNING: Don't shuffle this or access will be SLOW.
    """

    def __init__(
        self,
        embedding_path: Path,
        label_to_idx: Dict[str, int],
        nlabels: int,
        split_name: str,
        embedding_type: str,
        in_memory: bool,
        metadata: bool,
        task: str,
        is_test: bool,
        random_probe: bool,
    ):
        self.embedding_path = embedding_path
        self.label_to_idx = label_to_idx
        self.nlabels = nlabels
        self.split_name = split_name
        self.embedding_type = embedding_type
        self.task = task
        self.is_test = is_test
        self.random_probe = random_probe

        self.dim = tuple(
            json.load(
                open(embedding_path.joinpath(f"{split_name}.embedding-dimensions.json"))
            )
        )
        self.embeddings = np.memmap(
            filename=embedding_path.joinpath(f"{split_name}.embeddings.npy"),
            dtype=np.float32,
            mode="r",
            shape=self.dim,
        )
        if in_memory:
            self.embeddings = torch.stack(
                [torch.tensor(e) for e in tqdm(self.embeddings)]
            )
            nandim = self.embeddings.isnan().sum().tolist()
            infdim = self.embeddings.isinf().sum().tolist()
            assert nandim == 0 and infdim == 0

        self.labels = pickle.load(
            open(embedding_path.joinpath(f"{split_name}.target-labels.pkl"), "rb")
        )

        # Only used for event-based prediction, for validation and test scoring,
        # For timestamp (event) embedding tasks,
        # the metadata for each instance is {filename: , timestamp: }.
        if self.embedding_type == "event" and metadata:
            filename_timestamps_json = embedding_path.joinpath(
                f"{split_name}.filename-timestamps.json"
            )
            self.metadata = [
                {"filename": filename, "timestamp": timestamp}
                for filename, timestamp in json.load(open(filename_timestamps_json))
            ]
        else:
            self.metadata = [{}] * self.dim[0]
        assert len(self.labels) == self.dim[0]
        assert len(self.labels) == len(self.embeddings)
        assert len(self.labels) == len(self.metadata)
        assert self.embeddings[0].shape[0] == self.dim[1]

        """
        For all labels, return a multi or one-hot vector.
        This allows us to have tensors that are all the same shape.
        Later we reduce this with an argmax to get the vocabulary indices.
        """
        ys = []
        for idx in tqdm(range(len(self.labels))):
            # If we indeed have self.label_to_idx!
            if (
                ("cartesian" not in task)
                and ("polar" not in task)
                and ("distance" not in task)
            ):
                labels = [self.label_to_idx[str(label)] for label in self.labels[idx]]
                y = label_to_binary_vector(labels, self.nlabels)
            elif "distance" in task:
                y = torch.abs(torch.tensor(self.labels[idx]))
            else:
                y = torch.tensor(self.labels[idx])
            ys.append(y)
        self.y = torch.stack(ys)
        print(self.y.shape)
        print(len(self.labels))
        if self.y.ndim == 1:
            self.y = self.y.unsqueeze(1)
        assert self.y.shape == (len(self.labels), self.nlabels)

    def __len__(self) -> int:
        return self.dim[0]

    def __getitem__(self, idx) -> Tuple[torch.Tensor, torch.Tensor, Dict[str, Any]]:
        if self.random_probe:
            random_y_idx = random.randint(0, len(self) - 1)
            return (
                self.embeddings[idx],
                self.y[idx if self.is_test else random_y_idx],
                self.metadata[idx],
            )
        else:
            return self.embeddings[idx], self.y[idx], self.metadata[idx]


def create_events_from_prediction(
    prediction_dict: Dict[float, torch.Tensor],
    idx_to_label: Dict[int, str],
    threshold: float = 0.5,
    median_filter_ms: float = 150,
    min_duration: float = 60.0,
) -> List[Dict[str, Union[float, str]]]:
    """
    Takes a set of prediction tensors keyed on timestamps and generates events.
    (This is for one particular audio scene.)
    We convert the prediction tensor to a binary label based on the threshold value. Any
    events occurring at adjacent timestamps are considered to be part of the same event.
    This loops through and creates events for each label class.
    We optionally apply median filtering to predictions.
    We disregard events that are less than the min_duration milliseconds.

    Args:
        prediction_dict: A dictionary of predictions keyed on timestamp
            {timestamp -> prediction}. The prediction is a tensor of label
            probabilities.
        idx_to_label: Index to label mapping.
        threshold: Threshold for determining whether to apply a label
        min_duration: the minimum duration in milliseconds for an
                event to be included.

    Returns:
        A list of dicts withs keys "label", "start", and "end"
    """
    # Make sure the timestamps are in the correct order
    timestamps = np.array(sorted(prediction_dict.keys()))

    # Create a sorted numpy matrix of frame level predictions for this file. We convert
    # to a numpy array here before applying a median filter.
    predictions = np.stack(
        [prediction_dict[t].detach().cpu().numpy() for t in timestamps]
    )

    # Optionally apply a median filter here to smooth out events.
    ts_diff = np.mean(np.diff(timestamps))
    if median_filter_ms:
        filter_width = int(round(median_filter_ms / ts_diff))
        if filter_width:
            predictions = median_filter(predictions, size=(filter_width, 1))

    # Convert probabilities to binary vectors based on threshold
    predictions = (predictions > threshold).astype(np.int8)

    events = []
    for label in range(predictions.shape[1]):
        for group in more_itertools.consecutive_groups(
            np.where(predictions[:, label])[0]
        ):
            grouptuple = tuple(group)
            assert tuple(sorted(grouptuple)) == grouptuple, (
                f"{sorted(grouptuple)} != {grouptuple}"
            )
            startidx, endidx = (grouptuple[0], grouptuple[-1])

            start = timestamps[startidx]
            end = timestamps[endidx]
            # Add event if greater than the minimum duration threshold
            if end - start >= min_duration:
                events.append(
                    {"label": idx_to_label[label], "start": start, "end": end}
                )

    # This is just for pretty output, not really necessary
    events.sort(key=lambda k: k["start"])
    return events


def get_events_for_all_files(
    predictions: torch.Tensor,
    filenames: List[str],
    timestamps: torch.Tensor,
    idx_to_label: Dict[int, str],
    postprocessing_grid: Dict[str, List[float]],
    postprocessing: Optional[Tuple[Tuple[str, Any], ...]] = None,
) -> Dict[Tuple[Tuple[str, Any], ...], Dict[str, List[Dict[str, Union[str, float]]]]]:
    """
    Produces lists of events from a set of frame based label probabilities.
    The input prediction tensor may contain frame predictions from a set of different
    files concatenated together. file_timestamps has a list of filenames and
    timestamps for each frame in the predictions tensor.

    We split the predictions into separate tensors based on the filename and compute
    events based on those individually.

    If no postprocessing is specified (during training), we try a
    variety of ways of postprocessing the predictions into events,
    from the postprocessing_grid including median filtering and
    minimum event length.

    If postprocessing is specified (during test, chosen at the best
    validation epoch), we use this postprocessing.

    Args:
        predictions: a tensor of frame based multi-label predictions.
        filenames: a list of filenames where each entry corresponds
            to a frame in the predictions tensor.
        timestamps: a list of timestamps where each entry corresponds
            to a frame in the predictions tensor.
        idx_to_label: Index to label mapping.
        postprocessing: See above.

    Returns:
        A dictionary from filtering params to the following values:
        A dictionary of lists of events keyed on the filename slug.
        The event list is of dicts of the following format:
            {"label": str, "start": float ms, "end": float ms}
    """
    # This probably could be more efficient if we make the assumption that
    # timestamps are in sorted order. But this makes sure of it.
    assert predictions.shape[0] == len(filenames)
    assert predictions.shape[0] == len(timestamps)
    event_files: Dict[str, Dict[float, torch.Tensor]] = {}
    for i, (filename, timestamp) in enumerate(zip(filenames, timestamps)):
        slug = Path(filename).name

        # Key on the slug to be consistent with the ground truth
        if slug not in event_files:
            event_files[slug] = {}

        # Save the predictions for the file keyed on the timestamp
        event_files[slug][float(timestamp)] = predictions[i]

    # Create events for all the different files. Store all the events as a dictionary
    # with the same format as the ground truth from the luigi pipeline.
    # Ex) { slug -> [{"label" : "woof", "start": 0.0, "end": 2.32}, ...], ...}
    event_dict: Dict[
        Tuple[Tuple[str, Any], ...], Dict[str, List[Dict[str, Union[float, str]]]]
    ] = {}
    if postprocessing:
        postprocess = postprocessing
        event_dict[postprocess] = {}
        for slug, timestamp_predictions in event_files.items():
            event_dict[postprocess][slug] = create_events_from_prediction(
                timestamp_predictions, idx_to_label, **dict(postprocess)
            )
    else:
        postprocessing_confs = list(ParameterGrid(postprocessing_grid))
        for postprocess_dict in tqdm(postprocessing_confs):
            postprocess = tuple(postprocess_dict.items())
            event_dict[postprocess] = {}
            for slug, timestamp_predictions in event_files.items():
                event_dict[postprocess][slug] = create_events_from_prediction(
                    timestamp_predictions, idx_to_label, **postprocess_dict
                )

    return event_dict


def label_vocab_nlabels(embedding_path: Path) -> Tuple[pd.DataFrame, int]:
    # Reads the label vocab from the labelvocabulary.csv
    label_vocab = pd.read_csv(embedding_path.joinpath("labelvocabulary.csv"))
    nlabels = len(label_vocab)
    assert nlabels == label_vocab["idx"].max() + 1
    return (label_vocab, nlabels)


def dataloader_from_split_name(
    split_name: Union[str, List[str]],
    embedding_path: Path,
    label_to_idx: Dict[str, int],
    nlabels: int,
    embedding_type: str,
    in_memory: bool,
    task: str,
    metadata: bool = True,
    batch_size: int = 64,
    pin_memory: bool = True,
    is_test: bool = False,
    random_probe: bool = False,
) -> DataLoader:
    """
    Get the dataloader for a `split_name` or a list of `split_name`

    For a list of `split_name`, the dataset for each split will be concatenated.

    Case 1 - split_name is a string
        The Dataloader is built from a single data split.
    Case 2 - split_name is a list of string
        The Dataloader combines the data from the list of splits and
        returns a combined dataloader. This is useful when combining
        multiple folds of data to create the training or validation
        dataloader. For example, in k-fold, the training data-loader
        might be made from the first 4/5 folds, and calling this function
        with [fold00, fold01, fold02, fold03] will create the
        required dataloader
    """
    if isinstance(split_name, (list, set)):
        dataset = ConcatDataset(
            [
                # We need to do something with label_to_idx here...
                # Because we already have the labels!
                SplitMemmapDataset(
                    embedding_path=embedding_path,
                    label_to_idx=label_to_idx,
                    nlabels=nlabels,
                    split_name=name,
                    embedding_type=embedding_type,
                    in_memory=in_memory,
                    metadata=metadata,
                    task=task,
                    is_test=is_test,
                    random_probe=random_probe,
                )
                for name in split_name
            ]
        )
    elif isinstance(split_name, str):
        dataset = SplitMemmapDataset(
            embedding_path=embedding_path,
            label_to_idx=label_to_idx,
            nlabels=nlabels,
            split_name=split_name,
            embedding_type=embedding_type,
            in_memory=in_memory,
            metadata=metadata,
            task=task,
            is_test=is_test,
            random_probe=random_probe,
        )
    else:
        raise ValueError("split_name should be a list or string")

    print(
        f"Getting embeddings for split {split_name}, "
        + f"which has {len(dataset)} instances."
    )

    # It is not recommended to return CUDA tensors using multi-processing
    # If automatic memory pinning is set to True then the num_workers should be zero
    # https://pytorch.org/docs/stable/data.html#single-and-multi-process-data-loading
    if in_memory and not pin_memory:
        num_workers = NUM_WORKERS
    else:
        # We are disk bound or using automatic memory pinning,
        # so multiple workers might cause thrashing and slowdowns
        num_workers = 0

    if in_memory and split_name == "train":
        shuffle = True
    else:
        # We don't shuffle if we are memmap'ing from disk
        # We don't shuffle validation and test, to maintain the order
        # of the event metadata
        shuffle = False
    return DataLoader(
        dataset,
        batch_size=batch_size,
        shuffle=shuffle,
        pin_memory=pin_memory,
        num_workers=num_workers,
    )


class GridPointResult:
    def __init__(
        self,
        predictor,
        model_path: str,
        epoch: int,
        time_in_min: float,
        hparams: Dict[str, Any],
        postprocessing: Tuple[Tuple[str, Any], ...],
        trainer: pl.Trainer,
        validation_score: float,
        score_mode: str,
        conf: Dict,
    ):
        self.predictor = predictor
        self.model_path = model_path
        self.epoch = epoch
        self.time_in_min = time_in_min
        self.hparams = hparams
        self.postprocessing = postprocessing
        self.trainer = trainer
        self.validation_score = validation_score
        self.score_mode = score_mode
        self.conf = conf

    def __repr__(self):
        return json.dumps(
            (
                self.validation_score,
                self.epoch,
                hparams_to_json(self.hparams),
                self.postprocessing,
            )
        )


def task_predictions_train(
    embedding_path: Path,
    embedding_size: int,
    metadata: Dict[str, Any],
    data_splits: Dict[str, List[str]],
    label_to_idx: Dict[str, int],
    nlabels: int,
    scores: List[ScoreFunction],
    conf: Dict,
    use_scoring_for_early_stopping: bool,
    gpus: Any,
    in_memory: bool,
    deterministic: bool,
    task: str,
    random_probe: bool = False,
) -> GridPointResult:
    """
    Train a predictor for a specific task using pre-computed embeddings.
    """

    start = time.time()
    predictor: AbstractPredictionModel
    if metadata["embedding_type"] == "event":

        def _combine_target_events(split_names: List[str]):
            """
            This combines the target events from the list of splits and
            returns the combined target events. This is useful when combining
            multiple folds of data to create the training or validation
            dataloader. For example, in k-fold, the training data-loader
            might be made from the first 4/5 folds, and calling this function
            with [fold00, fold01, fold02, fold03] will return the
            aggregated target events across all the folds
            """
            combined_target_events: Dict = {}
            for split_name in split_names:
                target_events = json.load(
                    embedding_path.joinpath(f"{split_name}.json").open()
                )
                common_keys = set(combined_target_events.keys()).intersection(
                    target_events.keys()
                )
                assert len(common_keys) == 0, (
                    "Target events from one split should not override "
                    "target events from another. This is very unlikely as the "
                    "target_event is keyed on the files which are distinct for "
                    "each split"
                )
                combined_target_events.update(target_events)
            return combined_target_events

        validation_target_events: Dict = _combine_target_events(data_splits["valid"])
        test_target_events: Dict = _combine_target_events(data_splits["test"])

        # The postprocessing search space for getting the
        # best task specific postprocessing, can be task
        # specific, present in the task metadata in
        # evaluation_params.postprocessing_grid. If not, the default
        # EVENT_POSTPROCESSING_GRID will be used.
        if "event_postprocessing_grid" in metadata.get("evaluation_params", {}):
            postprocessing_grid = metadata["evaluation_params"][
                "event_postprocessing_grid"
            ]
        else:
            postprocessing_grid = EVENT_POSTPROCESSING_GRID

        predictor = EventPredictionModel(
            nfeatures=embedding_size,
            label_to_idx=label_to_idx,
            nlabels=nlabels,
            prediction_type=metadata["prediction_type"],
            scores=scores,
            validation_target_events=validation_target_events,
            test_target_events=test_target_events,
            postprocessing_grid=postprocessing_grid,
            conf=conf,
            use_scoring_for_early_stopping=use_scoring_for_early_stopping,
        )
    elif metadata["embedding_type"] == "scene":
        predictor = ScenePredictionModel(
            nfeatures=embedding_size,
            label_to_idx=label_to_idx,
            nlabels=nlabels,
            prediction_type=metadata["prediction_type"],
            scores=scores,
            conf=conf,
            use_scoring_for_early_stopping=use_scoring_for_early_stopping,
        )
    else:
        raise ValueError(f"Unknown embedding_type {metadata['embedding_type']}")

    if use_scoring_for_early_stopping:
        # First score is the target
        target_score = f"val_{str(scores[0])}"
        if scores[0].maximize:
            mode = "max"
        else:
            mode = "min"
    else:
        # This loss is much faster, but will give poorer scores
        target_score = "val_loss"
        mode = "min"
    checkpoint_callback = ModelCheckpoint(monitor=target_score, mode=mode)
    early_stop_callback = EarlyStopping(
        monitor=target_score,
        min_delta=0.00,
        patience=conf["patience"],
        check_on_train_epoch_end=False,
        verbose=False,
        mode=mode,
    )

    logger = CSVLogger(Path("logs").joinpath(embedding_path))
    logger.log_hyperparams(hparams_to_json(conf))

    trainer = pl.Trainer(
        callbacks=[checkpoint_callback]
        if random_probe
        else [checkpoint_callback, early_stop_callback],
        gpus=gpus,
        check_val_every_n_epoch=conf["check_val_every_n_epoch"],
        max_epochs=50 if random_probe else conf["max_epochs"],
        deterministic=deterministic,
        num_sanity_val_steps=0,
        # profiler=profiler,
        # profiler="pytorch",
        profiler="simple",
        logger=logger,
    )
    train_dataloader = dataloader_from_split_name(
        data_splits["train"],
        embedding_path,
        label_to_idx,
        nlabels,
        metadata["embedding_type"],
        batch_size=conf["batch_size"],
        in_memory=in_memory,
        metadata=False,
        task=task,
        is_test=False,
        random_probe=random_probe,
    )
    valid_dataloader = dataloader_from_split_name(
        data_splits["valid"],
        embedding_path,
        label_to_idx,
        nlabels,
        metadata["embedding_type"],
        batch_size=conf["batch_size"],
        in_memory=in_memory,
        task=task,
        is_test=True,
        random_probe=random_probe,
    )
    trainer.fit(predictor, train_dataloader, valid_dataloader)
    if checkpoint_callback.best_model_score is not None:
        sys.stdout.flush()
        end = time.time()
        time_in_min = (end - start) / 60
        epoch = torch.load(checkpoint_callback.best_model_path, weights_only=False)["epoch"]
        if metadata["embedding_type"] == "event":
            best_postprocessing = predictor.epoch_best_postprocessing_or_default(epoch)
        else:
            best_postprocessing = []
        # TODO: Postprocessing
        logger.log_metrics({"time_in_min": time_in_min})
        logger.finalize("success")
        logger.save()
        return GridPointResult(
            predictor=predictor,
            model_path=checkpoint_callback.best_model_path,
            epoch=epoch,
            time_in_min=time_in_min,
            hparams=dict(predictor.hparams),
            postprocessing=best_postprocessing,
            trainer=trainer,
            validation_score=checkpoint_callback.best_model_score.detach().cpu().item(),
            score_mode=mode,
            conf=conf,
        )
    else:
        raise ValueError(
            f"No score {checkpoint_callback.best_model_score} for this model"
        )


def task_predictions_test(
    embedding_path: Path,
    grid_point: GridPointResult,
    metadata: Dict[str, Any],
    data_splits: Dict[str, List[str]],
    label_to_idx: Dict[str, int],
    nlabels: int,
    in_memory: bool,
    task: str,
):
    """
    Test a pre-trained predictor using precomputed embeddings.
    """
    test_dataloader = dataloader_from_split_name(
        data_splits["test"],
        embedding_path,
        label_to_idx,
        nlabels,
        metadata["embedding_type"],
        batch_size=grid_point.conf["batch_size"],
        in_memory=in_memory,
        task=task,
        is_test=True,
        random_probe=False,
    )

    trainer = grid_point.trainer
    # This hack is necessary because we use the best validation epoch to
    # choose the event postprocessing
    trainer.fit_loop.current_epoch = grid_point.epoch

    # Run tests
    test_results = trainer.test(
        ckpt_path=grid_point.model_path, dataloaders=test_dataloader
    )
    assert len(test_results) == 1, "Should have only one test dataloader"
    test_results = test_results[0]
    return test_results


def serialize_value(v):
    if isinstance(v, str) or isinstance(v, float) or isinstance(v, int):
        return v
    else:
        return str(v)


def hparams_to_json(hparams):
    return {k: serialize_value(v) for k, v in hparams.items()}


def data_splits_from_folds(folds: List[str]) -> List[Dict[str, List[str]]]:
    """
    Create data splits by using Leave One Out Cross Validation strategy.

    folds is a list of dataset partitions created during pre-processing. For example,
    for 5-fold cross val: ["fold00", "fold01", ..., "fold04"]. This function will create
    k test, validation, and train splits using these folds. Each fold is successively
    treated as test split, the next split as validation, and the remaining as train.
    Folds will be sorted before applying the above strategy.

    With 5-fold, for example, we would have:
    test=fold00, val=fold01, train=fold02..04,
    test=fold01, val=fold02, train=fold03,04,01
    ...
    test=fold04, val=fold00, train=01..03
    """
    sorted_folds = tuple(sorted(folds))
    assert len(sorted_folds) == len(set(sorted_folds)), "Folds are not unique"
    num_folds = len(sorted_folds)
    all_data_splits: List[Dict[str, List[str]]] = []
    for fold_idx in range(num_folds):
        test_fold = sorted_folds[fold_idx]
        valid_fold = sorted_folds[(fold_idx + 1) % num_folds]
        train_folds = [f for f in sorted_folds if f not in (test_fold, valid_fold)]
        all_data_splits.append(
            {
                "train": train_folds,
                "valid": [valid_fold],
                "test": [test_fold],
            }
        )
        assert not set(train_folds).intersection({test_fold, valid_fold}), (
            "Train folds are not distinct from the dev and the test folds"
        )

    return all_data_splits


def aggregate_test_results(results: Dict[str, Dict[str, float]]) -> Dict[str, float]:
    """
    Aggregates test results over folds by mean and standard deviation
    """
    results_df = pd.DataFrame.from_dict(results, orient="index")
    aggregate_results = {}
    for score in results_df:
        aggregate_results[f"{score}_mean"] = results_df[score].mean()
        aggregate_results[f"{score}_std"] = results_df[score].std()

    return aggregate_results


def get_splits_from_metadata(metadata: Dict) -> List[Dict[str, List[str]]]:
    """
    Extracts the splits for training from the task metadata. If there are folds
    present then this creates a set of k splits for each fold.
    Args:
        metadata: The preprocessing task metadata
    Returns:
        list(dict): The `data_splits`, are created from the splits prepared by
            the hearpreprocess pipeline and represent the actual splits which
            will be used for training, testing and validation
            Each Data Split is a dict with the following keys and values:
                - train (list): The splits to be used for training
                - valid (list): The splits to be used for validation
                - test (list): The splits to be used for testing
            The data splits produced directly depend on the `split_mode`
                of the hearpreprocess task configuration
                - If the split mode is `new_split_kfold` or `presplit_kfold`,
                    each data split will be represent one out of the multiple
                    combination of LOOCV (refer function `data_splits_from_folds`)
                - If the split mode is `trainvaltest`, there is a predefined
                    data split and hence there will only be one data split which is
                    {
                        "train": ["train"],
                        "valid": ["valid"],
                        "test": ["test"],
                    }
                    This data split indicates that the splits (from hearpreprocess )
                    will be used for training,
                    validation and testing as defined by the name of the split

    Raises:
        AssertionError: If the `split_mode` of the metadata is invalid.
            Valid split modes are - `trainvaltest`, `new_split_kfold`, `presplit_kfold`

    """
    data_splits: List[Dict[str, List[str]]]
    if metadata["split_mode"] == "trainvaltest":
        # There are train/validation/test splits predefined. These are the only splits
        # that will be considered during training and testing.
        data_splits = [
            {
                "train": ["train"],
                "valid": ["valid"],
                "test": ["test"],
            }
        ]
    elif metadata["split_mode"] in ["new_split_kfold", "presplit_kfold"]:
        folds = metadata["splits"]
        # Folds should be a list of strings
        assert all(isinstance(x, str) for x in folds)
        # If we are using k-fold cross-validation then get a list of the
        # splits to test over. This expects that k data folds were generated
        # during pre-processing and the names of each of these folds is listed
        # in the metadata["folds"] variable.
        data_splits = data_splits_from_folds(folds)
    else:
        raise AssertionError(
            f"Unknown split_mode: {metadata['split_mode']} in task metadata."
        )

    return data_splits


def sort_grid_points(
    grid_point_results: List[GridPointResult],
) -> List[GridPointResult]:
    """
    Sort grid point results in place, so that the first result
    is the best.
    """
    assert len(set([g.score_mode for g in grid_point_results])) == 1, (
        "Score modes should be same for all the grid points"
    )
    mode: str = grid_point_results[0].score_mode
    # Pick the model with the best validation score
    if mode == "max":
        grid_point_results = sorted(
            grid_point_results, key=lambda g: g.validation_score, reverse=True
        )
    elif mode == "min":
        grid_point_results = sorted(
            grid_point_results, key=lambda g: g.validation_score
        )
    else:
        raise ValueError(f"mode = {mode}")

    return grid_point_results


def print_scores(
    grid_point_results: List[GridPointResult],
    embedding_path: Path,
    logger: logging.Logger,
):
    grid_point_results = sort_grid_points(grid_point_results)
    for g in grid_point_results:
        # Don't log this since it is diagnostic and repetitive
        print(f"Grid Point Summary: {g}")


def task_predictions(
    embedding_path: Path,
    embedding_size: int,
    grid_points: int,
    gpus: Optional[int],
    in_memory: bool,
    deterministic: bool,
    grid: str,
    logger: logging.Logger,
):
    # By setting workers=True in seed_everything(), Lightning derives
    # unique seeds across all dataloader workers and processes
    # for torch, numpy and stdlib random number generators.
    # Note that if you change the number of workers, determinism
    # might change.
    # However, it appears that workers=False does get deterministic
    # results on 4 multi-worker jobs I ran, probably because our
    # dataloader doesn't do any augmentation or use randomness.
    if deterministic:
        seed_everything(42, workers=True)

    metadata = json.load(embedding_path.joinpath("task_metadata.json").open())
    label_vocab, nlabels = label_vocab_nlabels(embedding_path)

    # wandb.init(project="heareval", tags=["predictions", embedding_path.name])

    label_to_idx = label_vocab_as_dict(label_vocab, key="label", value="idx")
    scores = [
        available_scores[score](label_to_idx=label_to_idx)
        for score in metadata["evaluation"]
    ]

    use_scoring_for_early_stopping = metadata.get(
        "use_scoring_for_early_stopping", True
    )

    # Data splits for training
    data_splits = get_splits_from_metadata(metadata)

    # Construct the grid points for model creation
    if grid == "default":
        final_grid = copy.copy(PARAM_GRID)
    elif grid == "fast":
        final_grid = copy.copy(FAST_PARAM_GRID)
    elif grid == "faster":
        final_grid = copy.copy(FASTER_PARAM_GRID)
    else:
        raise ValueError(
            f"Unknown grid type: {grid}. Please select default, fast, or faster"
        )

    # Update with task specific grid parameters
    # From the global TASK_SPECIFIC_PARAM_GRID
    if metadata["task_name"] in TASK_SPECIFIC_PARAM_GRID:
        final_grid.update(TASK_SPECIFIC_PARAM_GRID[metadata["task_name"]])

    # From task specific parameter grid in the task metadata
    # We add this option, so that task specific param grid can be used
    # for secret tasks, without mentioning them in the global
    # TASK_SPECIFIC_PARAM_GRID. Ideally one out of the two option should be
    # there
    if "task_specific_param_grid" in metadata.get("evaluation_params", {}):
        final_grid.update(metadata["evaluation_params"]["task_specific_param_grid"])

    # Model selection
    confs = list(ParameterGrid(final_grid))
    random.shuffle(confs)

    grid_point_results = []
    for confi, conf in tqdm(enumerate(confs[:grid_points]), desc="grid"):
        logger.info(f"Grid point {confi + 1} of {grid_points}: {conf}")
        grid_point_result = task_predictions_train(
            embedding_path=embedding_path,
            embedding_size=embedding_size,
            metadata=metadata,
            data_splits=data_splits[0],
            label_to_idx=label_to_idx,
            nlabels=nlabels,
            scores=scores,
            conf=conf,
            use_scoring_for_early_stopping=use_scoring_for_early_stopping,
            gpus=gpus,
            in_memory=in_memory,
            deterministic=deterministic,
            task=metadata["prediction_type"],
        )
        logger.info(f" result: {grid_point_result}")
        grid_point_results.append(grid_point_result)
        print_scores(grid_point_results, embedding_path, logger)

    # Use the best hyperparameters to train models for remaining folds,
    # then compute test scores using the resulting models
    grid_point_results = sort_grid_points(grid_point_results)
    best_grid_point = grid_point_results[0]
    logger.info(
        "Best Grid Point Validation Score: "
        f"{best_grid_point.validation_score}  "
        "Grid Point HyperParams: "
        f"{best_grid_point.hparams}  "
    )

    # Train predictors for the remaining splits using the hyperparameters selected
    # from the grid search.
    split_grid_points = [best_grid_point]
    for spliti, split in tqdm(enumerate(data_splits[1:]), desc="split"):
        logger.info(f"Training split {spliti + 2} of {len(data_splits)}: {split}")
        grid_point_result = task_predictions_train(
            embedding_path=embedding_path,
            embedding_size=embedding_size,
            metadata=metadata,
            data_splits=split,
            label_to_idx=label_to_idx,
            nlabels=nlabels,
            scores=scores,
            conf=best_grid_point.conf,
            use_scoring_for_early_stopping=use_scoring_for_early_stopping,
            gpus=gpus,
            in_memory=in_memory,
            deterministic=deterministic,
            task=metadata["prediction_type"],
        )
        split_grid_points.append(grid_point_result)
        logger.info(
            f"Validation Score for the Training Split: "
            f"{grid_point_result.validation_score}"
        )

    # Now test each of the trained models
    test_results = {}
    for i, split in enumerate(data_splits):
        test_fold_str = "|".join(split["test"])
        test_results[test_fold_str] = task_predictions_test(
            embedding_path=embedding_path,
            grid_point=split_grid_points[i],
            metadata=metadata,
            data_splits=split,
            label_to_idx=label_to_idx,
            nlabels=nlabels,
            in_memory=in_memory,
            task=metadata["prediction_type"],
        )

        # Cache predictions for detailed analysis
        prediction_file = embedding_path.joinpath(f"{test_fold_str}.predictions.pkl")
        with open(prediction_file, "wb") as fp:
            pickle.dump(split_grid_points[i].predictor.test_predictions, fp)

        # Add model training values relevant to this split model
        test_results[test_fold_str].update(
            {
                "validation_score": split_grid_points[i].validation_score,
                "epoch": split_grid_points[i].epoch,
                "time_in_min": split_grid_points[i].time_in_min,
            }
        )

    # Make sure we have a test score for each fold
    assert len(test_results) == len(data_splits)

    # Aggregate scores over folds
    if len(test_results) > 1:
        test_results["aggregated_scores"] = aggregate_test_results(test_results)

    # Update test results with values relevant for all split models
    test_results.update(
        {
            "hparams": hparams_to_json(best_grid_point.hparams),
            "postprocessing": best_grid_point.postprocessing,
            "score_mode": split_grid_points[i].score_mode,
            "embedding_path": str(embedding_path),
        }
    )

    # Save test scores
    open(embedding_path.joinpath("test.predicted-scores.json"), "wt").write(
        json.dumps(test_results, indent=4)
    )
    logger.info(f"Final Test Results: {json.dumps(test_results)}")

    # We no longer have best_predictor, the predictor is
    # loaded by trainer.test and then disappears
    """
    # Cache predictions for secondary sanity-check evaluation
    if metadata["embedding_type"] == "event":
        json.dump(
            best_predictor.test_predicted_events,
            embedding_path.joinpath("test.predictions.json").open("w"),
            indent=4,
        )
    pickle.dump(
        best_predictor.test_predicted_labels,
        open(embedding_path.joinpath("test.predicted-labels.pkl"), "wb"),
    )
    """


# This is for the RIR localization predictions.
def rir_localization_predictions(
    embedding_path: Path,
    embedding_size: int,
    grid_points: int,
    gpus: Optional[int],
    in_memory: bool,
    deterministic: bool,
    grid: str,
    logger: logging.Logger,
    prediction_type: str,
    random_probe: bool,
):
    # We keep the determinisim always for our tasks.
    if deterministic:
        seed_everything(42, workers=True)

    # This metadata is important for the normal task, but for RIR we need to change it up a bit.
    metadata = json.load(embedding_path.joinpath("task_metadata.json").open())
    metadata["prediction_type"] = prediction_type
    # We do not have labels, so passy a dummy label_to_idx variable.
    scores = [
        available_scores[score](label_to_idx=dict(), maximize=False)
        for score in metadata["evaluation"]
    ]

    use_scoring_for_early_stopping = metadata.get(
        "use_scoring_for_early_stopping", True
    )

    # Data splits for training
    data_splits = get_splits_from_metadata(metadata)

    # Construct the grid points for model creation
    if grid == "default":
        final_grid = copy.copy(PARAM_GRID)
    elif grid == "fast":
        final_grid = copy.copy(FAST_PARAM_GRID)
    elif grid == "faster":
        final_grid = copy.copy(FASTER_PARAM_GRID)
    else:
        raise ValueError(
            f"Unknown grid type: {grid}. Please select default, fast, or faster"
        )

    # Update with task specific grid parameters
    # From the global TASK_SPECIFIC_PARAM_GRID
    if metadata["task_name"] in TASK_SPECIFIC_PARAM_GRID:
        final_grid.update(TASK_SPECIFIC_PARAM_GRID[metadata["task_name"]])

    # From task specific parameter grid in the task metadata
    # We add this option, so that task specific param grid can be used
    # for secret tasks, without mentioning them in the global
    # TASK_SPECIFIC_PARAM_GRID. Ideally one out of the two option should be
    # there
    if "task_specific_param_grid" in metadata.get("evaluation_params", {}):
        final_grid.update(metadata["evaluation_params"]["task_specific_param_grid"])

    # Model selection
    confs = list(ParameterGrid(final_grid))
    random.shuffle(confs)

    grid_point_results = []
    for confi, conf in tqdm(enumerate(confs[:grid_points]), desc="grid"):
        logger.info(f"Grid point {confi + 1} of {grid_points}: {conf}")
        grid_point_result = task_predictions_train(
            embedding_path=embedding_path,
            embedding_size=embedding_size,
            metadata=metadata,
            data_splits=data_splits[0],
            label_to_idx=dict(),
            nlabels=3 if prediction_type == "cartesian-regression" else 2,
            scores=scores,
            conf=conf,
            use_scoring_for_early_stopping=use_scoring_for_early_stopping,
            gpus=gpus,
            in_memory=in_memory,
            deterministic=deterministic,
            task=metadata["prediction_type"],
            random_probe=random_probe,
        )
        logger.info(f" result: {grid_point_result}")
        grid_point_results.append(grid_point_result)
        print_scores(grid_point_results, embedding_path, logger)

    # Use the best hyperparameters to train models for remaining folds,
    # then compute test scores using the resulting models
    grid_point_results = sort_grid_points(grid_point_results)
    best_grid_point = grid_point_results[0]
    logger.info(
        "Best Grid Point Validation Score: "
        f"{best_grid_point.validation_score}  "
        "Grid Point HyperParams: "
        f"{best_grid_point.hparams}  "
    )

    # Train predictors for the remaining splits using the hyperparameters selected
    # from the grid search.
    split_grid_points = [best_grid_point]
    for spliti, split in tqdm(enumerate(data_splits[1:]), desc="split"):
        logger.info(f"Training split {spliti + 2} of {len(data_splits)}: {split}")
        grid_point_result = task_predictions_train(
            embedding_path=embedding_path,
            embedding_size=embedding_size,
            metadata=metadata,
            data_splits=split,
            label_to_idx=dict(),
            nlabels=3 if prediction_type == "cartesian-regression" else 2,
            scores=scores,
            conf=best_grid_point.conf,
            use_scoring_for_early_stopping=use_scoring_for_early_stopping,
            gpus=gpus,
            in_memory=in_memory,
            deterministic=deterministic,
            task=metadata["prediction_type"],
            random_probe=random_probe,
        )
        split_grid_points.append(grid_point_result)
        logger.info(
            f"Validation Score for the Training Split: "
            f"{grid_point_result.validation_score}"
        )

    # Now test each of the trained models
    test_results = {}
    for i, split in enumerate(data_splits):
        test_fold_str = "|".join(split["test"])
        test_results[test_fold_str] = task_predictions_test(
            embedding_path=embedding_path,
            grid_point=split_grid_points[i],
            metadata=metadata,
            data_splits=split,
            label_to_idx=dict(),
            nlabels=3 if prediction_type == "cartesian-regression" else 2,
            in_memory=in_memory,
            task=metadata["prediction_type"],
        )

        prediction_file = embedding_path.joinpath(
            f"{test_fold_str}.predictions-localization.pkl"
        )
        with open(prediction_file, "wb") as fp:
            pickle.dump(split_grid_points[i].predictor.test_predictions, fp)

        # Add model training values relevant to this split model
        test_results[test_fold_str].update(
            {
                "validation_score": split_grid_points[i].validation_score,
                "epoch": split_grid_points[i].epoch,
                "time_in_min": split_grid_points[i].time_in_min,
            }
        )

    # Make sure we have a test score for each fold
    assert len(test_results) == len(data_splits)

    # Aggregate scores over folds
    if len(test_results) > 1:
        test_results["aggregated_scores"] = aggregate_test_results(test_results)

    # Update test results with values relevant for all split models
    test_results.update(
        {
            "hparams": hparams_to_json(best_grid_point.hparams),
            "postprocessing": best_grid_point.postprocessing,
            "score_mode": split_grid_points[i].score_mode,
            "embedding_path": str(embedding_path),
        }
    )

    # Save test scores
    if random_probe:
        open(embedding_path.joinpath("test.predicted-scores-random.json"), "wt").write(
            json.dumps(test_results, indent=4)
        )
    else:
        open(
            embedding_path.joinpath("test.predicted-scores-localization.json"), "wt"
        ).write(json.dumps(test_results, indent=4))
    logger.info(f"Final Test Results: {json.dumps(test_results)}")
