import glob
import json
import numbers
import os

from .utils import colored_string
from .log_config_parser import ConfigParser


def read_server_config(config_path):
    """
    给定config的path，读取里面的config。如果config不存在，则按照默认的值创建
    :param config_path: str
    :return: dict, config的内容
    """
    config = ConfigParser(allow_no_value=True)
    if not os.path.exists(config_path):
        config_dir = os.path.dirname(config_path)
        if not os.path.isdir(config_dir):
            os.makedirs(config_dir)
        _read_default_config(config)
        with open(config_path, 'w', encoding='utf-8') as f:
            config.write(f)
    else:
        config.read(config_path, encoding='utf-8')
        # check configparser, 没有的话用默认值覆盖填写
        check_config(config)

    configs = {}
    # 读取settings
    settings = {}
    get_dict_from_config(config, 'frontend_settings', settings, 'bool')
    settings = {key.capitalize(): value for key, value in settings.items()}
    configs['settings'] = settings

    # 读取display_settings
    basic_settings = {}
    get_dict_from_config(config, 'basic_settings', basic_settings)
    for key in ['str_max_length', 'round_to']:
        basic_settings[key] = int(basic_settings[key])
    for key in ['ignore_unchanged_columns']:
        basic_settings[key] = basic_settings[key] == 'True'
    configs['basic_settings'] = basic_settings

    # 读取data_settings
    hidden_logs = read_list_from_config(config, 'data_settings', 'hidden_logs', ',')
    configs['hidden_rows'] = {log: 1 for log in hidden_logs}
    deleted_rows = read_list_from_config(config, 'data_settings', 'deleted_logs', ',')
    configs['deleted_rows'] = {log: 1 for log in deleted_rows}
    remote_log_servers = read_list_from_config(config, "data_settings", "remote_log_servers", ",")
    configs['remote_log_servers'] = list(remote_log_servers)
    if len(config.get('data_settings', 'filter_condition')) != 0:
        configs['filter_condition'] = json.loads(config.get('data_settings', 'filter_condition').replace("'", '"'))
        for key in list(configs['filter_condition'].keys()):
            delete = False
            if not isinstance(configs['filter_condition'][key], (str, list, numbers.Number)):
                print(colored_string("Unsupported type found in filter_condition in `{}`.".format(key), 'red'))
                delete = True
            if isinstance(configs['filter_condition'][key], list):
                for value in configs['filter_condition'][key]:
                    if not isinstance(value, (str, numbers.Number)):
                        print(colored_string("Unsupported type found in filter_condition in `{}`.".format(key), 'red'))
                        delete = True
            if delete:
                configs['filter_condition'].pop(key)
    else:
        configs['filter_condition'] = {}

    # 读取column_settings
    hidden_columns = read_list_from_config(config, 'column_settings', 'hidden_columns', ',')
    configs['hidden_columns'] = {column: 1 for column in hidden_columns}

    # 读取exclude columns
    exclude_columns = read_list_from_config(config, 'column_settings', 'exclude_columns', ',')
    configs['exclude_columns'] = {column: 1 for column in exclude_columns}

    # 读取editable_columns
    editable_columns = read_list_from_config(config, 'column_settings', 'editable_columns', ',')
    configs['editable_columns'] = {column: 1 for column in editable_columns}

    # 读取column_order
    column_order = config.get('column_settings', 'column_order')
    if column_order != '':
        column_order = json.loads(column_order)
    else:
        column_order = {}
    configs['column_order'] = column_order

    # 保存config对象, 因为需要保留下注释
    configs['config'] = config

    _dict = {}
    chart_exclude_columns = read_list_from_config(config, 'chart_settings', 'chart_exclude_columns', ',')
    _dict['chart_exclude_columns'] = {column: 1 for column in chart_exclude_columns}
    _dict['max_points'] = config.getint('chart_settings', 'max_points')
    _dict['update_every'] = config.getint('chart_settings', 'update_every')
    _dict['max_no_updates'] = config.getint('chart_settings', 'max_no_updates')
    configs['chart_settings'] = _dict

    _dict = {}
    if 'multi_chart_settings' in config:
        _dict['max_compare_metrics'] = config.getint('multi_chart_settings', 'max_compare_metrics')
    else:
        _dict['max_compare_metrics'] = 10
        config.add_section('multi_chart_settings')
        config.set('multi_chart_settings', 'max_compare_metrics', 10)
    configs['multi_chart_settings'] = _dict

    return configs


