@@ -15,7 +15,7 @@ use cbc::{Decryptor, Encryptor};
15
15
use hmac:: { Hmac , Mac } ;
16
16
use log:: info;
17
17
use rand:: Rng ;
18
- use serde_json:: json;
18
+ use serde_json:: { json, Value } ;
19
19
use sha2:: Sha256 ;
20
20
use std:: collections:: HashMap ;
21
21
use std:: sync:: Arc ;
@@ -28,14 +28,15 @@ pub use config::PusherConfig;
28
28
pub use error:: { PusherError , PusherResult } ;
29
29
pub use events:: { Event , SystemEvent } ;
30
30
31
- use websocket:: WebSocketClient ;
31
+ use websocket:: { WebSocketClient , WebSocketCommand } ;
32
32
33
33
/// This struct provides methods for connecting to Pusher, subscribing to channels,
34
34
/// triggering events, and handling incoming events.
35
35
pub struct PusherClient {
36
36
config : PusherConfig ,
37
37
auth : PusherAuth ,
38
- websocket : Option < WebSocketClient > ,
38
+ // websocket: Option<WebSocketClient>,
39
+ websocket_command_tx : Option < mpsc:: Sender < WebSocketCommand > > ,
39
40
channels : Arc < RwLock < HashMap < String , Channel > > > ,
40
41
event_handlers : Arc < RwLock < HashMap < String , Vec < Box < dyn Fn ( Event ) + Send + Sync + ' static > > > > > ,
41
42
state : Arc < RwLock < ConnectionState > > ,
@@ -73,29 +74,44 @@ impl PusherClient {
73
74
let auth = PusherAuth :: new ( & config. app_key , & config. app_secret ) ;
74
75
let ( event_tx, event_rx) = mpsc:: channel ( 100 ) ;
75
76
let state = Arc :: new ( RwLock :: new ( ConnectionState :: Disconnected ) ) ;
76
- let event_handlers = Arc :: new ( RwLock :: new ( HashMap :: new ( ) ) ) ;
77
- let encrypted_channels = Arc :: new ( RwLock :: new ( HashMap :: new ( ) ) ) ;
77
+ let event_handlers = Arc :: new ( RwLock :: new ( std :: collections :: HashMap :: new ( ) ) ) ;
78
+ let encrypted_channels = Arc :: new ( RwLock :: new ( std :: collections :: HashMap :: new ( ) ) ) ;
78
79
79
80
let client = Self {
80
81
config,
81
82
auth,
82
- websocket : None ,
83
- channels : Arc :: new ( RwLock :: new ( HashMap :: new ( ) ) ) ,
83
+ websocket_command_tx : None ,
84
+ channels : Arc :: new ( RwLock :: new ( std :: collections :: HashMap :: new ( ) ) ) ,
84
85
event_handlers : event_handlers. clone ( ) ,
85
86
state : state. clone ( ) ,
86
87
event_tx,
87
88
encrypted_channels,
88
89
} ;
89
90
90
- // Spawn the event handling task
91
91
tokio:: spawn ( Self :: handle_events ( event_rx, event_handlers) ) ;
92
92
93
93
Ok ( client)
94
94
}
95
+
96
+ async fn send ( & self , message : String ) -> PusherResult < ( ) > {
97
+ if let Some ( tx) = & self . websocket_command_tx {
98
+ tx. send ( WebSocketCommand :: Send ( message) )
99
+ . await
100
+ . map_err ( |e| {
101
+ PusherError :: WebSocketError ( format ! ( "Failed to send command: {}" , e) )
102
+ } ) ?;
103
+ Ok ( ( ) )
104
+ } else {
105
+ Err ( PusherError :: ConnectionError ( "Not connected" . into ( ) ) )
106
+ }
107
+ }
108
+
95
109
async fn handle_events (
96
110
mut event_rx : mpsc:: Receiver < Event > ,
97
111
event_handlers : Arc <
98
- RwLock < HashMap < String , Vec < Box < dyn Fn ( Event ) + Send + Sync + ' static > > > > ,
112
+ RwLock <
113
+ std:: collections:: HashMap < String , Vec < Box < dyn Fn ( Event ) + Send + Sync + ' static > > > ,
114
+ > ,
99
115
> ,
100
116
) {
101
117
while let Some ( event) = event_rx. recv ( ) . await {
@@ -115,18 +131,24 @@ impl PusherClient {
115
131
/// A `PusherResult` indicating success or failure.
116
132
pub async fn connect ( & mut self ) -> PusherResult < ( ) > {
117
133
let url = self . get_websocket_url ( ) ?;
118
- let mut websocket =
119
- WebSocketClient :: new ( url. clone ( ) , Arc :: clone ( & self . state ) , self . event_tx . clone ( ) ) ;
134
+ let ( command_tx, command_rx) = mpsc:: channel ( 100 ) ;
135
+
136
+ let mut websocket = WebSocketClient :: new (
137
+ url. clone ( ) ,
138
+ Arc :: clone ( & self . state ) ,
139
+ self . event_tx . clone ( ) ,
140
+ command_rx,
141
+ ) ;
142
+
120
143
log:: info!( "Connecting to Pusher using URL: {}" , url) ;
121
144
websocket. connect ( ) . await ?;
122
- self . websocket = Some ( websocket) ;
123
145
124
- // Start the WebSocket event loop
125
- let mut ws = self . websocket . take ( ) . unwrap ( ) ;
126
146
tokio:: spawn ( async move {
127
- ws . run ( ) . await ;
147
+ websocket . run ( ) . await ;
128
148
} ) ;
129
149
150
+ self . websocket_command_tx = Some ( command_tx) ;
151
+
130
152
Ok ( ( ) )
131
153
}
132
154
@@ -136,11 +158,12 @@ impl PusherClient {
136
158
///
137
159
/// A `PusherResult` indicating success or failure.
138
160
pub async fn disconnect ( & mut self ) -> PusherResult < ( ) > {
139
- if let Some ( websocket) = & self . websocket {
140
- websocket. close ( ) . await ?;
161
+ if let Some ( tx) = self . websocket_command_tx . take ( ) {
162
+ tx. send ( WebSocketCommand :: Close ) . await . map_err ( |e| {
163
+ PusherError :: WebSocketError ( format ! ( "Failed to send close command: {}" , e) )
164
+ } ) ?;
141
165
}
142
166
* self . state . write ( ) . await = ConnectionState :: Disconnected ;
143
- self . websocket = None ;
144
167
Ok ( ( ) )
145
168
}
146
169
@@ -158,21 +181,17 @@ impl PusherClient {
158
181
let mut channels = self . channels . write ( ) . await ;
159
182
channels. insert ( channel_name. to_string ( ) , channel) ;
160
183
161
- if let Some ( websocket) = & self . websocket {
162
- let data = json ! ( {
163
- "event" : "pusher:subscribe" ,
164
- "data" : {
165
- "channel" : channel_name
166
- }
167
- } ) ;
168
- websocket. send ( serde_json:: to_string ( & data) ?) . await ?;
169
- } else {
170
- return Err ( PusherError :: ConnectionError ( "Not connected" . into ( ) ) ) ;
171
- }
184
+ let data = json ! ( {
185
+ "event" : "pusher:subscribe" ,
186
+ "data" : {
187
+ "channel" : channel_name
188
+ }
189
+ } ) ;
172
190
173
- Ok ( ( ) )
191
+ self . send ( serde_json :: to_string ( & data ) ? ) . await
174
192
}
175
193
194
+
176
195
/// Subscribes to an encrypted channel.
177
196
///
178
197
/// # Arguments
@@ -208,6 +227,7 @@ impl PusherClient {
208
227
/// # Returns
209
228
///
210
229
/// A `PusherResult` indicating success or failure.
230
+ ///
211
231
pub async fn unsubscribe ( & mut self , channel_name : & str ) -> PusherResult < ( ) > {
212
232
{
213
233
let mut channels = self . channels . write ( ) . await ;
@@ -219,19 +239,14 @@ impl PusherClient {
219
239
encrypted_channels. remove ( channel_name) ;
220
240
}
221
241
222
- if let Some ( websocket) = & self . websocket {
223
- let data = json ! ( {
224
- "event" : "pusher:unsubscribe" ,
225
- "data" : {
226
- "channel" : channel_name
227
- }
228
- } ) ;
229
- websocket. send ( serde_json:: to_string ( & data) ?) . await ?;
230
- } else {
231
- return Err ( PusherError :: ConnectionError ( "Not connected" . into ( ) ) ) ;
232
- }
242
+ let data = json ! ( {
243
+ "event" : "pusher:unsubscribe" ,
244
+ "data" : {
245
+ "channel" : channel_name
246
+ }
247
+ } ) ;
233
248
234
- Ok ( ( ) )
249
+ self . send ( serde_json :: to_string ( & data ) ? ) . await
235
250
}
236
251
237
252
/// Triggers an event on a channel.
@@ -251,10 +266,14 @@ impl PusherClient {
251
266
self . config. cluster, self . config. app_id
252
267
) ;
253
268
269
+ // Validate that the data is valid JSON, but keep it as a string
270
+ serde_json:: from_str :: < serde_json:: Value > ( data)
271
+ . map_err ( |e| PusherError :: JsonError ( e) ) ?;
272
+
254
273
let body = json ! ( {
255
274
"name" : event,
256
275
"channel" : channel,
257
- "data" : data
276
+ "data" : data, // Keep data as a string
258
277
} ) ;
259
278
let path = format ! ( "/apps/{}/events" , self . config. app_id) ;
260
279
let auth_params = self . auth . authenticate_request ( "POST" , & path, & body) ?;
@@ -371,6 +390,7 @@ impl PusherClient {
371
390
/// # Returns
372
391
///
373
392
/// A `PusherResult` indicating success or failure.
393
+ ///
374
394
pub async fn bind < F > ( & self , event_name : & str , callback : F ) -> PusherResult < ( ) >
375
395
where
376
396
F : Fn ( Event ) + Send + Sync + ' static ,
@@ -535,7 +555,8 @@ mod tests {
535
555
536
556
#[ tokio:: test]
537
557
async fn test_trigger_batch ( ) {
538
- let config = PusherConfig :: from_env ( ) . expect ( "Failed to load Pusher configuration from environment" ) ;
558
+ let config =
559
+ PusherConfig :: from_env ( ) . expect ( "Failed to load Pusher configuration from environment" ) ;
539
560
let client = PusherClient :: new ( config) . unwrap ( ) ;
540
561
541
562
let batch_events = vec ! [
0 commit comments