1
0
Fork 0
mirror of https://github.com/denoland/deno.git synced 2025-01-20 20:42:19 -05:00

Compare commits

...

4 commits

Author SHA1 Message Date
Divy Srivastava
b0991fbde1
Merge 1fa8905401 into e4a16e91fa 2025-01-20 18:10:13 +05:30
ryu
e4a16e91fa
docs(readme): update redirected links (#27726) 2025-01-20 03:01:25 +00:00
Divy Srivastava
1fa8905401 Lint 2024-12-26 13:49:47 +05:30
Divy Srivastava
5f5cc5a707 fix(ext/websocket): cancel op_ws_next_event leak on close 2024-12-26 12:11:44 +05:30
5 changed files with 43 additions and 17 deletions

View file

@ -6,8 +6,8 @@
<img align="right" src="https://deno.land/logo.svg" height="150px" alt="the deno mascot dinosaur standing in the rain">
[Deno](https://www.deno.com)
([/ˈdiːnoʊ/](http://ipa-reader.xyz/?text=%CB%88di%CB%90no%CA%8A), pronounced
[Deno](https://deno.com)
([/ˈdiːnoʊ/](https://ipa-reader.com/?text=%CB%88di%CB%90no%CA%8A), pronounced
`dee-no`) is a JavaScript, TypeScript, and WebAssembly runtime with secure
defaults and a great developer experience. It's built on [V8](https://v8.dev/),
[Rust](https://www.rust-lang.org/), and [Tokio](https://tokio.rs/).

View file

@ -429,6 +429,7 @@ class WebSocket extends EventTarget {
const rid = this[_rid];
while (this[_readyState] !== CLOSED) {
const kind = await op_ws_next_event(rid);
/* close the connection if read was cancelled, and we didn't get a close frame */
if (
(this[_readyState] == CLOSING) &&
@ -442,6 +443,10 @@ class WebSocket extends EventTarget {
break;
}
if (kind == null) {
break;
}
switch (kind) {
case 0: {
/* string */

View file

@ -16,6 +16,7 @@ use deno_core::url;
use deno_core::AsyncMutFuture;
use deno_core::AsyncRefCell;
use deno_core::ByteString;
use deno_core::CancelFuture;
use deno_core::CancelHandle;
use deno_core::CancelTryFuture;
use deno_core::JsBuffer;
@ -552,6 +553,7 @@ pub struct ServerWebSocket {
string: Cell<Option<String>>,
ws_read: AsyncRefCell<FragmentCollectorRead<ReadHalf<WebSocketStream>>>,
ws_write: AsyncRefCell<WebSocketWrite<WriteHalf<WebSocketStream>>>,
cancel_handle: Rc<CancelHandle>,
}
impl ServerWebSocket {
@ -566,6 +568,7 @@ impl ServerWebSocket {
string: Cell::new(None),
ws_read: AsyncRefCell::new(FragmentCollectorRead::new(ws_read)),
ws_write: AsyncRefCell::new(ws_write),
cancel_handle: CancelHandle::new_rc(),
}
}
@ -769,7 +772,7 @@ pub async fn op_ws_close(
let Ok(resource) = state
.borrow_mut()
.resource_table
.get::<ServerWebSocket>(rid)
.take::<ServerWebSocket>(rid)
else {
return Ok(());
};
@ -784,6 +787,8 @@ pub async fn op_ws_close(
});
resource.closed.set(true);
resource.cancel_handle.cancel();
let lock = resource.reserve_lock();
resource.write_frame(lock, frame).await
}
@ -826,19 +831,19 @@ pub fn op_ws_get_error(state: &mut OpState, #[smi] rid: ResourceId) -> String {
pub async fn op_ws_next_event(
state: Rc<RefCell<OpState>>,
#[smi] rid: ResourceId,
) -> u16 {
) -> Option<u16> {
let Ok(resource) = state
.borrow_mut()
.resource_table
.get::<ServerWebSocket>(rid)
else {
// op_ws_get_error will correctly handle a bad resource
return MessageKind::Error as u16;
return Some(MessageKind::Error as u16);
};
// If there's a pending error, this always returns error
if resource.errored.get() {
return MessageKind::Error as u16;
return Some(MessageKind::Error as u16);
}
let mut ws = RcRef::map(&resource, |r| &r.ws_read).borrow_mut().await;
@ -847,19 +852,26 @@ pub async fn op_ws_next_event(
let writer = writer.clone();
async move { writer.borrow_mut().await.write_frame(frame).await }
};
let cancel_handle = resource.cancel_handle.clone();
loop {
let res = ws.read_frame(&mut sender).await;
let Ok(res) = ws
.read_frame(&mut sender)
.or_cancel(cancel_handle.clone())
.await
else {
return None;
};
let val = match res {
Ok(val) => val,
Err(err) => {
// No message was received, socket closed while we waited.
// Report closed status to JavaScript.
if resource.closed.get() {
return MessageKind::ClosedDefault as u16;
return Some(MessageKind::ClosedDefault as u16);
}
resource.set_error(Some(err.to_string()));
return MessageKind::Error as u16;
return Some(MessageKind::Error as u16);
}
};
@ -867,22 +879,22 @@ pub async fn op_ws_next_event(
OpCode::Text => match String::from_utf8(val.payload.to_vec()) {
Ok(s) => {
resource.string.set(Some(s));
MessageKind::Text as u16
Some(MessageKind::Text as u16)
}
Err(_) => {
resource.set_error(Some("Invalid string data".into()));
MessageKind::Error as u16
Some(MessageKind::Error as u16)
}
},
OpCode::Binary => {
resource.buffer.set(Some(val.payload.to_vec()));
MessageKind::Binary as u16
Some(MessageKind::Binary as u16)
}
OpCode::Close => {
// Close reason is returned through error
if val.payload.len() < 2 {
resource.set_error(None);
MessageKind::ClosedDefault as u16
Some(MessageKind::ClosedDefault as u16)
} else {
let close_code = CloseCode::from(u16::from_be_bytes([
val.payload[0],
@ -890,10 +902,10 @@ pub async fn op_ws_next_event(
]));
let reason = String::from_utf8(val.payload[2..].to_vec()).ok();
resource.set_error(reason);
close_code.into()
Some(close_code.into())
}
}
OpCode::Pong => MessageKind::Pong as u16,
OpCode::Pong => Some(MessageKind::Pong as u16),
OpCode::Continuation | OpCode::Ping => {
continue;
}

View file

@ -822,3 +822,12 @@ Deno.test("send to a closed socket", async () => {
};
await promise;
});
Deno.test(async function websocketDoesntLeak() {
const { promise, resolve } = Promise.withResolvers<void>();
const ws = new WebSocket(new URL("ws://localhost:4242/"));
assertEquals(ws.url, "ws://localhost:4242/");
ws.onopen = () => resolve();
await promise;
ws.close();
});

View file

@ -48,8 +48,8 @@ const packages: Package[] = [{
const markdownText = `# Deno
[Deno](https://www.deno.com)
([/ˈdiːnoʊ/](http://ipa-reader.xyz/?text=%CB%88di%CB%90no%CA%8A), pronounced
[Deno](https://deno.com)
([/ˈdiːnoʊ/](https://ipa-reader.com/?text=%CB%88di%CB%90no%CA%8A), pronounced
\`dee-no\`) is a JavaScript, TypeScript, and WebAssembly runtime with secure
defaults and a great developer experience. It's built on [V8](https://v8.dev/),
[Rust](https://www.rust-lang.org/), and [Tokio](https://tokio.rs/).