playwright_rs/protocol/web_socket_route.rs
1//! WebSocketRoute protocol object — represents an intercepted WebSocket connection.
2//!
3//! `WebSocketRoute` is created by the Playwright server when a WebSocket connection
4//! matches a pattern registered via [`crate::protocol::Page::route_web_socket`] or
5//! [`crate::protocol::BrowserContext::route_web_socket`].
6//!
7//! # Example
8//!
9//! ```ignore
10//! use playwright_rs::protocol::Playwright;
11//!
12//! #[tokio::main]
13//! async fn main() -> Result<(), Box<dyn std::error::Error>> {
14//! let playwright = Playwright::launch().await?;
15//! let browser = playwright.chromium().launch().await?;
16//! let page = browser.new_page().await?;
17//!
18//! // Intercept all WebSocket connections and proxy them to the real server
19//! page.route_web_socket("ws://**", |route| {
20//! Box::pin(async move {
21//! route.connect_to_server().await?;
22//! Ok(())
23//! })
24//! })
25//! .await?;
26//!
27//! browser.close().await?;
28//! Ok(())
29//! }
30//! ```
31//!
32//! See: <https://playwright.dev/docs/api/class-websocketroute>
33
34use crate::error::Result;
35use crate::server::channel::Channel;
36use crate::server::channel_owner::{ChannelOwner, ChannelOwnerImpl, ParentOrConnection};
37use serde_json::Value;
38use std::any::Any;
39use std::future::Future;
40use std::pin::Pin;
41use std::sync::{Arc, Mutex};
42
43/// Represents an intercepted WebSocket connection.
44///
45/// `WebSocketRoute` is passed to handlers registered via
46/// [`crate::protocol::Page::route_web_socket`] or [`crate::protocol::BrowserContext::route_web_socket`].
47/// The handler must call [`connect_to_server`](WebSocketRoute::connect_to_server)
48/// to forward the connection to the real server, or [`close`](WebSocketRoute::close)
49/// to terminate it.
50///
51/// See: <https://playwright.dev/docs/api/class-websocketroute>
52#[derive(Clone)]
53pub struct WebSocketRoute {
54 base: ChannelOwnerImpl,
55 /// The WebSocket URL being intercepted.
56 url: String,
57 /// Message handlers registered via on_message().
58 message_handlers: Arc<Mutex<Vec<WebSocketRouteMessageHandler>>>,
59 /// Close handlers registered via on_close().
60 close_handlers: Arc<Mutex<Vec<WebSocketRouteCloseHandler>>>,
61}
62
63/// Type alias for boxed WebSocketRoute message handler future.
64type WebSocketRouteHandlerFuture = Pin<Box<dyn Future<Output = Result<()>> + Send>>;
65
66/// Message handler type.
67type WebSocketRouteMessageHandler =
68 Arc<dyn Fn(String) -> WebSocketRouteHandlerFuture + Send + Sync>;
69
70/// Close handler type.
71type WebSocketRouteCloseHandler = Arc<dyn Fn() -> WebSocketRouteHandlerFuture + Send + Sync>;
72
73impl WebSocketRoute {
74 /// Creates a new `WebSocketRoute` object.
75 pub fn new(
76 parent: Arc<dyn ChannelOwner>,
77 type_name: String,
78 guid: Arc<str>,
79 initializer: Value,
80 ) -> Result<Self> {
81 let url = initializer["url"].as_str().unwrap_or("").to_string();
82 let base = ChannelOwnerImpl::new(
83 ParentOrConnection::Parent(parent),
84 type_name,
85 guid,
86 initializer,
87 );
88 Ok(Self {
89 base,
90 url,
91 message_handlers: Arc::new(Mutex::new(Vec::new())),
92 close_handlers: Arc::new(Mutex::new(Vec::new())),
93 })
94 }
95
96 /// Returns the URL of the intercepted WebSocket connection.
97 ///
98 /// See: <https://playwright.dev/docs/api/class-websocketroute#web-socket-route-url>
99 pub fn url(&self) -> &str {
100 &self.url
101 }
102
103 /// Connects this WebSocket to the actual server.
104 ///
105 /// After calling this method, all messages sent by the page are forwarded to
106 /// the server, and all messages sent by the server are forwarded to the page.
107 ///
108 /// # Errors
109 ///
110 /// Returns an error if the RPC call fails.
111 ///
112 /// See: <https://playwright.dev/docs/api/class-websocketroute#web-socket-route-connect-to-server>
113 pub async fn connect_to_server(&self) -> Result<()> {
114 self.base
115 .channel()
116 .send_no_result("connectToServer", serde_json::json!({}))
117 .await
118 }
119
120 /// Closes the WebSocket connection.
121 ///
122 /// # Arguments
123 ///
124 /// * `options` — Optional close code and reason.
125 ///
126 /// See: <https://playwright.dev/docs/api/class-websocketroute#web-socket-route-close>
127 pub async fn close(&self, options: Option<WebSocketRouteCloseOptions>) -> Result<()> {
128 let opts = options.unwrap_or_default();
129 let mut params = serde_json::Map::new();
130 if let Some(code) = opts.code {
131 params.insert("code".to_string(), serde_json::json!(code));
132 }
133 if let Some(reason) = opts.reason {
134 params.insert("reason".to_string(), serde_json::json!(reason));
135 }
136 self.base
137 .channel()
138 .send_no_result("close", Value::Object(params))
139 .await
140 }
141
142 /// Sends a text message to the page.
143 ///
144 /// # Arguments
145 ///
146 /// * `message` — The text message to send.
147 ///
148 /// See: <https://playwright.dev/docs/api/class-websocketroute#web-socket-route-send>
149 pub async fn send(&self, message: &str) -> Result<()> {
150 self.base
151 .channel()
152 .send_no_result(
153 "sendToPage",
154 serde_json::json!({ "message": message, "isBase64": false }),
155 )
156 .await
157 }
158
159 /// Registers a handler for messages sent from the page.
160 ///
161 /// # Arguments
162 ///
163 /// * `handler` — Async closure that receives the message payload as a `String`.
164 ///
165 /// See: <https://playwright.dev/docs/api/class-websocketroute#web-socket-route-on-message>
166 pub async fn on_message<F>(&self, handler: F) -> Result<()>
167 where
168 F: Fn(String) -> WebSocketRouteHandlerFuture + Send + Sync + 'static,
169 {
170 let handler_arc = Arc::new(handler);
171 self.message_handlers.lock().unwrap().push(handler_arc);
172 Ok(())
173 }
174
175 /// Registers a handler for when the WebSocket is closed by the page.
176 ///
177 /// See: <https://playwright.dev/docs/api/class-websocketroute#web-socket-route-on-close>
178 pub async fn on_close<F>(&self, handler: F) -> Result<()>
179 where
180 F: Fn() -> WebSocketRouteHandlerFuture + Send + Sync + 'static,
181 {
182 let handler_arc = Arc::new(handler);
183 self.close_handlers.lock().unwrap().push(handler_arc);
184 Ok(())
185 }
186
187 /// Dispatches an incoming server-side event to registered handlers.
188 pub(crate) fn handle_event(&self, event: &str, params: &Value) {
189 match event {
190 "messageFromPage" => {
191 let payload = params["message"].as_str().unwrap_or("").to_string();
192 let handlers = self.message_handlers.lock().unwrap().clone();
193 for handler in handlers {
194 let p = payload.clone();
195 tokio::spawn(async move {
196 let _ = handler(p).await;
197 });
198 }
199 }
200 "close" => {
201 let handlers = self.close_handlers.lock().unwrap().clone();
202 for handler in handlers {
203 tokio::spawn(async move {
204 let _ = handler().await;
205 });
206 }
207 }
208 _ => {}
209 }
210 }
211}
212
213/// Options for [`WebSocketRoute::close`].
214#[derive(Debug, Default, Clone)]
215pub struct WebSocketRouteCloseOptions {
216 /// WebSocket close code (e.g. 1000 for normal closure).
217 pub code: Option<u16>,
218 /// Human-readable close reason.
219 pub reason: Option<String>,
220}
221
222impl ChannelOwner for WebSocketRoute {
223 fn guid(&self) -> &str {
224 self.base.guid()
225 }
226
227 fn type_name(&self) -> &str {
228 self.base.type_name()
229 }
230
231 fn parent(&self) -> Option<Arc<dyn ChannelOwner>> {
232 self.base.parent()
233 }
234
235 fn connection(&self) -> Arc<dyn crate::server::connection::ConnectionLike> {
236 self.base.connection()
237 }
238
239 fn initializer(&self) -> &Value {
240 self.base.initializer()
241 }
242
243 fn channel(&self) -> &Channel {
244 self.base.channel()
245 }
246
247 fn dispose(&self, reason: crate::server::channel_owner::DisposeReason) {
248 self.base.dispose(reason)
249 }
250
251 fn adopt(&self, child: Arc<dyn ChannelOwner>) {
252 self.base.adopt(child)
253 }
254
255 fn add_child(&self, guid: Arc<str>, child: Arc<dyn ChannelOwner>) {
256 self.base.add_child(guid, child)
257 }
258
259 fn remove_child(&self, guid: &str) {
260 self.base.remove_child(guid)
261 }
262
263 fn on_event(&self, method: &str, params: Value) {
264 self.handle_event(method, ¶ms);
265 self.base.on_event(method, params)
266 }
267
268 fn was_collected(&self) -> bool {
269 self.base.was_collected()
270 }
271
272 fn as_any(&self) -> &dyn Any {
273 self
274 }
275}