-
Notifications
You must be signed in to change notification settings - Fork 0
Expand file tree
/
Copy pathws_client.py
More file actions
92 lines (75 loc) · 2.88 KB
/
ws_client.py
File metadata and controls
92 lines (75 loc) · 2.88 KB
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
import json
import threading
from typing import Callable, Optional
from websocket import WebSocketApp
class WebSocketManager:
def __init__(
self,
logger,
on_message: Optional[Callable[[str], None]] = None,
on_status_change: Optional[Callable[[bool], None]] = None,
) -> None:
self.logger = logger
self.on_message = on_message
self.on_status_change = on_status_change
self.ws_app = None
self.ws_thread = None
self.connected = False
def connect(self, ws_url: str) -> None:
if self.connected:
self.logger.log("WebSocket is already connected.")
return
if not ws_url:
self.logger.log("WebSocket URL is empty.")
return
self.logger.log(f"Connecting to WebSocket: {ws_url}")
def run_ws() -> None:
self.ws_app = WebSocketApp(
ws_url,
on_open=self._on_open,
on_message=self._on_message,
on_error=self._on_error,
on_close=self._on_close,
)
try:
self.ws_app.run_forever()
except Exception as exc:
self.logger.log(f"WebSocket run_forever exception: {exc}")
self.ws_thread = threading.Thread(target=run_ws, daemon=True)
self.ws_thread.start()
def disconnect(self) -> None:
if self.ws_app is None:
self.logger.log("No active WebSocket connection.")
return
self.logger.log("Disconnecting WebSocket...")
try:
self.ws_app.close()
except Exception as exc:
self.logger.log(f"Disconnect error: {exc}")
def send_json(self, payload: dict) -> None:
if not self.connected or self.ws_app is None:
self.logger.log("Cannot send via WebSocket: not connected.")
return
try:
raw_payload = json.dumps(payload)
self.ws_app.send(raw_payload)
self.logger.log(f"Sent WebSocket message: {raw_payload}")
except Exception as exc:
self.logger.log(f"WebSocket send error: {exc}")
def _set_connected(self, value: bool) -> None:
self.connected = value
if self.on_status_change:
self.on_status_change(value)
def _on_open(self, _ws) -> None:
self._set_connected(True)
self.logger.log("WebSocket connected.")
def _on_message(self, _ws, message: str) -> None:
if self.on_message:
self.on_message(message)
else:
self.logger.log(f"WebSocket received: {message}")
def _on_error(self, _ws, error) -> None:
self.logger.log(f"WebSocket error: {error}")
def _on_close(self, _ws, close_status_code, close_msg) -> None:
self._set_connected(False)
self.logger.log(f"WebSocket closed. code={close_status_code}, message={close_msg}")