import os
import torch
from torch.utils.data import ConcatDataset, DataLoader
from torchvision import transforms
import random

if __package__ is None:
    import sys
    from os import path

    sys.path.append(path.dirname(path.dirname(path.abspath(__file__))))

from data.meta_dataset import MetaDataset, GetDataLoaderDict
from configs.default import terrainc_path


terra_incognita_name_dict = {
    '100': 'location_100',
    '38': 'location_38',
    '43': 'location_43',
    '46': 'location_46',
}
transform_train = transforms.Compose(
            [transforms.RandomResizedCrop(224, scale=(0.7, 1.0)),
            transforms.RandomHorizontalFlip(),
            # transforms.RandomGrayscale( 0.1),
            transforms.ColorJitter(brightness=0.4, contrast=0.4, saturation=0.4, hue=0.4),
            transforms.RandomGrayscale(),
            transforms.ToTensor(),
            transforms.Normalize(mean=[0.485, 0.456, 0.406], std=[0.229, 0.224, 0.225])
            ])

transform_test = transforms.Compose(
            [transforms.Resize([224, 224]),
            transforms.ToTensor(),
            transforms.Normalize(mean=[0.485, 0.456, 0.406], std=[0.229, 0.224, 0.225])
            ])


def get_transforms():
    normalizer = transforms.Normalize(mean=[0.485, 0.456, 0.406], std=[0.229, 0.224, 0.225])
    base_aug = [
        transforms.RandomHorizontalFlip(), transforms.ColorJitter(brightness=0.4, contrast=0.4, saturation=0.4, hue=0.4),
        transforms.RandomGrayscale(), normalizer]
    global transform_train, transform_test
    transform_train = transforms.Compose([transforms.RandomResizedCrop(224, scale=(0.7, 1.0)), transforms.ToTensor()])
    transform_test = transforms.Compose([transforms.Resize([224, 224]), transforms.ToTensor(), normalizer])
    return base_aug, normalizer

def ra_transforms():
    global transform_train
    transform_train.transforms.insert(1, transforms.RandAugment())


def base_transforms(aug='standard'):
    if aug == 'aa':
        transform_train.transforms.insert(1, transforms.AutoAugment())
    elif aug == 'ra':
        transform_train.transforms.insert(1, transforms.RandAugment())
    elif aug == 'cutout':
        normalize_index = next(
            (i for i, t in enumerate(transform_train.transforms) if isinstance(t, transforms.Normalize)), None)
        if normalize_index is not None:
            transform_train.transforms.insert(normalize_index, transforms.RandomErasing())
        else:
            transform_train.transforms.append(transforms.RandomErasing())


class TerraInc_SingleDomain():
    def __init__(self, root_path=terrainc_path, domain_name='100', split='train', train_transform=None, seed=0):
        self.domain_name = domain_name
        assert domain_name in terra_incognita_name_dict.keys(), 'domain_name must be in {}'.format(terra_incognita_name_dict.keys())
        self.root_path = root_path
        self.domain = terra_incognita_name_dict[domain_name]
        self.domain_label = list(terra_incognita_name_dict.keys()).index(domain_name)
        self.txt_path = os.path.join(root_path, '{}_img_label_list.txt'.format(self.domain))
        
        self.split = split
        assert self.split in ['train', 'val', 'test'] , 'split must be train, val or test'
        
        if train_transform is not None:
            self.transform = train_transform
        else:
            self.transform = transform_test
        self.seed = seed
        
        self.imgs, self.labels = TerraInc_SingleDomain.read_txt(self.txt_path)
        
        if self.split == 'train' or self.split == 'val':
            random.seed(self.seed)
            train_img, val_img = TerraInc_SingleDomain.split_list(self.imgs, 0.8)
            random.seed(self.seed)
            train_label, val_label = TerraInc_SingleDomain.split_list(self.labels, 0.8)
            if self.split == 'train':
                self.imgs, self.labels = train_img, train_label
            elif self.split == 'val':
                self.imgs, self.labels = val_img, val_label
                
        self.dataset = MetaDataset(self.imgs, self.labels, self.domain_label, self.transform) # get数据集
    
    @staticmethod
    def split_list(l, ratio):
        assert ratio > 0 and ratio < 1
        random.shuffle(l)
        train_size = int(len(l)*ratio)
        train_l = l[:train_size]
        val_l = l[train_size:]
        return train_l, val_l
        
    @staticmethod
    def read_txt(txt_path):
        imgs = []
        labels = []
        with open(txt_path, 'r') as f:
            contents = f.readlines()
            
        for line_txt in contents:
            line_txt = line_txt.replace('\n', '')
            line_txt_list = line_txt.split(' ')
            imgs.append(line_txt_list[0])
            labels.append(int(line_txt_list[1]))
            
        return imgs, labels
    
