2021-02-26 15:39:35 +00:00
|
|
|
import json
|
2021-04-03 17:36:01 +00:00
|
|
|
import logging
|
2021-02-26 15:39:35 +00:00
|
|
|
|
2022-04-12 01:34:46 +00:00
|
|
|
from app.classes.shared.console import Console
|
|
|
|
|
2021-04-03 17:36:01 +00:00
|
|
|
logger = logging.getLogger(__name__)
|
|
|
|
|
2022-03-23 02:50:12 +00:00
|
|
|
|
2021-02-26 15:39:35 +00:00
|
|
|
class WebSocketHelper:
|
2022-04-11 05:23:55 +00:00
|
|
|
def __init__(self, helper):
|
|
|
|
self.helper = helper
|
2021-08-10 20:17:56 +00:00
|
|
|
self.clients = set()
|
2021-02-26 15:39:35 +00:00
|
|
|
|
2021-08-10 20:17:56 +00:00
|
|
|
def add_client(self, client):
|
2021-02-26 15:39:35 +00:00
|
|
|
self.clients.add(client)
|
2022-01-26 01:45:30 +00:00
|
|
|
|
2021-08-10 20:17:56 +00:00
|
|
|
def remove_client(self, client):
|
|
|
|
self.clients.remove(client)
|
2021-08-11 20:29:31 +00:00
|
|
|
|
2022-06-13 21:35:33 +00:00
|
|
|
def send_message(self, client, event_type: str, data):
|
2021-03-01 17:33:15 +00:00
|
|
|
if client.check_auth():
|
2022-03-23 02:50:12 +00:00
|
|
|
message = str(json.dumps({"event": event_type, "data": data}))
|
2021-08-11 20:29:31 +00:00
|
|
|
client.write_message_helper(message)
|
2021-02-26 15:39:35 +00:00
|
|
|
|
2021-08-10 20:17:56 +00:00
|
|
|
def broadcast(self, event_type: str, data):
|
2022-03-23 02:50:12 +00:00
|
|
|
logger.debug(
|
2022-03-23 06:06:13 +00:00
|
|
|
f"Sending to {len(self.clients)} clients: "
|
|
|
|
f"{json.dumps({'event': event_type, 'data': data})}"
|
2022-03-23 02:50:12 +00:00
|
|
|
)
|
2022-09-20 22:21:57 +00:00
|
|
|
for client in self.clients[:]: # pylint: disable=unsubscriptable-object
|
2021-02-26 15:39:35 +00:00
|
|
|
try:
|
2021-03-01 00:54:20 +00:00
|
|
|
self.send_message(client, event_type, data)
|
2021-08-11 20:29:31 +00:00
|
|
|
except Exception as e:
|
2022-03-23 02:50:12 +00:00
|
|
|
logger.exception(
|
2022-03-23 06:06:13 +00:00
|
|
|
f"Error caught while sending WebSocket message to "
|
|
|
|
f"{client.get_remote_ip()} {e}"
|
2022-03-23 02:50:12 +00:00
|
|
|
)
|
2021-08-10 20:17:56 +00:00
|
|
|
|
2022-09-20 22:21:57 +00:00
|
|
|
# Excempting 'unsubscriptable-object' warning as false positive from 'self' use
|
|
|
|
# Code has been tested by Andrew and functions as intended.
|
|
|
|
|
2021-08-10 20:17:56 +00:00
|
|
|
def broadcast_page(self, page: str, event_type: str, data):
|
|
|
|
def filter_fn(client):
|
|
|
|
return client.page == page
|
|
|
|
|
2021-11-27 22:10:43 +00:00
|
|
|
self.broadcast_with_fn(filter_fn, event_type, data)
|
2021-08-10 20:17:56 +00:00
|
|
|
|
2021-11-27 22:10:43 +00:00
|
|
|
def broadcast_user(self, user_id: str, event_type: str, data):
|
|
|
|
def filter_fn(client):
|
|
|
|
return client.get_user_id() == user_id
|
2021-08-10 20:17:56 +00:00
|
|
|
|
2021-11-27 22:10:43 +00:00
|
|
|
self.broadcast_with_fn(filter_fn, event_type, data)
|
|
|
|
|
|
|
|
def broadcast_user_page(self, page: str, user_id: str, event_type: str, data):
|
|
|
|
def filter_fn(client):
|
|
|
|
if client.get_user_id() != user_id:
|
|
|
|
return False
|
|
|
|
if client.page != page:
|
|
|
|
return False
|
|
|
|
return True
|
|
|
|
|
|
|
|
self.broadcast_with_fn(filter_fn, event_type, data)
|
|
|
|
|
2022-03-23 02:50:12 +00:00
|
|
|
def broadcast_user_page_params(
|
|
|
|
self, page: str, params: dict, user_id: str, event_type: str, data
|
|
|
|
):
|
2021-11-27 22:10:43 +00:00
|
|
|
def filter_fn(client):
|
|
|
|
if client.get_user_id() != user_id:
|
|
|
|
return False
|
|
|
|
if client.page != page:
|
|
|
|
return False
|
|
|
|
for key, param in params.items():
|
|
|
|
if param != client.page_query_params.get(key, None):
|
|
|
|
return False
|
|
|
|
return True
|
|
|
|
|
|
|
|
self.broadcast_with_fn(filter_fn, event_type, data)
|
2022-01-26 01:45:30 +00:00
|
|
|
|
2021-08-10 20:17:56 +00:00
|
|
|
def broadcast_page_params(self, page: str, params: dict, event_type: str, data):
|
|
|
|
def filter_fn(client):
|
|
|
|
if client.page != page:
|
|
|
|
return False
|
|
|
|
for key, param in params.items():
|
|
|
|
if param != client.page_query_params.get(key, None):
|
|
|
|
return False
|
|
|
|
return True
|
|
|
|
|
2021-11-27 22:10:43 +00:00
|
|
|
self.broadcast_with_fn(filter_fn, event_type, data)
|
|
|
|
|
|
|
|
def broadcast_with_fn(self, filter_fn, event_type: str, data):
|
2021-08-10 20:17:56 +00:00
|
|
|
clients = list(filter(filter_fn, self.clients))
|
2022-03-23 02:50:12 +00:00
|
|
|
logger.debug(
|
2022-03-23 06:06:13 +00:00
|
|
|
f"Sending to {len(clients)} out of {len(self.clients)} "
|
|
|
|
f"clients: {json.dumps({'event': event_type, 'data': data})}"
|
2022-03-23 02:50:12 +00:00
|
|
|
)
|
2021-08-10 20:17:56 +00:00
|
|
|
|
2022-09-16 17:09:10 +00:00
|
|
|
for client in clients[:]:
|
2021-08-10 20:17:56 +00:00
|
|
|
try:
|
|
|
|
self.send_message(client, event_type, data)
|
2021-08-11 20:29:31 +00:00
|
|
|
except Exception as e:
|
2022-03-23 02:50:12 +00:00
|
|
|
logger.exception(
|
2022-03-23 06:06:13 +00:00
|
|
|
f"Error catched while sending WebSocket message to "
|
|
|
|
f"{client.get_remote_ip()} {e}"
|
2022-03-23 02:50:12 +00:00
|
|
|
)
|
2022-01-26 01:45:30 +00:00
|
|
|
|
2021-02-26 15:39:35 +00:00
|
|
|
def disconnect_all(self):
|
2022-04-12 01:34:46 +00:00
|
|
|
Console.info("Disconnecting WebSocket clients")
|
2021-02-26 15:39:35 +00:00
|
|
|
for client in self.clients:
|
|
|
|
client.close()
|
2022-04-12 01:34:46 +00:00
|
|
|
Console.info("Disconnected WebSocket clients")
|