moonraker: manage subscriptions independently for each connection
This allows clients to "unsubscribe"by sending an empty dict. Each client will receive updates only for subscribed objects. Signed-off-by: Eric Callahan <arksine.code@gmail.com>
This commit is contained in:
parent
8d1239c316
commit
a6913a982a
|
@ -51,14 +51,7 @@ class Server:
|
||||||
self.init_handle = None
|
self.init_handle = None
|
||||||
self.init_attempts = 0
|
self.init_attempts = 0
|
||||||
self.klippy_state = "disconnected"
|
self.klippy_state = "disconnected"
|
||||||
|
self.subscriptions = {}
|
||||||
# XXX - currently moonraker maintains a superset of all
|
|
||||||
# subscriptions, the results of which are forwarded to all
|
|
||||||
# connected websockets. A better implementation would open a
|
|
||||||
# unique unix domain socket for each websocket client and
|
|
||||||
# allow Klipper to forward only those subscriptions back to
|
|
||||||
# correct client.
|
|
||||||
self.all_subscriptions = {}
|
|
||||||
|
|
||||||
# Server/IOLoop
|
# Server/IOLoop
|
||||||
self.server_running = False
|
self.server_running = False
|
||||||
|
@ -221,6 +214,7 @@ class Server:
|
||||||
for request in self.pending_requests.values():
|
for request in self.pending_requests.values():
|
||||||
request.notify(ServerError("Klippy Disconnected", 503))
|
request.notify(ServerError("Klippy Disconnected", 503))
|
||||||
self.pending_requests = {}
|
self.pending_requests = {}
|
||||||
|
self.subscriptions = {}
|
||||||
logging.info("Klippy Connection Removed")
|
logging.info("Klippy Connection Removed")
|
||||||
self.send_event("server:klippy_disconnect")
|
self.send_event("server:klippy_disconnect")
|
||||||
if self.init_handle is not None:
|
if self.init_handle is not None:
|
||||||
|
@ -236,8 +230,6 @@ class Server:
|
||||||
# Subscribe to "webhooks"
|
# Subscribe to "webhooks"
|
||||||
# Register "webhooks" subscription
|
# Register "webhooks" subscription
|
||||||
if "webhooks_sub" not in self.init_list:
|
if "webhooks_sub" not in self.init_list:
|
||||||
temp_subs = self.all_subscriptions
|
|
||||||
self.all_subscriptions = {}
|
|
||||||
try:
|
try:
|
||||||
await self.klippy_apis.subscribe_objects({'webhooks': None})
|
await self.klippy_apis.subscribe_objects({'webhooks': None})
|
||||||
except ServerError as e:
|
except ServerError as e:
|
||||||
|
@ -245,7 +237,6 @@ class Server:
|
||||||
else:
|
else:
|
||||||
logging.info("Webhooks Subscribed")
|
logging.info("Webhooks Subscribed")
|
||||||
self.init_list.append("webhooks_sub")
|
self.init_list.append("webhooks_sub")
|
||||||
self.all_subscriptions.update(temp_subs)
|
|
||||||
# Subscribe to Gcode Output
|
# Subscribe to Gcode Output
|
||||||
if "gcode_output_sub" not in self.init_list:
|
if "gcode_output_sub" not in self.init_list:
|
||||||
try:
|
try:
|
||||||
|
@ -318,10 +309,6 @@ class Server:
|
||||||
logging.info(
|
logging.info(
|
||||||
f"Unable to retreive Klipper Object List")
|
f"Unable to retreive Klipper Object List")
|
||||||
return
|
return
|
||||||
# Remove stale objects from the persistent subscription dict
|
|
||||||
for name in list(self.all_subscriptions.keys()):
|
|
||||||
if name not in result:
|
|
||||||
del self.all_subscriptions[name]
|
|
||||||
req_objs = set(["virtual_sdcard", "display_status", "pause_resume"])
|
req_objs = set(["virtual_sdcard", "display_status", "pause_resume"])
|
||||||
missing_objs = req_objs - set(result)
|
missing_objs = req_objs - set(result)
|
||||||
if missing_objs:
|
if missing_objs:
|
||||||
|
@ -360,28 +347,42 @@ class Server:
|
||||||
logging.info("Klippy has shutdown")
|
logging.info("Klippy has shutdown")
|
||||||
self.send_event("server:klippy_shutdown")
|
self.send_event("server:klippy_shutdown")
|
||||||
self.klippy_state = state
|
self.klippy_state = state
|
||||||
self.send_event("server:status_update", status)
|
for conn, sub in self.subscriptions.items():
|
||||||
|
conn_status = {}
|
||||||
|
for name, fields in sub.items():
|
||||||
|
if name in status:
|
||||||
|
val = status[name]
|
||||||
|
if fields is None:
|
||||||
|
conn_status[name] = dict(val)
|
||||||
|
else:
|
||||||
|
conn_status[name] = {
|
||||||
|
k: v for k, v in val.items() if k in fields}
|
||||||
|
conn.send_status(conn_status)
|
||||||
|
|
||||||
async def make_request(self, web_request):
|
async def make_request(self, web_request):
|
||||||
# XXX - This adds the "response_template" to a subscription
|
|
||||||
# request and tracks all subscriptions so that each
|
|
||||||
# client gets what its requesting. In the future we should
|
|
||||||
# track subscriptions per client and send clients only
|
|
||||||
# the data they are asking for.
|
|
||||||
rpc_method = web_request.get_endpoint()
|
rpc_method = web_request.get_endpoint()
|
||||||
args = web_request.get_args()
|
args = web_request.get_args()
|
||||||
if rpc_method == "objects/subscribe":
|
if rpc_method == "objects/subscribe":
|
||||||
for obj, items in args.get('objects', {}).items():
|
sub = args.get('objects', {})
|
||||||
if obj in self.all_subscriptions:
|
conn = web_request.get_connection()
|
||||||
pi = self.all_subscriptions[obj]
|
if conn is None:
|
||||||
|
raise self.error(
|
||||||
|
"No connection associated with subscription request")
|
||||||
|
self.subscriptions[conn] = sub
|
||||||
|
all_subs = {}
|
||||||
|
# request superset of all client subscriptions
|
||||||
|
for sub in self.subscriptions.values():
|
||||||
|
for obj, items in sub.items():
|
||||||
|
if obj in all_subs:
|
||||||
|
pi = all_subs[obj]
|
||||||
if items is None or pi is None:
|
if items is None or pi is None:
|
||||||
self.all_subscriptions[obj] = None
|
all_subs[obj] = None
|
||||||
else:
|
else:
|
||||||
uitems = list(set(pi) | set(items))
|
uitems = list(set(pi) | set(items))
|
||||||
self.all_subscriptions[obj] = uitems
|
all_subs[obj] = uitems
|
||||||
else:
|
else:
|
||||||
self.all_subscriptions[obj] = items
|
all_subs[obj] = items
|
||||||
args['objects'] = dict(self.all_subscriptions)
|
args['objects'] = all_subs
|
||||||
args['response_template'] = {'method': "process_status_update"}
|
args['response_template'] = {'method': "process_status_update"}
|
||||||
|
|
||||||
# Create a base klippy request
|
# Create a base klippy request
|
||||||
|
@ -392,6 +393,9 @@ class Server:
|
||||||
result = await base_request.wait()
|
result = await base_request.wait()
|
||||||
return result
|
return result
|
||||||
|
|
||||||
|
def remove_subscription(self, conn):
|
||||||
|
self.subscriptions.pop(conn, None)
|
||||||
|
|
||||||
async def _stop_server(self):
|
async def _stop_server(self):
|
||||||
self.server_running = False
|
self.server_running = False
|
||||||
for name, plugin in self.plugins.items():
|
for name, plugin in self.plugins.items():
|
||||||
|
|
|
@ -23,6 +23,10 @@ class KlippyAPI:
|
||||||
def __init__(self, config):
|
def __init__(self, config):
|
||||||
self.server = config.get_server()
|
self.server = config.get_server()
|
||||||
|
|
||||||
|
# Maintain a subscription for all moonraker requests, as
|
||||||
|
# we do not want to overwrite them
|
||||||
|
self.host_subscription = {}
|
||||||
|
|
||||||
# Register GCode Aliases
|
# Register GCode Aliases
|
||||||
self.server.register_endpoint(
|
self.server.register_endpoint(
|
||||||
"/printer/print/pause", ['POST'], self._gcode_pause)
|
"/printer/print/pause", ['POST'], self._gcode_pause)
|
||||||
|
@ -59,7 +63,7 @@ class KlippyAPI:
|
||||||
async def _send_klippy_request(self, method, params, default=Sentinel):
|
async def _send_klippy_request(self, method, params, default=Sentinel):
|
||||||
try:
|
try:
|
||||||
result = await self.server.make_request(
|
result = await self.server.make_request(
|
||||||
WebRequest(method, params))
|
WebRequest(method, params, conn=self))
|
||||||
except self.server.error as e:
|
except self.server.error as e:
|
||||||
if default == Sentinel:
|
if default == Sentinel:
|
||||||
raise
|
raise
|
||||||
|
@ -119,7 +123,17 @@ class KlippyAPI:
|
||||||
return result
|
return result
|
||||||
|
|
||||||
async def subscribe_objects(self, objects, default=Sentinel):
|
async def subscribe_objects(self, objects, default=Sentinel):
|
||||||
params = {'objects': objects}
|
for obj, items in objects.items():
|
||||||
|
if obj in self.host_subscription:
|
||||||
|
prev = self.host_subscription[obj]
|
||||||
|
if items is None or prev is None:
|
||||||
|
self.host_subscription[obj] = None
|
||||||
|
else:
|
||||||
|
uitems = list(set(prev) | set(items))
|
||||||
|
self.host_subscription[obj] = uitems
|
||||||
|
else:
|
||||||
|
self.host_subscription[obj] = items
|
||||||
|
params = {'objects': self.host_subscription}
|
||||||
result = await self._send_klippy_request(
|
result = await self._send_klippy_request(
|
||||||
SUBSCRIPTION_ENDPOINT, params, default)
|
SUBSCRIPTION_ENDPOINT, params, default)
|
||||||
if isinstance(result, dict) and 'status' in result:
|
if isinstance(result, dict) and 'status' in result:
|
||||||
|
@ -138,5 +152,8 @@ class KlippyAPI:
|
||||||
{'response_template': {"method": method_name},
|
{'response_template': {"method": method_name},
|
||||||
'remote_method': method_name})
|
'remote_method': method_name})
|
||||||
|
|
||||||
|
def send_status(self, status):
|
||||||
|
self.server.send_event("server:status_update", status)
|
||||||
|
|
||||||
def load_plugin(config):
|
def load_plugin(config):
|
||||||
return KlippyAPI(config)
|
return KlippyAPI(config)
|
||||||
|
|
|
@ -172,8 +172,6 @@ class WebsocketManager:
|
||||||
"server:klippy_disconnect", self._handle_klippy_disconnect)
|
"server:klippy_disconnect", self._handle_klippy_disconnect)
|
||||||
self.server.register_event_handler(
|
self.server.register_event_handler(
|
||||||
"server:gcode_response", self._handle_gcode_response)
|
"server:gcode_response", self._handle_gcode_response)
|
||||||
self.server.register_event_handler(
|
|
||||||
"server:status_update", self._handle_status_update)
|
|
||||||
self.server.register_event_handler(
|
self.server.register_event_handler(
|
||||||
"file_manager:filelist_changed", self._handle_filelist_changed)
|
"file_manager:filelist_changed", self._handle_filelist_changed)
|
||||||
self.server.register_event_handler(
|
self.server.register_event_handler(
|
||||||
|
@ -187,9 +185,6 @@ class WebsocketManager:
|
||||||
async def _handle_gcode_response(self, response):
|
async def _handle_gcode_response(self, response):
|
||||||
await self.notify_websockets("gcode_response", 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):
|
async def _handle_filelist_changed(self, flist):
|
||||||
await self.notify_websockets("filelist_changed", flist)
|
await self.notify_websockets("filelist_changed", flist)
|
||||||
|
|
||||||
|
@ -240,17 +235,17 @@ class WebsocketManager:
|
||||||
async with self.ws_lock:
|
async with self.ws_lock:
|
||||||
old_ws = self.websockets.pop(ws.uid, None)
|
old_ws = self.websockets.pop(ws.uid, None)
|
||||||
if old_ws is not None:
|
if old_ws is not None:
|
||||||
|
self.server.remove_subscription(old_ws)
|
||||||
logging.info(f"Websocket Removed: {ws.uid}")
|
logging.info(f"Websocket Removed: {ws.uid}")
|
||||||
|
|
||||||
async def notify_websockets(self, name, data=Sentinel):
|
async def notify_websockets(self, name, data=Sentinel):
|
||||||
msg = {'jsonrpc': "2.0", 'method': "notify_" + name}
|
msg = {'jsonrpc': "2.0", 'method': "notify_" + name}
|
||||||
if data != Sentinel:
|
if data != Sentinel:
|
||||||
msg['params'] = [data]
|
msg['params'] = [data]
|
||||||
notification = json.dumps(msg)
|
|
||||||
async with self.ws_lock:
|
async with self.ws_lock:
|
||||||
for ws in list(self.websockets.values()):
|
for ws in list(self.websockets.values()):
|
||||||
try:
|
try:
|
||||||
ws.write_message(notification)
|
ws.write_message(msg)
|
||||||
except WebSocketClosedError:
|
except WebSocketClosedError:
|
||||||
self.websockets.pop(ws.uid, None)
|
self.websockets.pop(ws.uid, None)
|
||||||
logging.info(f"Websocket Removed: {ws.uid}")
|
logging.info(f"Websocket Removed: {ws.uid}")
|
||||||
|
@ -286,6 +281,21 @@ class WebSocket(WebSocketHandler):
|
||||||
except Exception:
|
except Exception:
|
||||||
logging.exception("Websocket Command Error")
|
logging.exception("Websocket Command Error")
|
||||||
|
|
||||||
|
def send_status(self, status):
|
||||||
|
if not status:
|
||||||
|
return
|
||||||
|
try:
|
||||||
|
self.write_message({
|
||||||
|
'jsonrpc': "2.0",
|
||||||
|
'method': "notify_status_update",
|
||||||
|
'params': [status]})
|
||||||
|
except WebSocketClosedError:
|
||||||
|
self.websockets.pop(self.uid, None)
|
||||||
|
logging.info(f"Websocket Removed: {self.uid}")
|
||||||
|
except Exception:
|
||||||
|
logging.exception(
|
||||||
|
f"Error sending data over websocket: {self.uid}")
|
||||||
|
|
||||||
def on_close(self):
|
def on_close(self):
|
||||||
io_loop = IOLoop.current()
|
io_loop = IOLoop.current()
|
||||||
io_loop.spawn_callback(self.wsm.remove_websocket, self)
|
io_loop.spawn_callback(self.wsm.remove_websocket, self)
|
||||||
|
|
Loading…
Reference in New Issue