import os
import shutil
import traceback

from flask import Blueprint
from flask import make_response, send_file
from flask import render_template
from flask import request, jsonify
from werkzeug.utils import secure_filename

from .server.data_container import all_data
from .server.server_config import _get_config_names
from .server.table_utils import prepare_data, prepare_incremental_data
from .server.table_utils import save_all_data
from .server.utils import check_uuid
from .server.utils import replace_nan_inf

table_page = Blueprint("table_page", __name__, template_folder='templates')

first_time_access = True
FILE_NAME = 'file.log'


@table_page.route('/table/table')
def get_table():
    global first_time_access
    if not first_time_access:
        app_root_dir = all_data['root_log_dir']
        log_config_name = all_data['log_config_name']
        log_agent = all_data['log_agent']
        all_data.update(prepare_data(log_agent, app_root_dir, log_config_name, all_data))

    first_time_access = False
    data = all_data['data'].copy()  # all_data['data'] 只包含从硬盘的读取的log的信息
    # replace_with_extra_data(data, all_data['extra_data'], all_data['filter_condition'], all_data['deleted_rows'],
    #                          all_data['settings']['Ignore_filter_condition_not_exist_log'])
    replace_nan_inf(data, all_data['basic_settings']['round_to'])

    return jsonify(column_order=all_data['column_order'], column_dict=all_data['column_dict'],
                   hidden_columns=all_data['hidden_columns'],
                   data=data,
                   settings={key.replace('_', ' '): value for key, value in all_data['settings'].items()},
                   uuid=all_data['uuid'],
                   hidden_rows=list(all_data['hidden_rows'].keys()),
                   unchanged_columns=all_data['unchanged_columns'],
                   log_config_name=all_data['log_config_name'],
                   max_compare_metrics=all_data['config']['multi_chart_settings']['max_compare_metrics'])


@table_page.route('/table/refresh', methods=['POST'])
def refresh_table():
    res = check_uuid(all_data['uuid'], request.json['uuid'])
    if res is not None:
        return jsonify(res)
    log_agent = all_data['log_agent']
    try:
        new_logs = log_agent.read_logs(all_data['deleted_rows'])
        # 删除不满足条件的
        if len(new_logs) == 0:
            return jsonify(status='success', msg='Update successfully, no update found.', new_logs=[], updated_logs=[])
        else:
            new_logs, updated_logs = prepare_incremental_data(all_data['data'], new_logs, all_data['field_columns'],
                                                              all_data['filter_condition'],
                                                              all_data['settings']['Ignore_filter_condition_not_exist_log'])
            if len(new_logs) == 0 and len(updated_logs) == 0:
                return jsonify(status='success', msg='Update successfully, no update found.', new_logs=[], updated_logs=[])
            replace_nan_inf(new_logs, all_data['basic_settings']['round_to'])
            replace_nan_inf(updated_logs, all_data['basic_settings']['round_to'])
            return jsonify(status='success',
                           msg=f'Update successfully, {len(updated_logs)} log have updates, {len(new_logs)} newly added.',
                           new_logs=new_logs, updated_logs=updated_logs)
    except:
        return jsonify(status='fail', msg="Unknown error from server.")


@table_page.route('/table/delete_records', methods=['POST'])
def delete_records():
    res = check_uuid(all_data['uuid'], request.json['uuid'])
    if res is not None:
        return jsonify(res)
    ids = request.json['ids']
    for id in ids:
        if id in all_data['data']:
            all_data['deleted_rows'][id] = 1

    return jsonify(status='success', msg='')


@table_page.route('/table/erase_records', methods=['POST'])
def erase_records():
    res = check_uuid(all_data['uuid'], request.json['uuid'])
    if res is not None:
        return jsonify(res)
    ids = request.json['ids']
    fail_ids = []
    for index in range(len(ids) - 1, -1, -1):
        id = ids[index]
        if id in all_data['data']:
            all_data['data'].pop(id, None)
            all_data['deleted_rows'].pop(id, None)
            # 删除
            try:
                record_path = os.path.join(all_data['root_log_dir'], id)
                if os.path.isdir(record_path):
                    shutil.rmtree(record_path)
            except Exception as e:
                fail_ids.append(id)
        if id in all_data['extra_data']:
            all_data['extra_data'].pop(id)

    if len(fail_ids) != 0:
        return jsonify(status='fail', msg=fail_ids)
    else:
        return jsonify(status='success', msg='')


