2019-05-26 00:42:39 -04:00
|
|
|
# cython: language_level=3
|
2019-05-20 06:49:08 -04:00
|
|
|
import json
|
2019-05-21 07:54:52 -04:00
|
|
|
import logging
|
|
|
|
|
2019-05-26 00:42:39 -04:00
|
|
|
import aiojobs
|
2019-06-04 08:14:12 -04:00
|
|
|
import asyncio
|
2019-05-20 06:49:08 -04:00
|
|
|
from aiohttp import web
|
2019-05-21 07:54:52 -04:00
|
|
|
from aiojobs.aiohttp import atomic
|
2019-06-04 08:14:12 -04:00
|
|
|
from asyncio import coroutine
|
2019-05-21 07:54:52 -04:00
|
|
|
|
2019-06-06 04:43:47 -05:00
|
|
|
from src.database import DatabaseWriter
|
2019-05-21 15:13:08 -04:00
|
|
|
from src.database import get_claim_comments
|
|
|
|
from src.database import get_comments_by_id, get_comment_ids
|
|
|
|
from src.database import obtain_connection
|
2019-07-16 20:20:33 -04:00
|
|
|
from src.writes import create_comment_or_error
|
2019-05-21 06:56:27 -04:00
|
|
|
|
|
|
|
logger = logging.getLogger(__name__)
|
|
|
|
|
2019-05-20 06:49:08 -04:00
|
|
|
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'},
|
|
|
|
}
|
|
|
|
|
2019-07-16 20:20:33 -04:00
|
|
|
ID_LIST = {'claim_id', 'parent_id', 'comment_id', 'channel_id'}
|
2019-05-20 06:49:08 -04:00
|
|
|
|
2019-07-16 20:20:33 -04:00
|
|
|
|
|
|
|
def ping(*args, **kwargs):
|
2019-05-20 06:49:08 -04:00
|
|
|
return 'pong'
|
|
|
|
|
|
|
|
|
2019-07-16 20:20:33 -04:00
|
|
|
def handle_get_comment_ids(app, kwargs):
|
2019-05-21 05:00:01 -04:00
|
|
|
with obtain_connection(app['db_path']) as conn:
|
|
|
|
return get_comment_ids(conn, **kwargs)
|
2019-05-20 06:49:08 -04:00
|
|
|
|
|
|
|
|
2019-07-16 20:20:33 -04:00
|
|
|
def handle_get_claim_comments(app, kwargs):
|
2019-05-21 05:00:01 -04:00
|
|
|
with obtain_connection(app['db_path']) as conn:
|
|
|
|
return get_claim_comments(conn, **kwargs)
|
2019-05-20 06:49:08 -04:00
|
|
|
|
|
|
|
|
2019-07-16 20:20:33 -04:00
|
|
|
def handle_get_comments_by_id(app, kwargs):
|
2019-05-21 05:00:01 -04:00
|
|
|
with obtain_connection(app['db_path']) as conn:
|
|
|
|
return get_comments_by_id(conn, **kwargs)
|
2019-05-20 06:49:08 -04:00
|
|
|
|
|
|
|
|
2019-05-26 00:42:39 -04:00
|
|
|
async def create_comment_scheduler():
|
|
|
|
return await aiojobs.create_scheduler(limit=1, pending_limit=0)
|
|
|
|
|
|
|
|
|
2019-07-16 20:20:33 -04:00
|
|
|
async def write_comment(comment):
|
2019-05-26 00:42:39 -04:00
|
|
|
with DatabaseWriter._writer.connection as conn:
|
2019-07-16 20:20:33 -04:00
|
|
|
return await coroutine(create_comment_or_error)(conn, **comment)
|
2019-05-26 00:42:39 -04:00
|
|
|
|
|
|
|
|
2019-07-16 20:20:33 -04:00
|
|
|
async def handle_create_comment(scheduler, comment):
|
|
|
|
job = await scheduler.spawn(write_comment(comment))
|
2019-05-21 05:00:01 -04:00
|
|
|
return await job.wait()
|
2019-05-20 06:49:08 -04:00
|
|
|
|
|
|
|
|
|
|
|
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
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2019-07-16 20:20:33 -04:00
|
|
|
def clean_input_params(kwargs: dict):
|
|
|
|
for k, v in kwargs.items():
|
|
|
|
if type(v) is str:
|
|
|
|
kwargs[k] = v.strip()
|
|
|
|
if k in ID_LIST:
|
|
|
|
kwargs[k] = v.lower()
|
|
|
|
|
|
|
|
|
2019-05-21 05:00:01 -04:00
|
|
|
async def process_json(app, body: dict) -> dict:
|
2019-05-20 06:49:08 -04:00
|
|
|
response = {'jsonrpc': '2.0', 'id': body['id']}
|
|
|
|
if body['method'] in METHODS:
|
|
|
|
method = body['method']
|
|
|
|
params = body.get('params', {})
|
2019-07-16 20:20:33 -04:00
|
|
|
clean_input_params(params)
|
2019-05-20 06:49:08 -04:00
|
|
|
try:
|
2019-05-21 05:00:01 -04:00
|
|
|
if asyncio.iscoroutinefunction(METHODS[method]):
|
2019-07-16 20:20:33 -04:00
|
|
|
result = await METHODS[method](app['comment_scheduler'], params)
|
2019-05-20 06:49:08 -04:00
|
|
|
else:
|
2019-07-16 20:20:33 -04:00
|
|
|
result = METHODS[method](app, params)
|
2019-05-20 06:49:08 -04:00
|
|
|
response['result'] = result
|
|
|
|
except TypeError as te:
|
2019-05-21 06:56:27 -04:00
|
|
|
logger.exception('Got TypeError: %s', te)
|
2019-05-20 06:49:08 -04:00
|
|
|
response['error'] = ERRORS['INVALID_PARAMS']
|
2019-06-04 08:14:12 -04:00
|
|
|
except ValueError as ve:
|
|
|
|
logger.exception('Got ValueError: %s', ve)
|
|
|
|
response['error'] = ERRORS['INVALID_PARAMS']
|
2019-05-20 06:49:08 -04:00
|
|
|
else:
|
|
|
|
response['error'] = ERRORS['UNKNOWN']
|
|
|
|
return response
|
|
|
|
|
|
|
|
|
2019-05-21 05:00:01 -04:00
|
|
|
@atomic
|
|
|
|
async def api_endpoint(request: web.Request):
|
2019-05-20 06:49:08 -04:00
|
|
|
try:
|
2019-05-21 06:56:27 -04:00
|
|
|
logger.info('Received POST request from %s', request.remote)
|
2019-06-04 08:14:12 -04:00
|
|
|
body = await request.json()
|
2019-05-20 06:49:08 -04:00
|
|
|
if type(body) is list or type(body) is dict:
|
|
|
|
if type(body) is list:
|
2019-05-21 05:00:01 -04:00
|
|
|
return web.json_response(
|
|
|
|
[await process_json(request.app, part) for part in body]
|
|
|
|
)
|
2019-05-20 06:49:08 -04:00
|
|
|
else:
|
2019-05-21 05:00:01 -04:00
|
|
|
return web.json_response(await process_json(request.app, body))
|
2019-05-20 06:49:08 -04:00
|
|
|
else:
|
|
|
|
return web.json_response({'error': ERRORS['UNKNOWN']})
|
|
|
|
except json.decoder.JSONDecodeError as jde:
|
2019-05-21 06:56:27 -04:00
|
|
|
logger.exception('Received malformed JSON from %s: %s', request.remote, jde.msg)
|
|
|
|
logger.debug('Request headers: %s', request.headers)
|
2019-05-20 06:49:08 -04:00
|
|
|
return web.json_response({
|
|
|
|
'error': {'message': jde.msg, 'code': -1}
|
|
|
|
})
|