2020-07-02 04:21:35 +03:00
|
|
|
# Websocket Request/Response Handler
|
|
|
|
#
|
|
|
|
# Copyright (C) 2020 Eric Callahan <arksine.code@gmail.com>
|
|
|
|
#
|
|
|
|
# This file may be distributed under the terms of the GNU GPLv3 license
|
|
|
|
|
|
|
|
import logging
|
|
|
|
import tornado
|
|
|
|
import json
|
|
|
|
from tornado.ioloop import IOLoop
|
|
|
|
from tornado.websocket import WebSocketHandler, WebSocketClosedError
|
2020-07-31 12:54:45 +03:00
|
|
|
from utils import ServerError
|
2020-07-02 04:21:35 +03:00
|
|
|
|
|
|
|
class JsonRPC:
|
|
|
|
def __init__(self):
|
|
|
|
self.methods = {}
|
|
|
|
|
|
|
|
def register_method(self, name, method):
|
|
|
|
self.methods[name] = method
|
|
|
|
|
|
|
|
def remove_method(self, name):
|
2020-07-13 22:13:34 +03:00
|
|
|
self.methods.pop(name, None)
|
2020-07-02 04:21:35 +03:00
|
|
|
|
|
|
|
async def dispatch(self, data):
|
|
|
|
response = None
|
|
|
|
try:
|
|
|
|
request = json.loads(data)
|
|
|
|
except Exception:
|
2020-08-11 19:59:47 +03:00
|
|
|
msg = f"Websocket data not json: {data}"
|
2020-07-02 04:21:35 +03:00
|
|
|
logging.exception(msg)
|
|
|
|
response = self.build_error(-32700, "Parse error")
|
|
|
|
return json.dumps(response)
|
2020-07-31 12:54:45 +03:00
|
|
|
logging.debug("Websocket Request::" + data)
|
2020-07-02 04:21:35 +03:00
|
|
|
if isinstance(request, list):
|
|
|
|
response = []
|
|
|
|
for req in request:
|
|
|
|
resp = await self.process_request(req)
|
|
|
|
if resp is not None:
|
|
|
|
response.append(resp)
|
|
|
|
if not response:
|
|
|
|
response = None
|
|
|
|
else:
|
|
|
|
response = await self.process_request(request)
|
|
|
|
if response is not None:
|
|
|
|
response = json.dumps(response)
|
2020-07-31 12:54:45 +03:00
|
|
|
logging.debug("Websocket Response::" + response)
|
2020-07-02 04:21:35 +03:00
|
|
|
return response
|
|
|
|
|
|
|
|
async def process_request(self, request):
|
|
|
|
req_id = request.get('id', None)
|
|
|
|
rpc_version = request.get('jsonrpc', "")
|
|
|
|
method_name = request.get('method', None)
|
|
|
|
if rpc_version != "2.0" or not isinstance(method_name, str):
|
|
|
|
return self.build_error(-32600, "Invalid Request", req_id)
|
|
|
|
method = self.methods.get(method_name, None)
|
|
|
|
if method is None:
|
|
|
|
return self.build_error(-32601, "Method not found", req_id)
|
|
|
|
if 'params' in request:
|
|
|
|
params = request['params']
|
|
|
|
if isinstance(params, list):
|
|
|
|
response = await self.execute_method(method, req_id, *params)
|
|
|
|
elif isinstance(params, dict):
|
|
|
|
response = await self.execute_method(method, req_id, **params)
|
|
|
|
else:
|
|
|
|
return self.build_error(-32600, "Invalid Request", req_id)
|
|
|
|
else:
|
|
|
|
response = await self.execute_method(method, req_id)
|
|
|
|
return response
|
|
|
|
|
|
|
|
async def execute_method(self, method, req_id, *args, **kwargs):
|
|
|
|
try:
|
|
|
|
result = await method(*args, **kwargs)
|
|
|
|
except TypeError as e:
|
|
|
|
return self.build_error(-32603, "Invalid params", req_id)
|
|
|
|
except Exception as e:
|
|
|
|
return self.build_error(-31000, str(e), req_id)
|
|
|
|
if isinstance(result, ServerError):
|
|
|
|
return self.build_error(result.status_code, str(result), req_id)
|
|
|
|
elif req_id is None:
|
|
|
|
return None
|
|
|
|
else:
|
|
|
|
return self.build_result(result, req_id)
|
|
|
|
|
|
|
|
def build_result(self, result, req_id):
|
|
|
|
return {
|
|
|
|
'jsonrpc': "2.0",
|
|
|
|
'result': result,
|
|
|
|
'id': req_id
|
|
|
|
}
|
|
|
|
|
|
|
|
def build_error(self, code, msg, req_id=None):
|
|
|
|
return {
|
|
|
|
'jsonrpc': "2.0",
|
|
|
|
'error': {'code': code, 'message': msg},
|
|
|
|
'id': req_id
|
|
|
|
}
|
|
|
|
|
|
|
|
class WebsocketManager:
|
|
|
|
def __init__(self, server):
|
|
|
|
self.server = server
|
|
|
|
self.websockets = {}
|
|
|
|
self.ws_lock = tornado.locks.Lock()
|
|
|
|
self.rpc = JsonRPC()
|
|
|
|
|
|
|
|
# Register events
|
|
|
|
self.server.register_event_handler(
|
|
|
|
"server:klippy_state_changed", self._handle_klippy_state_changed)
|
|
|
|
self.server.register_event_handler(
|
|
|
|
"server:gcode_response", self._handle_gcode_response)
|
|
|
|
self.server.register_event_handler(
|
|
|
|
"server:status_update", self._handle_status_update)
|
|
|
|
self.server.register_event_handler(
|
2020-07-27 03:05:43 +03:00
|
|
|
"file_manager:filelist_changed", self._handle_filelist_changed)
|
2020-07-02 04:21:35 +03:00
|
|
|
|
|
|
|
async def _handle_klippy_state_changed(self, state):
|
|
|
|
await self.notify_websockets("klippy_state_changed", state)
|
|
|
|
|
|
|
|
async def _handle_gcode_response(self, response):
|
|
|
|
await self.notify_websockets("gcode_response", response)
|
|
|
|
|
|
|
|
async def _handle_status_update(self, status):
|
|
|
|
await self.notify_websockets("status_update", status)
|
|
|
|
|
|
|
|
async def _handle_filelist_changed(self, flist):
|
|
|
|
await self.notify_websockets("filelist_changed", flist)
|
|
|
|
|
|
|
|
def register_handler(self, api_def, callback=None):
|
|
|
|
for r_method in api_def.request_methods:
|
|
|
|
cmd = r_method.lower() + '_' + api_def.ws_method
|
|
|
|
if callback is not None:
|
|
|
|
# Callback is a local method
|
|
|
|
rpc_cb = self._generate_local_callback(
|
|
|
|
api_def.endpoint, r_method, callback)
|
|
|
|
else:
|
|
|
|
# Callback is a remote method
|
|
|
|
rpc_cb = self._generate_callback(api_def.endpoint, r_method)
|
|
|
|
self.rpc.register_method(cmd, rpc_cb)
|
|
|
|
|
|
|
|
def remove_handler(self, ws_method):
|
2020-07-04 21:33:17 +03:00
|
|
|
for prefix in ["get", "post", "delete"]:
|
2020-07-02 04:21:35 +03:00
|
|
|
self.rpc.remove_method(prefix + "_" + ws_method)
|
|
|
|
|
|
|
|
def _generate_callback(self, endpoint, request_method):
|
|
|
|
async def func(**kwargs):
|
|
|
|
request = self.server.make_request(
|
|
|
|
endpoint, request_method, kwargs)
|
|
|
|
result = await request.wait()
|
|
|
|
return result
|
|
|
|
return func
|
|
|
|
|
|
|
|
def _generate_local_callback(self, endpoint, request_method, callback):
|
|
|
|
async def func(**kwargs):
|
|
|
|
try:
|
|
|
|
result = await callback(
|
|
|
|
endpoint, request_method, kwargs)
|
|
|
|
except ServerError as e:
|
|
|
|
result = e
|
|
|
|
return result
|
|
|
|
return func
|
|
|
|
|
|
|
|
def has_websocket(self, ws_id):
|
|
|
|
return ws_id in self.websockets
|
|
|
|
|
|
|
|
async def add_websocket(self, ws):
|
|
|
|
async with self.ws_lock:
|
|
|
|
self.websockets[ws.uid] = ws
|
2020-08-11 19:59:47 +03:00
|
|
|
logging.info(f"New Websocket Added: {ws.uid}")
|
2020-07-02 04:21:35 +03:00
|
|
|
|
|
|
|
async def remove_websocket(self, ws):
|
|
|
|
async with self.ws_lock:
|
|
|
|
old_ws = self.websockets.pop(ws.uid, None)
|
|
|
|
if old_ws is not None:
|
2020-08-11 19:59:47 +03:00
|
|
|
logging.info(f"Websocket Removed: {ws.uid}")
|
2020-07-02 04:21:35 +03:00
|
|
|
|
|
|
|
async def notify_websockets(self, name, data):
|
|
|
|
notification = json.dumps({
|
|
|
|
'jsonrpc': "2.0",
|
|
|
|
'method': "notify_" + name,
|
|
|
|
'params': [data]})
|
|
|
|
async with self.ws_lock:
|
2020-08-04 17:17:05 +03:00
|
|
|
for ws in list(self.websockets.values()):
|
2020-07-02 04:21:35 +03:00
|
|
|
try:
|
|
|
|
ws.write_message(notification)
|
|
|
|
except WebSocketClosedError:
|
2020-07-13 22:13:34 +03:00
|
|
|
self.websockets.pop(ws.uid, None)
|
2020-08-11 19:59:47 +03:00
|
|
|
logging.info(f"Websocket Removed: {ws.uid}")
|
2020-07-02 04:21:35 +03:00
|
|
|
except Exception:
|
|
|
|
logging.exception(
|
2020-08-11 19:59:47 +03:00
|
|
|
f"Error sending data over websocket: {ws.uid}")
|
2020-07-02 04:21:35 +03:00
|
|
|
|
|
|
|
async def close(self):
|
|
|
|
async with self.ws_lock:
|
|
|
|
for ws in self.websockets.values():
|
|
|
|
ws.close()
|
|
|
|
self.websockets = {}
|
|
|
|
|
|
|
|
class WebSocket(WebSocketHandler):
|
|
|
|
def initialize(self, wsm, auth):
|
|
|
|
self.wsm = wsm
|
|
|
|
self.auth = auth
|
|
|
|
self.rpc = self.wsm.rpc
|
|
|
|
self.uid = id(self)
|
|
|
|
|
|
|
|
async def open(self):
|
|
|
|
await self.wsm.add_websocket(self)
|
|
|
|
|
|
|
|
def on_message(self, message):
|
|
|
|
io_loop = IOLoop.current()
|
|
|
|
io_loop.spawn_callback(self._process_message, message)
|
|
|
|
|
|
|
|
async def _process_message(self, message):
|
|
|
|
try:
|
|
|
|
response = await self.rpc.dispatch(message)
|
|
|
|
if response is not None:
|
|
|
|
self.write_message(response)
|
|
|
|
except Exception:
|
|
|
|
logging.exception("Websocket Command Error")
|
|
|
|
|
|
|
|
def on_close(self):
|
|
|
|
io_loop = IOLoop.current()
|
|
|
|
io_loop.spawn_callback(self.wsm.remove_websocket, self)
|
|
|
|
|
|
|
|
def check_origin(self, origin):
|
|
|
|
if self.settings['enable_cors']:
|
|
|
|
# allow CORS
|
|
|
|
return True
|
|
|
|
else:
|
|
|
|
return super(WebSocket, self).check_origin(origin)
|
|
|
|
|
|
|
|
# Check Authorized User
|
|
|
|
def prepare(self):
|
|
|
|
if not self.auth.check_authorized(self.request):
|
|
|
|
raise tornado.web.HTTPError(401, "Unauthorized")
|