import json
import logging

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

import lbry_comment_server.writes as writes
from lbry_comment_server.database import get_claim_comments
from lbry_comment_server.database import get_comments_by_id, get_comment_ids
from lbry_comment_server.database import obtain_connection

logger = logging.getLogger(__name__)

ERRORS = {
    'INVALID_PARAMS': {'code': -32602, 'message': 'Invalid parameters'},
    'INTERNAL': {'code': -32603, 'message': 'An internal error'},
    'UNKNOWN': {'code': -1, 'message': 'An unknown or very miscellaneous error'},
}


def ping(*args):
    return 'pong'


def handle_get_comment_ids(app, **kwargs):
    with obtain_connection(app['db_path']) as conn:
        return get_comment_ids(conn, **kwargs)


def handle_get_claim_comments(app, **kwargs):
    with obtain_connection(app['db_path']) as conn:
        return get_claim_comments(conn, **kwargs)


def handle_get_comments_by_id(app, **kwargs):
    with obtain_connection(app['db_path']) as conn:
        return get_comments_by_id(conn, **kwargs)


async def handle_create_comment(scheduler, **kwargs):
    job = await scheduler.spawn(writes.write_comment(**kwargs))
    return await job.wait()


METHODS = {
    'ping': ping,
    'get_claim_comments': handle_get_claim_comments,
    'get_comment_ids': handle_get_comment_ids,
    'get_comments_by_id': handle_get_comments_by_id,
    'create_comment': handle_create_comment
}


async def process_json(app, body: dict) -> dict:
    response = {'jsonrpc': '2.0', 'id': body['id']}
    if body['method'] in METHODS:
        method = body['method']
        params = body.get('params', {})
        try:
            if asyncio.iscoroutinefunction(METHODS[method]):
                result = await METHODS[method](app['comment_scheduler'], **params)
            else:
                result = METHODS[method](app, **params)
            response['result'] = result
        except TypeError as te:
            logger.exception('Got TypeError: %s', te)
            response['error'] = ERRORS['INVALID_PARAMS']
    else:
        response['error'] = ERRORS['UNKNOWN']
    return response


@atomic
async def api_endpoint(request: web.Request):
    try:
        body = await request.json()
        logger.info('Received POST request from %s', request.remote)
        if type(body) is list or type(body) is dict:
            if type(body) is list:
                return web.json_response(
                    [await process_json(request.app, part) for part in body]
                )
            else:
                return web.json_response(await process_json(request.app, body))
        else:
            return web.json_response({'error': ERRORS['UNKNOWN']})
    except json.decoder.JSONDecodeError as jde:
        logger.exception('Received malformed JSON from %s: %s', request.remote, jde.msg)
        logger.debug('Request headers: %s', request.headers)
        return web.json_response({
            'error': {'message': jde.msg, 'code': -1}
        })