#!/usr/bin/env python3
# Copyright (c) Facebook, Inc. and its affiliates. All Rights Reserved
import torch
import os
import detectron2.utils.comm as comm
from detectron2.checkpoint import DetectionCheckpointer
from detectron2.config import get_cfg
from detectron2.engine import default_argument_parser, default_setup, launch
from detectron2.modeling import GeneralizedRCNN
from croptrain.modeling.meta_arch.crop_rcnn import CropRCNN
from croptrain.engine.predictor import DroneDetPredictor

from croptrain import add_croptrainer_config
from croptrain.engine.trainer import BaselineTrainer
# hacky way to register
import croptrain.data.datasets.builtin
from croptrain.data.datasets.visdrone import register_visdrone
from croptrain.data.datasets.dota import register_dota
from croptrain.data.datasets.teldrone import register_teldrone

from detectron2.data import DatasetCatalog, MetadataCatalog
# from .missing_person_data import register_missing_person_dataset
# from missing_person_data import register_missing_person_dataset
from seasonal_data_loader import register_forestpersons_summer, register_forestpersons_winter, register_forestpersons_standing, register_forestpersons_combined, register_forestpersons_original


def setup(args):
    """
    Create configs and perform basic setups.
    """
    cfg = get_cfg()
    add_croptrainer_config(cfg)
    cfg.merge_from_file(args.config_file)
    cfg.merge_from_list(args.opts)
    cfg.freeze()
    default_setup(cfg, args)
    return cfg


def main(args):
    cfg = setup(args)
    if not torch.cuda.is_available():
        cfg.defrost()
        cfg.MODEL.DEVICE = 'cpu'
        cfg.freeze()
    Trainer = BaselineTrainer

    if cfg.CROPTRAIN.USE_CROPS:
        cfg.defrost()
        cfg.MODEL.ROI_HEADS.NUM_CLASSES += 1
        cfg.freeze()

    if "visdrone" in cfg.DATASETS.TRAIN[0] or "visdrone" in cfg.DATASETS.TEST[0]:
        # data_dir = os.path.join(os.environ['SLURM_TMPDIR'], "VisDrone")
        tmpdir = "/mnt/home/annonymous/layout_diffusion/yolov11_forestpersons/drone_detectron2/DroneDetectron2/croptrain/data/datasets"
        data_dir = os.path.join(tmpdir, "VisDrone")
        if not args.eval_only:
            register_visdrone(cfg.DATASETS.TRAIN[0], data_dir, cfg, True)
        register_visdrone(cfg.DATASETS.TEST[0], data_dir, cfg, False)
        breakpoint()


        # TMP
        # full_val_dicts = DatasetCatalog.get("visdrone_2019_val")
        # subset_val_dicts = full_val_dicts[:500]
        # DatasetCatalog.register("visdrone_2019_val_subset", lambda: subset_val_dicts)
        # MetadataCatalog.get("visdrone_2019_val_subset").set(
        #     thing_classes=MetadataCatalog.get("visdrone_2019_val").thing_classes,
        #     evaluator_type=MetadataCatalog.get("visdrone_2019_val").evaluator_type,
        #     image_root=MetadataCatalog.get("visdrone_2019_val").image_root,
        #     json_file=MetadataCatalog.get("visdrone_2019_val").json_file,
        #     thing_dataset_id_to_contiguous_id=MetadataCatalog.get("visdrone_2019_val").thing_dataset_id_to_contiguous_id,
        # )
        # cfg.defrost()
        # cfg.DATASETS.TEST = ("visdrone_2019_val_subset",)
        # cfg.freeze()
        # TMP

    if "dota" in cfg.DATASETS.TRAIN[0] or "dota" in cfg.DATASETS.TEST[0]:
        data_dir = os.path.join(os.environ['SLURM_TMPDIR'], "DOTA")
        if not args.eval_only:
            register_dota(cfg.DATASETS.TRAIN[0], data_dir, cfg, True)
        register_dota(cfg.DATASETS.TEST[0], data_dir, cfg, False)
    if "teldrone" in cfg.DATASETS.TRAIN[0] or "teldrone" in cfg.DATASETS.TEST[0]:
        data_dir = os.path.join(os.environ['SLURM_TMPDIR'], "TelDrone")
        if not args.eval_only:
            register_teldrone(cfg.DATASETS.TRAIN[0], data_dir, cfg, True)
        register_teldrone(cfg.DATASETS.TEST[0], data_dir, cfg, False)
    
    
    # register_missing_person_dataset()
    # cfg.dataloader.train.dataset.names = "missing_person_train"
    # cfg.dataloader.test.dataset.names = "missing_person_val"
    # Register ForestPersons Winter dataset
    # register_forestpersons_winter()
    # register_forestpersons_summer()
    # register_forestpersons_standing()
    # register_forestpersons_combined()
    register_forestpersons_original()

    if args.eval_only:
        model = Trainer.build_model(cfg)
        DetectionCheckpointer(model, save_dir=cfg.OUTPUT_DIR).resume_or_load(
            cfg.MODEL.WEIGHTS, resume=args.resume
        )
        if cfg.CROPTRAIN.USE_CROPS:
            res = Trainer.test_crop(cfg, model, 0)
        else:
            if "dota" in cfg.DATASETS.TEST[0]:
                res = Trainer.test_crop(cfg, model, 0)
            else:
                res = Trainer.test(cfg, model)
        return res
    if cfg.CROPTEST.PREDICT_ONLY:
        predictor = DroneDetPredictor(cfg)
        predictor("image")
        return

    trainer = Trainer(cfg)
    trainer.resume_or_load(resume=args.resume)

    return trainer.train()


if __name__ == "__main__":
    args = default_argument_parser().parse_args()
    print("No of gpus used: {}".format(args.num_gpus))
    print("Cuda detected {} gpus".format(torch.cuda.device_count()))

    print("Command Line Args:", args)
    launch(
        main,
        args.num_gpus,
        num_machines=args.num_machines,
        machine_rank=args.machine_rank,
        dist_url=args.dist_url,
        args=(args,),
    )
