from pathlib import Path
from vbench import VBench
from vbench.distributed import dist_init, print0
import argparse
import torch
from datetime import datetime
import json
import time

def parse_args():
    parser = argparse.ArgumentParser(description='VBench', formatter_class=argparse.RawTextHelpFormatter)
    parser.add_argument(
        "--output_path",
        type=str,
        default='./evaluation_results/',
        help="output path to save the evaluation results",
    )
    parser.add_argument(
        "--full_json_dir",
        type=str,
        default="",
        help="path to save the json file that contains the prompt and dimension information",
    )
    parser.add_argument(
        "--load_ckpt_from_local",
        type=bool,
        required=False,
        help="whether load checkpoints from local default paths (assuming you have downloaded the checkpoints locally",
    )
    parser.add_argument(
        "--read_frame",
        type=bool,
        required=False,
        help="whether directly read frames, or directly read videos",
    )
    parser.add_argument(
        "--mode",
        choices=['custom_input', 'vbench_standard', 'vbench_category'],
        default='custom_input',
        help="""This flags determine the mode of evaluations, choose one of the following:
        1. "custom_input": receive input prompt from either --prompt/--prompt_file flags or the filename
        2. "vbench_standard": evaluate on standard prompt suite of VBench
        3. "vbench_category": evaluate on specific category
        """,
    )
    parser.add_argument(
        "--prompt",
        type=str,
        default="None",
        help="""Specify the input prompt
        If not specified, filenames will be used as input prompts
        * Mutually exclusive to --prompt_file.
        ** This option must be used with --mode=custom_input flag
        """
    )
    parser.add_argument(
        "--prompt_file",
        type=str,
        required=False,
        help="""Specify the path of the file that contains prompt lists
        If not specified, filenames will be used as input prompts
        * Mutually exclusive to --prompt.
        ** This option must be used with --mode=custom_input flag
        """
    )
    parser.add_argument(
        "--category",
        type=str,
        required=False,
        help="""This is for mode=='vbench_category'
        The category to evaluate on, usage: --category=animal.
        """,
    )

    ## for dimension specific params ###
    parser.add_argument(
        "--imaging_quality_preprocessing_mode",
        type=str,
        required=False,
        default='longer',
        help="""This is for setting preprocessing in imaging_quality
        1. 'shorter': if the shorter side is more than 512, the image is resized so that the shorter side is 512.
        2. 'longer': if the longer side is more than 512, the image is resized so that the longer side is 512.
        3. 'shorter_centercrop': if the shorter side is more than 512, the image is resized so that the shorter side is 512. 
        Then the center 512 x 512 after resized is used for evaluation.
        4. 'None': no preprocessing
        """,
    )
    args = parser.parse_args()
    return args

def evaluate_videos(args, video_folder: Path, output_folder: Path):
    if any(output_folder.iterdir()):
        print("Results exists.")
        return
    
    current_hour = datetime.now().hour
    while not (current_hour >= 18 or current_hour <= 7):
        time.sleep(60)
        current_hour = datetime.now().hour
    
    print0(f'args: {args}')
    device = torch.device("cuda")
    my_VBench = VBench(device, args.full_json_dir, output_folder)
    print0(f'start evaluation')
    print0(f'video_folder: {video_folder}')
    print0(f'output_folder: {output_folder}')
    
    kwargs = {}

    prompt = []
    
    if (args.prompt_file is not None) and (args.prompt != "None"):
        raise Exception("--prompt_file and --prompt cannot be used together")
    if (args.prompt_file is not None or args.prompt != "None") and (args.mode!='custom_input'):
        raise Exception("must set --mode=custom_input for using external prompt")

    if args.prompt_file:
        with open(args.prompt_file, 'r') as f:
            prompt = json.load(f)
        assert type(prompt) == dict, "Invalid prompt file format. The correct format is {\"video_path\": prompt, ... }"
    elif args.prompt != "None":
        prompt = [args.prompt]

    if args.category != "":
        kwargs['category'] = args.category

    kwargs['imaging_quality_preprocessing_mode'] = args.imaging_quality_preprocessing_mode

    dimension_list = ['subject_consistency', 'background_consistency', 'motion_smoothness', 'dynamic_degree', 'aesthetic_quality', 'imaging_quality']

    my_VBench.evaluate(
        videos_path = video_folder,
        name = f'results',
        prompt_list=prompt, # pass in [] to read prompt from filename
        dimension_list = dimension_list,
        local=args.load_ckpt_from_local,
        read_frame=args.read_frame,
        mode=args.mode,
        **kwargs
    )


def main():
    args = parse_args()
    
    dist_init()

    database_folder = Path(__file__).resolve().parent.parent / "database"
    output_base_folder = Path(__file__).resolve().parent.parent / "vbench_output"
    llm_names = [folder.name for folder in database_folder.iterdir() if folder.is_dir() and not folder.name.endswith("sample")]
    for llm_name in llm_names:
        llm_folder = database_folder / llm_name
        scenario_ids = [folder.name for folder in llm_folder.iterdir() if folder.is_dir()]
        for scenario_id in scenario_ids:
            video_base_folder = llm_folder / scenario_id / "videos"
            for video_type in video_base_folder.iterdir():
                video_type_folder = video_base_folder / video_type.name
                for value in video_type_folder.iterdir():
                    value_folder = video_type_folder / value.name
                    for prompt_id in value_folder.iterdir():
                        prompt_folder = value_folder / prompt_id.name
                        output_folder = output_base_folder / llm_name / scenario_id / "videos" / video_type.name / value.name / prompt_id.name
                        output_folder.mkdir(exist_ok=True, parents=True)
                        evaluate_videos(args=args, video_folder=prompt_folder, output_folder=output_folder)

    print0('done')


if __name__ == "__main__":
    main()

