frigate/frigate/communication/ws.py

95 lines
3.0 KiB
Python
Raw Normal View History

2022-11-22 23:22:17 +03:00
"""Websocket communicator."""
2022-11-22 23:29:28 +03:00
import json
import logging
import threading
from wsgiref.simple_server import make_server
2022-11-22 23:22:17 +03:00
from ws4py.server.wsgirefserver import (
WebSocketWSGIHandler,
WebSocketWSGIRequestHandler,
WSGIServer,
)
2022-11-22 23:29:28 +03:00
from ws4py.server.wsgiutils import WebSocketWSGIApplication
from ws4py.websocket import WebSocket
2022-11-22 23:22:17 +03:00
from frigate.communication.dispatcher import Communicator
2022-11-22 23:40:31 +03:00
from frigate.config import FrigateConfig
2022-11-22 23:22:17 +03:00
2022-11-22 23:29:28 +03:00
logger = logging.getLogger(__name__)
class WebSocketClient(Communicator):
"""Frigate wrapper for ws client."""
2022-11-22 23:22:17 +03:00
2022-11-22 23:40:31 +03:00
def __init__(self, config: FrigateConfig) -> None:
self.config = config
2022-11-22 23:29:28 +03:00
2022-11-22 23:51:08 +03:00
def subscribe(self, receiver) -> None:
self._dispatcher = receiver
self.start()
2022-11-22 23:29:28 +03:00
def start(self):
2022-11-22 23:51:08 +03:00
"""Start the websocket client."""
class _WebSocketHandler(WebSocket):
receiver = self._dispatcher
def received_message(self, message):
try:
json_message = json.loads(message.data.decode("utf-8"))
json_message = {
"topic": f"{self.topic_prefix}/{json_message['topic']}",
"payload": json_message.get("payload"),
"retain": json_message.get("retain", False),
}
except Exception as e:
logger.warning("Unable to parse websocket message as valid json.")
return
logger.debug(
f"Publishing mqtt message from websockets at {json_message['topic']}."
)
self.receiver(
json_message["topic"],
json_message["payload"],
retain=json_message["retain"],
)
2022-11-22 23:22:17 +03:00
# start a websocket server on 5002
WebSocketWSGIHandler.http_version = "1.1"
self.websocket_server = make_server(
"127.0.0.1",
5002,
server_class=WSGIServer,
handler_class=WebSocketWSGIRequestHandler,
2022-11-22 23:40:31 +03:00
app=WebSocketWSGIApplication(handler_cls=_WebSocketHandler),
2022-11-22 23:22:17 +03:00
)
self.websocket_server.initialize_websockets_manager()
self.websocket_thread = threading.Thread(
target=self.websocket_server.serve_forever
)
2022-11-22 23:29:28 +03:00
self.websocket_thread.start()
2022-11-22 23:22:17 +03:00
2022-11-22 23:51:08 +03:00
def publish(self, topic: str, payload: str, _) -> None:
2022-11-22 23:29:28 +03:00
try:
ws_message = json.dumps(
{
"topic": topic,
"payload": payload,
}
)
except Exception as e:
# if the payload can't be decoded don't relay to clients
logger.debug(f"payload for {topic} wasn't text. Skipping...")
return
2022-11-22 23:22:17 +03:00
2022-11-22 23:29:28 +03:00
self.websocket_server.manager.broadcast(ws_message)
2022-11-22 23:22:17 +03:00
def stop(self):
self.websocket_server.manager.close_all()
self.websocket_server.manager.stop()
self.websocket_server.manager.join()
self.websocket_server.shutdown()
2022-11-22 23:29:28 +03:00
self.websocket_thread.join()