@table_page.route('/table/edit', methods=['POST'])
def table_edit():
    try:
        # 包含field, id, new_field_value三个值. 修改某个值
        res = check_uuid(all_data['uuid'], request.json['uuid'])
        if res is not None:
            return jsonify(res)
        id = request.json['id']
        field = request.json['field']
        new_field_value = request.json['new_field_value']
        if id in all_data['extra_data']:
            all_data['extra_data'][id][field] = new_field_value
        else:
            all_data['extra_data'][id] = {field: new_field_value}
        if id in all_data['data'] and field in all_data['data'][id]:
            all_data['data'][id][field] = new_field_value

        return jsonify(status='success', msg='')
    except Exception as e:
        print(e)
        import traceback
        traceback.print_exc()
        return jsonify(status='fail', msg='Unknown error fail to save edit results.')


@table_page.route('/table/settings', methods=['POST'])
def settings():
    res = check_uuid(all_data['uuid'], request.json['uuid'])
    if res is not None:
        return jsonify(res)
    settings = request.json['settings']
    all_data['settings'].update({key.replace(' ', '_'): value for key, value in settings.items()})

    return jsonify(status='success', msg='')


@table_page.route('/table/save_config_name', methods=['POST'])
def save_config_name():
    res = check_uuid(all_data['uuid'], request.json['uuid'])
    if res is not None:
        return jsonify(res)
    log_config_name = request.json['save_config_name'].replace(' ', '_')
    log_config_name = secure_filename(log_config_name)
    if len(log_config_name) != 0:
        if all_data['log_config_name'] != log_config_name:
            all_data['log_config_name'] = log_config_name
        log_dir = all_data['root_log_dir']
        save_all_data(all_data, log_dir, log_config_name)
        return jsonify(status='success', msg=log_config_name)
    else:
        return jsonify(status='fail', msg='Invalid file name')


@table_page.route('/table/hidden_rows', methods=['POST'])
def hidden_ids():
    res = check_uuid(all_data['uuid'], request.json['uuid'])
    if res is not None:
        return jsonify(res)
    ids = request.json['ids']
    all_data['hidden_rows'].clear()
    for id in ids:
        all_data['hidden_rows'][id] = 1

    return jsonify(status='success', msg='')


@table_page.route('/table/hidden_columns', methods=['POST'])
def hidden_columns():
    res = check_uuid(all_data['uuid'], request.json['uuid'])
    if res is not None:
        return jsonify(res)
    hidden_columns = request.json['hidden_columns']
    all_data['hidden_columns'] = hidden_columns
    return jsonify(status='success', msg='')


@table_page.route('/table/column_order', methods=['POST'])
def column_order():
    res = check_uuid(all_data['uuid'], request.json['uuid'])
    if res is not None:
        return jsonify(res)
    column_order = request.json['column_order']
    all_data['column_order'] = column_order
    return jsonify(status='success', msg='')


@table_page.route('/table/row', methods=['POST'])
def add_row():
    res = check_uuid(all_data['uuid'], request.json['uuid'])
    if res is not None:
        return jsonify(res)
    new_row = request.json['row']
    if new_row['id'] not in all_data['data']:
        all_data['data'][new_row['id']] = new_row
        all_data['extra_data'][new_row['id']] = new_row
        return jsonify(status='success', msg='')
    else:
        return jsonify(status='fail', msg='Duplicated id.')


@table_page.route('/table/save_settings', methods=['POST'])
def save():
    res = check_uuid(all_data['uuid'], request.json['uuid'])
    if res is not None:
        return jsonify(res)
    if 'condition' in request.json:
        condition = request.json['condition']  # {'key': 'value'}
        all_data['filter_condition'].update(condition)
    app_root_dir = all_data['root_log_dir']
    log_config_name = all_data['log_config_name']
    save_all_data(all_data, app_root_dir, log_config_name, force_save=True)
    return jsonify(status='success', msg='')


