from models.base_arch import BaseArch
from models.dgcnn import DGCNN
from models.modules import MLP, MLP_PointCloud
import torch
import os 
from omegaconf import OmegaConf
from models.foldingnet.models_by_AnTao import FoldNet_Encoder, DGCNN_Cls_Encoder, DGCNN_Seg_Encoder, FoldNet_Decoder
from models.pointnet import PointNet
from models.pointnet2 import PointNet2
#from models.atlasnet.atlasnet import Atlasnet
#from models.atlasnet.atlasnet import PointNet as Atlas_Pointnet
#from models.pointcapsnet.pointcapsnet import PointCapsNet_Encoder, PointCapsNet_Decoder

def get_model(cfg, *args, version=None, **kwargs):
    if 'model' in cfg:
        model_dict = cfg['model']
    elif 'arch' in cfg:
        model_dict = cfg
    else:
        raise ValueError(f'Invalid model configuration dictionary: {cfg}')

    name = model_dict["arch"]
    model = _get_model_instance(name)
    return model(**model_dict)

def _get_model_instance(name):
    try:
        return {
            "base_arch": get_base_arch
        }[name]
    except:
        raise ("Model {} not available".format(name))

def get_base_arch(**cfg):
    encoder_model = get_encoder(**cfg['encoder'])
    decoder_model = get_decoder(**cfg['decoder'])
    MED = cfg.get('MED', 0.0)
    k = cfg.get('k', 0.0)
    kl_reg = cfg.get('kl_reg', None)
    fm_reg = cfg.get('fm_reg', None)
    alpha_0 = cfg.get('alpha_0', 0.2)
    relaxation = cfg.get('relaxation', False)
    visualization = cfg.get('visualization', False)
    fm_val_show = bool(cfg.get('fm_val_show', False))
    latent_show = bool(cfg.get('latent_show', False))
    latent_G_show = bool(cfg.get('latent_G_show', False))
    use_identity = bool(cfg.get('use_identity', False))
    
    ### newly added ###
    contractive_reg = cfg.get('contractive_reg', None)
    contractive_use_identity = bool(cfg.get('contractive_use_identity', False))
    denoise_reg = cfg.get('denoise_reg', None)
    denoise_use_identity = bool(cfg.get('denoise_use_identity', False))
    eps = cfg.get('eps', 1e-12)
    svm_dual = cfg.get('svm_dual', True)
    ################
    
    model = BaseArch(
        encoder_model, decoder_model, kl_reg=kl_reg, fm_reg=fm_reg, 
        alpha_0=alpha_0, relaxation=relaxation, sigma=MED * k, 
        fm_val_show=fm_val_show, visualization=visualization, 
        latent_show=latent_show, latent_G_show=latent_G_show,
        use_identity=use_identity, contractive_reg=contractive_reg, 
        contractive_use_identity=contractive_use_identity, 
        denoise_reg=denoise_reg, 
        denoise_use_identity=denoise_use_identity, eps = eps,
        svm_dual = svm_dual)
    return model


# def get_backbone(**kwargs):
#     name = kwargs['arch']
#     backbone = _get_backbone_instance(name)
#     net = backbone(**kwargs)
#     return net

def get_encoder(**kwargs):
    net_list = []
    for key in kwargs.keys():
        module_dict = kwargs[key]
        name = module_dict['arch']
        module_nn = _get_encoder_instance(name)
        module = module_nn(**module_dict)
        net_list.append(module)
    return net_list

def get_decoder(**kwargs):
    net_list = []
    for key in kwargs.keys():
        module_dict = kwargs[key]
        name = module_dict['arch']
        module_nn = _get_decoder_instance(name)
        module = module_nn(**module_dict)
        net_list.append(module)
    return net_list

def _get_encoder_instance(name):
    try:
        return {
            'dgcnn': get_dgcnn,
            'mlp': get_mlp,
            'foldnet': get_lspc,
            'dgcnn_cls': get_lspc,
            'dgcnn_seg': get_lspc,
            'atlas_pointnet': get_atlas_pointnet,
            'pointcapsnet': get_pointcapsnet_encoder,
        }[name]
    except:
        raise ("Model {} not available".format(name))