def save_config(all_data, config_path):
    config = all_data['config']
    if not os.path.exists(config_path):
        config_dir = os.path.dirname(config_path)
        if not os.path.isdir(config_dir):
            os.makedirs(config_dir)

    # frontend_settings
    settings = all_data['settings']
    settings = {key.replace(' ', '_'): value for key, value in settings.items()}
    save_dict_to_config(config, 'frontend_settings', settings)

    # basic_settings
    basic_settings = all_data['basic_settings']
    save_dict_to_config(config, 'basic_settings', basic_settings)

    # data settings
    hidden_logs = all_data['hidden_rows'].keys()
    config.set('data_settings', 'hidden_logs', ','.join(hidden_logs))
    deleted_logs = all_data['deleted_rows'].keys()
    config.set('data_settings', 'deleted_logs', ','.join(deleted_logs))
    if len(all_data['filter_condition']) != 0:
        filter_condition = json.dumps(all_data['filter_condition'])
    else:
        filter_condition = ''
    config.set('data_settings', 'filter_condition', filter_condition)

    # column_settings
    for option in ['hidden_columns', 'editable_columns']:
        save_list_to_config(config, 'column_settings', option, all_data[option])
    column_order = refine_column_order(all_data['column_order'])
    column_order = json.dumps(column_order)
    config.set('column_settings', 'column_order', column_order)

    with open(config_path, 'w', encoding='utf-8') as f:
        config.write(f)


def refine_column_order(column_order):
    # 重新生成column_order
    new_column_order = {}
    if 'OrderKeys' in column_order:  # 还有下一级
        keys = column_order['OrderKeys']
        for key in keys:
            value = column_order[key]
            child = refine_column_order(value)
            if len(child) == 0:
                new_column_order[key] = 'EndOfOrder'
            else:
                new_column_order[key] = child
    return new_column_order


def save_list_to_config(config, section, option, container, sep=','):
    if not config.has_section(section):
        config.add_section(section)
    if len(container) == 0:
        str_ = ''
    else:
        str_ = sep.join(container.keys())
    config.set(section, option, str_)


def save_dict_to_config(config, section, container):
    if not config.has_section(section):
        config.add_section(section)
    for key, value in container.items():
        config.set(section, key, str(value))


def read_list_from_config(config, section, option, sep):
    str_ = config.get(section, option)
    items = []
    if str_ != '':
        items = str_.split(sep)
        items = [item.strip() for item in items]
    return items


def _read_default_config(config):
    """

    :param config: ConfigParser对象
    :return:
    """
    default_cfg_fp = os.path.join(os.path.realpath(__file__)[:-len("server_config.py")], '..',
                                  '..', 'fastgit', 'normal', 'logs', 'default.cfg')
    config.read(default_cfg_fp, encoding='utf-8')
    return config


def check_config(config):
    # 检查config是否拥有所有的值，如果没有的话，使用默认值填写
    default_config = ConfigParser()
    default_config = _read_default_config(default_config)
    for section in default_config.sections():
        if not config.has_section(section):
            # raise KeyError("Section:`{}` is not in config.".format(section))
            config.add_section(section)
            for option, value in default_config.items(section):
                config.set(section, option, value)
        for opt in default_config.options(section):
            if not config.has_option(section, opt):
                # raise KeyError("Option:`{}` is not in Section:`{}` is not in config.".format(opt, section))
                config.set(section, opt, default_config.get(section, opt))


def get_dict_from_config(config, section, container, dtype=None):
    if dtype == 'int':
        func = config.getint
    elif dtype == 'bool' or dtype == 'boolean':
        func = config.getboolean
    else:
        func = config.get
    for option in config.options(section):
        container[option] = func(section, option)


def read_extra_data(path):
    with open(path, 'r', encoding='utf-8') as f:
        data = json.load(f)
    return data


def save_extra_data(path, data):
    with open(path, 'w', encoding='utf-8') as f:
        json.dump(data, f)


def _get_config_names(app_root_dir):
    """
    给定log的路径, 返回下面所有结尾为.cfg的文件

    :param str, app_root_dir:
    :return: list(dir)
    """
    configs = glob.glob(os.path.join(app_root_dir, '*.cfg'))
    configs = [os.path.basename(config) for config in configs]
    return configs
