|
@ -15,38 +15,45 @@ import time |
|
|
from lib.util import LoggedClass |
|
|
from lib.util import LoggedClass |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
def json_result_payload(result, id_): |
|
|
def json_response_payload(result, id_): |
|
|
# We should not respond to notifications |
|
|
# We should not respond to notifications |
|
|
assert id_ is not None |
|
|
assert id_ is not None |
|
|
return {'jsonrpc': '2.0', 'error': None, 'result': result, 'id': id_} |
|
|
return {'jsonrpc': '2.0', 'result': result, 'id': id_} |
|
|
|
|
|
|
|
|
def json_error_payload(message, code, id_=None): |
|
|
def json_error_payload(message, code, id_=None): |
|
|
error = {'message': message, 'code': code} |
|
|
error = {'message': message, 'code': code} |
|
|
return {'jsonrpc': '2.0', 'error': error, 'result': None, 'id': id_} |
|
|
return {'jsonrpc': '2.0', 'error': error, 'id': id_} |
|
|
|
|
|
|
|
|
def json_notification_payload(method, params): |
|
|
def json_request_payload(method, id_, params=None): |
|
|
return {'jsonrpc': '2.0', 'id': None, 'method': method, 'params': params} |
|
|
payload = {'jsonrpc': '2.0', 'id': id_, 'method': method} |
|
|
|
|
|
if params: |
|
|
|
|
|
payload['params'] = params |
|
|
|
|
|
return payload |
|
|
|
|
|
|
|
|
|
|
|
def json_notification_payload(method, params=None): |
|
|
|
|
|
return json_request_payload(method, None, params) |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
class JSONRPC(asyncio.Protocol, LoggedClass): |
|
|
class JSONRPC(asyncio.Protocol, LoggedClass): |
|
|
'''Manages a JSONRPC connection. |
|
|
'''Manages a JSONRPC connection. |
|
|
|
|
|
|
|
|
Assumes JSON messages are newline-separated and that newlines |
|
|
Assumes JSON messages are newline-separated and that newlines |
|
|
cannot appear in the JSON other than to separate lines. |
|
|
cannot appear in the JSON other than to separate lines. Incoming |
|
|
|
|
|
messages are queued on the messages queue for later asynchronous |
|
|
Derived classes need to implement the synchronous functions |
|
|
processing, and should be passed to the handle_message() function. |
|
|
on_json_request() and method_handler(). They probably also want |
|
|
|
|
|
to override connection_made() and connection_lost() but should be |
|
|
Derived classes may want to override connection_made() and |
|
|
sure to call the implementation in this base class first. |
|
|
connection_lost() but should be sure to call the implementation in |
|
|
|
|
|
this base class first. They will also want to implement some or |
|
|
on_json_request() is passed a JSON request as a python object |
|
|
all of the asynchronous functions handle_notification(), |
|
|
after decoding. It should arrange to pass on to the asynchronous |
|
|
handle_response() and handle_request(). |
|
|
handle_json_request() method. |
|
|
|
|
|
|
|
|
handle_request() returns the result to pass over the network, and |
|
|
method_handler() takes a method string and should return a function |
|
|
must raise an RPCError if there is an error. |
|
|
that can be passed a parameters array, or None for an unknown method. |
|
|
handle_notification() and handle_response() should not return |
|
|
|
|
|
anything or raise any exceptions. All three functions have |
|
|
|
|
|
default "ignore" implementations supplied by this class. |
|
|
|
|
|
|
|
|
Handlers should raise an RPCError on error. |
|
|
|
|
|
''' |
|
|
''' |
|
|
|
|
|
|
|
|
# See http://www.jsonrpc.org/specification |
|
|
# See http://www.jsonrpc.org/specification |
|
@ -54,7 +61,7 @@ class JSONRPC(asyncio.Protocol, LoggedClass): |
|
|
INVALID_REQUEST = -32600 |
|
|
INVALID_REQUEST = -32600 |
|
|
METHOD_NOT_FOUND = -32601 |
|
|
METHOD_NOT_FOUND = -32601 |
|
|
INVALID_PARAMS = -32602 |
|
|
INVALID_PARAMS = -32602 |
|
|
INTERAL_ERROR = -32603 |
|
|
INTERNAL_ERROR = -32603 |
|
|
|
|
|
|
|
|
ID_TYPES = (type(None), str, numbers.Number) |
|
|
ID_TYPES = (type(None), str, numbers.Number) |
|
|
|
|
|
|
|
@ -65,7 +72,6 @@ class JSONRPC(asyncio.Protocol, LoggedClass): |
|
|
self.msg = msg |
|
|
self.msg = msg |
|
|
self.code = code |
|
|
self.code = code |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
def __init__(self): |
|
|
def __init__(self): |
|
|
super().__init__() |
|
|
super().__init__() |
|
|
self.start = time.time() |
|
|
self.start = time.time() |
|
@ -80,6 +86,7 @@ class JSONRPC(asyncio.Protocol, LoggedClass): |
|
|
self.send_size = 0 |
|
|
self.send_size = 0 |
|
|
self.error_count = 0 |
|
|
self.error_count = 0 |
|
|
self.peer_info = None |
|
|
self.peer_info = None |
|
|
|
|
|
self.messages = asyncio.Queue() |
|
|
|
|
|
|
|
|
def connection_made(self, transport): |
|
|
def connection_made(self, transport): |
|
|
'''Handle an incoming client connection.''' |
|
|
'''Handle an incoming client connection.''' |
|
@ -132,15 +139,20 @@ class JSONRPC(asyncio.Protocol, LoggedClass): |
|
|
self.transport.close() |
|
|
self.transport.close() |
|
|
return |
|
|
return |
|
|
|
|
|
|
|
|
self.on_json_request(message) |
|
|
'''Queue the request for asynchronous handling.''' |
|
|
|
|
|
self.messages.put_nowait(message) |
|
|
|
|
|
|
|
|
def send_json_notification(self, method, params): |
|
|
def send_json_notification(self, method, params): |
|
|
'''Create a json notification.''' |
|
|
'''Create a json notification.''' |
|
|
self.send_json(json_notification_payload(method, params)) |
|
|
self.send_json(json_notification_payload(method, params)) |
|
|
|
|
|
|
|
|
def send_json_result(self, result, id_): |
|
|
def send_json_request(self, method, id_, params=None): |
|
|
|
|
|
'''Send a JSON request.''' |
|
|
|
|
|
self.send_json(json_request_payload(method, id_, params)) |
|
|
|
|
|
|
|
|
|
|
|
def send_json_response(self, result, id_): |
|
|
'''Send a JSON result.''' |
|
|
'''Send a JSON result.''' |
|
|
self.send_json(json_result_payload(result, id_)) |
|
|
self.send_json(json_response_payload(result, id_)) |
|
|
|
|
|
|
|
|
def send_json_error(self, message, code, id_=None): |
|
|
def send_json_error(self, message, code, id_=None): |
|
|
'''Send a JSON error.''' |
|
|
'''Send a JSON error.''' |
|
@ -167,20 +179,20 @@ class JSONRPC(asyncio.Protocol, LoggedClass): |
|
|
self.send_size += len(data) |
|
|
self.send_size += len(data) |
|
|
self.transport.write(data) |
|
|
self.transport.write(data) |
|
|
|
|
|
|
|
|
async def handle_json_request(self, request): |
|
|
async def handle_message(self, message): |
|
|
'''Asynchronously handle a JSON request. |
|
|
'''Asynchronously handle a JSON request or response. |
|
|
|
|
|
|
|
|
Handles batch requests. |
|
|
Handles batches according to the JSON 2.0 spec. |
|
|
''' |
|
|
''' |
|
|
if isinstance(request, list): |
|
|
if isinstance(message, list): |
|
|
payload = await self.batch_request_payload(request) |
|
|
payload = await self.batch_payload(message) |
|
|
else: |
|
|
else: |
|
|
payload = await self.single_request_payload(request) |
|
|
payload = await self.single_payload(message) |
|
|
|
|
|
|
|
|
if payload: |
|
|
if payload: |
|
|
self.send_json(payload) |
|
|
self.send_json(payload) |
|
|
|
|
|
|
|
|
async def batch_request_payload(self, batch): |
|
|
async def batch_payload(self, batch): |
|
|
'''Return the JSON payload corresponding to a batch JSON request.''' |
|
|
'''Return the JSON payload corresponding to a batch JSON request.''' |
|
|
# Batches must have at least one request. |
|
|
# Batches must have at least one request. |
|
|
if not batch: |
|
|
if not batch: |
|
@ -189,38 +201,39 @@ class JSONRPC(asyncio.Protocol, LoggedClass): |
|
|
|
|
|
|
|
|
# PYTHON 3.6: use asynchronous comprehensions when supported |
|
|
# PYTHON 3.6: use asynchronous comprehensions when supported |
|
|
payload = [] |
|
|
payload = [] |
|
|
for item in request: |
|
|
for message in batch: |
|
|
item_payload = await self.single_request_payload(item) |
|
|
message_payload = await self.single_payload(message) |
|
|
if item_payload: |
|
|
if message_payload: |
|
|
payload.append(item_payload) |
|
|
payload.append(message_payload) |
|
|
return payload |
|
|
return payload |
|
|
|
|
|
|
|
|
async def single_request_payload(self, request): |
|
|
async def single_payload(self, message): |
|
|
'''Return the JSON payload corresponding to a single JSON request. |
|
|
'''Return the JSON payload corresponding to a single JSON request, |
|
|
|
|
|
response or notification. |
|
|
|
|
|
|
|
|
Return None if the request is a notification. |
|
|
Return None if the request is a notification or a response. |
|
|
''' |
|
|
''' |
|
|
if not isinstance(request, dict): |
|
|
if not isinstance(message, dict): |
|
|
return json_error_payload('request must be a dict', |
|
|
return json_error_payload('request must be a dict', |
|
|
self.INVALID_REQUEST) |
|
|
self.INVALID_REQUEST) |
|
|
|
|
|
|
|
|
id_ = request.get('id') |
|
|
if not 'id' in message: |
|
|
|
|
|
return await self.json_notification(message) |
|
|
|
|
|
|
|
|
|
|
|
id_ = message['id'] |
|
|
if not isinstance(id_, self.ID_TYPES): |
|
|
if not isinstance(id_, self.ID_TYPES): |
|
|
return json_error_payload('invalid id: {}'.format(id_), |
|
|
return json_error_payload('invalid id: {}'.format(id_), |
|
|
self.INVALID_REQUEST) |
|
|
self.INVALID_REQUEST) |
|
|
|
|
|
|
|
|
try: |
|
|
if 'method' in message: |
|
|
result = await self.method_result(request.get('method'), |
|
|
return await self.json_request(message) |
|
|
request.get('params', [])) |
|
|
|
|
|
if id_ is None: |
|
|
return await self.json_response(message) |
|
|
return None |
|
|
|
|
|
return json_result_payload(result, id_) |
|
|
def method_and_params(self, message): |
|
|
except self.RPCError as e: |
|
|
method = message.get('method') |
|
|
if id_ is None: |
|
|
params = message.get('params', []) |
|
|
return None |
|
|
|
|
|
return json_error_payload(e.msg, e.code, id_) |
|
|
|
|
|
|
|
|
|
|
|
async def method_result(self, method, params): |
|
|
|
|
|
if not isinstance(method, str): |
|
|
if not isinstance(method, str): |
|
|
raise self.RPCError('invalid method: {}'.format(method), |
|
|
raise self.RPCError('invalid method: {}'.format(method), |
|
|
self.INVALID_REQUEST) |
|
|
self.INVALID_REQUEST) |
|
@ -229,17 +242,46 @@ class JSONRPC(asyncio.Protocol, LoggedClass): |
|
|
raise self.RPCError('params should be an array', |
|
|
raise self.RPCError('params should be an array', |
|
|
self.INVALID_REQUEST) |
|
|
self.INVALID_REQUEST) |
|
|
|
|
|
|
|
|
handler = self.method_handler(method) |
|
|
return method, params |
|
|
if not handler: |
|
|
|
|
|
raise self.RPCError('unknown method: "{}"'.format(method), |
|
|
|
|
|
self.METHOD_NOT_FOUND) |
|
|
|
|
|
|
|
|
|
|
|
return await handler(params) |
|
|
|
|
|
|
|
|
|
|
|
def on_json_request(self, request): |
|
|
async def json_notification(self, message): |
|
|
raise NotImplementedError('on_json_request in class {}'. |
|
|
try: |
|
|
format(self.__class__.__name__)) |
|
|
method, params = self.method_and_params(message) |
|
|
|
|
|
except RCPError: |
|
|
|
|
|
pass |
|
|
|
|
|
else: |
|
|
|
|
|
self.handle_notification(method, params) |
|
|
|
|
|
return None |
|
|
|
|
|
|
|
|
def method_handler(self, method): |
|
|
async def json_request(self, message): |
|
|
raise NotImplementedError('method_handler in class {}'. |
|
|
try: |
|
|
format(self.__class__.__name__)) |
|
|
method, params = self.method_and_params(message) |
|
|
|
|
|
result = await self.handle_request(method, params) |
|
|
|
|
|
return json_response_payload(result, message['id']) |
|
|
|
|
|
except self.RPCError as e: |
|
|
|
|
|
return json_error_payload(e.msg, e.code, message['id']) |
|
|
|
|
|
|
|
|
|
|
|
async def json_response(self, message): |
|
|
|
|
|
# Only one of result and error should exist; we go with 'error' |
|
|
|
|
|
# if both are supplied. |
|
|
|
|
|
if 'error' in message: |
|
|
|
|
|
await self.handle_response(None, message['error'], message['id']) |
|
|
|
|
|
elif 'result' in message: |
|
|
|
|
|
await self.handle_response(message['result'], None, message['id']) |
|
|
|
|
|
return None |
|
|
|
|
|
|
|
|
|
|
|
def raise_unknown_method(method): |
|
|
|
|
|
'''Respond to a request with an unknown method.''' |
|
|
|
|
|
raise self.RPCError('unknown method: "{}"'.format(method), |
|
|
|
|
|
self.METHOD_NOT_FOUND) |
|
|
|
|
|
|
|
|
|
|
|
# --- derived classes are intended to override these functions |
|
|
|
|
|
async def handle_notification(self, method, params): |
|
|
|
|
|
'''Handle a notification.''' |
|
|
|
|
|
|
|
|
|
|
|
async def handle_request(self, method, params): |
|
|
|
|
|
'''Handle a request.''' |
|
|
|
|
|
return None |
|
|
|
|
|
|
|
|
|
|
|
async def handle_response(self, result, error, id_): |
|
|
|
|
|
'''Handle a response.''' |
|
|