import numpy as np
import os
import collections
from os.path import dirname, abspath
from copy import deepcopy
from sacred import Experiment, SETTINGS
from sacred.observers import FileStorageObserver
from sacred.utils import apply_backspaces_and_linefeeds
import sys
import torch as th
from utils.logging import get_logger
import yaml

# import run program
from run import run as run
from meta_train_run import run as meta_train_run
from meta_test_run import run as meta_test_run
from collect_run import run as collect_run

SETTINGS['CAPTURE_MODE'] = "fd" # set to "no" if you want to see stdout/stderr in console
logger = get_logger()

ex = Experiment("pymarl")
ex.logger = logger
ex.captured_out_filter = apply_backspaces_and_linefeeds

results_path = os.path.join(dirname(dirname(abspath(__file__))), "results")


@ex.main
def my_main(_run, _config, _log):
    # Setting the random seed throughout the modules
    config = config_copy(_config)
    np.random.seed(config["seed"])
    th.manual_seed(config["seed"])
    config['env_args']['seed'] = config["seed"]

    # run the framework
    if config.get("meta_train", False):
        meta_train_run(_run, config, _log)
    elif config.get("meta_test", False):
        meta_test_run(_run, config, _log)
    elif config.get("collect_run", False):
        collect_run(_run, config, _log)
    else:
        run(_run, config, _log)


def _get_config(params, arg_name, subfolder):
    config_name = None
    for _i, _v in enumerate(params):
        if _v.split("=")[0] == arg_name:
            config_name = _v.split("=")[1]
            del params[_i]
            break

    if config_name is not None:
        with open(os.path.join(os.path.dirname(__file__), "config", subfolder, "{}.yaml".format(config_name)), "r") as f:
            try:
                config_dict = yaml.load(f)
            except yaml.YAMLError as exc:
                assert False, "{}.yaml error: {}".format(config_name, exc)

        return config_dict


def _get_env_config(env_name):
    with open(os.path.join(os.path.dirname(__file__), "config", "envs", "{}.yaml".format(env_name)), "r") as f:
        try:
            config_dict = yaml.load(f)
        except yaml.YAMLError as exc:
            assert False, "{}.yaml error: {}".format(env_name, exc)
    return config_dict         
    

def recursive_dict_update(d, u):
    for k, v in u.items():
        if isinstance(v, collections.Mapping):
            d[k] = recursive_dict_update(d.get(k, {}), v)
        else:
            d[k] = v
    return d


def config_copy(config):
    if isinstance(config, dict):
        return {k: config_copy(v) for k, v in config.items()}
    elif isinstance(config, list):
        return [config_copy(v) for v in config]
    else:
        return deepcopy(config)


if __name__ == '__main__':
    params = deepcopy(sys.argv)

    # Get the defaults from default.yaml
    with open(os.path.join(os.path.dirname(__file__), "config", "default.yaml"), "r") as f:
        try:
            config_dict = yaml.load(f)
        except yaml.YAMLError as exc:
            assert False, "default.yaml error: {}".format(exc)

    # Load algorithm base configs
    alg_config = _get_config(params, "--config", "algs")
    # config_dict = {**config_dict, **alg_config}
    config_dict = recursive_dict_update(config_dict, alg_config)
    
    # check whether is meta_train
    meta_train = config_dict.get("meta_train", False)
    meta_test = config_dict.get("meta_test", False)
    if meta_train:
        # recursively load task config
        task_config = _get_config(params, "--task-config", "tasks")
        config_dict = recursive_dict_update(config_dict, task_config)
        # get env type and load env config
        env_name = config_dict["env"]
        env_config = _get_env_config(env_name)
        config_dict = recursive_dict_update(config_dict, env_config)
    elif meta_test:
        # recursively load task config
        task_config = _get_config(params, "--task-config", "tasks")
        config_dict = recursive_dict_update(config_dict, task_config)
        # get env type and load env config
        env_name = config_dict["env"]
        env_config = _get_env_config(env_name)
        config_dict = recursive_dict_update(config_dict, env_config)
        if env_name == "sc2":
            config_dict["env_args"]["map_name"] = config_dict["test_task"]
        else:
            config_dict["env_args"]["task_id"] = config_dict["test_task"]
    elif collect_run:
        # recursively load task config
        task_config = _get_config(params, "--task-config", "tasks")
        config_dict = recursive_dict_update(config_dict, task_config)
        # get env type and load env config
        env_name = config_dict["env"]
        env_config = _get_env_config(env_name)
        config_dict = recursive_dict_update(config_dict, env_config)
        if env_name == "sc2":
            config_dict["env_args"]["map_name"] = config_dict["test_task"]
        else:
            config_dict["env_args"]["task_id"] = config_dict["test_task"] 
    else:
        env_config = _get_config(params, "--env-config", "envs")
        config_dict = recursive_dict_update(config_dict, env_config)

    # get config from argv, such as "remark"
    def _get_argv_config(params):
        config = {}
        to_del = []
        for _i, _v in enumerate(params):
            item = _v.split("=")[0]
            if item[:2] == "--" and item not in ["envs", "algs"]:
                config_v = _v.split("=")[1]
                try:
                    config_v = eval(config_v)
                except:
                    pass
                config[item[2:]] = config_v
                to_del.append(_v)
        for _v in to_del:
            params.remove(_v)
        return config

    config_dict = recursive_dict_update(config_dict, _get_argv_config(params))
    
    # if set map_name, we overwrite it
    if "map_name" in config_dict:
        assert not meta_train and meta_test, "Unexpected scenario!!!"
        config_dict["env_args"]["map_name"] = config_dict["map_name"]

    if "task_id" in config_dict:
        assert not meta_train and (meta_test or collect_run), "Unexpected scenario!!!"
        config_dict["env_args"]["task_id"] = config_dict["task_id"]

    # now add all the config to sacred
    ex.add_config(config_dict)

    # Save to disk by default for sacred
    logger.info("Saving to FileStorageObserver in results/sacred.")
    file_obs_path = os.path.join(results_path, "sacred")
    ex.observers.append(FileStorageObserver.create(file_obs_path))

    ex.run_commandline(params)
