import logging
import wandb
from time import time
import os.path as osp

import torch

from utils.loss import LpLoss
from utils.helper import save_code
from datasets import CarraDataset

from trainers import TRAINER_DICT


def carra_procedure(args):
    save_code(osp.abspath(__file__), args['saving_path'], with_path=True)
    if args['model_name'] not in TRAINER_DICT.keys():
        raise NotImplementedError("Model {} not implemented".format(args['model_name']))
    
    if args['verbose']:
        logger = logging.info if args['log'] else print

    if args['wandb']:
        wandb.init(
            project=args['wandb_project'], 
            name=args['saving_name'],
            tags=[args['model'], args['dataset']],
            config=args)
    
    if args['verbose']:
        logger("Loading {} dataset".format(args['dataset']))
    start = time()
    dataset = CarraDataset(
        data_path=args['data_path'],
        sample_factor=args['sample_factor'],
        train_ratio=args['train_ratio'],
        valid_ratio=args['valid_ratio'],
        test_ratio=args['test_ratio'],
        train_batchsize=args['train_batchsize'],
        eval_batchsize=args['eval_batchsize'],
        normalize=args['normalize'],
        normalizer_type=args['normalizer_type'],
        prop=args['prop'],
    )
    save_code(dataset, args['saving_path'], with_dir=False)
    train_loader = dataset.train_loader
    valid_loader = dataset.valid_loader
    test_loader = dataset.test_loader
    if args['verbose']:
        logger("Loading data costs {: .2f}s".format(time() - start))
        
    if args['verbose']:
        logger("Building models")
    start = time()
    trainer = TRAINER_DICT[args['model_name']](args)
    save_code(trainer, args['saving_path'], with_dir=False)
    model = trainer.build_model(args)
    save_code(model, args['saving_path'], with_dir=True)
    model = model.to(args['device'])
    optimizer = trainer.build_optimizer(model, args)
    scheduler = trainer.build_scheduler(optimizer, args)
    criterion = LpLoss(d=2, p=2, size_average=False)
    if args['verbose']:
        logger("Model: {}".format(model))
        logger("Criterion: {}".format(criterion))
        logger("Optimizer: {}".format(optimizer))
        logger("Scheduler: {}".format(scheduler))
        logger("Building models costs {: .2f}s".format(time() - start))

    trainer.process(
        model=model,
        train_loader=train_loader,
        valid_loader=valid_loader,
        test_loader=test_loader,
        optimizer=optimizer,
        scheduler=scheduler,
        criterion=criterion,
        prop=args['prop'],
    )