def _get_decoder_instance(name):
    try:
        return {
            'mlp_pointcloud': get_mlp_pointcloud,
            'foldnet': get_fold_decoder,
            'atlasnet': get_atlasnet,
            'pointcapsnet': get_pointcapsnet_decoder,
        }[name]
    except:
        raise ("Model {} not available".format(name))

# encoder
def get_dgcnn(**cfg):
    model = DGCNN(**cfg)
    return model

def get_lspc(**cfg):
    if cfg['arch'] == "foldnet":
        model = FoldNet_Encoder(cfg)
    elif cfg['arch'] == "dgcnn_cls":
        model = DGCNN_Cls_Encoder(cfg)
    elif cfg['arch'] == "dgcnn_seg":
        model = DGCNN_Seg_Encoder(cfg)
    return model

def get_atlas_pointnet(**cfg):
    nlatent = cfg["nlatent"]
    model = Atlas_Pointnet(nlatent=nlatent)
    return model 

def get_pointcapsnet_encoder(**cfg):
    prim_caps_size = cfg.get('prim_caps_size', 1024)
    prim_vec_size = cfg.get('prim_vec_size', 16)
    latent_caps_size = cfg.get('latent_caps_size', 32)
    latent_vec_size = cfg.get('latent_vec_size', 16)
    num_points = cfg.get('num_points', 2048)
    model = PointCapsNet_Encoder(prim_caps_size,prim_vec_size,latent_caps_size,latent_vec_size,num_points)
    return model 

# decoder
def get_mlp(**cfg):
    model = MLP(**cfg)
    return model

def get_mlp_pointcloud(**cfg):
    model = MLP_PointCloud(**cfg)
    return model

def get_fold_decoder(**cfg):
    model = FoldNet_Decoder(cfg)
    return model

def get_atlasnet(**cfg):
    bottleneck_size = cfg["bottleneck_size"]
    opt_dict = {
        "nb_primitives": 25,
        "template_type": "SQUARE",
        "lambda_conformal_regul": 0.001,
        "device": 'cuda:0',
        "number_points": 2048,
        "number_points_eval": 2048,
        "remove_all_batchNorms": True,
        "bottleneck_size": bottleneck_size,
        "dim_template": 3,
        "hidden_neurons": 512,
        "num_layers": 2,
        "activation": 'relu',
        "template_type": "SPHERE"
    }
    class Struct:
        def __init__(self, **entries):
            self.__dict__.update(entries)
    opt = Struct(**opt_dict)
    model = Atlasnet(opt)
    return model

def get_pointcapsnet_decoder(**cfg):
    latent_caps_size = cfg.get('latent_caps_size', 32)
    latent_vec_size = cfg.get('latent_vec_size', 16)
    num_points = cfg.get('num_points', 2048)
    model = PointCapsNet_Decoder(latent_caps_size,latent_vec_size,num_points)
    return model 

def load_pretrained(identifier, config_file, ckpt_file, root='pretrained', **kwargs):
    """
    load pre-trained model.
    identifier: '<model name>/<run name>'. e.g. 'ae_mnist/z16'
    config_file: name of a config file. e.g. 'ae.yml'
    ckpt_file: name of a model checkpoint file. e.g. 'model_best.pth'
    root: path to pretrained directory
    """
    config_path = os.path.join(root, identifier, config_file)
    ckpt_path = os.path.join(root, identifier, ckpt_file)
    cfg = OmegaConf.load(config_path)
    if "model" in cfg:
        model_name = cfg["model"]['arch']
    else:
        model_name = cfg['arch']
    
    model = get_model(cfg)
    ckpt = torch.load(ckpt_path, map_location='cpu')
    if 'model_state' in ckpt:
        ckpt = ckpt['model_state']
    model.load_state_dict(ckpt)

    return model, cfg