import argparse
import functools
import os
import sys
import traceback

import ignite
import ignite.utils as ignite_utils
import torch
import torch.nn.functional as F
import yaml
from ignite.contrib.handlers.tqdm_logger import ProgressBar
from ignite.engine import Engine, Events
from ignite.handlers import ModelCheckpoint
from ignite.handlers.param_scheduler import LRScheduler, PiecewiseLinear
from ignite.metrics import Accuracy, Loss
from ignite.metrics.running_average import RunningAverage
from torch.backends import cudnn
from torch.optim.lr_scheduler import MultiStepLR

sys.path.append(os.path.join(os.path.dirname(__file__), ".."))
from evaluator.classification import ClassifierEvaluator
from model.ema import ModelEMA
from util import train_util, yaml_utils

import extensions


def log_basic_info(logger, config):
    logger.info(f"- PyTorch version: {torch.__version__}")
    logger.info(f"- Ignite version: {ignite.__version__}")
    if torch.cuda.is_available():
        logger.info(f"- GPU Device: {torch.cuda.get_device_name(0)}")
        logger.info(f"- CUDA version: {torch.version.cuda}")
        logger.info(f"- CUDNN version: {cudnn.version()}")
    logger.info("--------------")
    logger.info("Configuration:")
    for key, value in config.items():
        logger.info(f"\t{key}: {value}")
    logger.info("--------------")


