import sqlite3
import typing
import json

import sql_requests
import utils
from EDMCLogging import get_main_logger

logger = get_main_logger()
logger.propagate = False

db: sqlite3.Connection = sqlite3.connect('squads_stat.sqlite3', check_same_thread=False)

db.executescript(sql_requests.schema_create)  # schema creation

# thx https://stackoverflow.com/a/48789604
db.row_factory = lambda c, r: dict(zip([col[0] for col in c.description], r))


def get_activity_changes(platform: str, leaderboard_type: str, limit: int, low_timestamp, high_timestamp) -> list:
    cache_key: str = f'{platform}_{leaderboard_type}_{limit}_{low_timestamp}_{high_timestamp}'
    cached_result: typing.Union[str, None] = cache.get(cache_key)

    if cached_result is not None:
        logger.debug(f'Cached result for {cache_key}')
        return json.loads(cached_result)

    logger.debug(f'Not cached result for {cache_key}')

    sql_req: sqlite3.Cursor = db.execute(sql_requests.select_activity_pretty_names, {
        'LB_type': utils.LeaderboardTypes(leaderboard_type.lower()).value,
        'platform': utils.Platform(platform.upper()).value,
        'limit': limit,
        'high_timestamp': high_timestamp,
        'low_timestamp': low_timestamp
    })

    result: list = sql_req.fetchall()
    cache.set(cache_key, json.dumps(result))

    return result


def insert_leaderboard_db(leaderboard_list: dict) -> None:
    """
    Takes leaderboard as list, it platform, type, db connection and insert leaderboard to DB

    :param leaderboard_list: list from request_leaderboard
    :return:
    """

    platform: str = leaderboard_list['platform']
    LB_type: str = leaderboard_list['type']
    leaderboard: list = leaderboard_list['leaderboard']

    action_id: int  # not last, current that we will use

    sql_req_action_id: sqlite3.Cursor = db.execute(sql_requests.select_last_action_id)
    action_id_fetch_one: typing.Union[None, dict[str, int]] = sql_req_action_id.fetchone()
    if action_id_fetch_one is None:
        # i.e. first launch
        action_id = 1  # yep, not 0

    else:
        action_id = action_id_fetch_one['action_id'] + 1

    # Patch for additional values
    for squad in leaderboard:
        squad.update({'action_id': action_id, 'LB_type': LB_type, 'platform': platform})

    with db:
        db.executemany(
            sql_requests.insert_leader_board,
            leaderboard)

    cache.delete_all()  # drop cache


def get_diff_action_id(action_id: int) -> list:
    """
    Takes action_id and returns which squadrons has been changed in leaderboard as in action_id and
    experience they got in compassion to action_id - 1 for the same leaderboard and platform

    :param action_id:
    :return:
    """
    cache_key: str = f'{action_id}'
    cached_result: typing.Union[str, None] = cache.get(cache_key)

    if cached_result is not None:
        logger.debug(f'Cached result for {cache_key}')
        return json.loads(cached_result)

    logger.debug(f'Not cached result for {cache_key}')
    sql_req: sqlite3.Cursor = db.execute(sql_requests.select_diff_by_action_id, {'action_id': action_id})
    result: list = sql_req.fetchall()
    cache.set(cache_key, json.dumps(result))

    return result


class Cache:
    def __init__(self, db_conn: sqlite3.Connection, disabled: bool = False):
        self.disabled = disabled
        self.db: sqlite3.Connection = db_conn
        with self.db:
            self.db.execute("create table if not exists cache (key text unique, value text);")

    def set(self, key, value) -> None:
        if self.disabled:
            return

        with db:
            if self.db.execute('select count(key) as count from cache where key = ?;', [key]).fetchone()['count'] == 1:

                # key exists, just need to update value
                self.db.execute('update cache set value = ? where key = ?;', [value, key])
            else:

                # key doesn't exists, need to insert new row
                self.db.execute('insert into cache (key, value) values (?, ?);', [key, value])

    def get(self, key, default=None):
        if self.disabled:
            return

        res = self.db.execute('select value from cache where key = ?;', [key]).fetchone()
        if res is None:
            return default

        return res['value']

    def delete(self, key):
        self.db.execute('delete from cache where key = ?;', [key])

    def delete_all(self):
        logger.debug('Dropping cache')
        with self.db:
            self.db.execute('delete from cache;')


cache = Cache(db)