class TerraInc_FedDG():
    def __init__(self, test_domain='100', batch_size=64, seed=0):
        self.batch_size = batch_size
        self.domain_list = list(terra_incognita_name_dict.keys())
        self.test_domain = test_domain
        self.train_domain_list = self.domain_list.copy()
        self.train_domain_list.remove(self.test_domain)  
        self.seed = seed
        
        self.site_dataset_dict = {}
        self.site_dataloader_dict = {}
        for domain_name in self.domain_list:
            self.site_dataloader_dict[domain_name], self.site_dataset_dict[domain_name] = TerraInc_FedDG.SingleSite(domain_name, self.batch_size, self.seed)

        self.val_dataset = ConcatDataset([self.site_dataset_dict[i]['val'] for i in self.train_domain_list])
        self.val_dataloader = torch.utils.data.DataLoader(self.val_dataset, batch_size=batch_size, shuffle=False, drop_last=False, num_workers=4, pin_memory=True)
        
        self.test_dataset = self.site_dataset_dict[self.test_domain]['test']
        self.test_dataloader = self.site_dataloader_dict[self.test_domain]['test']
        
          
    @staticmethod
    def SingleSite(domain_name, batch_size=64, seed=0):
        dataset_dict = {
            'train': TerraInc_SingleDomain(domain_name=domain_name, split='train', train_transform=transform_train, seed=seed).dataset,
            'val': TerraInc_SingleDomain(domain_name=domain_name, split='val', seed=seed).dataset,
            'test': TerraInc_SingleDomain(domain_name=domain_name, split='test', seed=seed).dataset,
        }
        dataloader_dict = GetDataLoaderDict(dataset_dict, batch_size)
        return dataloader_dict, dataset_dict
        
    def GetData(self):
        return self.site_dataloader_dict, self.site_dataset_dict

def GenFileList():
    '''
    get path label list on all domains
    '''
    total_class_list = None
    for domain_name in terra_incognita_name_dict.keys():
        domain = terra_incognita_name_dict[domain_name]
        domain_path = os.path.join(terrainc_path, domain)
        class_list = os.listdir(domain_path)
        class_list.sort()
        if total_class_list is None:
            total_class_list = class_list
        
        assert total_class_list == class_list, 'class_list must be same'
    
    domain_file_dict = {}
    for domain_name in terra_incognita_name_dict.keys():
        domain_file_dict[domain_name] = []
        domain = terra_incognita_name_dict[domain_name]
        domain_path = os.path.join(terrainc_path, domain)
        for label_idx, class_name in enumerate(total_class_list):
            class_path = os.path.join(domain_path, class_name)
            file_list = os.listdir(class_path)
            
            for file_name in file_list:
                file_path = os.path.join(class_path, file_name)
                domain_file_dict[domain_name].append(file_path + ' ' + str(label_idx) + '\n')
        
    return domain_file_dict

if __name__ == '__main__':
    domain_file_dict = GenFileList()
    for (domain_abb, domain_name), domain_list in zip(terra_incognita_name_dict.items(), domain_file_dict.values()):
        file_name = os.path.join(terrainc_path, '{}_img_label_list.txt'.format(domain_name))
        with open(file_name, 'w') as file:
            # 将列表中的每个元素写入文件
            file.writelines(domain_list)