def main(config):

    device = "cuda:0" if (torch.cuda.is_available()) else "cpu"

    logger = ignite_utils.setup_logger(name=config["pattern"])
    log_basic_info(logger, config)

    # Create output directory
    logger.info("Creating output directory")
    out = os.path.join(config["results_dir"], config["pattern"], f"experiment{config['experiment_id']}")
    train_util.create_result_dir(out, config["config_path"])

    # DataLoader
    logger.info("Constructing data loaders")
    train_loader, val_loader, test_loader = train_util.setup_dataloaders(config)
    u_loader = train_util.setup_unsupervised_dataloaders(config)

    # Model
    logger.info("Constructing models")
    classifier = train_util.load_models(config["models"]["classifier"])
    if torch.cuda.device_count() > 1:
        classifier = torch.nn.DataParallel(classifier)
    classifier.to(device)
    # EMA model for evaluation
    ema_model = None
    if "use_ema" in config and config["use_ema"]:
        ema_model = ModelEMA(device, classifier, config["ema_decay"])

    # Optimizer
    logger.info("Constructing optimizers")
    opt = train_util.make_optimizer(classifier, config)

    if config["resume"]:
        logger.info("Resume training with snapshot:{}".format(config["resume"]))
        if os.path.isfile(config["resume"]):
            checkpoint = torch.load(config["resume"])
            classifier.load_state_dict(checkpoint["model"])
            opt.load_state_dict(checkpoint["optimizer"])
            config["start_epoch"] = opt.param_groups[0]["epoch"]

    # Updater
    logger.info("Constructing updater and evaluators")
    kwargs = config["updater"]["args"] if "args" in config["updater"] else {}
    kwargs.update(
        {
            "classifier": classifier,
            "loader": train_loader,
            "optimizer": opt,
            "device": device,
            "ema_model": ema_model,
            "max_iter": len(train_loader) * config["epoch"],
            "u_loader": u_loader,
            "ubatch_ratio": config["ubatch_ratio"],
        }
    )
    updater = yaml_utils.load_updater_class(config)
    updater = updater(**kwargs)

    # Trainer := Ignite.Engine
    trainer = Engine(updater)
    monitoring_metrics = ["train_accuracy", "train_loss"]
    RunningAverage(Accuracy(output_transform=lambda x: [x["y_pred"], x["y"]])).attach(trainer, "train_accuracy")
    RunningAverage(output_transform=lambda x: x["loss"]).attach(trainer, "train_loss")
    if "log_metrics" in config:
        for key in config["log_metrics"]:
            monitoring_metrics.append(key)
            RunningAverage(output_transform=lambda x: x[key]).attach(trainer, key)
    logger.info(f"Monitoring Metrics: {monitoring_metrics}")
    pbar = ProgressBar()
    pbar.attach(trainer, metric_names=monitoring_metrics)

    # Evaluator
    evaluator = Engine(ClassifierEvaluator(classifier=classifier, device=device, ema_model=ema_model))
    eval_metrics = {"accuracy": Accuracy(), "loss": Loss(F.cross_entropy)}
    extensions.register_metrics(evaluator, eval_metrics)

    # Event Handlers
    logger.info("Constructing event handlers")
    # Log Handler
    log = {"running": [], "best_val_accuracy": 0.0, "test_accuracy": 0.0}
    log = log if not config["resume"] else extensions.load_log(out)
    logger_train = functools.partial(extensions.log_training_results, log=log, pbar=pbar)
    logger_val = functools.partial(
        extensions.log_validation_results,
        evaluator=evaluator,
        val_loader=val_loader,
        test_loader=test_loader,
        log=log,
        pbar=pbar,
        dist=str(out),
    )

    # Check Point Handler
    check_pointer = ModelCheckpoint(str(out), filename_prefix="model", n_saved=1)
    best_check_pointer = ModelCheckpoint(
        str(out), filename_prefix="best", score_function=extensions.check_accuracy, n_saved=1, score_name="val_accuracy"
    )

    # Learning Rate Schedule Handler
    milestone = config["optimizer"]["lr_milestone"]
    if milestone == "linear":
        init_lr = config["optimizer"]["args"]["lr"]
        max_iter = int(len(train_loader) * config["epoch"])
        lr_scheduler = PiecewiseLinear(opt, "lr", milestones_values=[(0, init_lr), (max_iter, 0)], save_history=True)
        trainer.add_event_handler(Events.ITERATION_STARTED, lr_scheduler)
    if milestone == "cosine":
        max_iter = int(len(train_loader) * config["epoch"])
        lr_scheduler = LRScheduler(train_util.CosineAnnealingLR(opt, max_iter))
        trainer.add_event_handler(Events.ITERATION_STARTED, lr_scheduler)
    else:
        gamma = config["optimizer"]["lr_drop_rate"]
        lr_scheduler = LRScheduler(MultiStepLR(opt, milestones=milestone, gamma=gamma), save_history=True)
        trainer.add_event_handler(Events.EPOCH_STARTED, lr_scheduler)

    # Append handlers to trainer/evaluator engine
    trainer.add_event_handler(Events.EPOCH_COMPLETED, logger_train)
    trainer.add_event_handler(Events.EPOCH_COMPLETED, logger_val)
    trainer.add_event_handler(
        Events.EPOCH_COMPLETED(every=config["snapshot_interval"]),
        check_pointer,
        {"model": classifier, "optimizer": opt},
    )
    evaluator.add_event_handler(
        Events.COMPLETED(every=config["snapshot_interval"]), best_check_pointer, {"model": classifier}
    )

    if config["resume"]:
        resumer = functools.partial(extensions.resume_training, resume_epoch=config["start_epoch"])
        trainer.add_event_handler(Events.EPOCH_STARTED, resumer)

    # Run the training
    logger.info("Running train script")
    try:
        trainer.run(train_loader, max_epochs=config["epoch"])
    except Exception as e:
        logger.error(e)
        logger.error(traceback.format_exc())
    finally:
        log.update({"best_model": str(best_check_pointer.last_checkpoint)})
        extensions.dump_log(log, str(out))


def run():
    parser = argparse.ArgumentParser()
    parser.add_argument("--config_path", type=str, default="configs/base.yml", help="path to config file")
    parser.add_argument("--results_dir", type=str, default="./result", help="directory to save the results to")
    parser.add_argument("--resume", type=str, default="", help="path to the snapshot")
    parser.add_argument("--experiment_id", type=int, default=0)
    parser.add_argument("--num_worker", type=int, default=16)
    parser.add_argument("--backend", type=str, default="nccl")
    parser.add_argument("--seed", type=int, default=42)

    args = parser.parse_args()
    config = yaml_utils.Config(yaml.load(open(args.config_path), Loader=yaml.SafeLoader))
    config.config_dict["pattern"] = yaml_utils.make_pattern(config)
    config.config_dict.update(vars(args))
    main(config.config_dict)


if __name__ == "__main__":
    run()
