mirror of
https://github.com/denoland/deno.git
synced 2024-12-22 07:14:47 -05:00
6261c89e04
Add support for transferring `MessagePort`s between workers.
81 lines
2.3 KiB
Rust
81 lines
2.3 KiB
Rust
// Copyright 2018-2021 the Deno authors. All rights reserved. MIT license.
|
|
|
|
use crate::web_worker::WebWorkerInternalHandle;
|
|
use crate::web_worker::WorkerControlEvent;
|
|
use deno_core::error::generic_error;
|
|
use deno_core::error::AnyError;
|
|
use deno_core::op_async;
|
|
use deno_core::op_sync;
|
|
use deno_core::CancelFuture;
|
|
use deno_core::Extension;
|
|
use deno_core::OpState;
|
|
use deno_web::JsMessageData;
|
|
use std::cell::RefCell;
|
|
use std::rc::Rc;
|
|
|
|
pub fn init() -> Extension {
|
|
Extension::builder()
|
|
.ops(vec![
|
|
("op_worker_post_message", op_sync(op_worker_post_message)),
|
|
("op_worker_recv_message", op_async(op_worker_recv_message)),
|
|
// Notify host that guest worker closes.
|
|
("op_worker_close", op_sync(op_worker_close)),
|
|
// Notify host that guest worker has unhandled error.
|
|
(
|
|
"op_worker_unhandled_error",
|
|
op_sync(op_worker_unhandled_error),
|
|
),
|
|
])
|
|
.build()
|
|
}
|
|
|
|
fn op_worker_post_message(
|
|
state: &mut OpState,
|
|
data: JsMessageData,
|
|
_: (),
|
|
) -> Result<(), AnyError> {
|
|
let handle = state.borrow::<WebWorkerInternalHandle>().clone();
|
|
handle.port.send(state, data)?;
|
|
Ok(())
|
|
}
|
|
|
|
async fn op_worker_recv_message(
|
|
state: Rc<RefCell<OpState>>,
|
|
_: (),
|
|
_: (),
|
|
) -> Result<Option<JsMessageData>, AnyError> {
|
|
let handle = {
|
|
let state = state.borrow();
|
|
state.borrow::<WebWorkerInternalHandle>().clone()
|
|
};
|
|
handle
|
|
.port
|
|
.recv(state.clone())
|
|
.or_cancel(handle.cancel)
|
|
.await?
|
|
}
|
|
|
|
fn op_worker_close(state: &mut OpState, _: (), _: ()) -> Result<(), AnyError> {
|
|
// Notify parent that we're finished
|
|
let mut handle = state.borrow_mut::<WebWorkerInternalHandle>().clone();
|
|
|
|
handle.terminate();
|
|
Ok(())
|
|
}
|
|
|
|
/// A worker that encounters an uncaught error will pass this error
|
|
/// to its parent worker using this op. The parent worker will use
|
|
/// this same op to pass the error to its own parent (in case
|
|
/// `e.preventDefault()` was not called in `worker.onerror`). This
|
|
/// is done until the error reaches the root/ main worker.
|
|
fn op_worker_unhandled_error(
|
|
state: &mut OpState,
|
|
message: String,
|
|
_: (),
|
|
) -> Result<(), AnyError> {
|
|
let sender = state.borrow::<WebWorkerInternalHandle>().clone();
|
|
sender
|
|
.post_event(WorkerControlEvent::Error(generic_error(message)))
|
|
.expect("Failed to propagate error event to parent worker");
|
|
Ok(())
|
|
}
|