import torch
import pytorch_lightning as L
import json
from typing import Sequence
from pathlib import Path
from PIL import Image
from collections import defaultdict
import numpy as np
import torch.utils

from .common import default_collate_fn, default_transform

def _label_names_from_split_info(split_info):
    idx_to_classname = {x[1]: x[2] for x in split_info}
    return [idx_to_classname[x] for x in sorted(idx_to_classname.keys())]

def _load_images(data_dir, image_paths):
    imgs = []
    for path in image_paths:
        with Image.open(data_dir / path) as img:
            imgs.append(img.copy())
    
    return imgs

class EuroSATDataset(torch.utils.data.Dataset):
    def __init__(
            self, 
            image_dir: str,
            split_info: list,
            text_prompt: str, 
            transform=None,
            use_few_shot = False,
            shots_per_class = 5,
            few_shot_sample_seed = 0,
        ):
        self._data_dir = Path(image_dir)
        self._split_info = split_info
        self._label_names = _label_names_from_split_info(split_info)
        self._text_prompt = text_prompt
        self._transform = transform
        self._images = _load_images(image_dir, [x[0] for x in split_info])

        self.use_few_shot = use_few_shot
        if self.use_few_shot:
            self.shots_per_class = shots_per_class
            self.few_shot_sample_seed = few_shot_sample_seed

            # get the index for each class
            self.class_index = defaultdict(list)
            for i in range(self.__len__()):
                class_id = self.get_class_id(i)
                self.class_index[class_id].append(i)
            
            # create few-shot dataset through sampling
            selected_data = []
            for indices in self.class_index.values():
                np.random.seed(self.few_shot_sample_seed)
                selected_data.extend(np.random.choice(indices, self.shots_per_class, replace=False))
            self.selected_data = selected_data

    def __len__(self):
        return len(self._split_info)

    def get_class_id(self, idx):
        class_id = self._split_info[idx][1]
            
        return class_id
    
    def __getitem__(self, idx):
        class_id = self._split_info[idx][1]

        text = self._text_prompt.format(
            class_name=self._label_names[class_id]
        )

        image = self._images[idx] 
        if self._transform is not None:
            image = self._transform(image)
            
        return dict(image=image, text=text, class_id=class_id, image_id=idx)


class EuroSATDataModule(L.LightningDataModule):
    DATASET_SUBDIR = 'eurosat'

    def __init__(
        self,
        data_dir: str,
        batch_size: int = 32,
        num_workers: int = 4,
        text_prompt: str = "An image of a {class_name}",
        train_transform=default_transform(image_size=244),
        test_transform=default_transform(image_size=244),
        shuffle_train: bool = True,
        subset_indices: Sequence[int] = None,
        shots_per_class: int = 10,
        use_few_shot: bool = False,
        few_shot_sample_seed: int = 42,
    ) -> None:
        super().__init__()
        self.batch_size = batch_size
        self.num_workers = num_workers
        self.data_dir = data_dir
        self.text_prompt = text_prompt
        self.train_transform = train_transform
        self.test_transform = test_transform
        self.shuffle_train = shuffle_train
        self.subset_indices = subset_indices
        
        self.use_few_shot = use_few_shot
        self.shots_per_class = shots_per_class
        self.few_shot_sample_seed = few_shot_sample_seed

    def setup(self, stage: str = None):
        splits_file = Path(self.data_dir) / 'split_zhou_EuroSAT.json'
        with open(splits_file) as f:
            splits_info = json.load(f)

        if self.use_few_shot:
            self.train_ds = EuroSATDataset(
                image_dir=self.data_dir / '2750',
                split_info=splits_info['train'],
                text_prompt=self.text_prompt,
                transform=self.train_transform,
                use_few_shot = True,
                shots_per_class = self.shots_per_class,
                few_shot_sample_seed = self.few_shot_sample_seed
            )
            self.train_ds = torch.utils.data.Subset(self.train_ds, self.train_ds.selected_data)
        else:
            self.train_ds = EuroSATDataset(
                image_dir=self.data_dir / '2750',
                split_info=splits_info['train'],
                text_prompt=self.text_prompt,
                transform=self.train_transform,
            )
        if self.subset_indices is not None:
            self.train_ds = torch.utils.data.Subset(self.train_ds, self.subset_indices)

        self.val_ds = EuroSATDataset(
            image_dir=self.data_dir / '2750',
            split_info=splits_info['val'],
            text_prompt=self.text_prompt,
            transform=self.test_transform,
        )
        
        self.test_ds = EuroSATDataset(
            image_dir=self.data_dir / '2750',
            split_info=splits_info['test'],
            text_prompt=self.text_prompt,
            transform=self.test_transform,
        )

    def train_dataloader(self):
        return torch.utils.data.DataLoader(
            self.train_ds,
            batch_size=self.batch_size,
            num_workers=self.num_workers,
            collate_fn=default_collate_fn,
            shuffle=self.shuffle_train,
            persistent_workers=True,
        )
    
    def val_dataloader(self):
        return torch.utils.data.DataLoader(
            self.val_ds,
            batch_size=self.batch_size,
            num_workers=self.num_workers,
            collate_fn=default_collate_fn,
            persistent_workers=True,
            shuffle=False,
        )
    
    def test_dataloader(self):
        return torch.utils.data.DataLoader(
            self.test_ds,
            batch_size=self.batch_size,
            num_workers=self.num_workers,
            collate_fn=default_collate_fn,
            persistent_workers=True,
            shuffle=False,
        )

    @property
    def class_prompts(self):
        if self.use_few_shot:
            return [self.text_prompt.format(class_name=name) for name in self.test_ds._label_names]
        else:
            return [self.text_prompt.format(class_name=name) for name in self.train_ds._label_names]