@table_page.route('/table')
def table():
    if all_data['token'] is not None:
        return jsonify(msg='This url needs a token to access. If you did not specify a token when start the server, '
                           'you may access the wrong ip or port. If you specify a token when start the server, '
                           'use http://{your server ip}:{port}/table/{token} to access.')
    return render_template('table.html')


@table_page.route('/table/<token>')
def table_with_token(token):
    if token != all_data['token'] or all_data['token'] is None:
        return jsonify(msg='Wrong token or you are not using a token')
    return render_template('table.html')


@table_page.route('/table/configs', methods=['POST'])
def table_configs():
    # 显示有哪些config文件的接口
    res = check_uuid(all_data['uuid'], request.json['uuid'])
    if res is not None:
        return jsonify(res)
    # 读取当前可以使用的config有哪些
    config_names = _get_config_names(all_data['root_log_dir'])
    if len(config_names) == 0:
        return jsonify(status='fail', msg='There is no valid config file in {}.'.format(all_data['root_log_dir']))
    if config_names.index(all_data['log_config_name']) == -1:
        return jsonify(status='fail', msg='There current config file:{} is not exist.'.format(all_data['log_config_name']))
    configs = {}
    for config in config_names:
        configs[config] = config == all_data['log_config_name']
    return jsonify(status='success', msg='', configs=configs)


@table_page.route('/table/change_config', methods=['POST'])
def table_change_config():
    # 改变config的接口
    res = check_uuid(all_data['uuid'], request.json['uuid'])
    if res is not None:
        return jsonify(res)
    if 'config_name' in request.json:
        config_names = _get_config_names(all_data['root_log_dir'])
        if config_names.index(request.json['config_name']) != -1:
            log_dir = all_data['root_log_dir']
            log_config_name = all_data['log_config_name']
            save_all_data(all_data, log_dir, log_config_name)
            all_data['log_config_name'] = request.json['config_name']
            return jsonify(status='success', msg='')
        else:
            return jsonify(status='fail', msg='There is no config named:{}.'.format(request.json['config_name']))
    else:
        return jsonify(status='fail', msg='There is no config_name in your request.')


@table_page.route('/table/delete_config', methods=['POST'])
def table_delete_config():
    res = check_uuid(all_data['uuid'], request.json['uuid'])
    if res is not None:
        return jsonify(res)
    if 'config_name' in request.json:
        if request.json['config_name'] != all_data['log_config_name']:
            log_dir = all_data['root_log_dir']
            path = os.path.join(log_dir, request.json['config_name'])
            if os.path.isfile(path):
                try:
                    os.remove(path)
                except Exception as e:
                    print("Error occurs when delete config:{}.".format(request.json['config_name']))
                    traceback.print_exc()
                    return jsonify(status='fail', msg='Error happens when delete.')
                return jsonify(status='success')
            else:
                jsonify(status='success')
        else:
            return jsonify(status='fail', msg='Cannot delete config being used.')
    else:
        return jsonify(status='fail', msg='There is no config_name in your request.')


@table_page.route('/table/is_file_exist', methods=['POST'])
def is_file_exist():
    res = check_uuid(all_data['uuid'], request.json['uuid'])
    if res is not None:
        return jsonify(res)
    if 'id' in request.json:
        try:
            id = request.json['id']
            log_dir = all_data['root_log_dir']
            path = os.path.join(log_dir, id, FILE_NAME)
            if not os.path.exists(path):
                return jsonify(status='fail', msg='There is no record file for {}.'.format(id))
            else:
                return jsonify(status='success', filename=FILE_NAME)
        except BaseException as e:
            traceback.print_exc()
            return jsonify(status='fail', msg='Following exception occurs:{}'.format(e))
    else:
        return jsonify(status='fail', msg='There is no id in your request.')


@table_page.route('/table/get_file', methods=['POST'])
def get_file():
    res = check_uuid(all_data['uuid'], request.values['uuid'])
    if res is not None:
        return jsonify(res)
    if 'id' in request.values and 'filename' in request.values:
        id = request.values['id']
        log_dir = all_data['root_log_dir']
        filename = request.values['filename']
        resp = make_response(send_file(os.path.join(log_dir, id, filename)))
        resp.headers["Content-type"] = "text/plan;charset=UTF-8"
        return resp
    else:
        return jsonify(status='fail', msg="The request lacks id or filename.")
