# cython: language_level=3
import logging
import pathlib
import signal
import time
import queue


import aiojobs
import aiojobs.aiohttp
import asyncio
import aiohttp
from aiohttp import web

from src.database.queries import setup_database, backup_database
from src.database.queries import obtain_connection, DatabaseWriter
from src.server.handles import api_endpoint, get_api_endpoint

logger = logging.getLogger(__name__)


async def setup_db_schema(app):
    if not pathlib.Path(app['db_path']).exists():
        logger.info(f'Setting up schema in {app["db_path"]}')
        setup_database(app['db_path'])
    else:
        logger.info(f'Database already exists in {app["db_path"]}, skipping setup')


async def database_backup_routine(app):
    try:
        while True:
            await asyncio.sleep(app['config']['backup_int'])
            with app['reader'] as conn:
                logger.debug('backing up database')
                backup_database(conn, app['backup'])
    except asyncio.CancelledError:
        pass


async def post_errors_to_slack(client, app):
    while app['errors'].qsize() > 0:
        msg = app['errors'].get_nowait()
        if 'slack_webhook' in app['config']:
            await client.post(app['config']['slack_webhook'], json=msg)


async def report_errors_to_slack_webhook(app):
    async with aiohttp.ClientSession() as client:
        try:
            while True:
                await asyncio.shield(post_errors_to_slack(client, app))
                await asyncio.sleep(10)

        except asyncio.CancelledError:
            await asyncio.shield(post_errors_to_slack(client, app))


async def start_background_tasks(app):
    app['reader'] = obtain_connection(app['db_path'], True)
    app['waitful_backup'] = asyncio.create_task(database_backup_routine(app))
    app['comment_scheduler'] = await aiojobs.create_scheduler(limit=1, pending_limit=0)
    app['db_writer'] = DatabaseWriter(app['db_path'])
    app['writer'] = app['db_writer'].connection

    app['errors'] = queue.Queue()
    app['slack_webhook'] = asyncio.create_task(report_errors_to_slack_webhook(app))


async def close_database_connections(app):
    logger.info('Ending background backup loop')
    app['waitful_backup'].cancel()
    await app['waitful_backup']
    app['reader'].close()
    app['writer'].close()
    app['db_writer'].cleanup()


async def stop_reporting_errors(app):
    app['slack_webhook'].cancel()
    await app['slack_webhook']


async def close_comment_scheduler(app):
    logger.info('Closing comment_scheduler')
    await app['comment_scheduler'].close()


class CommentDaemon:
    def __init__(self, config, db_file=None, backup=None, **kwargs):
        app = web.Application()
        app['config'] = config
        self.config = app['config']
        if db_file:
            app['db_path'] = db_file
            app['backup'] = backup
        else:
            app['db_path'] = config['path']['database']
            app['backup'] = backup or (app['db_path'] + '.backup')
        app.on_startup.append(setup_db_schema)
        app.on_startup.append(start_background_tasks)
        app.on_shutdown.append(close_comment_scheduler)
        app.on_cleanup.append(close_database_connections)
        app.on_cleanup.append(stop_reporting_errors)
        aiojobs.aiohttp.setup(app, **kwargs)
        app.add_routes([
            web.post('/api', api_endpoint),
            web.get('/', get_api_endpoint),
            web.get('/api', get_api_endpoint)
        ])
        self.app = app
        self.app_runner = None
        self.app_site = None

    async def start(self, host=None, port=None):
        self.app['start_time'] = time.time()
        self.app_runner = web.AppRunner(self.app)
        await self.app_runner.setup()
        self.app_site = web.TCPSite(
            runner=self.app_runner,
            host=host or self.config['host'],
            port=port or self.config['port'],
        )
        await self.app_site.start()
        logger.info(f'Comment Server is running on {self.config["host"]}:{self.config["port"]}')

    async def stop(self):
        await self.app_runner.shutdown()
        await self.app_runner.cleanup()


def run_app(config, db_file=None):
    comment_app = CommentDaemon(config=config, db_file=db_file, close_timeout=5.0)

    loop = asyncio.get_event_loop()

    def __exit():
        raise web.GracefulExit()

    loop.add_signal_handler(signal.SIGINT, __exit)
    loop.add_signal_handler(signal.SIGTERM, __exit)

    try:
        loop.run_until_complete(comment_app.start())
        loop.run_forever()
    except (web.GracefulExit, KeyboardInterrupt, asyncio.CancelledError, ValueError):
        logging.warning('Server going down, asyncio loop raised cancelled error:')
    finally:
        loop.run_until_complete(comment_app.stop())