import sqlite3

import click
from flask import current_app, g
from flask.cli import with_appcontext

from tlapbot.redeems import milestone_complete


def get_db():
    if 'db' not in g:
        g.db = sqlite3.connect(
            current_app.config['DATABASE'],
            detect_types=sqlite3.PARSE_DECLTYPES
        )
        g.db.row_factory = sqlite3.Row

    return g.db


def close_db(e=None):
    db = g.pop('db', None)

    if db is not None:
        db.close()


def insert_counters(db):
    for redeem, redeem_info in current_app.config['REDEEMS'].items():
        if redeem_info["type"] == "counter":
            try:
                db.execute(
                    "INSERT INTO counters(name, count) VALUES(?, 0)",
                    (redeem,)
                )
                db.commit()
            except sqlite3.Error as e:
                print("Failed inserting counters to db:", e.args[0])
                return False
    return True


def init_db():
    db = get_db()

    with current_app.open_resource('schema.sql') as f:
        db.executescript(f.read().decode('utf8'))

    if insert_counters(db):
        return True


def clear_redeem_queue():
    db = get_db()

    try:
        cursor = db.execute(
            "DELETE FROM redeem_queue"
        )
        cursor.execute(
            """UPDATE counters SET count = 0"""
        )
        db.commit()
    except sqlite3.Error as e:
        print("Error occurred deleting redeem queue:", e.args[0])
        return False
    return True


def refresh_counters():
    db = get_db()

    try:
        db.execute("DELETE FROM counters")
        db.commit()
    except sqlite3.Error as e:
        print("Error occurred deleting old counters:", e.args[0])
        return False
    if insert_counters(db):
        return True


def refresh_milestones():
    db = get_db()
    # delete old milestones
    try:
        cursor = db.execute("SELECT name FROM milestones")
        milestones = cursor.fetchall()
        to_delete = []
        for milestone in milestones:
            milestone = milestone[0]
            if milestone not in current_app.config['REDEEMS'].keys():
                to_delete.append(milestone)
            elif current_app.config['REDEEMS'][milestone]['type'] != "milestone":
                to_delete.append(milestone)
        for milestone in to_delete:
            cursor.execute("DELETE FROM milestones WHERE name = ?", (milestone,))
        db.commit()
    except sqlite3.Error as e:
        print("Failed deleting old milestones from db:", e.args[0])
        return False

    # add new milestones
    try:
        for redeem, redeem_info in current_app.config['REDEEMS'].items():
            if redeem_info["type"] == "milestone":
                cursor = db.execute(
                    "SELECT goal FROM milestones WHERE name = ?",
                    (redeem,)
                )
                result = cursor.fetchone()
                if result is None:
                    cursor.execute(
                        "INSERT INTO milestones(name, progress, goal, complete) VALUES(?, 0, ?, FALSE)",
                        (redeem, redeem_info['goal'])
                    )
                # update existing milestone to new goal
                elif result != redeem_info["goal"]:
                    cursor.execute(
                        "UPDATE milestones SET goal = ? WHERE name = ?",
                        (redeem_info["goal"], redeem)
                    )
        db.commit()
    except sqlite3.Error as e:
        print("Failed inserting milestones to db:", e.args[0])
        return False
    return True


def reset_milestone(milestone):
    if milestone not in current_app.config['REDEEMS']:
        print(f"Failed resetting milestone, {milestone} not in redeems file.")
        return False
    try:
        db = get_db()
        db.execute(
            "DELETE FROM milestones WHERE name = ?",
            (milestone,)
        )
        db.execute(
            "INSERT INTO milestones(name, progress, goal, complete) VALUES(?, ?, ?, FALSE)",
            (milestone, 0, current_app.config['REDEEMS'][milestone]['goal'])
        )
        db.commit()
        return True
    except sqlite3.Error as e:
        current_app.logger.error(f"Error occurred adding a milestone: {e.args[0]}")
        return False


@click.command('init-db')
@with_appcontext
def init_db_command():
    """Clear the existing data and create new tables."""
    if init_db():
        click.echo('Initialized the database.')


@click.command('clear-queue')
@with_appcontext
def clear_queue_command():
    """Remove all redeems from the redeem queue."""
    if clear_redeem_queue():
        click.echo('Cleared redeem queue.')


@click.command('refresh-counters')
@with_appcontext
def refresh_counters_command():
    """Refresh counters from current config file.
    (Remove old ones, add new ones.)"""
    if refresh_counters():
        click.echo('Counters refreshed.')


@click.command('clear-refresh')
@with_appcontext
def refresh_and_clear_command():
    """Refresh counters and clear queue."""
    if refresh_counters() and clear_redeem_queue():
        click.echo('Counters refreshed and queue cleared.')


@click.command('refresh-milestones')
@with_appcontext
def refresh_milestones_command():
    """Initialize all milestones from the redeems file,
    delete milestones not in redeem file."""
    if refresh_milestones():
        click.echo('Refreshed milestones.')


@click.command('reset-milestone')
@click.argument('milestone')
def reset_milestone_command(milestone):
    """Resets a completed milestone back to zero."""
    if milestone_complete(get_db(), milestone):
        if reset_milestone(milestone):
            click.echo(f"Reset milestone {milestone}.")
    else:
        click.echo(f"Could not reset milestone {milestone}, milestone not completed.")
        click.echo("(You can hard-reset-milestone if you really want to reset it.)")


@click.command('hard-reset-milestone')
@click.argument('milestone')
def hard_reset_milestone_command(milestone):
    """Resets any milestone back to zero."""
    if reset_milestone(milestone):
        click.echo(f"Hard reset milestone {milestone}.")


def init_app(app):
    app.teardown_appcontext(close_db)
    app.cli.add_command(init_db_command)