#!/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
# Import seasonal dataset registration
from seasonal_data_loader import register_forestpersons_winter


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)

    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 ForestPersons Winter dataset
    register_forestpersons_winter()

    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,),
    )