import argparse
import os
import torch
import torch.optim
import torch.nn as nn
import torch.nn.functional as F
import torch.utils.data
from torchvision import datasets, transforms
from tqdm.contrib import tqdm
from models.model_utils import get_model
#from data.datasets import load_dataset
from utils.utils import checkpoint_summary
from torch.utils.data import DataLoader, Dataset, Subset, SubsetRandomSampler
from pathlib import Path
import numpy as np
from sklearn.model_selection import train_test_split
from retrieve_any_layer import ModelWrapper

class Feature_Extractor:
    def __init__(self, cfg:argparse.Namespace, ckpt_dir: Path=None, feature_dir: Path=None, model=None) -> None:
        self.cfg = cfg
        self.dataset_name = cfg.set
        # get model
        assert ckpt_dir is not None or model is not None, "Either ckpt_dir or model should be active."
        self.feature_dir = feature_dir
        
        self.model = self.load_model(ckpt_dir, model)
        for param in self.model.parameters():
            param.requires_grad_(False)
            
        self.model.cuda().eval()
        self.hooks = self.attach_hooks()
        self.keys = list(self.hooks.keys())
        
        ### ImageNet-100
        cfg.data_dir='/data/datasets/ImageNet-100'
        train_loader, val_loader = self.get_flex_imageNet100_dataloader(cfg, resolution_size=cfg.input_size, 
        batch_size=cfg.batch_size, class_num=50, sample_per_class=200)
        test_loader = None

        #train_loader, val_loader, test_loader = load_dataset(self.cfg)
        self.loaders = {'train':train_loader, 'val':val_loader, 'test':test_loader}
        
        self.activations = {}
        # self._dummy_batch()

    def build_transform(self, cfg):
        resize_im = cfg.input_size > 30
        t = []
        if resize_im:
            size = int((256 / 224) * cfg.input_size)
            t.append(
                transforms.Resize(size, interpolation=3),  # to maintain same ratio w.r.t. 224 images
            )
            t.append(transforms.CenterCrop(cfg.input_size))

        t.append(transforms.ToTensor())
        t.append(transforms.Normalize(mean=[0.485, 0.456, 0.406],
                                     std=[0.229, 0.224, 0.225]))
        return transforms.Compose(t)

    def filter_by_class(self, labels, min_class, max_class):
        return list(np.where(np.logical_and(labels >= min_class, labels < max_class))[0])

    def get_flex_imageNet100_dataloader(self, cfg, resolution_size=32, batch_size=512, class_num=100, 
        sample_per_class=100):
        IMAGENET_100_DIR='/data/datasets/ImageNet-100'
        np.random.seed(cfg.seed)
        print(f"loading ImageNet100 data with resolution {resolution_size}")
        train_dir = os.path.join(IMAGENET_100_DIR, "train")
        test_dir = os.path.join(IMAGENET_100_DIR, "val")

        transform = self.build_transform(cfg)

        train_dataset = datasets.ImageFolder(root=train_dir, transform=transform)
        test_dataset = datasets.ImageFolder(root=test_dir, transform=transform)
        ## get indices

        train_indices = []
        for class_idx in range(class_num):
            class_indices = np.where(np.array(train_dataset.targets) == class_idx)[0]
            class_indices = np.random.choice(class_indices, sample_per_class, replace=False)
            train_indices.extend(class_indices)

        test_indices = self.filter_by_class(np.array(test_dataset.targets), 0, class_num)

        print(f"training samples: {len(train_indices)}")
        print(f"test samples: {len(test_indices)}")

        # create subset with indices
        train_dataset = Subset(train_dataset, train_indices)
        test_dataset = Subset(test_dataset, test_indices)

        train_dataloader = DataLoader(train_dataset, batch_size=batch_size, shuffle=True, num_workers=8, pin_memory=True)
        test_dataloader = DataLoader(test_dataset, batch_size=batch_size, shuffle=False, num_workers=8, pin_memory=True)

        return train_dataloader, test_dataloader
        
    def load_model(self, ckpt_dir, model=None):
        if model is not None:
            return model
        
        print(f"loading model from {ckpt_dir}")
        _model = get_model(self.cfg)
        #print(_model)
        # sys.exit()
        _model_dict = _model.state_dict()
        #print(_model_dict)

        ckpt0 = torch.load(ckpt_dir)
        _state_dict_key = 'state_dict'

        ckpt = {}
        ckpt[_state_dict_key] = ckpt0 #state_dict

        layers_to_drop = [k for k in ckpt[_state_dict_key].keys() if 'layer.' in k or 'mask' in k]
        #if 'transfer' in self.cfg.task:
        if self.cfg.arch.lower() == 'resnet18':
            layers_to_drop.extend(['fc.weight', 'fc.bias'])
        elif self.cfg.arch.lower() == 'vgg13':
            layers_to_drop.extend(['classifier.weight', 'classifier.bias'])
        elif self.cfg.arch.lower() == 'vgg19':
            layers_to_drop.extend(['classifier.weight', 'classifier.bias'])
        
        for k in layers_to_drop:
            ckpt[_state_dict_key].pop(k)
        
            
        _model_dict.update(ckpt[_state_dict_key])
        
        _model.load_state_dict(_model_dict)

        model_state_dict = _model.state_dict()
        for k in ckpt[_state_dict_key]:
            #if not k.startswith('fc.'):
            assert torch.equal(model_state_dict[k].cpu(), ckpt[_state_dict_key][k].cpu()), k
        print("\nSuccessfully performed sanity-check on loading pre-trained ckpt")
        
        if 'best_val_acc' in ckpt:
            self.best_val_acc1 = ckpt['best_val_acc']
            if 'best_test_acc' in ckpt.keys():
                self.best_test_acc1 = ckpt['best_test_acc']

            checkpoint_summary(ckpt_dir, ckpt)
        # print(f"checkpoint's best accuracy is {ckpt['best_val_acc']}")
        return _model
        
    def attach_hooks(self):
        hooks = {}
        patch_size=2
        def _getActivation(name):
            # the hook signature
            def hook(model, input, output):
                self.activations[name.replace('.', '')] = torch.flatten(F.adaptive_avg_pool2d(input[0], 2).squeeze(), 1).detach()
                
            return hook
        
        for name, module in self.model.named_modules():
            if name in self.cfg.hook_layers:
                hooks[name.replace('.', '')] = module.register_forward_hook(_getActivation(name))
        
        return hooks
    
    def _dummy_batch(self):
        # print("Forwarding a dummy batch")
        x, _ = next(iter(self.loaders['train']))
        y = self.model(x.cuda())
        self.activations['out'] = y
        pool = nn.AdaptiveAvgPool2d(1)
    
    def check_features_exist(self, mode):
        self._dummy_batch()
        
        feature_dir = self.feature_dir.parent / mode
        feature_files = list(feature_dir.glob('*.pt'))
        
        # check which features exist. This line drops the _POSTFIX
        feature_files = ['_'.join(f.stem.split('_')[:-1]) for f in feature_files]
        feature_dir.mkdir(parents=True, exist_ok=True) 
        
        for k in self.activations:
            if f"{k}_{self.dataset_name}" not in feature_files:
                # Something didn't exist
                return False, None
            
        return True, feature_dir
                
    def save_features(self, features, targets, mode, postfix):
        feature_dir = self.feature_dir.parent / mode
        feature_dir.mkdir(parents=True, exist_ok=True) 
        
        dict_to_save = {}
        for k in features:
            dict_to_save[k] = {'features':features[k], 'targets':targets[k]}
            
            feature_file = feature_dir / f"{k}_{self.dataset_name}_{postfix}.pt"
            torch.save(dict_to_save[k], feature_file)
            print(f'saved at {feature_file}')
            
        dict_to_save = {}
            
    
    @torch.no_grad()
    def _extract(self, loader, mode='train', progressbar=True):
        # If already extracted, don't extract
        already_extracted, _fd = self.check_features_exist(mode)
        if already_extracted:
            print(f"All files are alreay extracted! To force feature_extraction, remove all features")
            return
        
        # dicts to store features and targets for saving on disk
        features={}
        targets={}
        total_processed = 0

        wrapper = ModelWrapper(self.model, self.cfg.hook_layers)
        
        with tqdm(loader, disable=not progressbar) as t:
            for batch_idx, (inputs, _targets) in enumerate(t): #TODO wrap it with tqdm
                total_processed += inputs.size(0)
                self.activations = {}
                inputs, _targets = inputs.cuda(),_targets.long().squeeze().cuda()
                #out = self.model(inputs)
                out = wrapper(inputs)
                # Get output of the network as a feature
                #self.activations['out'] = out
                #exit()
                self.activations = out
                
                # Append features and targets to the dictionaries
                for k in self.activations:
                    if k not in features:
                        features[k] = self.activations[k]
                        targets[k] = _targets
                    else:
                        features[k] = torch.cat([features[k], self.activations[k]], dim=0)
                        targets[k] = torch.cat([targets[k], _targets], dim=0)
                
                # Log progress in TQDM
                t.set_postfix(progress=f"{len(features[k])}/{len(loader.dataset)}")
                
                # Save features periodically to avoid OOM
                if len(features[k]) + self.cfg.batch_size > self.cfg.features_per_file:
                    self.save_features(features, targets, mode, total_processed)
                    features = {}
                    targets = {}
        
        if len(features) > 0: # the leftovers 
            self.save_features(features, targets, mode, total_processed)
            features = {}
            targets = {}
                
    def extract(self, progressbar=True):
        modes = ['train', 'val']
        if self.cfg.eval_tst:
            modes.append('test')
            
        for m in modes:
            print(f"extracting features for {m}")
            loader = self.loaders[m]
            self._extract(loader, m, progressbar)
        