import json
import sys
from pycocotools.coco import COCO
# from pycocotools.cocoeval import COCOeval
from detectron2.evaluation.fast_eval_api import COCOeval_opt

def read_coco_ann_as_results(coco_ann_file):
    """
    This function reads a coco annotation file and converts it to a results file with a given score
    :param coco_ann_file: path to coco annotation file
    :param score: score to give the annotations
    :return: results
    """
    coco = COCO(coco_ann_file)

    results = []

    for img_id in coco.getImgIds():
        ann_ids = coco.getAnnIds(imgIds=img_id)
        anns = coco.loadAnns(ann_ids)
        for ann in anns:
            if "score" in ann:
                score = ann["score"]
            else:
                score = 1.0
            if "segmentation" in ann:
                segmentation = ann["segmentation"]
            else:
                segmentation = []
            results.append({
                "image_id": img_id,
                "category_id": ann["category_id"],
                "score": score,
                "bbox": [round(b) for b in ann["bbox"]],
                "segmentation": segmentation
            })

    return results

def derive_coco_results(coco_eval, iou_type):
    metrics = {
        "bbox": ["AP", "AP50", "AP75", "APs", "APm", "APl"],
        "segm": ["AP", "AP50", "AP75", "APs", "APm", "APl"],
        "keypoints": ["AP", "AP50", "AP75", "APm", "APl"],
    }[iou_type]

    ar_metrics = ["AR@1", "AR@10", "AR@100", "ARs", "ARm", "ARl"]

    results = {
        metric: round(coco_eval.stats[idx] * 100, 1) if coco_eval.stats[idx] >= 0 else float("nan")
        for idx, metric in enumerate(metrics)
    }

    for idx, metric in enumerate(ar_metrics, start=6):
        results[metric] = round(coco_eval.stats[idx] * 100, 1) if coco_eval.stats[idx] >= 0 else float("nan")

    print(f'Task {iou_type}')
    print(",".join(results.keys()))
    print(",".join(f"{v}" for v in results.values()))
    print('AP,AP50,AP75,AR@100')
    print(",".join(f"{results[k]}" for k in ['AP', 'AP50', 'AP75', 'AR@100']))

def detect_json_top_structure(path):
    with open(path, 'r', encoding='utf-8') as f:
        data = json.load(f)
    
    if isinstance(data, list):
        return 'list'
    elif isinstance(data, dict):
        return 'dict'
    else:
        return None

def determine_iou_type(coco_annotations):
    """
    Check segmentation field in COCO annotations to determine iou_type
    
    Args:
        coco_annotations: COCO annotation data or annotation list
        
    Returns:
        str: 'bbox' or 'segm' based on segmentation field presence
    """
    # Check if annotations have segmentation field
    if 'annotations' in coco_annotations:
        annotations = coco_annotations['annotations']
    else:
        annotations = coco_annotations
    
    # Check if any annotation has segmentation data
    for ann in annotations:
        if 'segmentation' in ann and ann['segmentation']:
            # Check if segmentation is not empty
            if isinstance(ann['segmentation'], list) and len(ann['segmentation']) > 0:
                return ['segm', 'bbox']
            elif isinstance(ann['segmentation'], dict) and ann['segmentation'].get('counts'):
                return ['segm', 'bbox']  # RLE format
    
    return ['bbox']  # Default to bbox if no segmentation found

def eval_json(gt_path, dt_path):
    coco_gt = COCO(gt_path)

    json_top_structure = detect_json_top_structure(dt_path)
    if json_top_structure == 'dict':
        res = read_coco_ann_as_results(dt_path)
        coco_dt = coco_gt.loadRes(res)
    elif json_top_structure == 'list':
        coco_dt = coco_gt.loadRes(dt_path)
    else:
        raise ValueError("Not support")

    iou_types = determine_iou_type(coco_gt.dataset)
    # print(iou_types)
    # exit()
    for iou_type in iou_types:
        # coco_eval = COCOeval(coco_gt, coco_dt, iouType=iou_type)
        coco_eval = COCOeval_opt(coco_gt, coco_dt, iouType=iou_type)
        coco_eval.evaluate()
        coco_eval.accumulate()
        coco_eval.summarize()

        derive_coco_results(coco_eval, iou_type)

DATASET_ROOT=f"/data/xxx/datasets/"
EVAL_ROOT="coler_official_eval/cls_agnostic"
result_eval_dict={
    "coco20k": [f"{DATASET_ROOT}coco/annotations/coco20k_trainval_gt.json", f"{EVAL_ROOT}_coco20k/inference/coco_instances_results.json"],
    "voc": [f"{DATASET_ROOT}voc/annotations/trainvaltest_2007_cls_agnostic.json", f"{EVAL_ROOT}_voc/inference/coco_instances_results.json"],
    "coco": [f"{DATASET_ROOT}coco/annotations/coco_cls_agnostic_instances_val2017.json", f"{EVAL_ROOT}_coco/inference/coco_instances_results.json"],
    "imagenet": [f"{DATASET_ROOT}imagenet/annotations/imagenet_val_cls_agnostic_gt.json", f"{EVAL_ROOT}_imagenet/inference/coco_instances_results.json"],
    "lvis": [f"{DATASET_ROOT}coco/annotations/lvis1.0_cocofied_val_cls_agnostic.json", f"{EVAL_ROOT}_lvis/inference/coco_instances_results.json"],
    "clipart": [f"{DATASET_ROOT}clipart/annotations/traintest_clipart_cls_agnostic.json", f"{EVAL_ROOT}_clipart/inference/coco_instances_results.json"],
    "watercolor": [f"{DATASET_ROOT}watercolor/annotations/traintest_watercolor_cls_agnostic.json", f"{EVAL_ROOT}_watercolor/inference/coco_instances_results.json"],
    "comic": [f"{DATASET_ROOT}comic/annotations/traintest_comic_cls_agnostic.json", f"{EVAL_ROOT}_comic/inference/coco_instances_results.json"],
    "kitti": [f"{DATASET_ROOT}kitti/annotations/trainval_cls_agnostic.json", f"{EVAL_ROOT}_kitti/inference/coco_instances_results.json"],
    "openimages": [f"{DATASET_ROOT}openimages-v7/annotations/openimages_val_cls_agnostic.json", f"{EVAL_ROOT}_openimages/inference/coco_instances_results.json"],
    "objects365": [f"{DATASET_ROOT}objects365/annotations/zhiyuan_objv2_val_cls_agnostic.json", f"{EVAL_ROOT}_objects365/inference/coco_instances_results.json"],
}

def eval_coco_json(dataset_name, predict_path):
    gt = result_eval_dict[dataset_name][0]
    dt = predict_path
    eval_json(gt,dt)

def eval_all():
    dataset_names = ["coco", "coco20k", "lvis", "voc", "clipart", "watercolor", "comic", "kitti", "openimages", "objects365"]
    for dataset_name in dataset_names:
        gt = result_eval_dict[dataset_name][0]
        dt = result_eval_dict[dataset_name][1]
        try:
            print(f"=== Evaluating {dataset_name} ===")
            eval_json(gt, dt)
        except Exception as e:
            print(f"Error in {dataset_name}: {e}")
            continue

if __name__ == "__main__":
    # if len(sys.argv) != 3:
    #     print(f"usage: {sys.argv[0]} <dataset> <predict_path>")
    #     exit(0)
    if len(sys.argv) == 3:
        print("=== eval a json ===")
        param1 = sys.argv[1]
        param2 = sys.argv[2]
        eval_coco_json(sys.argv[1], sys.argv[2])
    else:
        print("=== eval all jsons ===")
        eval_all()

