import os
import sys
import yaml
import psutil
import datetime
import traceback
import ray
import ray.tune as tune

sys.path.append(os.path.abspath(".."))

from config.benchmark import ROOT_DIR, BENCHMARK_DIR, benchmark_dict, benchmark_type_dict, benchmark_path_dict
THIRDPARTY_DIR = os.path.join(ROOT_DIR, "thirdparty")
SOURCE_DIR = os.path.join(ROOT_DIR, "src")

from types import SimpleNamespace
from logger import Logger
from utils.debug import *
from utils.random_parser import set_seed
from utils.res2sheet import res2sheet

sys.path.append(ROOT_DIR)
sys.path.append(THIRDPARTY_DIR)
sys.path.append(SOURCE_DIR)
sys.path.append(BENCHMARK_DIR)

os.environ["PYTHONPATH"] = ":".join(sys.path)

cpus = psutil.cpu_count(logical=True)

import logging
logging.root.name = 'BBO4Placement'
logging.basicConfig(level=logging.INFO,
                    format='[%(levelname)-7s] %(name)s - %(message)s',
                    stream=sys.stdout)


def process_benchmark_path(benchmark):
    is_benchmark_registered = False
    for benchmark_base in benchmark_dict:
        if benchmark in benchmark_dict[benchmark_base]:
            is_benchmark_registered = True
            break
    if not is_benchmark_registered:
        assert0("benchmark was not registered in config/benchmark.py")

    benchmark_path = os.path.join(BENCHMARK_DIR, benchmark_base, benchmark)
    benchmark_type = benchmark_type_dict[benchmark_base]
    
    return benchmark_path, benchmark_type, benchmark_base


def set_error_log(file):
    error_log = open(file, 'a')
    os.dup2(error_log.fileno(), 2)


def process_args():
    # cmd config
    params = [arg.lstrip("--") for arg in sys.argv if arg.startswith("--")]

    cmd_config_dict = {}
    for arg in params:
        key, value = arg.split('=')
        try:
            cmd_config_dict[key] = eval(value)
        except:
            cmd_config_dict[key] = value


    # default config
    config_path = os.path.abspath("../config/default.yaml")
    with open(config_path, 'r') as f:
        config_dict = yaml.load(f, Loader=yaml.FullLoader)
    
    for key, value in cmd_config_dict.items():
        config_dict[key] = value

    # placer config
    with open(f"../config/placer/{config_dict['placer']}.yaml", 'r') as f:
        try:
            config_dict.update(yaml.load(f, Loader=yaml.FullLoader))
        except:
            pass
    
    # algo config
    with open(f"../config/algorithm/{config_dict['algorithm']}.yaml", 'r') as f:
        try:
            config_dict.update(yaml.load(f, Loader=yaml.FullLoader))
        except:
            pass

    for key, value in cmd_config_dict.items():
        config_dict[key] = value
    
    args = SimpleNamespace(**config_dict)
    args.benchmark_path, args.benchmark_type, args.benchmark_base = process_benchmark_path(config_dict["benchmark"])

    setattr(args, "ROOT_DIR", ROOT_DIR)
    setattr(args, "THIRDPARTY_DIR", THIRDPARTY_DIR)
    setattr(args, "SOURCE_DIR", SOURCE_DIR)

    return args

    
def single_run(args):
    
    # set seed
    set_seed(args.seed)

    # set unique token
    unique_token = "seed_{}_{}".format(args.seed, datetime.datetime.now().strftime("%Y-%m-%d_%H-%M-%S"))
    args.unique_token = unique_token

    # set result path
    args.result_path = os.path.join(ROOT_DIR, 
                                    f"results/{args.benchmark}/{args.name}/{args.placer}/{args.algorithm}/{args.unique_token}")
    os.makedirs(args.result_path, exist_ok=True)

    # set error log
    error_log_file = os.path.join(args.result_path, "error.log")
    set_error_log(file=error_log_file)
    args.error_log_file = error_log_file
    

    logger = Logger(args=args)
    placedb = PlaceDB(args=args)
    placer = PLACER_REGISTRY[args.placer.lower()](args=args, placedb=placedb)
    runner = ALGO_REGISTRY[args.algorithm.lower()](args=args, placer=placer, logger=logger)
    runner.run()
    logging.info("Exit single run")



if __name__ == "__main__":
    args = process_args()
    os.environ["CUDA_VISIBLE_DEVICES"] = str(args.gpu)

    from placedb import PlaceDB
    from placer import REGISTRY as PLACER_REGISTRY
    from algorithm import REGISTRY as ALGO_REGISTRY
    
    cpus = min(args.n_cpu_max, cpus)

    ray.init(num_cpus=cpus,
            num_gpus=1,
            include_dashboard=False,
            logging_level=logging.ERROR,
            _temp_dir=os.path.expanduser('~/tmp'),
            ignore_reinit_error=True)


    if args.run_mode == "single":
        single_run(args)
    elif args.run_mode == "result":
        sheet_path = os.path.join(ROOT_DIR, "sheets")
        os.makedirs(sheet_path, exist_ok=True)
        result_path = os.path.join(ROOT_DIR, f"results/{args.benchmark}/{args.name}/{args.placer}/{args.algorithm}")
        res2sheet(args=args, sheet_path=sheet_path, res_path=result_path)
    else:
        raise NotImplementedError
    logging.info("Exit Main")
